disk_solaris.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. // +build solaris
  2. package disk
  3. import (
  4. "bufio"
  5. "context"
  6. "fmt"
  7. "math"
  8. "os"
  9. "strings"
  10. "github.com/shirou/gopsutil/internal/common"
  11. "golang.org/x/sys/unix"
  12. )
  13. const (
  14. // _DEFAULT_NUM_MOUNTS is set to `cat /etc/mnttab | wc -l` rounded up to the
  15. // nearest power of two.
  16. _DEFAULT_NUM_MOUNTS = 32
  17. // _MNTTAB default place to read mount information
  18. _MNTTAB = "/etc/mnttab"
  19. )
  20. var (
  21. // A blacklist of read-only virtual filesystems. Writable filesystems are of
  22. // operational concern and must not be included in this list.
  23. fsTypeBlacklist = map[string]struct{}{
  24. "ctfs": struct{}{},
  25. "dev": struct{}{},
  26. "fd": struct{}{},
  27. "lofs": struct{}{},
  28. "lxproc": struct{}{},
  29. "mntfs": struct{}{},
  30. "objfs": struct{}{},
  31. "proc": struct{}{},
  32. }
  33. )
  34. func Partitions(all bool) ([]PartitionStat, error) {
  35. return PartitionsWithContext(context.Background(), all)
  36. }
  37. func PartitionsWithContext(ctx context.Context, all bool) ([]PartitionStat, error) {
  38. ret := make([]PartitionStat, 0, _DEFAULT_NUM_MOUNTS)
  39. // Scan mnttab(4)
  40. f, err := os.Open(_MNTTAB)
  41. if err != nil {
  42. }
  43. defer func() {
  44. if err == nil {
  45. err = f.Close()
  46. } else {
  47. f.Close()
  48. }
  49. }()
  50. scanner := bufio.NewScanner(f)
  51. for scanner.Scan() {
  52. fields := strings.Split(scanner.Text(), "\t")
  53. if _, found := fsTypeBlacklist[fields[2]]; found {
  54. continue
  55. }
  56. ret = append(ret, PartitionStat{
  57. // NOTE(seanc@): Device isn't exactly accurate: from mnttab(4): "The name
  58. // of the resource that has been mounted." Ideally this value would come
  59. // from Statvfs_t.Fsid but I'm leaving it to the caller to traverse
  60. // unix.Statvfs().
  61. Device: fields[0],
  62. Mountpoint: fields[1],
  63. Fstype: fields[2],
  64. Opts: fields[3],
  65. })
  66. }
  67. if err := scanner.Err(); err != nil {
  68. return nil, fmt.Errorf("unable to scan %q: %v", _MNTTAB, err)
  69. }
  70. return ret, err
  71. }
  72. func IOCounters(names ...string) (map[string]IOCountersStat, error) {
  73. return IOCountersWithContext(context.Background(), names...)
  74. }
  75. func IOCountersWithContext(ctx context.Context, names ...string) (map[string]IOCountersStat, error) {
  76. return nil, common.ErrNotImplementedError
  77. }
  78. func Usage(path string) (*UsageStat, error) {
  79. return UsageWithContext(context.Background(), path)
  80. }
  81. func UsageWithContext(ctx context.Context, path string) (*UsageStat, error) {
  82. statvfs := unix.Statvfs_t{}
  83. if err := unix.Statvfs(path, &statvfs); err != nil {
  84. return nil, fmt.Errorf("unable to call statvfs(2) on %q: %v", path, err)
  85. }
  86. usageStat := &UsageStat{
  87. Path: path,
  88. Fstype: common.IntToString(statvfs.Basetype[:]),
  89. Total: statvfs.Blocks * statvfs.Frsize,
  90. Free: statvfs.Bfree * statvfs.Frsize,
  91. Used: (statvfs.Blocks - statvfs.Bfree) * statvfs.Frsize,
  92. // NOTE: ZFS (and FreeBZSD's UFS2) use dynamic inode/dnode allocation.
  93. // Explicitly return a near-zero value for InodesUsedPercent so that nothing
  94. // attempts to garbage collect based on a lack of available inodes/dnodes.
  95. // Similarly, don't use the zero value to prevent divide-by-zero situations
  96. // and inject a faux near-zero value. Filesystems evolve. Has your
  97. // filesystem evolved? Probably not if you care about the number of
  98. // available inodes.
  99. InodesTotal: 1024.0 * 1024.0,
  100. InodesUsed: 1024.0,
  101. InodesFree: math.MaxUint64,
  102. InodesUsedPercent: (1024.0 / (1024.0 * 1024.0)) * 100.0,
  103. }
  104. usageStat.UsedPercent = (float64(usageStat.Used) / float64(usageStat.Total)) * 100.0
  105. return usageStat, nil
  106. }