2021-11-04 08:14:51 +00:00
|
|
|
/*
|
|
|
|
Copyright 2021 The Kubernetes 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package utils
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2022-09-08 10:23:49 +00:00
|
|
|
"os"
|
2021-11-04 08:14:51 +00:00
|
|
|
"path/filepath"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
2022-10-14 12:28:52 +00:00
|
|
|
corev1 "k8s.io/api/core/v1"
|
2021-11-04 08:14:51 +00:00
|
|
|
"k8s.io/apimachinery/pkg/api/resource"
|
|
|
|
"k8s.io/klog/v2"
|
|
|
|
resourcehelper "k8s.io/kubernetes/pkg/apis/core/helper"
|
|
|
|
|
2022-10-06 11:05:11 +00:00
|
|
|
"sigs.k8s.io/node-feature-discovery/pkg/utils/hostpath"
|
2021-11-04 08:14:51 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2022-10-06 11:05:11 +00:00
|
|
|
sysBusNodeBasepath = hostpath.SysfsDir.Path("bus/node/devices")
|
2021-11-04 08:14:51 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// NumaMemoryResources contains information of the memory resources per NUMA
|
|
|
|
// nodes of the system.
|
|
|
|
type NumaMemoryResources map[int]MemoryResourceInfo
|
|
|
|
|
|
|
|
// MemoryResourceInfo holds information of memory resources per resource type.
|
2022-10-14 12:28:52 +00:00
|
|
|
type MemoryResourceInfo map[corev1.ResourceName]int64
|
2021-11-04 08:14:51 +00:00
|
|
|
|
|
|
|
// GetNumaMemoryResources returns total amount of memory and hugepages under NUMA nodes
|
|
|
|
func GetNumaMemoryResources() (NumaMemoryResources, error) {
|
2022-09-08 10:23:49 +00:00
|
|
|
nodes, err := os.ReadDir(sysBusNodeBasepath)
|
2021-11-04 08:14:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
memoryResources := make(NumaMemoryResources, len(nodes))
|
|
|
|
for _, n := range nodes {
|
|
|
|
numaNode := n.Name()
|
|
|
|
nodeID, err := strconv.Atoi(numaNode[4:])
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to parse NUMA node ID of %q", numaNode)
|
|
|
|
}
|
|
|
|
|
|
|
|
info := make(MemoryResourceInfo)
|
|
|
|
|
|
|
|
// Get total memory
|
|
|
|
nodeTotalMemory, err := readTotalMemoryFromMeminfo(filepath.Join(sysBusNodeBasepath, numaNode, "meminfo"))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-10-14 12:28:52 +00:00
|
|
|
info[corev1.ResourceMemory] = nodeTotalMemory
|
2021-11-04 08:14:51 +00:00
|
|
|
|
|
|
|
// Get hugepages
|
|
|
|
hugepageBytes, err := getHugepagesBytes(filepath.Join(sysBusNodeBasepath, numaNode, "hugepages"))
|
|
|
|
if err != nil {
|
2023-08-08 07:14:44 +00:00
|
|
|
if os.IsNotExist(err) {
|
|
|
|
continue
|
|
|
|
} else {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-11-04 08:14:51 +00:00
|
|
|
}
|
|
|
|
for n, s := range hugepageBytes {
|
|
|
|
info[n] = s
|
|
|
|
}
|
|
|
|
|
|
|
|
memoryResources[nodeID] = info
|
|
|
|
}
|
|
|
|
|
|
|
|
return memoryResources, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getHugepagesBytes(path string) (MemoryResourceInfo, error) {
|
2022-09-08 10:23:49 +00:00
|
|
|
entries, err := os.ReadDir(path)
|
2021-11-04 08:14:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
hugepagesBytes := make(MemoryResourceInfo)
|
|
|
|
for _, entry := range entries {
|
|
|
|
split := strings.SplitN(entry.Name(), "-", 2)
|
|
|
|
if len(split) != 2 || split[0] != "hugepages" {
|
2023-05-12 11:55:14 +00:00
|
|
|
klog.InfoS("malformed hugepages entry", "hugepagesEntry", entry.Name())
|
2021-11-04 08:14:51 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use Ki instead of kB
|
|
|
|
q, err := resource.ParseQuantity(strings.Replace(split[1], "kB", "Ki", 1))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-09-08 10:23:49 +00:00
|
|
|
data, err := os.ReadFile(filepath.Join(path, entry.Name(), "nr_hugepages"))
|
2021-11-04 08:14:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
nr, err := strconv.ParseInt(strings.TrimSpace(string(data)), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
size, _ := q.AsInt64()
|
2022-10-14 12:28:52 +00:00
|
|
|
name := corev1.ResourceName(resourcehelper.HugePageResourceName(q))
|
2021-11-04 08:14:51 +00:00
|
|
|
hugepagesBytes[name] = nr * size
|
|
|
|
}
|
|
|
|
|
|
|
|
return hugepagesBytes, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func readTotalMemoryFromMeminfo(path string) (int64, error) {
|
2022-09-08 10:23:49 +00:00
|
|
|
data, err := os.ReadFile(path)
|
2021-11-04 08:14:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, line := range strings.Split(string(data), "\n") {
|
|
|
|
split := strings.SplitN(line, ":", 2)
|
|
|
|
if len(split) != 2 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if strings.Contains(split[0], "MemTotal") {
|
|
|
|
memValue := strings.Trim(split[1], "\t\n kB")
|
|
|
|
convertedValue, err := strconv.ParseInt(memValue, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return -1, fmt.Errorf("failed to convert value: %v", memValue)
|
|
|
|
}
|
|
|
|
|
|
|
|
// return information in bytes
|
|
|
|
return 1024 * convertedValue, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1, fmt.Errorf("failed to find MemTotal field under the file %q", path)
|
|
|
|
}
|