cpu_solaris.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. package cpu
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "os/exec"
  7. "regexp"
  8. "sort"
  9. "strconv"
  10. "strings"
  11. "github.com/shirou/gopsutil/internal/common"
  12. )
  13. var ClocksPerSec = float64(128)
  14. func init() {
  15. getconf, err := exec.LookPath("/usr/bin/getconf")
  16. if err != nil {
  17. return
  18. }
  19. out, err := invoke.Command(getconf, "CLK_TCK")
  20. // ignore errors
  21. if err == nil {
  22. i, err := strconv.ParseFloat(strings.TrimSpace(string(out)), 64)
  23. if err == nil {
  24. ClocksPerSec = float64(i)
  25. }
  26. }
  27. }
  28. func Times(percpu bool) ([]TimesStat, error) {
  29. return TimesWithContext(context.Background(), percpu)
  30. }
  31. func TimesWithContext(ctx context.Context, percpu bool) ([]TimesStat, error) {
  32. return []TimesStat{}, common.ErrNotImplementedError
  33. }
  34. func Info() ([]InfoStat, error) {
  35. return InfoWithContext(context.Background())
  36. }
  37. func InfoWithContext(ctx context.Context) ([]InfoStat, error) {
  38. psrInfo, err := exec.LookPath("/usr/sbin/psrinfo")
  39. if err != nil {
  40. return nil, fmt.Errorf("cannot find psrinfo: %s", err)
  41. }
  42. psrInfoOut, err := invoke.CommandWithContext(ctx, psrInfo, "-p", "-v")
  43. if err != nil {
  44. return nil, fmt.Errorf("cannot execute psrinfo: %s", err)
  45. }
  46. isaInfo, err := exec.LookPath("/usr/bin/isainfo")
  47. if err != nil {
  48. return nil, fmt.Errorf("cannot find isainfo: %s", err)
  49. }
  50. isaInfoOut, err := invoke.CommandWithContext(ctx, isaInfo, "-b", "-v")
  51. if err != nil {
  52. return nil, fmt.Errorf("cannot execute isainfo: %s", err)
  53. }
  54. procs, err := parseProcessorInfo(string(psrInfoOut))
  55. if err != nil {
  56. return nil, fmt.Errorf("error parsing psrinfo output: %s", err)
  57. }
  58. flags, err := parseISAInfo(string(isaInfoOut))
  59. if err != nil {
  60. return nil, fmt.Errorf("error parsing isainfo output: %s", err)
  61. }
  62. result := make([]InfoStat, 0, len(flags))
  63. for _, proc := range procs {
  64. procWithFlags := proc
  65. procWithFlags.Flags = flags
  66. result = append(result, procWithFlags)
  67. }
  68. return result, nil
  69. }
  70. var flagsMatch = regexp.MustCompile(`[\w\.]+`)
  71. func parseISAInfo(cmdOutput string) ([]string, error) {
  72. words := flagsMatch.FindAllString(cmdOutput, -1)
  73. // Sanity check the output
  74. if len(words) < 4 || words[1] != "bit" || words[3] != "applications" {
  75. return nil, errors.New("attempted to parse invalid isainfo output")
  76. }
  77. flags := make([]string, len(words)-4)
  78. for i, val := range words[4:] {
  79. flags[i] = val
  80. }
  81. sort.Strings(flags)
  82. return flags, nil
  83. }
  84. var psrInfoMatch = regexp.MustCompile(`The physical processor has (?:([\d]+) virtual processor \(([\d]+)\)|([\d]+) cores and ([\d]+) virtual processors[^\n]+)\n(?:\s+ The core has.+\n)*\s+.+ \((\w+) ([\S]+) family (.+) model (.+) step (.+) clock (.+) MHz\)\n[\s]*(.*)`)
  85. const (
  86. psrNumCoresOffset = 1
  87. psrNumCoresHTOffset = 3
  88. psrNumHTOffset = 4
  89. psrVendorIDOffset = 5
  90. psrFamilyOffset = 7
  91. psrModelOffset = 8
  92. psrStepOffset = 9
  93. psrClockOffset = 10
  94. psrModelNameOffset = 11
  95. )
  96. func parseProcessorInfo(cmdOutput string) ([]InfoStat, error) {
  97. matches := psrInfoMatch.FindAllStringSubmatch(cmdOutput, -1)
  98. var infoStatCount int32
  99. result := make([]InfoStat, 0, len(matches))
  100. for physicalIndex, physicalCPU := range matches {
  101. var step int32
  102. var clock float64
  103. if physicalCPU[psrStepOffset] != "" {
  104. stepParsed, err := strconv.ParseInt(physicalCPU[psrStepOffset], 10, 32)
  105. if err != nil {
  106. return nil, fmt.Errorf("cannot parse value %q for step as 32-bit integer: %s", physicalCPU[9], err)
  107. }
  108. step = int32(stepParsed)
  109. }
  110. if physicalCPU[psrClockOffset] != "" {
  111. clockParsed, err := strconv.ParseInt(physicalCPU[psrClockOffset], 10, 64)
  112. if err != nil {
  113. return nil, fmt.Errorf("cannot parse value %q for clock as 32-bit integer: %s", physicalCPU[10], err)
  114. }
  115. clock = float64(clockParsed)
  116. }
  117. var err error
  118. var numCores int64
  119. var numHT int64
  120. switch {
  121. case physicalCPU[psrNumCoresOffset] != "":
  122. numCores, err = strconv.ParseInt(physicalCPU[psrNumCoresOffset], 10, 32)
  123. if err != nil {
  124. return nil, fmt.Errorf("cannot parse value %q for core count as 32-bit integer: %s", physicalCPU[1], err)
  125. }
  126. for i := 0; i < int(numCores); i++ {
  127. result = append(result, InfoStat{
  128. CPU: infoStatCount,
  129. PhysicalID: strconv.Itoa(physicalIndex),
  130. CoreID: strconv.Itoa(i),
  131. Cores: 1,
  132. VendorID: physicalCPU[psrVendorIDOffset],
  133. ModelName: physicalCPU[psrModelNameOffset],
  134. Family: physicalCPU[psrFamilyOffset],
  135. Model: physicalCPU[psrModelOffset],
  136. Stepping: step,
  137. Mhz: clock,
  138. })
  139. infoStatCount++
  140. }
  141. case physicalCPU[psrNumCoresHTOffset] != "":
  142. numCores, err = strconv.ParseInt(physicalCPU[psrNumCoresHTOffset], 10, 32)
  143. if err != nil {
  144. return nil, fmt.Errorf("cannot parse value %q for core count as 32-bit integer: %s", physicalCPU[3], err)
  145. }
  146. numHT, err = strconv.ParseInt(physicalCPU[psrNumHTOffset], 10, 32)
  147. if err != nil {
  148. return nil, fmt.Errorf("cannot parse value %q for hyperthread count as 32-bit integer: %s", physicalCPU[4], err)
  149. }
  150. for i := 0; i < int(numCores); i++ {
  151. result = append(result, InfoStat{
  152. CPU: infoStatCount,
  153. PhysicalID: strconv.Itoa(physicalIndex),
  154. CoreID: strconv.Itoa(i),
  155. Cores: int32(numHT) / int32(numCores),
  156. VendorID: physicalCPU[psrVendorIDOffset],
  157. ModelName: physicalCPU[psrModelNameOffset],
  158. Family: physicalCPU[psrFamilyOffset],
  159. Model: physicalCPU[psrModelOffset],
  160. Stepping: step,
  161. Mhz: clock,
  162. })
  163. infoStatCount++
  164. }
  165. default:
  166. return nil, errors.New("values for cores with and without hyperthreading are both set")
  167. }
  168. }
  169. return result, nil
  170. }