mem_openbsd.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. // +build openbsd
  2. package mem
  3. import (
  4. "bytes"
  5. "context"
  6. "encoding/binary"
  7. "errors"
  8. "fmt"
  9. "os/exec"
  10. "github.com/shirou/gopsutil/internal/common"
  11. )
  12. func GetPageSize() (uint64, error) {
  13. return GetPageSizeWithContext(context.Background())
  14. }
  15. func GetPageSizeWithContext(ctx context.Context) (uint64, error) {
  16. mib := []int32{CTLVm, VmUvmexp}
  17. buf, length, err := common.CallSyscall(mib)
  18. if err != nil {
  19. return 0, err
  20. }
  21. if length < sizeOfUvmexp {
  22. return 0, fmt.Errorf("short syscall ret %d bytes", length)
  23. }
  24. var uvmexp Uvmexp
  25. br := bytes.NewReader(buf)
  26. err = common.Read(br, binary.LittleEndian, &uvmexp)
  27. if err != nil {
  28. return 0, err
  29. }
  30. return uint64(uvmexp.Pagesize), nil
  31. }
  32. func VirtualMemory() (*VirtualMemoryStat, error) {
  33. return VirtualMemoryWithContext(context.Background())
  34. }
  35. func VirtualMemoryWithContext(ctx context.Context) (*VirtualMemoryStat, error) {
  36. mib := []int32{CTLVm, VmUvmexp}
  37. buf, length, err := common.CallSyscall(mib)
  38. if err != nil {
  39. return nil, err
  40. }
  41. if length < sizeOfUvmexp {
  42. return nil, fmt.Errorf("short syscall ret %d bytes", length)
  43. }
  44. var uvmexp Uvmexp
  45. br := bytes.NewReader(buf)
  46. err = common.Read(br, binary.LittleEndian, &uvmexp)
  47. if err != nil {
  48. return nil, err
  49. }
  50. p := uint64(uvmexp.Pagesize)
  51. ret := &VirtualMemoryStat{
  52. Total: uint64(uvmexp.Npages) * p,
  53. Free: uint64(uvmexp.Free) * p,
  54. Active: uint64(uvmexp.Active) * p,
  55. Inactive: uint64(uvmexp.Inactive) * p,
  56. Cached: 0, // not available
  57. Wired: uint64(uvmexp.Wired) * p,
  58. }
  59. ret.Available = ret.Inactive + ret.Cached + ret.Free
  60. ret.Used = ret.Total - ret.Available
  61. ret.UsedPercent = float64(ret.Used) / float64(ret.Total) * 100.0
  62. mib = []int32{CTLVfs, VfsGeneric, VfsBcacheStat}
  63. buf, length, err = common.CallSyscall(mib)
  64. if err != nil {
  65. return nil, err
  66. }
  67. if length < sizeOfBcachestats {
  68. return nil, fmt.Errorf("short syscall ret %d bytes", length)
  69. }
  70. var bcs Bcachestats
  71. br = bytes.NewReader(buf)
  72. err = common.Read(br, binary.LittleEndian, &bcs)
  73. if err != nil {
  74. return nil, err
  75. }
  76. ret.Buffers = uint64(bcs.Numbufpages) * p
  77. return ret, nil
  78. }
  79. // Return swapctl summary info
  80. func SwapMemory() (*SwapMemoryStat, error) {
  81. return SwapMemoryWithContext(context.Background())
  82. }
  83. func SwapMemoryWithContext(ctx context.Context) (*SwapMemoryStat, error) {
  84. swapctl, err := exec.LookPath("swapctl")
  85. if err != nil {
  86. return nil, err
  87. }
  88. out, err := invoke.CommandWithContext(ctx, swapctl, "-sk")
  89. if err != nil {
  90. return &SwapMemoryStat{}, nil
  91. }
  92. line := string(out)
  93. var total, used, free uint64
  94. _, err = fmt.Sscanf(line,
  95. "total: %d 1K-blocks allocated, %d used, %d available",
  96. &total, &used, &free)
  97. if err != nil {
  98. return nil, errors.New("failed to parse swapctl output")
  99. }
  100. percent := float64(used) / float64(total) * 100
  101. return &SwapMemoryStat{
  102. Total: total * 1024,
  103. Used: used * 1024,
  104. Free: free * 1024,
  105. UsedPercent: percent,
  106. }, nil
  107. }