| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138 |
- //go:build libipmctl && cgo
- // +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"
- "sync"
- info "github.com/google/cadvisor/info/v1"
- "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
- }
|