2018-07-17 09:26:11 +00:00
|
|
|
/*
|
2021-02-23 08:05:13 +00:00
|
|
|
Copyright 2018-2021 The Kubernetes Authors.
|
2018-07-17 09:26:11 +00:00
|
|
|
|
|
|
|
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 (
|
|
|
|
"io/ioutil"
|
2021-02-23 08:05:13 +00:00
|
|
|
|
|
|
|
"k8s.io/klog/v2"
|
2018-07-17 09:26:11 +00:00
|
|
|
|
|
|
|
"sigs.k8s.io/node-feature-discovery/source"
|
2020-08-11 06:01:20 +00:00
|
|
|
"sigs.k8s.io/node-feature-discovery/source/internal/cpuidutils"
|
2018-07-17 09:26:11 +00:00
|
|
|
)
|
|
|
|
|
2019-05-13 14:10:55 +00:00
|
|
|
// Configuration file options
|
|
|
|
type cpuidConfig struct {
|
|
|
|
AttributeBlacklist []string `json:"attributeBlacklist,omitempty"`
|
|
|
|
AttributeWhitelist []string `json:"attributeWhitelist,omitempty"`
|
|
|
|
}
|
|
|
|
|
2020-04-21 19:03:37 +00:00
|
|
|
type Config struct {
|
2019-05-13 14:10:55 +00:00
|
|
|
Cpuid cpuidConfig `json:"cpuid,omitempty"`
|
|
|
|
}
|
|
|
|
|
2020-04-21 19:03:37 +00:00
|
|
|
// newDefaultConfig returns a new config with pre-populated defaults
|
|
|
|
func newDefaultConfig() *Config {
|
|
|
|
return &Config{
|
|
|
|
cpuidConfig{
|
|
|
|
AttributeBlacklist: []string{
|
|
|
|
"BMI1",
|
|
|
|
"BMI2",
|
|
|
|
"CLMUL",
|
|
|
|
"CMOV",
|
|
|
|
"CX16",
|
|
|
|
"ERMS",
|
|
|
|
"F16C",
|
|
|
|
"HTT",
|
|
|
|
"LZCNT",
|
|
|
|
"MMX",
|
|
|
|
"MMXEXT",
|
|
|
|
"NX",
|
|
|
|
"POPCNT",
|
|
|
|
"RDRAND",
|
|
|
|
"RDSEED",
|
|
|
|
"RDTSCP",
|
|
|
|
"SGX",
|
|
|
|
"SGXLC",
|
|
|
|
"SSE",
|
|
|
|
"SSE2",
|
|
|
|
"SSE3",
|
|
|
|
"SSE4.1",
|
|
|
|
"SSE4.2",
|
|
|
|
"SSSE3",
|
|
|
|
},
|
|
|
|
AttributeWhitelist: []string{},
|
2019-05-13 14:10:55 +00:00
|
|
|
},
|
2020-04-21 19:03:37 +00:00
|
|
|
}
|
2019-05-13 14:10:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Filter for cpuid labels
|
|
|
|
type keyFilter struct {
|
|
|
|
keys map[string]struct{}
|
|
|
|
whitelist bool
|
|
|
|
}
|
|
|
|
|
2018-07-17 09:26:11 +00:00
|
|
|
// Implement FeatureSource interface
|
2020-04-21 19:03:37 +00:00
|
|
|
type Source struct {
|
|
|
|
config *Config
|
|
|
|
cpuidFilter *keyFilter
|
|
|
|
}
|
2018-07-17 09:26:11 +00:00
|
|
|
|
|
|
|
func (s Source) Name() string { return "cpu" }
|
|
|
|
|
2020-04-21 19:03:37 +00:00
|
|
|
// NewConfig method of the FeatureSource interface
|
|
|
|
func (s *Source) NewConfig() source.Config { return newDefaultConfig() }
|
2018-07-17 09:26:11 +00:00
|
|
|
|
2020-04-21 19:03:37 +00:00
|
|
|
// GetConfig method of the FeatureSource interface
|
|
|
|
func (s *Source) GetConfig() source.Config { return s.config }
|
|
|
|
|
|
|
|
// SetConfig method of the FeatureSource interface
|
|
|
|
func (s *Source) SetConfig(conf source.Config) {
|
|
|
|
switch v := conf.(type) {
|
|
|
|
case *Config:
|
|
|
|
s.config = v
|
|
|
|
s.initCpuidFilter()
|
|
|
|
default:
|
2021-02-23 08:05:13 +00:00
|
|
|
klog.Fatalf("invalid config type: %T", conf)
|
2019-05-13 14:10:55 +00:00
|
|
|
}
|
2020-04-21 19:03:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Source) Discover() (source.Features, error) {
|
|
|
|
features := source.Features{}
|
2019-05-13 14:10:55 +00:00
|
|
|
|
2018-07-17 09:26:11 +00:00
|
|
|
// Check if hyper-threading seems to be enabled
|
|
|
|
found, err := haveThreadSiblings()
|
|
|
|
if err != nil {
|
2021-02-23 08:05:13 +00:00
|
|
|
klog.Errorf("failed to detect hyper-threading: %v", err)
|
2018-07-17 09:26:11 +00:00
|
|
|
} else if found {
|
|
|
|
features["hardware_multithreading"] = true
|
|
|
|
}
|
2019-04-12 11:30:53 +00:00
|
|
|
|
|
|
|
// Check SST-BF
|
|
|
|
found, err = discoverSSTBF()
|
|
|
|
if err != nil {
|
2021-02-23 08:05:13 +00:00
|
|
|
klog.Errorf("failed to detect SST-BF: %v", err)
|
2019-04-12 11:30:53 +00:00
|
|
|
} else if found {
|
|
|
|
features["power.sst_bf.enabled"] = true
|
|
|
|
}
|
|
|
|
|
2019-02-19 08:52:46 +00:00
|
|
|
// Detect CPUID
|
2020-08-11 06:01:20 +00:00
|
|
|
cpuidFlags := cpuidutils.GetCpuidFlags()
|
2019-02-19 08:52:46 +00:00
|
|
|
for _, f := range cpuidFlags {
|
2020-04-21 19:03:37 +00:00
|
|
|
if s.cpuidFilter.unmask(f) {
|
2019-05-13 14:10:55 +00:00
|
|
|
features["cpuid."+f] = true
|
|
|
|
}
|
2019-02-19 08:52:46 +00:00
|
|
|
}
|
|
|
|
|
2019-08-29 13:11:27 +00:00
|
|
|
// Detect pstate features
|
|
|
|
pstate, err := detectPstate()
|
2019-02-19 08:52:46 +00:00
|
|
|
if err != nil {
|
2021-02-23 08:05:13 +00:00
|
|
|
klog.Error(err)
|
2019-08-29 13:11:27 +00:00
|
|
|
} else {
|
|
|
|
for k, v := range pstate {
|
|
|
|
features["pstate."+k] = v
|
|
|
|
}
|
2019-02-19 08:52:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Detect RDT features
|
|
|
|
rdt := discoverRDT()
|
|
|
|
for _, f := range rdt {
|
|
|
|
features["rdt."+f] = true
|
|
|
|
}
|
|
|
|
|
2018-07-17 09:26:11 +00:00
|
|
|
return features, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if any (online) CPUs have thread siblings
|
|
|
|
func haveThreadSiblings() (bool, error) {
|
2020-05-20 11:32:07 +00:00
|
|
|
|
|
|
|
files, err := ioutil.ReadDir(source.SysfsDir.Path("bus/cpu/devices"))
|
2018-07-17 09:26:11 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, file := range files {
|
|
|
|
// Try to read siblings from topology
|
2020-05-20 11:32:07 +00:00
|
|
|
siblings, err := ioutil.ReadFile(source.SysfsDir.Path("bus/cpu/devices", file.Name(), "topology/thread_siblings_list"))
|
2018-07-17 09:26:11 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
for _, char := range siblings {
|
|
|
|
// If list separator found, we determine that there are multiple siblings
|
|
|
|
if char == ',' || char == '-' {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// No siblings were found
|
|
|
|
return false, nil
|
|
|
|
}
|
2019-05-13 14:10:55 +00:00
|
|
|
|
2020-04-21 19:03:37 +00:00
|
|
|
func (s *Source) initCpuidFilter() {
|
2019-05-13 14:10:55 +00:00
|
|
|
newFilter := keyFilter{keys: map[string]struct{}{}}
|
2020-04-21 19:03:37 +00:00
|
|
|
if len(s.config.Cpuid.AttributeWhitelist) > 0 {
|
|
|
|
for _, k := range s.config.Cpuid.AttributeWhitelist {
|
2019-05-13 14:10:55 +00:00
|
|
|
newFilter.keys[k] = struct{}{}
|
|
|
|
}
|
|
|
|
newFilter.whitelist = true
|
|
|
|
} else {
|
2020-04-21 19:03:37 +00:00
|
|
|
for _, k := range s.config.Cpuid.AttributeBlacklist {
|
2019-05-13 14:10:55 +00:00
|
|
|
newFilter.keys[k] = struct{}{}
|
|
|
|
}
|
|
|
|
newFilter.whitelist = false
|
|
|
|
}
|
2020-04-21 19:03:37 +00:00
|
|
|
s.cpuidFilter = &newFilter
|
2019-05-13 14:10:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f keyFilter) unmask(k string) bool {
|
|
|
|
if f.whitelist {
|
|
|
|
if _, ok := f.keys[k]; ok {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if _, ok := f.keys[k]; !ok {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|