disk_openbsd.go 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. // +build openbsd
  2. package disk
  3. import (
  4. "bytes"
  5. "context"
  6. "encoding/binary"
  7. "path"
  8. "unsafe"
  9. "github.com/shirou/gopsutil/internal/common"
  10. "golang.org/x/sys/unix"
  11. )
  12. func Partitions(all bool) ([]PartitionStat, error) {
  13. return PartitionsWithContext(context.Background(), all)
  14. }
  15. func PartitionsWithContext(ctx context.Context, all bool) ([]PartitionStat, error) {
  16. var ret []PartitionStat
  17. // get length
  18. count, err := unix.Getfsstat(nil, MNT_WAIT)
  19. if err != nil {
  20. return ret, err
  21. }
  22. fs := make([]Statfs, count)
  23. if _, err = Getfsstat(fs, MNT_WAIT); err != nil {
  24. return ret, err
  25. }
  26. for _, stat := range fs {
  27. opts := "rw"
  28. if stat.F_flags&MNT_RDONLY != 0 {
  29. opts = "ro"
  30. }
  31. if stat.F_flags&MNT_SYNCHRONOUS != 0 {
  32. opts += ",sync"
  33. }
  34. if stat.F_flags&MNT_NOEXEC != 0 {
  35. opts += ",noexec"
  36. }
  37. if stat.F_flags&MNT_NOSUID != 0 {
  38. opts += ",nosuid"
  39. }
  40. if stat.F_flags&MNT_NODEV != 0 {
  41. opts += ",nodev"
  42. }
  43. if stat.F_flags&MNT_ASYNC != 0 {
  44. opts += ",async"
  45. }
  46. d := PartitionStat{
  47. Device: common.IntToString(stat.F_mntfromname[:]),
  48. Mountpoint: common.IntToString(stat.F_mntonname[:]),
  49. Fstype: common.IntToString(stat.F_fstypename[:]),
  50. Opts: opts,
  51. }
  52. if all == false {
  53. if !path.IsAbs(d.Device) || !common.PathExists(d.Device) {
  54. continue
  55. }
  56. }
  57. ret = append(ret, d)
  58. }
  59. return ret, nil
  60. }
  61. func IOCounters(names ...string) (map[string]IOCountersStat, error) {
  62. return IOCountersWithContext(context.Background(), names...)
  63. }
  64. func IOCountersWithContext(ctx context.Context, names ...string) (map[string]IOCountersStat, error) {
  65. ret := make(map[string]IOCountersStat)
  66. r, err := unix.SysctlRaw("hw.diskstats")
  67. if err != nil {
  68. return nil, err
  69. }
  70. buf := []byte(r)
  71. length := len(buf)
  72. count := int(uint64(length) / uint64(sizeOfDiskstats))
  73. // parse buf to Diskstats
  74. for i := 0; i < count; i++ {
  75. b := buf[i*sizeOfDiskstats : i*sizeOfDiskstats+sizeOfDiskstats]
  76. d, err := parseDiskstats(b)
  77. if err != nil {
  78. continue
  79. }
  80. name := common.IntToString(d.Name[:])
  81. if len(names) > 0 && !common.StringsHas(names, name) {
  82. continue
  83. }
  84. ds := IOCountersStat{
  85. ReadCount: d.Rxfer,
  86. WriteCount: d.Wxfer,
  87. ReadBytes: d.Rbytes,
  88. WriteBytes: d.Wbytes,
  89. Name: name,
  90. }
  91. ret[name] = ds
  92. }
  93. return ret, nil
  94. }
  95. // BT2LD(time) ((long double)(time).sec + (time).frac * BINTIME_SCALE)
  96. // Getfsstat is borrowed from pkg/syscall/syscall_freebsd.go
  97. // change Statfs_t to Statfs in order to get more information
  98. func Getfsstat(buf []Statfs, flags int) (n int, err error) {
  99. return GetfsstatWithContext(context.Background(), buf, flags)
  100. }
  101. func GetfsstatWithContext(ctx context.Context, buf []Statfs, flags int) (n int, err error) {
  102. var _p0 unsafe.Pointer
  103. var bufsize uintptr
  104. if len(buf) > 0 {
  105. _p0 = unsafe.Pointer(&buf[0])
  106. bufsize = unsafe.Sizeof(Statfs{}) * uintptr(len(buf))
  107. }
  108. r0, _, e1 := unix.Syscall(unix.SYS_GETFSSTAT, uintptr(_p0), bufsize, uintptr(flags))
  109. n = int(r0)
  110. if e1 != 0 {
  111. err = e1
  112. }
  113. return
  114. }
  115. func parseDiskstats(buf []byte) (Diskstats, error) {
  116. var ds Diskstats
  117. br := bytes.NewReader(buf)
  118. // err := binary.Read(br, binary.LittleEndian, &ds)
  119. err := common.Read(br, binary.LittleEndian, &ds)
  120. if err != nil {
  121. return ds, err
  122. }
  123. return ds, nil
  124. }
  125. func Usage(path string) (*UsageStat, error) {
  126. return UsageWithContext(context.Background(), path)
  127. }
  128. func UsageWithContext(ctx context.Context, path string) (*UsageStat, error) {
  129. stat := unix.Statfs_t{}
  130. err := unix.Statfs(path, &stat)
  131. if err != nil {
  132. return nil, err
  133. }
  134. bsize := stat.F_bsize
  135. ret := &UsageStat{
  136. Path: path,
  137. Fstype: getFsType(stat),
  138. Total: (uint64(stat.F_blocks) * uint64(bsize)),
  139. Free: (uint64(stat.F_bavail) * uint64(bsize)),
  140. InodesTotal: (uint64(stat.F_files)),
  141. InodesFree: (uint64(stat.F_ffree)),
  142. }
  143. ret.InodesUsed = (ret.InodesTotal - ret.InodesFree)
  144. ret.InodesUsedPercent = (float64(ret.InodesUsed) / float64(ret.InodesTotal)) * 100.0
  145. ret.Used = (uint64(stat.F_blocks) - uint64(stat.F_bfree)) * uint64(bsize)
  146. ret.UsedPercent = (float64(ret.Used) / float64(ret.Total)) * 100.0
  147. return ret, nil
  148. }
  149. func getFsType(stat unix.Statfs_t) string {
  150. return common.IntToString(stat.F_fstypename[:])
  151. }