cpu.go 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. package cpu
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "runtime"
  7. "strconv"
  8. "strings"
  9. "sync"
  10. "time"
  11. "github.com/shirou/gopsutil/internal/common"
  12. )
  13. // TimesStat contains the amounts of time the CPU has spent performing different
  14. // kinds of work. Time units are in USER_HZ or Jiffies (typically hundredths of
  15. // a second). It is based on linux /proc/stat file.
  16. type TimesStat struct {
  17. CPU string `json:"cpu"`
  18. User float64 `json:"user"`
  19. System float64 `json:"system"`
  20. Idle float64 `json:"idle"`
  21. Nice float64 `json:"nice"`
  22. Iowait float64 `json:"iowait"`
  23. Irq float64 `json:"irq"`
  24. Softirq float64 `json:"softirq"`
  25. Steal float64 `json:"steal"`
  26. Guest float64 `json:"guest"`
  27. GuestNice float64 `json:"guestNice"`
  28. Stolen float64 `json:"stolen"`
  29. }
  30. type InfoStat struct {
  31. CPU int32 `json:"cpu"`
  32. VendorID string `json:"vendorId"`
  33. Family string `json:"family"`
  34. Model string `json:"model"`
  35. Stepping int32 `json:"stepping"`
  36. PhysicalID string `json:"physicalId"`
  37. CoreID string `json:"coreId"`
  38. Cores int32 `json:"cores"`
  39. ModelName string `json:"modelName"`
  40. Mhz float64 `json:"mhz"`
  41. CacheSize int32 `json:"cacheSize"`
  42. Flags []string `json:"flags"`
  43. Microcode string `json:"microcode"`
  44. }
  45. type lastPercent struct {
  46. sync.Mutex
  47. lastCPUTimes []TimesStat
  48. lastPerCPUTimes []TimesStat
  49. }
  50. var lastCPUPercent lastPercent
  51. var invoke common.Invoker = common.Invoke{}
  52. func init() {
  53. lastCPUPercent.Lock()
  54. lastCPUPercent.lastCPUTimes, _ = Times(false)
  55. lastCPUPercent.lastPerCPUTimes, _ = Times(true)
  56. lastCPUPercent.Unlock()
  57. }
  58. func Counts(logical bool) (int, error) {
  59. return CountsWithContext(context.Background(), logical)
  60. }
  61. func CountsWithContext(ctx context.Context, logical bool) (int, error) {
  62. return runtime.NumCPU(), nil
  63. }
  64. func (c TimesStat) String() string {
  65. v := []string{
  66. `"cpu":"` + c.CPU + `"`,
  67. `"user":` + strconv.FormatFloat(c.User, 'f', 1, 64),
  68. `"system":` + strconv.FormatFloat(c.System, 'f', 1, 64),
  69. `"idle":` + strconv.FormatFloat(c.Idle, 'f', 1, 64),
  70. `"nice":` + strconv.FormatFloat(c.Nice, 'f', 1, 64),
  71. `"iowait":` + strconv.FormatFloat(c.Iowait, 'f', 1, 64),
  72. `"irq":` + strconv.FormatFloat(c.Irq, 'f', 1, 64),
  73. `"softirq":` + strconv.FormatFloat(c.Softirq, 'f', 1, 64),
  74. `"steal":` + strconv.FormatFloat(c.Steal, 'f', 1, 64),
  75. `"guest":` + strconv.FormatFloat(c.Guest, 'f', 1, 64),
  76. `"guestNice":` + strconv.FormatFloat(c.GuestNice, 'f', 1, 64),
  77. `"stolen":` + strconv.FormatFloat(c.Stolen, 'f', 1, 64),
  78. }
  79. return `{` + strings.Join(v, ",") + `}`
  80. }
  81. // Total returns the total number of seconds in a CPUTimesStat
  82. func (c TimesStat) Total() float64 {
  83. total := c.User + c.System + c.Nice + c.Iowait + c.Irq + c.Softirq + c.Steal +
  84. c.Guest + c.GuestNice + c.Idle + c.Stolen
  85. return total
  86. }
  87. func (c InfoStat) String() string {
  88. s, _ := json.Marshal(c)
  89. return string(s)
  90. }
  91. func getAllBusy(t TimesStat) (float64, float64) {
  92. busy := t.User + t.System + t.Nice + t.Iowait + t.Irq +
  93. t.Softirq + t.Steal + t.Guest + t.GuestNice + t.Stolen
  94. return busy + t.Idle, busy
  95. }
  96. func calculateBusy(t1, t2 TimesStat) float64 {
  97. t1All, t1Busy := getAllBusy(t1)
  98. t2All, t2Busy := getAllBusy(t2)
  99. if t2Busy <= t1Busy {
  100. return 0
  101. }
  102. if t2All <= t1All {
  103. return 1
  104. }
  105. return (t2Busy - t1Busy) / (t2All - t1All) * 100
  106. }
  107. func calculateAllBusy(t1, t2 []TimesStat) ([]float64, error) {
  108. // Make sure the CPU measurements have the same length.
  109. if len(t1) != len(t2) {
  110. return nil, fmt.Errorf(
  111. "received two CPU counts: %d != %d",
  112. len(t1), len(t2),
  113. )
  114. }
  115. ret := make([]float64, len(t1))
  116. for i, t := range t2 {
  117. ret[i] = calculateBusy(t1[i], t)
  118. }
  119. return ret, nil
  120. }
  121. // Percent calculates the percentage of cpu used either per CPU or combined.
  122. // If an interval of 0 is given it will compare the current cpu times against the last call.
  123. // Returns one value per cpu, or a single value if percpu is set to false.
  124. func Percent(interval time.Duration, percpu bool) ([]float64, error) {
  125. return PercentWithContext(context.Background(), interval, percpu)
  126. }
  127. func PercentWithContext(ctx context.Context, interval time.Duration, percpu bool) ([]float64, error) {
  128. if interval <= 0 {
  129. return percentUsedFromLastCall(percpu)
  130. }
  131. // Get CPU usage at the start of the interval.
  132. cpuTimes1, err := Times(percpu)
  133. if err != nil {
  134. return nil, err
  135. }
  136. time.Sleep(interval)
  137. // And at the end of the interval.
  138. cpuTimes2, err := Times(percpu)
  139. if err != nil {
  140. return nil, err
  141. }
  142. return calculateAllBusy(cpuTimes1, cpuTimes2)
  143. }
  144. func percentUsedFromLastCall(percpu bool) ([]float64, error) {
  145. cpuTimes, err := Times(percpu)
  146. if err != nil {
  147. return nil, err
  148. }
  149. lastCPUPercent.Lock()
  150. defer lastCPUPercent.Unlock()
  151. var lastTimes []TimesStat
  152. if percpu {
  153. lastTimes = lastCPUPercent.lastPerCPUTimes
  154. lastCPUPercent.lastPerCPUTimes = cpuTimes
  155. } else {
  156. lastTimes = lastCPUPercent.lastCPUTimes
  157. lastCPUPercent.lastCPUTimes = cpuTimes
  158. }
  159. if lastTimes == nil {
  160. return nil, fmt.Errorf("error getting times for cpu percent. lastTimes was nil")
  161. }
  162. return calculateAllBusy(lastTimes, cpuTimes)
  163. }