disk_freebsd.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. // +build freebsd
  2. package disk
  3. import (
  4. "bytes"
  5. "context"
  6. "encoding/binary"
  7. "path"
  8. "strconv"
  9. "unsafe"
  10. "golang.org/x/sys/unix"
  11. "github.com/shirou/gopsutil/internal/common"
  12. )
  13. func Partitions(all bool) ([]PartitionStat, error) {
  14. return PartitionsWithContext(context.Background(), all)
  15. }
  16. func PartitionsWithContext(ctx context.Context, all bool) ([]PartitionStat, error) {
  17. var ret []PartitionStat
  18. // get length
  19. count, err := unix.Getfsstat(nil, MNT_WAIT)
  20. if err != nil {
  21. return ret, err
  22. }
  23. fs := make([]Statfs, count)
  24. if _, err = Getfsstat(fs, MNT_WAIT); err != nil {
  25. return ret, err
  26. }
  27. for _, stat := range fs {
  28. opts := "rw"
  29. if stat.Flags&MNT_RDONLY != 0 {
  30. opts = "ro"
  31. }
  32. if stat.Flags&MNT_SYNCHRONOUS != 0 {
  33. opts += ",sync"
  34. }
  35. if stat.Flags&MNT_NOEXEC != 0 {
  36. opts += ",noexec"
  37. }
  38. if stat.Flags&MNT_NOSUID != 0 {
  39. opts += ",nosuid"
  40. }
  41. if stat.Flags&MNT_UNION != 0 {
  42. opts += ",union"
  43. }
  44. if stat.Flags&MNT_ASYNC != 0 {
  45. opts += ",async"
  46. }
  47. if stat.Flags&MNT_SUIDDIR != 0 {
  48. opts += ",suiddir"
  49. }
  50. if stat.Flags&MNT_SOFTDEP != 0 {
  51. opts += ",softdep"
  52. }
  53. if stat.Flags&MNT_NOSYMFOLLOW != 0 {
  54. opts += ",nosymfollow"
  55. }
  56. if stat.Flags&MNT_GJOURNAL != 0 {
  57. opts += ",gjounalc"
  58. }
  59. if stat.Flags&MNT_MULTILABEL != 0 {
  60. opts += ",multilabel"
  61. }
  62. if stat.Flags&MNT_ACLS != 0 {
  63. opts += ",acls"
  64. }
  65. if stat.Flags&MNT_NOATIME != 0 {
  66. opts += ",noattime"
  67. }
  68. if stat.Flags&MNT_NOCLUSTERR != 0 {
  69. opts += ",nocluster"
  70. }
  71. if stat.Flags&MNT_NOCLUSTERW != 0 {
  72. opts += ",noclusterw"
  73. }
  74. if stat.Flags&MNT_NFS4ACLS != 0 {
  75. opts += ",nfs4acls"
  76. }
  77. d := PartitionStat{
  78. Device: common.IntToString(stat.Mntfromname[:]),
  79. Mountpoint: common.IntToString(stat.Mntonname[:]),
  80. Fstype: common.IntToString(stat.Fstypename[:]),
  81. Opts: opts,
  82. }
  83. if all == false {
  84. if !path.IsAbs(d.Device) || !common.PathExists(d.Device) {
  85. continue
  86. }
  87. }
  88. ret = append(ret, d)
  89. }
  90. return ret, nil
  91. }
  92. func IOCounters(names ...string) (map[string]IOCountersStat, error) {
  93. return IOCountersWithContext(context.Background(), names...)
  94. }
  95. func IOCountersWithContext(ctx context.Context, names ...string) (map[string]IOCountersStat, error) {
  96. // statinfo->devinfo->devstat
  97. // /usr/include/devinfo.h
  98. ret := make(map[string]IOCountersStat)
  99. r, err := unix.Sysctl("kern.devstat.all")
  100. if err != nil {
  101. return nil, err
  102. }
  103. buf := []byte(r)
  104. length := len(buf)
  105. count := int(uint64(length) / uint64(sizeOfDevstat))
  106. buf = buf[8:] // devstat.all has version in the head.
  107. // parse buf to Devstat
  108. for i := 0; i < count; i++ {
  109. b := buf[i*sizeOfDevstat : i*sizeOfDevstat+sizeOfDevstat]
  110. d, err := parseDevstat(b)
  111. if err != nil {
  112. continue
  113. }
  114. un := strconv.Itoa(int(d.Unit_number))
  115. name := common.IntToString(d.Device_name[:]) + un
  116. if len(names) > 0 && !common.StringsHas(names, name) {
  117. continue
  118. }
  119. ds := IOCountersStat{
  120. ReadCount: d.Operations[DEVSTAT_READ],
  121. WriteCount: d.Operations[DEVSTAT_WRITE],
  122. ReadBytes: d.Bytes[DEVSTAT_READ],
  123. WriteBytes: d.Bytes[DEVSTAT_WRITE],
  124. ReadTime: uint64(d.Duration[DEVSTAT_READ].Compute() * 1000),
  125. WriteTime: uint64(d.Duration[DEVSTAT_WRITE].Compute() * 1000),
  126. IoTime: uint64(d.Busy_time.Compute() * 1000),
  127. Name: name,
  128. }
  129. ret[name] = ds
  130. }
  131. return ret, nil
  132. }
  133. func (b Bintime) Compute() float64 {
  134. BINTIME_SCALE := 5.42101086242752217003726400434970855712890625e-20
  135. return float64(b.Sec) + float64(b.Frac)*BINTIME_SCALE
  136. }
  137. // BT2LD(time) ((long double)(time).sec + (time).frac * BINTIME_SCALE)
  138. // Getfsstat is borrowed from pkg/syscall/syscall_freebsd.go
  139. // change Statfs_t to Statfs in order to get more information
  140. func Getfsstat(buf []Statfs, flags int) (n int, err error) {
  141. return GetfsstatWithContext(context.Background(), buf, flags)
  142. }
  143. func GetfsstatWithContext(ctx context.Context, buf []Statfs, flags int) (n int, err error) {
  144. var _p0 unsafe.Pointer
  145. var bufsize uintptr
  146. if len(buf) > 0 {
  147. _p0 = unsafe.Pointer(&buf[0])
  148. bufsize = unsafe.Sizeof(Statfs{}) * uintptr(len(buf))
  149. }
  150. r0, _, e1 := unix.Syscall(unix.SYS_GETFSSTAT, uintptr(_p0), bufsize, uintptr(flags))
  151. n = int(r0)
  152. if e1 != 0 {
  153. err = e1
  154. }
  155. return
  156. }
  157. func parseDevstat(buf []byte) (Devstat, error) {
  158. var ds Devstat
  159. br := bytes.NewReader(buf)
  160. // err := binary.Read(br, binary.LittleEndian, &ds)
  161. err := common.Read(br, binary.LittleEndian, &ds)
  162. if err != nil {
  163. return ds, err
  164. }
  165. return ds, nil
  166. }
  167. func getFsType(stat unix.Statfs_t) string {
  168. return common.IntToString(stat.Fstypename[:])
  169. }