Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
481 lines
12 KiB
481 lines
12 KiB
// Copyright 2019 The Prometheus Authors |
|
// Licensed under the Apache License, Version 2.0 (the "License"); |
|
// you may not use this file except in compliance with the License. |
|
// You may obtain a copy of the License at |
|
// |
|
// http://www.apache.org/licenses/LICENSE-2.0 |
|
// |
|
// Unless required by applicable law or agreed to in writing, software |
|
// distributed under the License is distributed on an "AS IS" BASIS, |
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
// See the License for the specific language governing permissions and |
|
// limitations under the License. |
|
|
|
// +build linux |
|
|
|
package procfs |
|
|
|
import ( |
|
"bufio" |
|
"bytes" |
|
"errors" |
|
"fmt" |
|
"regexp" |
|
"strconv" |
|
"strings" |
|
|
|
"github.com/prometheus/procfs/internal/util" |
|
) |
|
|
|
// CPUInfo contains general information about a system CPU found in /proc/cpuinfo |
|
type CPUInfo struct { |
|
Processor uint |
|
VendorID string |
|
CPUFamily string |
|
Model string |
|
ModelName string |
|
Stepping string |
|
Microcode string |
|
CPUMHz float64 |
|
CacheSize string |
|
PhysicalID string |
|
Siblings uint |
|
CoreID string |
|
CPUCores uint |
|
APICID string |
|
InitialAPICID string |
|
FPU string |
|
FPUException string |
|
CPUIDLevel uint |
|
WP string |
|
Flags []string |
|
Bugs []string |
|
BogoMips float64 |
|
CLFlushSize uint |
|
CacheAlignment uint |
|
AddressSizes string |
|
PowerManagement string |
|
} |
|
|
|
var ( |
|
cpuinfoClockRegexp = regexp.MustCompile(`([\d.]+)`) |
|
cpuinfoS390XProcessorRegexp = regexp.MustCompile(`^processor\s+(\d+):.*`) |
|
) |
|
|
|
// CPUInfo returns information about current system CPUs. |
|
// See https://www.kernel.org/doc/Documentation/filesystems/proc.txt |
|
func (fs FS) CPUInfo() ([]CPUInfo, error) { |
|
data, err := util.ReadFileNoStat(fs.proc.Path("cpuinfo")) |
|
if err != nil { |
|
return nil, err |
|
} |
|
return parseCPUInfo(data) |
|
} |
|
|
|
func parseCPUInfoX86(info []byte) ([]CPUInfo, error) { |
|
scanner := bufio.NewScanner(bytes.NewReader(info)) |
|
|
|
// find the first "processor" line |
|
firstLine := firstNonEmptyLine(scanner) |
|
if !strings.HasPrefix(firstLine, "processor") || !strings.Contains(firstLine, ":") { |
|
return nil, fmt.Errorf("invalid cpuinfo file: %q", firstLine) |
|
} |
|
field := strings.SplitN(firstLine, ": ", 2) |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
firstcpu := CPUInfo{Processor: uint(v)} |
|
cpuinfo := []CPUInfo{firstcpu} |
|
i := 0 |
|
|
|
for scanner.Scan() { |
|
line := scanner.Text() |
|
if !strings.Contains(line, ":") { |
|
continue |
|
} |
|
field := strings.SplitN(line, ": ", 2) |
|
switch strings.TrimSpace(field[0]) { |
|
case "processor": |
|
cpuinfo = append(cpuinfo, CPUInfo{}) // start of the next processor |
|
i++ |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].Processor = uint(v) |
|
case "vendor", "vendor_id": |
|
cpuinfo[i].VendorID = field[1] |
|
case "cpu family": |
|
cpuinfo[i].CPUFamily = field[1] |
|
case "model": |
|
cpuinfo[i].Model = field[1] |
|
case "model name": |
|
cpuinfo[i].ModelName = field[1] |
|
case "stepping": |
|
cpuinfo[i].Stepping = field[1] |
|
case "microcode": |
|
cpuinfo[i].Microcode = field[1] |
|
case "cpu MHz": |
|
v, err := strconv.ParseFloat(field[1], 64) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].CPUMHz = v |
|
case "cache size": |
|
cpuinfo[i].CacheSize = field[1] |
|
case "physical id": |
|
cpuinfo[i].PhysicalID = field[1] |
|
case "siblings": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].Siblings = uint(v) |
|
case "core id": |
|
cpuinfo[i].CoreID = field[1] |
|
case "cpu cores": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].CPUCores = uint(v) |
|
case "apicid": |
|
cpuinfo[i].APICID = field[1] |
|
case "initial apicid": |
|
cpuinfo[i].InitialAPICID = field[1] |
|
case "fpu": |
|
cpuinfo[i].FPU = field[1] |
|
case "fpu_exception": |
|
cpuinfo[i].FPUException = field[1] |
|
case "cpuid level": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].CPUIDLevel = uint(v) |
|
case "wp": |
|
cpuinfo[i].WP = field[1] |
|
case "flags": |
|
cpuinfo[i].Flags = strings.Fields(field[1]) |
|
case "bugs": |
|
cpuinfo[i].Bugs = strings.Fields(field[1]) |
|
case "bogomips": |
|
v, err := strconv.ParseFloat(field[1], 64) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].BogoMips = v |
|
case "clflush size": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].CLFlushSize = uint(v) |
|
case "cache_alignment": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].CacheAlignment = uint(v) |
|
case "address sizes": |
|
cpuinfo[i].AddressSizes = field[1] |
|
case "power management": |
|
cpuinfo[i].PowerManagement = field[1] |
|
} |
|
} |
|
return cpuinfo, nil |
|
} |
|
|
|
func parseCPUInfoARM(info []byte) ([]CPUInfo, error) { |
|
scanner := bufio.NewScanner(bytes.NewReader(info)) |
|
|
|
firstLine := firstNonEmptyLine(scanner) |
|
match, _ := regexp.MatchString("^[Pp]rocessor", firstLine) |
|
if !match || !strings.Contains(firstLine, ":") { |
|
return nil, fmt.Errorf("invalid cpuinfo file: %q", firstLine) |
|
} |
|
field := strings.SplitN(firstLine, ": ", 2) |
|
cpuinfo := []CPUInfo{} |
|
featuresLine := "" |
|
commonCPUInfo := CPUInfo{} |
|
i := 0 |
|
if strings.TrimSpace(field[0]) == "Processor" { |
|
commonCPUInfo = CPUInfo{ModelName: field[1]} |
|
i = -1 |
|
} else { |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
firstcpu := CPUInfo{Processor: uint(v)} |
|
cpuinfo = []CPUInfo{firstcpu} |
|
} |
|
|
|
for scanner.Scan() { |
|
line := scanner.Text() |
|
if !strings.Contains(line, ":") { |
|
continue |
|
} |
|
field := strings.SplitN(line, ": ", 2) |
|
switch strings.TrimSpace(field[0]) { |
|
case "processor": |
|
cpuinfo = append(cpuinfo, commonCPUInfo) // start of the next processor |
|
i++ |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].Processor = uint(v) |
|
case "BogoMIPS": |
|
if i == -1 { |
|
cpuinfo = append(cpuinfo, commonCPUInfo) // There is only one processor |
|
i++ |
|
cpuinfo[i].Processor = 0 |
|
} |
|
v, err := strconv.ParseFloat(field[1], 64) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].BogoMips = v |
|
case "Features": |
|
featuresLine = line |
|
case "model name": |
|
cpuinfo[i].ModelName = field[1] |
|
} |
|
} |
|
fields := strings.SplitN(featuresLine, ": ", 2) |
|
for i := range cpuinfo { |
|
cpuinfo[i].Flags = strings.Fields(fields[1]) |
|
} |
|
return cpuinfo, nil |
|
|
|
} |
|
|
|
func parseCPUInfoS390X(info []byte) ([]CPUInfo, error) { |
|
scanner := bufio.NewScanner(bytes.NewReader(info)) |
|
|
|
firstLine := firstNonEmptyLine(scanner) |
|
if !strings.HasPrefix(firstLine, "vendor_id") || !strings.Contains(firstLine, ":") { |
|
return nil, fmt.Errorf("invalid cpuinfo file: %q", firstLine) |
|
} |
|
field := strings.SplitN(firstLine, ": ", 2) |
|
cpuinfo := []CPUInfo{} |
|
commonCPUInfo := CPUInfo{VendorID: field[1]} |
|
|
|
for scanner.Scan() { |
|
line := scanner.Text() |
|
if !strings.Contains(line, ":") { |
|
continue |
|
} |
|
field := strings.SplitN(line, ": ", 2) |
|
switch strings.TrimSpace(field[0]) { |
|
case "bogomips per cpu": |
|
v, err := strconv.ParseFloat(field[1], 64) |
|
if err != nil { |
|
return nil, err |
|
} |
|
commonCPUInfo.BogoMips = v |
|
case "features": |
|
commonCPUInfo.Flags = strings.Fields(field[1]) |
|
} |
|
if strings.HasPrefix(line, "processor") { |
|
match := cpuinfoS390XProcessorRegexp.FindStringSubmatch(line) |
|
if len(match) < 2 { |
|
return nil, fmt.Errorf("invalid cpuinfo file: %q", firstLine) |
|
} |
|
cpu := commonCPUInfo |
|
v, err := strconv.ParseUint(match[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpu.Processor = uint(v) |
|
cpuinfo = append(cpuinfo, cpu) |
|
} |
|
if strings.HasPrefix(line, "cpu number") { |
|
break |
|
} |
|
} |
|
|
|
i := 0 |
|
for scanner.Scan() { |
|
line := scanner.Text() |
|
if !strings.Contains(line, ":") { |
|
continue |
|
} |
|
field := strings.SplitN(line, ": ", 2) |
|
switch strings.TrimSpace(field[0]) { |
|
case "cpu number": |
|
i++ |
|
case "cpu MHz dynamic": |
|
clock := cpuinfoClockRegexp.FindString(strings.TrimSpace(field[1])) |
|
v, err := strconv.ParseFloat(clock, 64) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].CPUMHz = v |
|
case "physical id": |
|
cpuinfo[i].PhysicalID = field[1] |
|
case "core id": |
|
cpuinfo[i].CoreID = field[1] |
|
case "cpu cores": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].CPUCores = uint(v) |
|
case "siblings": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].Siblings = uint(v) |
|
} |
|
} |
|
|
|
return cpuinfo, nil |
|
} |
|
|
|
func parseCPUInfoMips(info []byte) ([]CPUInfo, error) { |
|
scanner := bufio.NewScanner(bytes.NewReader(info)) |
|
|
|
// find the first "processor" line |
|
firstLine := firstNonEmptyLine(scanner) |
|
if !strings.HasPrefix(firstLine, "system type") || !strings.Contains(firstLine, ":") { |
|
return nil, fmt.Errorf("invalid cpuinfo file: %q", firstLine) |
|
} |
|
field := strings.SplitN(firstLine, ": ", 2) |
|
cpuinfo := []CPUInfo{} |
|
systemType := field[1] |
|
|
|
i := 0 |
|
|
|
for scanner.Scan() { |
|
line := scanner.Text() |
|
if !strings.Contains(line, ":") { |
|
continue |
|
} |
|
field := strings.SplitN(line, ": ", 2) |
|
switch strings.TrimSpace(field[0]) { |
|
case "processor": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
i = int(v) |
|
cpuinfo = append(cpuinfo, CPUInfo{}) // start of the next processor |
|
cpuinfo[i].Processor = uint(v) |
|
cpuinfo[i].VendorID = systemType |
|
case "cpu model": |
|
cpuinfo[i].ModelName = field[1] |
|
case "BogoMIPS": |
|
v, err := strconv.ParseFloat(field[1], 64) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].BogoMips = v |
|
} |
|
} |
|
return cpuinfo, nil |
|
} |
|
|
|
func parseCPUInfoPPC(info []byte) ([]CPUInfo, error) { |
|
scanner := bufio.NewScanner(bytes.NewReader(info)) |
|
|
|
firstLine := firstNonEmptyLine(scanner) |
|
if !strings.HasPrefix(firstLine, "processor") || !strings.Contains(firstLine, ":") { |
|
return nil, fmt.Errorf("invalid cpuinfo file: %q", firstLine) |
|
} |
|
field := strings.SplitN(firstLine, ": ", 2) |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
firstcpu := CPUInfo{Processor: uint(v)} |
|
cpuinfo := []CPUInfo{firstcpu} |
|
i := 0 |
|
|
|
for scanner.Scan() { |
|
line := scanner.Text() |
|
if !strings.Contains(line, ":") { |
|
continue |
|
} |
|
field := strings.SplitN(line, ": ", 2) |
|
switch strings.TrimSpace(field[0]) { |
|
case "processor": |
|
cpuinfo = append(cpuinfo, CPUInfo{}) // start of the next processor |
|
i++ |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].Processor = uint(v) |
|
case "cpu": |
|
cpuinfo[i].VendorID = field[1] |
|
case "clock": |
|
clock := cpuinfoClockRegexp.FindString(strings.TrimSpace(field[1])) |
|
v, err := strconv.ParseFloat(clock, 64) |
|
if err != nil { |
|
return nil, err |
|
} |
|
cpuinfo[i].CPUMHz = v |
|
} |
|
} |
|
return cpuinfo, nil |
|
} |
|
|
|
func parseCPUInfoRISCV(info []byte) ([]CPUInfo, error) { |
|
scanner := bufio.NewScanner(bytes.NewReader(info)) |
|
|
|
firstLine := firstNonEmptyLine(scanner) |
|
if !strings.HasPrefix(firstLine, "processor") || !strings.Contains(firstLine, ":") { |
|
return nil, fmt.Errorf("invalid cpuinfo file: %q", firstLine) |
|
} |
|
field := strings.SplitN(firstLine, ": ", 2) |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
firstcpu := CPUInfo{Processor: uint(v)} |
|
cpuinfo := []CPUInfo{firstcpu} |
|
i := 0 |
|
|
|
for scanner.Scan() { |
|
line := scanner.Text() |
|
if !strings.Contains(line, ":") { |
|
continue |
|
} |
|
field := strings.SplitN(line, ": ", 2) |
|
switch strings.TrimSpace(field[0]) { |
|
case "processor": |
|
v, err := strconv.ParseUint(field[1], 0, 32) |
|
if err != nil { |
|
return nil, err |
|
} |
|
i = int(v) |
|
cpuinfo = append(cpuinfo, CPUInfo{}) // start of the next processor |
|
cpuinfo[i].Processor = uint(v) |
|
case "hart": |
|
cpuinfo[i].CoreID = field[1] |
|
case "isa": |
|
cpuinfo[i].ModelName = field[1] |
|
} |
|
} |
|
return cpuinfo, nil |
|
} |
|
|
|
func parseCPUInfoDummy(_ []byte) ([]CPUInfo, error) { // nolint:unused,deadcode |
|
return nil, errors.New("not implemented") |
|
} |
|
|
|
// firstNonEmptyLine advances the scanner to the first non-empty line |
|
// and returns the contents of that line |
|
func firstNonEmptyLine(scanner *bufio.Scanner) string { |
|
for scanner.Scan() { |
|
line := scanner.Text() |
|
if strings.TrimSpace(line) != "" { |
|
return line |
|
} |
|
} |
|
return "" |
|
}
|
|
|