cpu_linux.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. // +build linux
  2. package cpu
  3. import (
  4. "context"
  5. "errors"
  6. "fmt"
  7. "os/exec"
  8. "strconv"
  9. "strings"
  10. "github.com/shirou/gopsutil/internal/common"
  11. )
  12. var cpu_tick = float64(100)
  13. func init() {
  14. getconf, err := exec.LookPath("/usr/bin/getconf")
  15. if err != nil {
  16. return
  17. }
  18. out, err := invoke.CommandWithContext(context.Background(), getconf, "CLK_TCK")
  19. // ignore errors
  20. if err == nil {
  21. i, err := strconv.ParseFloat(strings.TrimSpace(string(out)), 64)
  22. if err == nil {
  23. cpu_tick = float64(i)
  24. }
  25. }
  26. }
  27. func Times(percpu bool) ([]TimesStat, error) {
  28. return TimesWithContext(context.Background(), percpu)
  29. }
  30. func TimesWithContext(ctx context.Context, percpu bool) ([]TimesStat, error) {
  31. filename := common.HostProc("stat")
  32. var lines = []string{}
  33. if percpu {
  34. statlines, err := common.ReadLines(filename)
  35. if err != nil || len(statlines) < 2 {
  36. return []TimesStat{}, nil
  37. }
  38. for _, line := range statlines[1:] {
  39. if !strings.HasPrefix(line, "cpu") {
  40. break
  41. }
  42. lines = append(lines, line)
  43. }
  44. } else {
  45. lines, _ = common.ReadLinesOffsetN(filename, 0, 1)
  46. }
  47. ret := make([]TimesStat, 0, len(lines))
  48. for _, line := range lines {
  49. ct, err := parseStatLine(line)
  50. if err != nil {
  51. continue
  52. }
  53. ret = append(ret, *ct)
  54. }
  55. return ret, nil
  56. }
  57. func sysCPUPath(cpu int32, relPath string) string {
  58. return common.HostSys(fmt.Sprintf("devices/system/cpu/cpu%d", cpu), relPath)
  59. }
  60. func finishCPUInfo(c *InfoStat) error {
  61. var lines []string
  62. var err error
  63. var value float64
  64. if len(c.CoreID) == 0 {
  65. lines, err = common.ReadLines(sysCPUPath(c.CPU, "topology/core_id"))
  66. if err == nil {
  67. c.CoreID = lines[0]
  68. }
  69. }
  70. // override the value of c.Mhz with cpufreq/cpuinfo_max_freq regardless
  71. // of the value from /proc/cpuinfo because we want to report the maximum
  72. // clock-speed of the CPU for c.Mhz, matching the behaviour of Windows
  73. lines, err = common.ReadLines(sysCPUPath(c.CPU, "cpufreq/cpuinfo_max_freq"))
  74. // if we encounter errors below such as there are no cpuinfo_max_freq file,
  75. // we just ignore. so let Mhz is 0.
  76. if err != nil {
  77. return nil
  78. }
  79. value, err = strconv.ParseFloat(lines[0], 64)
  80. if err != nil {
  81. return nil
  82. }
  83. c.Mhz = value / 1000.0 // value is in kHz
  84. if c.Mhz > 9999 {
  85. c.Mhz = c.Mhz / 1000.0 // value in Hz
  86. }
  87. return nil
  88. }
  89. // CPUInfo on linux will return 1 item per physical thread.
  90. //
  91. // CPUs have three levels of counting: sockets, cores, threads.
  92. // Cores with HyperThreading count as having 2 threads per core.
  93. // Sockets often come with many physical CPU cores.
  94. // For example a single socket board with two cores each with HT will
  95. // return 4 CPUInfoStat structs on Linux and the "Cores" field set to 1.
  96. func Info() ([]InfoStat, error) {
  97. return InfoWithContext(context.Background())
  98. }
  99. func InfoWithContext(ctx context.Context) ([]InfoStat, error) {
  100. filename := common.HostProc("cpuinfo")
  101. lines, _ := common.ReadLines(filename)
  102. var ret []InfoStat
  103. var processorName string
  104. c := InfoStat{CPU: -1, Cores: 1}
  105. for _, line := range lines {
  106. fields := strings.Split(line, ":")
  107. if len(fields) < 2 {
  108. continue
  109. }
  110. key := strings.TrimSpace(fields[0])
  111. value := strings.TrimSpace(fields[1])
  112. switch key {
  113. case "Processor":
  114. processorName = value
  115. case "processor":
  116. if c.CPU >= 0 {
  117. err := finishCPUInfo(&c)
  118. if err != nil {
  119. return ret, err
  120. }
  121. ret = append(ret, c)
  122. }
  123. c = InfoStat{Cores: 1, ModelName: processorName}
  124. t, err := strconv.ParseInt(value, 10, 64)
  125. if err != nil {
  126. return ret, err
  127. }
  128. c.CPU = int32(t)
  129. case "vendorId", "vendor_id":
  130. c.VendorID = value
  131. case "cpu family":
  132. c.Family = value
  133. case "model":
  134. c.Model = value
  135. case "model name", "cpu":
  136. c.ModelName = value
  137. if strings.Contains(value, "POWER8") ||
  138. strings.Contains(value, "POWER7") {
  139. c.Model = strings.Split(value, " ")[0]
  140. c.Family = "POWER"
  141. c.VendorID = "IBM"
  142. }
  143. case "stepping", "revision":
  144. val := value
  145. if key == "revision" {
  146. val = strings.Split(value, ".")[0]
  147. }
  148. t, err := strconv.ParseInt(val, 10, 64)
  149. if err != nil {
  150. return ret, err
  151. }
  152. c.Stepping = int32(t)
  153. case "cpu MHz", "clock":
  154. // treat this as the fallback value, thus we ignore error
  155. if t, err := strconv.ParseFloat(strings.Replace(value, "MHz", "", 1), 64); err == nil {
  156. c.Mhz = t
  157. }
  158. case "cache size":
  159. t, err := strconv.ParseInt(strings.Replace(value, " KB", "", 1), 10, 64)
  160. if err != nil {
  161. return ret, err
  162. }
  163. c.CacheSize = int32(t)
  164. case "physical id":
  165. c.PhysicalID = value
  166. case "core id":
  167. c.CoreID = value
  168. case "flags", "Features":
  169. c.Flags = strings.FieldsFunc(value, func(r rune) bool {
  170. return r == ',' || r == ' '
  171. })
  172. case "microcode":
  173. c.Microcode = value
  174. }
  175. }
  176. if c.CPU >= 0 {
  177. err := finishCPUInfo(&c)
  178. if err != nil {
  179. return ret, err
  180. }
  181. ret = append(ret, c)
  182. }
  183. return ret, nil
  184. }
  185. func parseStatLine(line string) (*TimesStat, error) {
  186. fields := strings.Fields(line)
  187. if len(fields) == 0 {
  188. return nil, errors.New("stat does not contain cpu info")
  189. }
  190. if strings.HasPrefix(fields[0], "cpu") == false {
  191. // return CPUTimesStat{}, e
  192. return nil, errors.New("not contain cpu")
  193. }
  194. cpu := fields[0]
  195. if cpu == "cpu" {
  196. cpu = "cpu-total"
  197. }
  198. user, err := strconv.ParseFloat(fields[1], 64)
  199. if err != nil {
  200. return nil, err
  201. }
  202. nice, err := strconv.ParseFloat(fields[2], 64)
  203. if err != nil {
  204. return nil, err
  205. }
  206. system, err := strconv.ParseFloat(fields[3], 64)
  207. if err != nil {
  208. return nil, err
  209. }
  210. idle, err := strconv.ParseFloat(fields[4], 64)
  211. if err != nil {
  212. return nil, err
  213. }
  214. iowait, err := strconv.ParseFloat(fields[5], 64)
  215. if err != nil {
  216. return nil, err
  217. }
  218. irq, err := strconv.ParseFloat(fields[6], 64)
  219. if err != nil {
  220. return nil, err
  221. }
  222. softirq, err := strconv.ParseFloat(fields[7], 64)
  223. if err != nil {
  224. return nil, err
  225. }
  226. ct := &TimesStat{
  227. CPU: cpu,
  228. User: float64(user) / cpu_tick,
  229. Nice: float64(nice) / cpu_tick,
  230. System: float64(system) / cpu_tick,
  231. Idle: float64(idle) / cpu_tick,
  232. Iowait: float64(iowait) / cpu_tick,
  233. Irq: float64(irq) / cpu_tick,
  234. Softirq: float64(softirq) / cpu_tick,
  235. }
  236. if len(fields) > 8 { // Linux >= 2.6.11
  237. steal, err := strconv.ParseFloat(fields[8], 64)
  238. if err != nil {
  239. return nil, err
  240. }
  241. ct.Steal = float64(steal) / cpu_tick
  242. }
  243. if len(fields) > 9 { // Linux >= 2.6.24
  244. guest, err := strconv.ParseFloat(fields[9], 64)
  245. if err != nil {
  246. return nil, err
  247. }
  248. ct.Guest = float64(guest) / cpu_tick
  249. }
  250. if len(fields) > 10 { // Linux >= 3.2.0
  251. guestNice, err := strconv.ParseFloat(fields[10], 64)
  252. if err != nil {
  253. return nil, err
  254. }
  255. ct.GuestNice = float64(guestNice) / cpu_tick
  256. }
  257. return ct, nil
  258. }