cpu_windows.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. // +build windows
  2. package cpu
  3. import (
  4. "context"
  5. "fmt"
  6. "unsafe"
  7. "github.com/StackExchange/wmi"
  8. "github.com/shirou/gopsutil/internal/common"
  9. "golang.org/x/sys/windows"
  10. )
  11. type Win32_Processor struct {
  12. LoadPercentage *uint16
  13. Family uint16
  14. Manufacturer string
  15. Name string
  16. NumberOfLogicalProcessors uint32
  17. ProcessorID *string
  18. Stepping *string
  19. MaxClockSpeed uint32
  20. }
  21. // Win32_PerfFormattedData_Counters_ProcessorInformation stores instance value of the perf counters
  22. type Win32_PerfFormattedData_Counters_ProcessorInformation struct {
  23. Name string
  24. PercentDPCTime uint64
  25. PercentIdleTime uint64
  26. PercentUserTime uint64
  27. PercentProcessorTime uint64
  28. PercentInterruptTime uint64
  29. PercentPriorityTime uint64
  30. PercentPrivilegedTime uint64
  31. InterruptsPerSec uint32
  32. ProcessorFrequency uint32
  33. DPCRate uint32
  34. }
  35. // Win32_PerfFormattedData_PerfOS_System struct to have count of processes and processor queue length
  36. type Win32_PerfFormattedData_PerfOS_System struct {
  37. Processes uint32
  38. ProcessorQueueLength uint32
  39. }
  40. // Times returns times stat per cpu and combined for all CPUs
  41. func Times(percpu bool) ([]TimesStat, error) {
  42. return TimesWithContext(context.Background(), percpu)
  43. }
  44. func TimesWithContext(ctx context.Context, percpu bool) ([]TimesStat, error) {
  45. if percpu {
  46. return perCPUTimes()
  47. }
  48. var ret []TimesStat
  49. var lpIdleTime common.FILETIME
  50. var lpKernelTime common.FILETIME
  51. var lpUserTime common.FILETIME
  52. r, _, _ := common.ProcGetSystemTimes.Call(
  53. uintptr(unsafe.Pointer(&lpIdleTime)),
  54. uintptr(unsafe.Pointer(&lpKernelTime)),
  55. uintptr(unsafe.Pointer(&lpUserTime)))
  56. if r == 0 {
  57. return ret, windows.GetLastError()
  58. }
  59. LOT := float64(0.0000001)
  60. HIT := (LOT * 4294967296.0)
  61. idle := ((HIT * float64(lpIdleTime.DwHighDateTime)) + (LOT * float64(lpIdleTime.DwLowDateTime)))
  62. user := ((HIT * float64(lpUserTime.DwHighDateTime)) + (LOT * float64(lpUserTime.DwLowDateTime)))
  63. kernel := ((HIT * float64(lpKernelTime.DwHighDateTime)) + (LOT * float64(lpKernelTime.DwLowDateTime)))
  64. system := (kernel - idle)
  65. ret = append(ret, TimesStat{
  66. CPU: "cpu-total",
  67. Idle: float64(idle),
  68. User: float64(user),
  69. System: float64(system),
  70. })
  71. return ret, nil
  72. }
  73. func Info() ([]InfoStat, error) {
  74. return InfoWithContext(context.Background())
  75. }
  76. func InfoWithContext(ctx context.Context) ([]InfoStat, error) {
  77. var ret []InfoStat
  78. var dst []Win32_Processor
  79. q := wmi.CreateQuery(&dst, "")
  80. if err := common.WMIQueryWithContext(ctx, q, &dst); err != nil {
  81. return ret, err
  82. }
  83. var procID string
  84. for i, l := range dst {
  85. procID = ""
  86. if l.ProcessorID != nil {
  87. procID = *l.ProcessorID
  88. }
  89. cpu := InfoStat{
  90. CPU: int32(i),
  91. Family: fmt.Sprintf("%d", l.Family),
  92. VendorID: l.Manufacturer,
  93. ModelName: l.Name,
  94. Cores: int32(l.NumberOfLogicalProcessors),
  95. PhysicalID: procID,
  96. Mhz: float64(l.MaxClockSpeed),
  97. Flags: []string{},
  98. }
  99. ret = append(ret, cpu)
  100. }
  101. return ret, nil
  102. }
  103. // PerfInfo returns the performance counter's instance value for ProcessorInformation.
  104. // Name property is the key by which overall, per cpu and per core metric is known.
  105. func PerfInfo() ([]Win32_PerfFormattedData_Counters_ProcessorInformation, error) {
  106. return PerfInfoWithContext(context.Background())
  107. }
  108. func PerfInfoWithContext(ctx context.Context) ([]Win32_PerfFormattedData_Counters_ProcessorInformation, error) {
  109. var ret []Win32_PerfFormattedData_Counters_ProcessorInformation
  110. q := wmi.CreateQuery(&ret, "")
  111. err := common.WMIQueryWithContext(ctx, q, &ret)
  112. if err != nil {
  113. return []Win32_PerfFormattedData_Counters_ProcessorInformation{}, err
  114. }
  115. return ret, err
  116. }
  117. // ProcInfo returns processes count and processor queue length in the system.
  118. // There is a single queue for processor even on multiprocessors systems.
  119. func ProcInfo() ([]Win32_PerfFormattedData_PerfOS_System, error) {
  120. return ProcInfoWithContext(context.Background())
  121. }
  122. func ProcInfoWithContext(ctx context.Context) ([]Win32_PerfFormattedData_PerfOS_System, error) {
  123. var ret []Win32_PerfFormattedData_PerfOS_System
  124. q := wmi.CreateQuery(&ret, "")
  125. err := common.WMIQueryWithContext(ctx, q, &ret)
  126. if err != nil {
  127. return []Win32_PerfFormattedData_PerfOS_System{}, err
  128. }
  129. return ret, err
  130. }
  131. // perCPUTimes returns times stat per cpu, per core and overall for all CPUs
  132. func perCPUTimes() ([]TimesStat, error) {
  133. var ret []TimesStat
  134. stats, err := PerfInfo()
  135. if err != nil {
  136. return nil, err
  137. }
  138. for _, v := range stats {
  139. c := TimesStat{
  140. CPU: v.Name,
  141. User: float64(v.PercentUserTime),
  142. System: float64(v.PercentPrivilegedTime),
  143. Idle: float64(v.PercentIdleTime),
  144. Irq: float64(v.PercentInterruptTime),
  145. }
  146. ret = append(ret, c)
  147. }
  148. return ret, nil
  149. }