1
0
mirror of https://github.com/shirou/gopsutil.git synced 2025-04-24 13:48:56 +08:00
shirou_gopsutil/cpu/cpu_linux.go

376 lines
9.3 KiB
Go
Raw Normal View History

2014-04-18 16:34:47 +09:00
// +build linux
2014-12-30 22:09:05 +09:00
package cpu
2014-04-18 16:34:47 +09:00
import (
2017-12-31 15:25:49 +09:00
"context"
2014-04-24 16:15:57 +09:00
"errors"
"fmt"
"path/filepath"
2014-04-18 16:34:47 +09:00
"strconv"
"strings"
2014-11-27 10:25:14 +09:00
"github.com/shirou/gopsutil/internal/common"
"github.com/tklauser/go-sysconf"
2014-04-18 16:34:47 +09:00
)
var ClocksPerSec = float64(100)
2015-07-17 21:46:26 +09:00
func init() {
clkTck, err := sysconf.Sysconf(sysconf.SC_CLK_TCK)
2015-07-17 21:46:26 +09:00
// ignore errors
if err == nil {
ClocksPerSec = float64(clkTck)
2015-07-17 21:46:26 +09:00
}
}
func Times(percpu bool) ([]TimesStat, error) {
2017-12-31 15:25:49 +09:00
return TimesWithContext(context.Background(), percpu)
}
func TimesWithContext(ctx context.Context, percpu bool) ([]TimesStat, error) {
filename := common.HostProc("stat")
var lines = []string{}
2014-11-02 01:14:26 +01:00
if percpu {
statlines, err := common.ReadLines(filename)
if err != nil || len(statlines) < 2 {
return []TimesStat{}, nil
}
for _, line := range statlines[1:] {
if !strings.HasPrefix(line, "cpu") {
break
}
lines = append(lines, line)
}
2014-11-02 01:14:26 +01:00
} else {
2014-11-27 10:25:14 +09:00
lines, _ = common.ReadLinesOffsetN(filename, 0, 1)
2014-11-02 01:14:26 +01:00
}
2014-05-01 12:47:43 +09:00
ret := make([]TimesStat, 0, len(lines))
2014-05-01 12:47:43 +09:00
2014-04-18 16:34:47 +09:00
for _, line := range lines {
2014-04-24 16:15:57 +09:00
ct, err := parseStatLine(line)
if err != nil {
2014-04-18 16:34:47 +09:00
continue
}
ret = append(ret, *ct)
2014-04-18 16:34:47 +09:00
2014-04-24 16:15:57 +09:00
}
return ret, nil
}
2014-04-18 16:34:47 +09:00
2016-04-01 21:34:39 +09:00
func sysCPUPath(cpu int32, relPath string) string {
return common.HostSys(fmt.Sprintf("devices/system/cpu/cpu%d", cpu), relPath)
}
func finishCPUInfo(c *InfoStat) error {
var lines []string
var err error
var value float64
if len(c.CoreID) == 0 {
lines, err = common.ReadLines(sysCPUPath(c.CPU, "topology/core_id"))
if err == nil {
c.CoreID = lines[0]
}
}
// override the value of c.Mhz with cpufreq/cpuinfo_max_freq regardless
// of the value from /proc/cpuinfo because we want to report the maximum
// clock-speed of the CPU for c.Mhz, matching the behaviour of Windows
lines, err = common.ReadLines(sysCPUPath(c.CPU, "cpufreq/cpuinfo_max_freq"))
// if we encounter errors below such as there are no cpuinfo_max_freq file,
// we just ignore. so let Mhz is 0.
2019-11-14 21:47:42 +09:00
if err != nil || len(lines) == 0 {
return nil
}
value, err = strconv.ParseFloat(lines[0], 64)
if err != nil {
return nil
}
2017-02-01 23:05:29 +00:00
c.Mhz = value / 1000.0 // value is in kHz
2017-03-23 21:36:54 +09:00
if c.Mhz > 9999 {
c.Mhz = c.Mhz / 1000.0 // value in Hz
}
return nil
}
// CPUInfo on linux will return 1 item per physical thread.
//
// CPUs have three levels of counting: sockets, cores, threads.
// Cores with HyperThreading count as having 2 threads per core.
// Sockets often come with many physical CPU cores.
// For example a single socket board with two cores each with HT will
// return 4 CPUInfoStat structs on Linux and the "Cores" field set to 1.
func Info() ([]InfoStat, error) {
2017-12-31 15:25:49 +09:00
return InfoWithContext(context.Background())
}
func InfoWithContext(ctx context.Context) ([]InfoStat, error) {
filename := common.HostProc("cpuinfo")
2014-11-27 10:25:14 +09:00
lines, _ := common.ReadLines(filename)
2014-05-16 18:12:18 +09:00
var ret []InfoStat
var processorName string
2014-05-16 18:12:18 +09:00
c := InfoStat{CPU: -1, Cores: 1}
2014-05-16 18:12:18 +09:00
for _, line := range lines {
fields := strings.Split(line, ":")
2014-05-16 18:39:17 +09:00
if len(fields) < 2 {
2014-05-16 18:12:18 +09:00
continue
}
key := strings.TrimSpace(fields[0])
value := strings.TrimSpace(fields[1])
switch key {
case "Processor":
processorName = value
2014-05-16 18:12:18 +09:00
case "processor":
if c.CPU >= 0 {
err := finishCPUInfo(&c)
if err != nil {
return ret, err
}
ret = append(ret, c)
}
c = InfoStat{Cores: 1, ModelName: processorName}
t, err := strconv.ParseInt(value, 10, 64)
if err != nil {
return ret, err
}
c.CPU = int32(t)
case "vendorId", "vendor_id":
2014-05-16 18:39:17 +09:00
c.VendorID = value
case "CPU implementer":
if value == "0x41" {
c.VendorID = "ARM"
}
2014-05-16 18:12:18 +09:00
case "cpu family":
c.Family = value
case "model", "CPU part":
2014-05-16 18:12:18 +09:00
c.Model = value
case "model name", "cpu":
2014-05-16 18:12:18 +09:00
c.ModelName = value
if strings.Contains(value, "POWER8") ||
2017-02-01 23:05:29 +00:00
strings.Contains(value, "POWER7") {
c.Model = strings.Split(value, " ")[0]
c.Family = "POWER"
c.VendorID = "IBM"
}
case "stepping", "revision", "CPU revision":
val := value
if key == "revision" {
val = strings.Split(value, ".")[0]
}
t, err := strconv.ParseInt(val, 10, 64)
if err != nil {
return ret, err
}
c.Stepping = int32(t)
case "cpu MHz", "clock":
// treat this as the fallback value, thus we ignore error
if t, err := strconv.ParseFloat(strings.Replace(value, "MHz", "", 1), 64); err == nil {
c.Mhz = t
}
2014-05-16 18:12:18 +09:00
case "cache size":
t, err := strconv.ParseInt(strings.Replace(value, " KB", "", 1), 10, 64)
if err != nil {
return ret, err
}
c.CacheSize = int32(t)
2014-05-16 18:12:18 +09:00
case "physical id":
c.PhysicalID = value
case "core id":
c.CoreID = value
case "flags", "Features":
c.Flags = strings.FieldsFunc(value, func(r rune) bool {
return r == ',' || r == ' '
})
case "microcode":
c.Microcode = value
}
}
if c.CPU >= 0 {
err := finishCPUInfo(&c)
if err != nil {
return ret, err
2014-05-16 18:12:18 +09:00
}
ret = append(ret, c)
2014-05-16 18:12:18 +09:00
}
return ret, nil
}
func parseStatLine(line string) (*TimesStat, error) {
2014-04-24 16:15:57 +09:00
fields := strings.Fields(line)
2017-02-15 11:25:49 -05:00
if len(fields) == 0 {
return nil, errors.New("stat does not contain cpu info")
}
2014-04-24 16:15:57 +09:00
if strings.HasPrefix(fields[0], "cpu") == false {
return nil, errors.New("not contain cpu")
2014-04-18 16:34:47 +09:00
}
2014-04-24 16:15:57 +09:00
cpu := fields[0]
if cpu == "cpu" {
cpu = "cpu-total"
}
user, err := strconv.ParseFloat(fields[1], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
nice, err := strconv.ParseFloat(fields[2], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
system, err := strconv.ParseFloat(fields[3], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
idle, err := strconv.ParseFloat(fields[4], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
iowait, err := strconv.ParseFloat(fields[5], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
irq, err := strconv.ParseFloat(fields[6], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
softirq, err := strconv.ParseFloat(fields[7], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
2015-02-13 14:55:42 +09:00
ct := &TimesStat{
2014-04-30 16:16:07 +09:00
CPU: cpu,
User: user / ClocksPerSec,
Nice: nice / ClocksPerSec,
System: system / ClocksPerSec,
Idle: idle / ClocksPerSec,
Iowait: iowait / ClocksPerSec,
Irq: irq / ClocksPerSec,
Softirq: softirq / ClocksPerSec,
2014-04-24 16:15:57 +09:00
}
if len(fields) > 8 { // Linux >= 2.6.11
steal, err := strconv.ParseFloat(fields[8], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
ct.Steal = steal / ClocksPerSec
2014-04-24 16:15:57 +09:00
}
if len(fields) > 9 { // Linux >= 2.6.24
guest, err := strconv.ParseFloat(fields[9], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
ct.Guest = guest / ClocksPerSec
2014-04-24 16:15:57 +09:00
}
if len(fields) > 10 { // Linux >= 3.2.0
guestNice, err := strconv.ParseFloat(fields[10], 64)
2014-09-20 10:22:41 +09:00
if err != nil {
return nil, err
}
ct.GuestNice = guestNice / ClocksPerSec
2014-04-24 16:15:57 +09:00
}
return ct, nil
2014-04-18 16:34:47 +09:00
}
func CountsWithContext(ctx context.Context, logical bool) (int, error) {
if logical {
ret := 0
// https://github.com/giampaolo/psutil/blob/d01a9eaa35a8aadf6c519839e987a49d8be2d891/psutil/_pslinux.py#L599
procCpuinfo := common.HostProc("cpuinfo")
lines, err := common.ReadLines(procCpuinfo)
if err == nil {
for _, line := range lines {
line = strings.ToLower(line)
if strings.HasPrefix(line, "processor") {
_, err = strconv.Atoi(strings.TrimSpace(line[strings.IndexByte(line, ':')+1:]))
if err == nil {
ret++
}
}
}
}
if ret == 0 {
procStat := common.HostProc("stat")
lines, err = common.ReadLines(procStat)
if err != nil {
return 0, err
}
for _, line := range lines {
if len(line) >= 4 && strings.HasPrefix(line, "cpu") && '0' <= line[3] && line[3] <= '9' { // `^cpu\d` regexp matching
ret++
}
}
}
return ret, nil
}
// physical cores
// https://github.com/giampaolo/psutil/blob/8415355c8badc9c94418b19bdf26e622f06f0cce/psutil/_pslinux.py#L615-L628
var threadSiblingsLists = make(map[string]bool)
// These 2 files are the same but */core_cpus_list is newer while */thread_siblings_list is deprecated and may disappear in the future.
// https://www.kernel.org/doc/Documentation/admin-guide/cputopology.rst
// https://github.com/giampaolo/psutil/pull/1727#issuecomment-707624964
// https://lkml.org/lkml/2019/2/26/41
for _, glob := range []string{"devices/system/cpu/cpu[0-9]*/topology/core_cpus_list", "devices/system/cpu/cpu[0-9]*/topology/thread_siblings_list"} {
if files, err := filepath.Glob(common.HostSys(glob)); err == nil {
for _, file := range files {
lines, err := common.ReadLines(file)
if err != nil || len(lines) != 1 {
continue
}
threadSiblingsLists[lines[0]] = true
}
ret := len(threadSiblingsLists)
if ret != 0 {
return ret, nil
}
}
}
// https://github.com/giampaolo/psutil/blob/122174a10b75c9beebe15f6c07dcf3afbe3b120d/psutil/_pslinux.py#L631-L652
filename := common.HostProc("cpuinfo")
lines, err := common.ReadLines(filename)
if err != nil {
return 0, err
}
mapping := make(map[int]int)
currentInfo := make(map[string]int)
for _, line := range lines {
line = strings.ToLower(strings.TrimSpace(line))
if line == "" {
// new section
id, okID := currentInfo["physical id"]
cores, okCores := currentInfo["cpu cores"]
if okID && okCores {
mapping[id] = cores
}
currentInfo = make(map[string]int)
continue
}
fields := strings.Split(line, ":")
if len(fields) < 2 {
continue
}
fields[0] = strings.TrimSpace(fields[0])
if fields[0] == "physical id" || fields[0] == "cpu cores" {
val, err := strconv.Atoi(strings.TrimSpace(fields[1]))
if err != nil {
continue
}
currentInfo[fields[0]] = val
}
}
ret := 0
for _, v := range mapping {
ret += v
}
return ret, nil
}