mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-31 22:01:06 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			320 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			320 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2015 Google Inc. All Rights Reserved.
 | |
| //
 | |
| // 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.
 | |
| 
 | |
| // The machine package contains functions that extract machine-level specs.
 | |
| package machine
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io/ioutil"
 | |
| 	"regexp"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	// s390/s390x changes
 | |
| 	"runtime"
 | |
| 	"syscall"
 | |
| 
 | |
| 	info "github.com/google/cadvisor/info/v1"
 | |
| 	"github.com/google/cadvisor/utils"
 | |
| 	"github.com/google/cadvisor/utils/sysfs"
 | |
| 	"github.com/google/cadvisor/utils/sysinfo"
 | |
| 
 | |
| 	"github.com/golang/glog"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	cpuRegExp  = regexp.MustCompile(`^processor\s*:\s*([0-9]+)$`)
 | |
| 	coreRegExp = regexp.MustCompile(`^core id\s*:\s*([0-9]+)$`)
 | |
| 	nodeRegExp = regexp.MustCompile(`^physical id\s*:\s*([0-9]+)$`)
 | |
| 	// Power systems have a different format so cater for both
 | |
| 	cpuClockSpeedMHz     = regexp.MustCompile(`(?:cpu MHz|clock)\s*:\s*([0-9]+\.[0-9]+)(?:MHz)?`)
 | |
| 	memoryCapacityRegexp = regexp.MustCompile(`MemTotal:\s*([0-9]+) kB`)
 | |
| 	swapCapacityRegexp   = regexp.MustCompile(`SwapTotal:\s*([0-9]+) kB`)
 | |
| )
 | |
| 
 | |
| const maxFreqFile = "/sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_max_freq"
 | |
| 
 | |
| // GetClockSpeed returns the CPU clock speed, given a []byte formatted as the /proc/cpuinfo file.
 | |
| func GetClockSpeed(procInfo []byte) (uint64, error) {
 | |
| 	// s390/s390x, aarch64 and arm32 changes
 | |
| 	if isSystemZ() || isAArch64() || isArm32() {
 | |
| 		return 0, nil
 | |
| 	}
 | |
| 
 | |
| 	// First look through sys to find a max supported cpu frequency.
 | |
| 	if utils.FileExists(maxFreqFile) {
 | |
| 		val, err := ioutil.ReadFile(maxFreqFile)
 | |
| 		if err != nil {
 | |
| 			return 0, err
 | |
| 		}
 | |
| 		var maxFreq uint64
 | |
| 		n, err := fmt.Sscanf(string(val), "%d", &maxFreq)
 | |
| 		if err != nil || n != 1 {
 | |
| 			return 0, fmt.Errorf("could not parse frequency %q", val)
 | |
| 		}
 | |
| 		return maxFreq, nil
 | |
| 	}
 | |
| 	// Fall back to /proc/cpuinfo
 | |
| 	matches := cpuClockSpeedMHz.FindSubmatch(procInfo)
 | |
| 	if len(matches) != 2 {
 | |
| 		return 0, fmt.Errorf("could not detect clock speed from output: %q", string(procInfo))
 | |
| 	}
 | |
| 
 | |
| 	speed, err := strconv.ParseFloat(string(matches[1]), 64)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	// Convert to kHz
 | |
| 	return uint64(speed * 1000), nil
 | |
| }
 | |
| 
 | |
| // GetMachineMemoryCapacity returns the machine's total memory from /proc/meminfo.
 | |
| // Returns the total memory capacity as an uint64 (number of bytes).
 | |
| func GetMachineMemoryCapacity() (uint64, error) {
 | |
| 	out, err := ioutil.ReadFile("/proc/meminfo")
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 
 | |
| 	memoryCapacity, err := parseCapacity(out, memoryCapacityRegexp)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	return memoryCapacity, err
 | |
| }
 | |
| 
 | |
| // GetMachineSwapCapacity returns the machine's total swap from /proc/meminfo.
 | |
| // Returns the total swap capacity as an uint64 (number of bytes).
 | |
| func GetMachineSwapCapacity() (uint64, error) {
 | |
| 	out, err := ioutil.ReadFile("/proc/meminfo")
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 
 | |
| 	swapCapacity, err := parseCapacity(out, swapCapacityRegexp)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	return swapCapacity, err
 | |
| }
 | |
| 
 | |
| // parseCapacity matches a Regexp in a []byte, returning the resulting value in bytes.
 | |
| // Assumes that the value matched by the Regexp is in KB.
 | |
| func parseCapacity(b []byte, r *regexp.Regexp) (uint64, error) {
 | |
| 	matches := r.FindSubmatch(b)
 | |
| 	if len(matches) != 2 {
 | |
| 		return 0, fmt.Errorf("failed to match regexp in output: %q", string(b))
 | |
| 	}
 | |
| 	m, err := strconv.ParseUint(string(matches[1]), 10, 64)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 
 | |
| 	// Convert to bytes.
 | |
| 	return m * 1024, err
 | |
| }
 | |
| 
 | |
| func GetTopology(sysFs sysfs.SysFs, cpuinfo string) ([]info.Node, int, error) {
 | |
| 	nodes := []info.Node{}
 | |
| 
 | |
| 	// s390/s390x changes
 | |
| 	if true == isSystemZ() {
 | |
| 		return nodes, getNumCores(), nil
 | |
| 	}
 | |
| 
 | |
| 	numCores := 0
 | |
| 	lastThread := -1
 | |
| 	lastCore := -1
 | |
| 	lastNode := -1
 | |
| 	for _, line := range strings.Split(cpuinfo, "\n") {
 | |
| 		if line == "" {
 | |
| 			continue
 | |
| 		}
 | |
| 		ok, val, err := extractValue(line, cpuRegExp)
 | |
| 		if err != nil {
 | |
| 			return nil, -1, fmt.Errorf("could not parse cpu info from %q: %v", line, err)
 | |
| 		}
 | |
| 		if ok {
 | |
| 			thread := val
 | |
| 			numCores++
 | |
| 			if lastThread != -1 {
 | |
| 				// New cpu section. Save last one.
 | |
| 				nodeIdx, err := addNode(&nodes, lastNode)
 | |
| 				if err != nil {
 | |
| 					return nil, -1, fmt.Errorf("failed to add node %d: %v", lastNode, err)
 | |
| 				}
 | |
| 				nodes[nodeIdx].AddThread(lastThread, lastCore)
 | |
| 				lastCore = -1
 | |
| 				lastNode = -1
 | |
| 			}
 | |
| 			lastThread = thread
 | |
| 			continue
 | |
| 		}
 | |
| 		ok, val, err = extractValue(line, coreRegExp)
 | |
| 		if err != nil {
 | |
| 			return nil, -1, fmt.Errorf("could not parse core info from %q: %v", line, err)
 | |
| 		}
 | |
| 		if ok {
 | |
| 			lastCore = val
 | |
| 			continue
 | |
| 		}
 | |
| 		ok, val, err = extractValue(line, nodeRegExp)
 | |
| 		if err != nil {
 | |
| 			return nil, -1, fmt.Errorf("could not parse node info from %q: %v", line, err)
 | |
| 		}
 | |
| 		if ok {
 | |
| 			lastNode = val
 | |
| 			continue
 | |
| 		}
 | |
| 	}
 | |
| 	nodeIdx, err := addNode(&nodes, lastNode)
 | |
| 	if err != nil {
 | |
| 		return nil, -1, fmt.Errorf("failed to add node %d: %v", lastNode, err)
 | |
| 	}
 | |
| 	nodes[nodeIdx].AddThread(lastThread, lastCore)
 | |
| 	if numCores < 1 {
 | |
| 		return nil, numCores, fmt.Errorf("could not detect any cores")
 | |
| 	}
 | |
| 	for idx, node := range nodes {
 | |
| 		caches, err := sysinfo.GetCacheInfo(sysFs, node.Cores[0].Threads[0])
 | |
| 		if err != nil {
 | |
| 			glog.Errorf("failed to get cache information for node %d: %v", node.Id, err)
 | |
| 			continue
 | |
| 		}
 | |
| 		numThreadsPerCore := len(node.Cores[0].Threads)
 | |
| 		numThreadsPerNode := len(node.Cores) * numThreadsPerCore
 | |
| 		for _, cache := range caches {
 | |
| 			c := info.Cache{
 | |
| 				Size:  cache.Size,
 | |
| 				Level: cache.Level,
 | |
| 				Type:  cache.Type,
 | |
| 			}
 | |
| 			if cache.Cpus == numThreadsPerNode && cache.Level > 2 {
 | |
| 				// Add a node-level cache.
 | |
| 				nodes[idx].AddNodeCache(c)
 | |
| 			} else if cache.Cpus == numThreadsPerCore {
 | |
| 				// Add to each core.
 | |
| 				nodes[idx].AddPerCoreCache(c)
 | |
| 			}
 | |
| 			// Ignore unknown caches.
 | |
| 		}
 | |
| 	}
 | |
| 	return nodes, numCores, nil
 | |
| }
 | |
| 
 | |
| func extractValue(s string, r *regexp.Regexp) (bool, int, error) {
 | |
| 	matches := r.FindSubmatch([]byte(s))
 | |
| 	if len(matches) == 2 {
 | |
| 		val, err := strconv.ParseInt(string(matches[1]), 10, 32)
 | |
| 		if err != nil {
 | |
| 			return false, -1, err
 | |
| 		}
 | |
| 		return true, int(val), nil
 | |
| 	}
 | |
| 	return false, -1, nil
 | |
| }
 | |
| 
 | |
| func findNode(nodes []info.Node, id int) (bool, int) {
 | |
| 	for i, n := range nodes {
 | |
| 		if n.Id == id {
 | |
| 			return true, i
 | |
| 		}
 | |
| 	}
 | |
| 	return false, -1
 | |
| }
 | |
| 
 | |
| func addNode(nodes *[]info.Node, id int) (int, error) {
 | |
| 	var idx int
 | |
| 	if id == -1 {
 | |
| 		// Some VMs don't fill topology data. Export single package.
 | |
| 		id = 0
 | |
| 	}
 | |
| 
 | |
| 	ok, idx := findNode(*nodes, id)
 | |
| 	if !ok {
 | |
| 		// New node
 | |
| 		node := info.Node{Id: id}
 | |
| 		// Add per-node memory information.
 | |
| 		meminfo := fmt.Sprintf("/sys/devices/system/node/node%d/meminfo", id)
 | |
| 		out, err := ioutil.ReadFile(meminfo)
 | |
| 		// Ignore if per-node info is not available.
 | |
| 		if err == nil {
 | |
| 			m, err := parseCapacity(out, memoryCapacityRegexp)
 | |
| 			if err != nil {
 | |
| 				return -1, err
 | |
| 			}
 | |
| 			node.Memory = uint64(m)
 | |
| 		}
 | |
| 		*nodes = append(*nodes, node)
 | |
| 		idx = len(*nodes) - 1
 | |
| 	}
 | |
| 	return idx, nil
 | |
| }
 | |
| 
 | |
| // s390/s390x changes
 | |
| func getMachineArch() (string, error) {
 | |
| 	uname := syscall.Utsname{}
 | |
| 	err := syscall.Uname(&uname)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	var arch string
 | |
| 	for _, val := range uname.Machine {
 | |
| 		arch += string(int(val))
 | |
| 	}
 | |
| 
 | |
| 	return arch, nil
 | |
| }
 | |
| 
 | |
| // arm32 chanes
 | |
| func isArm32() bool {
 | |
| 	arch, err := getMachineArch()
 | |
| 	if err == nil {
 | |
| 		return strings.Contains(arch, "arm")
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| // aarch64 changes
 | |
| func isAArch64() bool {
 | |
| 	arch, err := getMachineArch()
 | |
| 	if err == nil {
 | |
| 		return strings.Contains(arch, "aarch64")
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| // s390/s390x changes
 | |
| func isSystemZ() bool {
 | |
| 	arch, err := getMachineArch()
 | |
| 	if err == nil {
 | |
| 		return strings.Contains(arch, "390")
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| // s390/s390x changes
 | |
| func getNumCores() int {
 | |
| 	maxProcs := runtime.GOMAXPROCS(0)
 | |
| 	numCPU := runtime.NumCPU()
 | |
| 
 | |
| 	if maxProcs < numCPU {
 | |
| 		return maxProcs
 | |
| 	}
 | |
| 
 | |
| 	return numCPU
 | |
| }
 |