disk_windows.go 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. // +build windows
  2. package disk
  3. import (
  4. "bytes"
  5. "context"
  6. "unsafe"
  7. "github.com/shirou/gopsutil/internal/common"
  8. "golang.org/x/sys/windows"
  9. )
  10. var (
  11. procGetDiskFreeSpaceExW = common.Modkernel32.NewProc("GetDiskFreeSpaceExW")
  12. procGetLogicalDriveStringsW = common.Modkernel32.NewProc("GetLogicalDriveStringsW")
  13. procGetDriveType = common.Modkernel32.NewProc("GetDriveTypeW")
  14. provGetVolumeInformation = common.Modkernel32.NewProc("GetVolumeInformationW")
  15. )
  16. var (
  17. FileFileCompression = int64(16) // 0x00000010
  18. FileReadOnlyVolume = int64(524288) // 0x00080000
  19. )
  20. type Win32_PerfFormattedData struct {
  21. Name string
  22. AvgDiskBytesPerRead uint64
  23. AvgDiskBytesPerWrite uint64
  24. AvgDiskReadQueueLength uint64
  25. AvgDiskWriteQueueLength uint64
  26. AvgDisksecPerRead uint64
  27. AvgDisksecPerWrite uint64
  28. }
  29. const WaitMSec = 500
  30. func Usage(path string) (*UsageStat, error) {
  31. return UsageWithContext(context.Background(), path)
  32. }
  33. func UsageWithContext(ctx context.Context, path string) (*UsageStat, error) {
  34. ret := &UsageStat{}
  35. lpFreeBytesAvailable := int64(0)
  36. lpTotalNumberOfBytes := int64(0)
  37. lpTotalNumberOfFreeBytes := int64(0)
  38. diskret, _, err := procGetDiskFreeSpaceExW.Call(
  39. uintptr(unsafe.Pointer(windows.StringToUTF16Ptr(path))),
  40. uintptr(unsafe.Pointer(&lpFreeBytesAvailable)),
  41. uintptr(unsafe.Pointer(&lpTotalNumberOfBytes)),
  42. uintptr(unsafe.Pointer(&lpTotalNumberOfFreeBytes)))
  43. if diskret == 0 {
  44. return nil, err
  45. }
  46. ret = &UsageStat{
  47. Path: path,
  48. Total: uint64(lpTotalNumberOfBytes),
  49. Free: uint64(lpTotalNumberOfFreeBytes),
  50. Used: uint64(lpTotalNumberOfBytes) - uint64(lpTotalNumberOfFreeBytes),
  51. UsedPercent: (float64(lpTotalNumberOfBytes) - float64(lpTotalNumberOfFreeBytes)) / float64(lpTotalNumberOfBytes) * 100,
  52. // InodesTotal: 0,
  53. // InodesFree: 0,
  54. // InodesUsed: 0,
  55. // InodesUsedPercent: 0,
  56. }
  57. return ret, nil
  58. }
  59. func Partitions(all bool) ([]PartitionStat, error) {
  60. return PartitionsWithContext(context.Background(), all)
  61. }
  62. func PartitionsWithContext(ctx context.Context, all bool) ([]PartitionStat, error) {
  63. var ret []PartitionStat
  64. lpBuffer := make([]byte, 254)
  65. diskret, _, err := procGetLogicalDriveStringsW.Call(
  66. uintptr(len(lpBuffer)),
  67. uintptr(unsafe.Pointer(&lpBuffer[0])))
  68. if diskret == 0 {
  69. return ret, err
  70. }
  71. for _, v := range lpBuffer {
  72. if v >= 65 && v <= 90 {
  73. path := string(v) + ":"
  74. if path == "A:" || path == "B:" { // skip floppy drives
  75. continue
  76. }
  77. typepath, _ := windows.UTF16PtrFromString(path)
  78. typeret, _, _ := procGetDriveType.Call(uintptr(unsafe.Pointer(typepath)))
  79. if typeret == 0 {
  80. return ret, windows.GetLastError()
  81. }
  82. // 2: DRIVE_REMOVABLE 3: DRIVE_FIXED 4: DRIVE_REMOTE 5: DRIVE_CDROM
  83. if typeret == 2 || typeret == 3 || typeret == 4 || typeret == 5 {
  84. lpVolumeNameBuffer := make([]byte, 256)
  85. lpVolumeSerialNumber := int64(0)
  86. lpMaximumComponentLength := int64(0)
  87. lpFileSystemFlags := int64(0)
  88. lpFileSystemNameBuffer := make([]byte, 256)
  89. volpath, _ := windows.UTF16PtrFromString(string(v) + ":/")
  90. driveret, _, err := provGetVolumeInformation.Call(
  91. uintptr(unsafe.Pointer(volpath)),
  92. uintptr(unsafe.Pointer(&lpVolumeNameBuffer[0])),
  93. uintptr(len(lpVolumeNameBuffer)),
  94. uintptr(unsafe.Pointer(&lpVolumeSerialNumber)),
  95. uintptr(unsafe.Pointer(&lpMaximumComponentLength)),
  96. uintptr(unsafe.Pointer(&lpFileSystemFlags)),
  97. uintptr(unsafe.Pointer(&lpFileSystemNameBuffer[0])),
  98. uintptr(len(lpFileSystemNameBuffer)))
  99. if driveret == 0 {
  100. if typeret == 5 || typeret == 2 {
  101. continue //device is not ready will happen if there is no disk in the drive
  102. }
  103. return ret, err
  104. }
  105. opts := "rw"
  106. if lpFileSystemFlags&FileReadOnlyVolume != 0 {
  107. opts = "ro"
  108. }
  109. if lpFileSystemFlags&FileFileCompression != 0 {
  110. opts += ".compress"
  111. }
  112. d := PartitionStat{
  113. Mountpoint: path,
  114. Device: path,
  115. Fstype: string(bytes.Replace(lpFileSystemNameBuffer, []byte("\x00"), []byte(""), -1)),
  116. Opts: opts,
  117. }
  118. ret = append(ret, d)
  119. }
  120. }
  121. }
  122. return ret, nil
  123. }
  124. func IOCounters(names ...string) (map[string]IOCountersStat, error) {
  125. return IOCountersWithContext(context.Background(), names...)
  126. }
  127. func IOCountersWithContext(ctx context.Context, names ...string) (map[string]IOCountersStat, error) {
  128. ret := make(map[string]IOCountersStat, 0)
  129. var dst []Win32_PerfFormattedData
  130. err := common.WMIQueryWithContext(ctx, "SELECT * FROM Win32_PerfFormattedData_PerfDisk_LogicalDisk", &dst)
  131. if err != nil {
  132. return ret, err
  133. }
  134. for _, d := range dst {
  135. if len(d.Name) > 3 { // not get _Total or Harddrive
  136. continue
  137. }
  138. if len(names) > 0 && !common.StringsHas(names, d.Name) {
  139. continue
  140. }
  141. ret[d.Name] = IOCountersStat{
  142. Name: d.Name,
  143. ReadCount: uint64(d.AvgDiskReadQueueLength),
  144. WriteCount: d.AvgDiskWriteQueueLength,
  145. ReadBytes: uint64(d.AvgDiskBytesPerRead),
  146. WriteBytes: uint64(d.AvgDiskBytesPerWrite),
  147. ReadTime: d.AvgDisksecPerRead,
  148. WriteTime: d.AvgDisksecPerWrite,
  149. }
  150. }
  151. return ret, nil
  152. }