mirror of https://github.com/k3s-io/k3s
137 lines
4.3 KiB
Go
137 lines
4.3 KiB
Go
// +build libipmctl,cgo
|
|
|
|
// Copyright 2020 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.
|
|
|
|
package nvm
|
|
|
|
// #cgo pkg-config: libipmctl
|
|
// #include <nvm_management.h>
|
|
import "C"
|
|
import (
|
|
"fmt"
|
|
info "github.com/google/cadvisor/info/v1"
|
|
"sync"
|
|
|
|
"k8s.io/klog/v2"
|
|
)
|
|
|
|
var (
|
|
isNVMLibInitialized = false
|
|
nvmLibMutex = sync.Mutex{}
|
|
)
|
|
|
|
func init() {
|
|
nvmLibMutex.Lock()
|
|
defer nvmLibMutex.Unlock()
|
|
cErr := C.nvm_init()
|
|
if cErr != C.NVM_SUCCESS {
|
|
// Unfortunately klog does not seem to work here. I believe it's better to
|
|
// output information using fmt rather then let it disappear silently.
|
|
fmt.Printf("libipmctl initialization failed with status %d", cErr)
|
|
return
|
|
}
|
|
isNVMLibInitialized = true
|
|
}
|
|
|
|
// getAvgPowerBudget retrieves configured power budget
|
|
// (in watts) for NVM devices. When libipmct is not available
|
|
// zero is returned.
|
|
func getAvgPowerBudget() (uint, error) {
|
|
// Get number of devices on the platform
|
|
// see: https://github.com/intel/ipmctl/blob/v01.00.00.3497/src/os/nvm_api/nvm_management.h#L1478
|
|
count := C.uint(0)
|
|
err := C.nvm_get_number_of_devices(&count)
|
|
if err != C.NVM_SUCCESS {
|
|
klog.Warningf("Unable to get number of NVM devices. Status code: %d", err)
|
|
return uint(0), fmt.Errorf("Unable to get number of NVM devices. Status code: %d", err)
|
|
}
|
|
|
|
if count == 0 {
|
|
klog.Warningf("There are no NVM devices!")
|
|
return uint(0), nil
|
|
}
|
|
|
|
// Load basic device information for all the devices
|
|
// to obtain UID of the first one.
|
|
devices := make([]C.struct_device_discovery, count)
|
|
err = C.nvm_get_devices(&devices[0], C.uchar(count))
|
|
if err != C.NVM_SUCCESS {
|
|
klog.Warningf("Unable to get all NVM devices. Status code: %d", err)
|
|
return uint(0), fmt.Errorf("Unable to get all NVM devices. Status code: %d", err)
|
|
}
|
|
|
|
// Power budget is same for all the devices
|
|
// so we can rely on any of them.
|
|
device := C.struct_device_details{}
|
|
err = C.nvm_get_device_details(&devices[0].uid[0], &device)
|
|
if err != C.NVM_SUCCESS {
|
|
uid := C.GoString(&devices[0].uid[0])
|
|
klog.Warningf("Unable to get details of NVM device %q. Status code: %d", uid, err)
|
|
return uint(0), fmt.Errorf("Unable to get details of NVM device %q. Status code: %d", uid, err)
|
|
}
|
|
|
|
return uint(device.avg_power_budget / 1000), nil
|
|
}
|
|
|
|
// getCapacities retrieves the total NVM capacity in bytes for memory mode and app direct mode
|
|
func getCapacities() (uint64, uint64, error) {
|
|
caps := C.struct_device_capacities{}
|
|
err := C.nvm_get_nvm_capacities(&caps)
|
|
if err != C.NVM_SUCCESS {
|
|
klog.Warningf("Unable to get NVM capacity. Status code: %d", err)
|
|
return uint64(0), uint64(0), fmt.Errorf("Unable to get NVM capacity. Status code: %d", err)
|
|
}
|
|
return uint64(caps.memory_capacity), uint64(caps.app_direct_capacity), nil
|
|
}
|
|
|
|
// GetInfo returns information specific for non-volatile memory modules
|
|
func GetInfo() (info.NVMInfo, error) {
|
|
nvmLibMutex.Lock()
|
|
defer nvmLibMutex.Unlock()
|
|
|
|
nvmInfo := info.NVMInfo{}
|
|
if !isNVMLibInitialized {
|
|
klog.V(1).Info("libipmctl has not been initialized. NVM information will not be available")
|
|
return nvmInfo, nil
|
|
}
|
|
|
|
var err error
|
|
nvmInfo.MemoryModeCapacity, nvmInfo.AppDirectModeCapacity, err = getCapacities()
|
|
if err != nil {
|
|
return info.NVMInfo{}, fmt.Errorf("Unable to get NVM capacities, err: %s", err)
|
|
}
|
|
|
|
nvmInfo.AvgPowerBudget, err = getAvgPowerBudget()
|
|
if err != nil {
|
|
return info.NVMInfo{}, fmt.Errorf("Unable to get NVM average power budget, err: %s", err)
|
|
}
|
|
return nvmInfo, nil
|
|
}
|
|
|
|
// Finalize un-initializes libipmctl. See https://github.com/google/cadvisor/issues/2457.
|
|
func Finalize() {
|
|
nvmLibMutex.Lock()
|
|
defer nvmLibMutex.Unlock()
|
|
|
|
klog.V(1).Info("Attempting to un-initialize libipmctl")
|
|
if !isNVMLibInitialized {
|
|
klog.V(1).Info("libipmctl has not been initialized; not un-initializing.")
|
|
return
|
|
}
|
|
|
|
C.nvm_uninit()
|
|
isNVMLibInitialized = false
|
|
}
|