stat.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. package procfs
  2. import (
  3. "bufio"
  4. "fmt"
  5. "io"
  6. "os"
  7. "strconv"
  8. "strings"
  9. )
  10. // CPUStat shows how much time the cpu spend in various stages.
  11. type CPUStat struct {
  12. User float64
  13. Nice float64
  14. System float64
  15. Idle float64
  16. Iowait float64
  17. IRQ float64
  18. SoftIRQ float64
  19. Steal float64
  20. Guest float64
  21. GuestNice float64
  22. }
  23. // SoftIRQStat represent the softirq statistics as exported in the procfs stat file.
  24. // A nice introduction can be found at https://0xax.gitbooks.io/linux-insides/content/interrupts/interrupts-9.html
  25. // It is possible to get per-cpu stats by reading /proc/softirqs
  26. type SoftIRQStat struct {
  27. Hi uint64
  28. Timer uint64
  29. NetTx uint64
  30. NetRx uint64
  31. Block uint64
  32. BlockIoPoll uint64
  33. Tasklet uint64
  34. Sched uint64
  35. Hrtimer uint64
  36. Rcu uint64
  37. }
  38. // Stat represents kernel/system statistics.
  39. type Stat struct {
  40. // Boot time in seconds since the Epoch.
  41. BootTime uint64
  42. // Summed up cpu statistics.
  43. CPUTotal CPUStat
  44. // Per-CPU statistics.
  45. CPU []CPUStat
  46. // Number of times interrupts were handled, which contains numbered and unnumbered IRQs.
  47. IRQTotal uint64
  48. // Number of times a numbered IRQ was triggered.
  49. IRQ []uint64
  50. // Number of times a context switch happened.
  51. ContextSwitches uint64
  52. // Number of times a process was created.
  53. ProcessCreated uint64
  54. // Number of processes currently running.
  55. ProcessesRunning uint64
  56. // Number of processes currently blocked (waiting for IO).
  57. ProcessesBlocked uint64
  58. // Number of times a softirq was scheduled.
  59. SoftIRQTotal uint64
  60. // Detailed softirq statistics.
  61. SoftIRQ SoftIRQStat
  62. }
  63. // NewStat returns kernel/system statistics read from /proc/stat.
  64. func NewStat() (Stat, error) {
  65. fs, err := NewFS(DefaultMountPoint)
  66. if err != nil {
  67. return Stat{}, err
  68. }
  69. return fs.NewStat()
  70. }
  71. // Parse a cpu statistics line and returns the CPUStat struct plus the cpu id (or -1 for the overall sum).
  72. func parseCPUStat(line string) (CPUStat, int64, error) {
  73. cpuStat := CPUStat{}
  74. var cpu string
  75. count, err := fmt.Sscanf(line, "%s %f %f %f %f %f %f %f %f %f %f",
  76. &cpu,
  77. &cpuStat.User, &cpuStat.Nice, &cpuStat.System, &cpuStat.Idle,
  78. &cpuStat.Iowait, &cpuStat.IRQ, &cpuStat.SoftIRQ, &cpuStat.Steal,
  79. &cpuStat.Guest, &cpuStat.GuestNice)
  80. if err != nil && err != io.EOF {
  81. return CPUStat{}, -1, fmt.Errorf("couldn't parse %s (cpu): %s", line, err)
  82. }
  83. if count == 0 {
  84. return CPUStat{}, -1, fmt.Errorf("couldn't parse %s (cpu): 0 elements parsed", line)
  85. }
  86. cpuStat.User /= userHZ
  87. cpuStat.Nice /= userHZ
  88. cpuStat.System /= userHZ
  89. cpuStat.Idle /= userHZ
  90. cpuStat.Iowait /= userHZ
  91. cpuStat.IRQ /= userHZ
  92. cpuStat.SoftIRQ /= userHZ
  93. cpuStat.Steal /= userHZ
  94. cpuStat.Guest /= userHZ
  95. cpuStat.GuestNice /= userHZ
  96. if cpu == "cpu" {
  97. return cpuStat, -1, nil
  98. }
  99. cpuID, err := strconv.ParseInt(cpu[3:], 10, 64)
  100. if err != nil {
  101. return CPUStat{}, -1, fmt.Errorf("couldn't parse %s (cpu/cpuid): %s", line, err)
  102. }
  103. return cpuStat, cpuID, nil
  104. }
  105. // Parse a softirq line.
  106. func parseSoftIRQStat(line string) (SoftIRQStat, uint64, error) {
  107. softIRQStat := SoftIRQStat{}
  108. var total uint64
  109. var prefix string
  110. _, err := fmt.Sscanf(line, "%s %d %d %d %d %d %d %d %d %d %d %d",
  111. &prefix, &total,
  112. &softIRQStat.Hi, &softIRQStat.Timer, &softIRQStat.NetTx, &softIRQStat.NetRx,
  113. &softIRQStat.Block, &softIRQStat.BlockIoPoll,
  114. &softIRQStat.Tasklet, &softIRQStat.Sched,
  115. &softIRQStat.Hrtimer, &softIRQStat.Rcu)
  116. if err != nil {
  117. return SoftIRQStat{}, 0, fmt.Errorf("couldn't parse %s (softirq): %s", line, err)
  118. }
  119. return softIRQStat, total, nil
  120. }
  121. // NewStat returns an information about current kernel/system statistics.
  122. func (fs FS) NewStat() (Stat, error) {
  123. // See https://www.kernel.org/doc/Documentation/filesystems/proc.txt
  124. f, err := os.Open(fs.Path("stat"))
  125. if err != nil {
  126. return Stat{}, err
  127. }
  128. defer f.Close()
  129. stat := Stat{}
  130. scanner := bufio.NewScanner(f)
  131. for scanner.Scan() {
  132. line := scanner.Text()
  133. parts := strings.Fields(scanner.Text())
  134. // require at least <key> <value>
  135. if len(parts) < 2 {
  136. continue
  137. }
  138. switch {
  139. case parts[0] == "btime":
  140. if stat.BootTime, err = strconv.ParseUint(parts[1], 10, 64); err != nil {
  141. return Stat{}, fmt.Errorf("couldn't parse %s (btime): %s", parts[1], err)
  142. }
  143. case parts[0] == "intr":
  144. if stat.IRQTotal, err = strconv.ParseUint(parts[1], 10, 64); err != nil {
  145. return Stat{}, fmt.Errorf("couldn't parse %s (intr): %s", parts[1], err)
  146. }
  147. numberedIRQs := parts[2:]
  148. stat.IRQ = make([]uint64, len(numberedIRQs))
  149. for i, count := range numberedIRQs {
  150. if stat.IRQ[i], err = strconv.ParseUint(count, 10, 64); err != nil {
  151. return Stat{}, fmt.Errorf("couldn't parse %s (intr%d): %s", count, i, err)
  152. }
  153. }
  154. case parts[0] == "ctxt":
  155. if stat.ContextSwitches, err = strconv.ParseUint(parts[1], 10, 64); err != nil {
  156. return Stat{}, fmt.Errorf("couldn't parse %s (ctxt): %s", parts[1], err)
  157. }
  158. case parts[0] == "processes":
  159. if stat.ProcessCreated, err = strconv.ParseUint(parts[1], 10, 64); err != nil {
  160. return Stat{}, fmt.Errorf("couldn't parse %s (processes): %s", parts[1], err)
  161. }
  162. case parts[0] == "procs_running":
  163. if stat.ProcessesRunning, err = strconv.ParseUint(parts[1], 10, 64); err != nil {
  164. return Stat{}, fmt.Errorf("couldn't parse %s (procs_running): %s", parts[1], err)
  165. }
  166. case parts[0] == "procs_blocked":
  167. if stat.ProcessesBlocked, err = strconv.ParseUint(parts[1], 10, 64); err != nil {
  168. return Stat{}, fmt.Errorf("couldn't parse %s (procs_blocked): %s", parts[1], err)
  169. }
  170. case parts[0] == "softirq":
  171. softIRQStats, total, err := parseSoftIRQStat(line)
  172. if err != nil {
  173. return Stat{}, err
  174. }
  175. stat.SoftIRQTotal = total
  176. stat.SoftIRQ = softIRQStats
  177. case strings.HasPrefix(parts[0], "cpu"):
  178. cpuStat, cpuID, err := parseCPUStat(line)
  179. if err != nil {
  180. return Stat{}, err
  181. }
  182. if cpuID == -1 {
  183. stat.CPUTotal = cpuStat
  184. } else {
  185. for int64(len(stat.CPU)) <= cpuID {
  186. stat.CPU = append(stat.CPU, CPUStat{})
  187. }
  188. stat.CPU[cpuID] = cpuStat
  189. }
  190. }
  191. }
  192. if err := scanner.Err(); err != nil {
  193. return Stat{}, fmt.Errorf("couldn't parse %s: %s", f.Name(), err)
  194. }
  195. return stat, nil
  196. }