mirror of
https://github.com/containers/podman.git
synced 2025-12-01 18:49:18 +08:00
189 lines
4.2 KiB
Go
189 lines
4.2 KiB
Go
// Copyright 2018 Tobias Klauser
|
|
//
|
|
// 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.
|
|
|
|
package numcpus
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"golang.org/x/sys/unix"
|
|
)
|
|
|
|
const (
|
|
sysfsCPUBasePath = "/sys/devices/system/cpu"
|
|
|
|
offline = "offline"
|
|
online = "online"
|
|
possible = "possible"
|
|
present = "present"
|
|
)
|
|
|
|
func getFromCPUAffinity() (int, error) {
|
|
var cpuSet unix.CPUSet
|
|
if err := unix.SchedGetaffinity(0, &cpuSet); err != nil {
|
|
return 0, err
|
|
}
|
|
return cpuSet.Count(), nil
|
|
}
|
|
|
|
func readCPURangeWith[T any](file string, f func(cpus string) (T, error)) (T, error) {
|
|
var zero T
|
|
buf, err := os.ReadFile(filepath.Join(sysfsCPUBasePath, file))
|
|
if err != nil {
|
|
return zero, err
|
|
}
|
|
return f(strings.Trim(string(buf), "\n "))
|
|
}
|
|
|
|
func countCPURange(cpus string) (int, error) {
|
|
// Treat empty file as valid. This might be the case if there are no offline CPUs in which
|
|
// case /sys/devices/system/cpu/offline is empty.
|
|
if cpus == "" {
|
|
return 0, nil
|
|
}
|
|
|
|
n := int(0)
|
|
for _, cpuRange := range strings.Split(cpus, ",") {
|
|
if cpuRange == "" {
|
|
return 0, fmt.Errorf("empty CPU range in CPU string %q", cpus)
|
|
}
|
|
from, to, found := strings.Cut(cpuRange, "-")
|
|
first, err := strconv.ParseUint(from, 10, 32)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
if !found {
|
|
n++
|
|
continue
|
|
}
|
|
last, err := strconv.ParseUint(to, 10, 32)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
if last < first {
|
|
return 0, fmt.Errorf("last CPU in range (%d) less than first (%d)", last, first)
|
|
}
|
|
n += int(last - first + 1)
|
|
}
|
|
return n, nil
|
|
}
|
|
|
|
func listCPURange(cpus string) ([]int, error) {
|
|
// See comment in countCPURange.
|
|
if cpus == "" {
|
|
return []int{}, nil
|
|
}
|
|
|
|
list := []int{}
|
|
for _, cpuRange := range strings.Split(cpus, ",") {
|
|
if cpuRange == "" {
|
|
return nil, fmt.Errorf("empty CPU range in CPU string %q", cpus)
|
|
}
|
|
from, to, found := strings.Cut(cpuRange, "-")
|
|
first, err := strconv.ParseUint(from, 10, 32)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if !found {
|
|
// range containing a single element
|
|
list = append(list, int(first))
|
|
continue
|
|
}
|
|
last, err := strconv.ParseUint(to, 10, 32)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if last < first {
|
|
return nil, fmt.Errorf("last CPU in range (%d) less than first (%d)", last, first)
|
|
}
|
|
for cpu := int(first); cpu <= int(last); cpu++ {
|
|
list = append(list, cpu)
|
|
}
|
|
}
|
|
return list, nil
|
|
}
|
|
|
|
func getConfigured() (int, error) {
|
|
d, err := os.Open(sysfsCPUBasePath)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
defer d.Close()
|
|
fis, err := d.Readdir(-1)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
count := 0
|
|
for _, fi := range fis {
|
|
if name := fi.Name(); fi.IsDir() && strings.HasPrefix(name, "cpu") {
|
|
_, err := strconv.ParseInt(name[3:], 10, 64)
|
|
if err == nil {
|
|
count++
|
|
}
|
|
}
|
|
}
|
|
return count, nil
|
|
}
|
|
|
|
func getKernelMax() (int, error) {
|
|
buf, err := os.ReadFile(filepath.Join(sysfsCPUBasePath, "kernel_max"))
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
n, err := strconv.ParseInt(strings.Trim(string(buf), "\n "), 10, 32)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return int(n), nil
|
|
}
|
|
|
|
func getOffline() (int, error) {
|
|
return readCPURangeWith(offline, countCPURange)
|
|
}
|
|
|
|
func getOnline() (int, error) {
|
|
if n, err := getFromCPUAffinity(); err == nil {
|
|
return n, nil
|
|
}
|
|
return readCPURangeWith(online, countCPURange)
|
|
}
|
|
|
|
func getPossible() (int, error) {
|
|
return readCPURangeWith(possible, countCPURange)
|
|
}
|
|
|
|
func getPresent() (int, error) {
|
|
return readCPURangeWith(present, countCPURange)
|
|
}
|
|
|
|
func listOffline() ([]int, error) {
|
|
return readCPURangeWith(offline, listCPURange)
|
|
}
|
|
|
|
func listOnline() ([]int, error) {
|
|
return readCPURangeWith(online, listCPURange)
|
|
}
|
|
|
|
func listPossible() ([]int, error) {
|
|
return readCPURangeWith(possible, listCPURange)
|
|
}
|
|
|
|
func listPresent() ([]int, error) {
|
|
return readCPURangeWith(present, listCPURange)
|
|
}
|