mirror of
https://github.com/kubernetes-sigs/node-feature-discovery.git
synced 2025-03-05 16:27:05 +00:00
Support for additional cpu features
This adds additional cpu features: - pstate status from status of intel_pstate driver - pstate scaling settings from scaling_governor - cstate enable from max_cstates in intel_idle driver
This commit is contained in:
parent
e370429970
commit
a65f73e834
4 changed files with 129 additions and 6 deletions
|
@ -61,7 +61,10 @@ The **cpu** feature source supports the following labels:
|
||||||
| cpuid | <cpuid flag> | CPU capability is supported
|
| cpuid | <cpuid flag> | CPU capability is supported
|
||||||
| hardware_multithreading | | Hardware multithreading, such as Intel HTT, enabled (number of logical CPUs is greater than physical CPUs)
|
| hardware_multithreading | | Hardware multithreading, such as Intel HTT, enabled (number of logical CPUs is greater than physical CPUs)
|
||||||
| power | sst_bf.enabled | Intel SST-BF ([Intel Speed Select Technology][intel-sst] - Base frequency) enabled
|
| power | sst_bf.enabled | Intel SST-BF ([Intel Speed Select Technology][intel-sst] - Base frequency) enabled
|
||||||
| [pstate][intel-pstate] | turbo | Set to 'true' if turbo frequencies are enabled in Intel pstate driver, set to 'false' if they have been disabled.
|
| [pstate][intel-pstate] | status | The status of the Intel pstate driver when in use and enabled, either 'active' or 'passive'.
|
||||||
|
| | turbo | Set to 'true' if turbo frequencies are enabled in Intel pstate driver, set to 'false' if they have been disabled.
|
||||||
|
| | scaling_governor | The value of the Intel pstate scaling_governor when in use, either 'powersave' or 'performance'.
|
||||||
|
| cstate | enabled | Set to 'true' if cstates are set in the intel_idle driver, otherwise set to 'false'.
|
||||||
| [rdt][intel-rdt] | RDTMON | Intel RDT Monitoring Technology
|
| [rdt][intel-rdt] | RDTMON | Intel RDT Monitoring Technology
|
||||||
| | RDTCMT | Intel Cache Monitoring (CMT)
|
| | RDTCMT | Intel Cache Monitoring (CMT)
|
||||||
| | RDTMBM | Intel Memory Bandwidth Monitoring (MBM)
|
| | RDTMBM | Intel Memory Bandwidth Monitoring (MBM)
|
||||||
|
|
|
@ -144,6 +144,14 @@ func (s *Source) Discover() (source.Features, error) {
|
||||||
features["rdt."+f] = true
|
features["rdt."+f] = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Detect cstate configuration
|
||||||
|
cstate, err := detectCstate()
|
||||||
|
if err != nil {
|
||||||
|
klog.Errorf("failed to detect cstate: %v", err)
|
||||||
|
} else {
|
||||||
|
features["cstate.enabled"] = cstate
|
||||||
|
}
|
||||||
|
|
||||||
return features, nil
|
return features, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
51
source/cpu/cstate.go
Normal file
51
source/cpu/cstate.go
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
/*
|
||||||
|
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 cpu
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"sigs.k8s.io/node-feature-discovery/source"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Discover if c-states are enabled
|
||||||
|
func detectCstate() (bool, error) {
|
||||||
|
// When the intel_idle driver is in use (default), check setting of max_cstates
|
||||||
|
driver, err := ioutil.ReadFile(source.SysfsDir.Path("devices/system/cpu/cpuidle/current_driver"))
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("cannot get driver for cpuidle: %s", err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.TrimSpace(string(driver)) != "intel_idle" {
|
||||||
|
// Currently only checking intel_idle driver for cstates
|
||||||
|
return false, fmt.Errorf("intel_idle driver is not in use: %s", string(driver))
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := ioutil.ReadFile(source.SysfsDir.Path("module/intel_idle/parameters/max_cstate"))
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("cannot determine cstate from max_cstates: %s", err.Error())
|
||||||
|
}
|
||||||
|
cstates, err := strconv.Atoi(strings.TrimSpace(string(data)))
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("non-integer value of cstates: %s", err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
return cstates > 0, nil
|
||||||
|
}
|
|
@ -20,6 +20,9 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"runtime"
|
"runtime"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"k8s.io/klog/v2"
|
||||||
|
|
||||||
"sigs.k8s.io/node-feature-discovery/source"
|
"sigs.k8s.io/node-feature-discovery/source"
|
||||||
)
|
)
|
||||||
|
@ -32,14 +35,72 @@ func detectPstate() (map[string]string, error) {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Only looking for turbo boost for now...
|
// Get global pstate status
|
||||||
|
data, err := ioutil.ReadFile(source.SysfsDir.Path("devices/system/cpu/intel_pstate/status"))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("could not read pstate status: %s", err.Error())
|
||||||
|
}
|
||||||
|
status := strings.TrimSpace(string(data))
|
||||||
|
if status == "off" {
|
||||||
|
// No need to check other pstate features
|
||||||
|
klog.Infof("intel_pstate driver is not in use")
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
features := map[string]string{"status": status}
|
||||||
|
|
||||||
|
// Check turbo boost
|
||||||
bytes, err := ioutil.ReadFile(source.SysfsDir.Path("devices/system/cpu/intel_pstate/no_turbo"))
|
bytes, err := ioutil.ReadFile(source.SysfsDir.Path("devices/system/cpu/intel_pstate/no_turbo"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("can't detect whether turbo boost is enabled: %s", err.Error())
|
klog.Errorf("can't detect whether turbo boost is enabled: %s", err.Error())
|
||||||
|
} else {
|
||||||
|
features["turbo"] = "false"
|
||||||
|
if bytes[0] == byte('0') {
|
||||||
|
features["turbo"] = "true"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
features := map[string]string{"turbo": "false"}
|
|
||||||
if bytes[0] == byte('0') {
|
if status != "active" {
|
||||||
features["turbo"] = "true"
|
// Don't check other features which depend on active state
|
||||||
|
return features, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine scaling governor that is being used
|
||||||
|
policies, err := ioutil.ReadDir(source.SysfsDir.Path("devices/system/cpu/cpufreq"))
|
||||||
|
if err != nil {
|
||||||
|
klog.Errorf("failed to read cpufreq directory: %s", err.Error())
|
||||||
|
return features, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
scaling := ""
|
||||||
|
for _, policy := range policies {
|
||||||
|
// Ensure at least one cpu is using this policy
|
||||||
|
cpus, err := ioutil.ReadFile(source.SysfsDir.Path("devices/system/cpu/cpufreq", policy.Name(), "affected_cpus"))
|
||||||
|
if err != nil {
|
||||||
|
klog.Errorf("could not read cpufreq policy %s affected_cpus", policy.Name())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if strings.TrimSpace(string(cpus)) == "" {
|
||||||
|
klog.Infof("policy %s has no associated cpus", policy.Name())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := ioutil.ReadFile(source.SysfsDir.Path("devices/system/cpu/cpufreq", policy.Name(), "scaling_governor"))
|
||||||
|
if err != nil {
|
||||||
|
klog.Errorf("could not read cpufreq policy %s scaling_governor", policy.Name())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
policy_scaling := strings.TrimSpace(string(data))
|
||||||
|
// Check that all of the policies have the same scaling governor, if not don't set feature
|
||||||
|
if scaling != "" && scaling != policy_scaling {
|
||||||
|
klog.Infof("scaling_governor for policy %s doesn't match prior policy", policy.Name())
|
||||||
|
scaling = ""
|
||||||
|
break
|
||||||
|
}
|
||||||
|
scaling = policy_scaling
|
||||||
|
}
|
||||||
|
|
||||||
|
if scaling != "" {
|
||||||
|
features["scaling_governor"] = scaling
|
||||||
}
|
}
|
||||||
|
|
||||||
return features, nil
|
return features, nil
|
||||||
|
|
Loading…
Add table
Reference in a new issue