cpu_darwin.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. // +build darwin
  2. package cpu
  3. import (
  4. "context"
  5. "os/exec"
  6. "strconv"
  7. "strings"
  8. )
  9. // sys/resource.h
  10. const (
  11. CPUser = 0
  12. CPNice = 1
  13. CPSys = 2
  14. CPIntr = 3
  15. CPIdle = 4
  16. CPUStates = 5
  17. )
  18. // default value. from time.h
  19. var ClocksPerSec = float64(128)
  20. func Times(percpu bool) ([]TimesStat, error) {
  21. return TimesWithContext(context.Background(), percpu)
  22. }
  23. func TimesWithContext(ctx context.Context, percpu bool) ([]TimesStat, error) {
  24. if percpu {
  25. return perCPUTimes()
  26. }
  27. return allCPUTimes()
  28. }
  29. // Returns only one CPUInfoStat on FreeBSD
  30. func Info() ([]InfoStat, error) {
  31. return InfoWithContext(context.Background())
  32. }
  33. func InfoWithContext(ctx context.Context) ([]InfoStat, error) {
  34. var ret []InfoStat
  35. sysctl, err := exec.LookPath("/usr/sbin/sysctl")
  36. if err != nil {
  37. return ret, err
  38. }
  39. out, err := invoke.CommandWithContext(ctx, sysctl, "machdep.cpu")
  40. if err != nil {
  41. return ret, err
  42. }
  43. c := InfoStat{}
  44. for _, line := range strings.Split(string(out), "\n") {
  45. values := strings.Fields(line)
  46. if len(values) < 1 {
  47. continue
  48. }
  49. t, err := strconv.ParseInt(values[1], 10, 64)
  50. // err is not checked here because some value is string.
  51. if strings.HasPrefix(line, "machdep.cpu.brand_string") {
  52. c.ModelName = strings.Join(values[1:], " ")
  53. } else if strings.HasPrefix(line, "machdep.cpu.family") {
  54. c.Family = values[1]
  55. } else if strings.HasPrefix(line, "machdep.cpu.model") {
  56. c.Model = values[1]
  57. } else if strings.HasPrefix(line, "machdep.cpu.stepping") {
  58. if err != nil {
  59. return ret, err
  60. }
  61. c.Stepping = int32(t)
  62. } else if strings.HasPrefix(line, "machdep.cpu.features") {
  63. for _, v := range values[1:] {
  64. c.Flags = append(c.Flags, strings.ToLower(v))
  65. }
  66. } else if strings.HasPrefix(line, "machdep.cpu.leaf7_features") {
  67. for _, v := range values[1:] {
  68. c.Flags = append(c.Flags, strings.ToLower(v))
  69. }
  70. } else if strings.HasPrefix(line, "machdep.cpu.extfeatures") {
  71. for _, v := range values[1:] {
  72. c.Flags = append(c.Flags, strings.ToLower(v))
  73. }
  74. } else if strings.HasPrefix(line, "machdep.cpu.core_count") {
  75. if err != nil {
  76. return ret, err
  77. }
  78. c.Cores = int32(t)
  79. } else if strings.HasPrefix(line, "machdep.cpu.cache.size") {
  80. if err != nil {
  81. return ret, err
  82. }
  83. c.CacheSize = int32(t)
  84. } else if strings.HasPrefix(line, "machdep.cpu.vendor") {
  85. c.VendorID = values[1]
  86. }
  87. }
  88. // Use the rated frequency of the CPU. This is a static value and does not
  89. // account for low power or Turbo Boost modes.
  90. out, err = invoke.CommandWithContext(ctx, sysctl, "hw.cpufrequency")
  91. if err != nil {
  92. return ret, err
  93. }
  94. values := strings.Fields(string(out))
  95. hz, err := strconv.ParseFloat(values[1], 64)
  96. if err != nil {
  97. return ret, err
  98. }
  99. c.Mhz = hz / 1000000.0
  100. return append(ret, c), nil
  101. }