gvisor/runsc/mitigate/mitigate.go

454 lines
12 KiB
Go

// Copyright 2021 The gVisor 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 mitigate provides libraries for the mitigate command. The
// mitigate command mitigates side channel attacks such as MDS. Mitigate
// shuts down CPUs via /sys/devices/system/cpu/cpu{N}/online.
package mitigate
import (
"fmt"
"io/ioutil"
"os"
"regexp"
"sort"
"strconv"
"strings"
)
const (
// mds is the only bug we care about.
mds = "mds"
// Constants for parsing /proc/cpuinfo.
processorKey = "processor"
vendorIDKey = "vendor_id"
cpuFamilyKey = "cpu family"
modelKey = "model"
physicalIDKey = "physical id"
coreIDKey = "core id"
bugsKey = "bugs"
// Path to shutdown a CPU.
cpuOnlineTemplate = "/sys/devices/system/cpu/cpu%d/online"
)
// CPUSet contains a map of all CPUs on the system, mapped
// by Physical ID and CoreIDs. threads with the same
// Core and Physical ID are Hyperthread pairs.
type CPUSet map[threadID]*ThreadGroup
// NewCPUSet creates a CPUSet from data read from /proc/cpuinfo.
func NewCPUSet(data []byte, vulnerable func(Thread) bool) (CPUSet, error) {
processors, err := getThreads(string(data))
if err != nil {
return nil, err
}
set := make(CPUSet)
for _, p := range processors {
// Each ID is of the form physicalID:coreID. Hyperthread pairs
// have identical physical and core IDs. We need to match
// Hyperthread pairs so that we can shutdown all but one per
// pair.
core, ok := set[p.id]
if !ok {
core = &ThreadGroup{}
set[p.id] = core
}
core.isVulnerable = core.isVulnerable || vulnerable(p)
core.threads = append(core.threads, p)
}
// We need to make sure we shutdown the lowest number processor per
// thread group.
for _, tg := range set {
sort.Slice(tg.threads, func(i, j int) bool {
return tg.threads[i].processorNumber < tg.threads[j].processorNumber
})
}
return set, nil
}
// NewCPUSetFromPossible makes a cpuSet data read from
// /sys/devices/system/cpu/possible. This is used in enable operations
// where the caller simply wants to enable all CPUS.
func NewCPUSetFromPossible(data []byte) (CPUSet, error) {
threads, err := GetThreadsFromPossible(data)
if err != nil {
return nil, err
}
// We don't care if a CPU is vulnerable or not, we just
// want to return a list of all CPUs on the host.
set := CPUSet{
threads[0].id: &ThreadGroup{
threads: threads,
isVulnerable: false,
},
}
return set, nil
}
// String implements the String method for CPUSet.
func (c CPUSet) String() string {
ret := ""
for _, tg := range c {
ret += fmt.Sprintf("%s\n", tg)
}
return ret
}
// GetRemainingList returns the list of threads that will remain active
// after mitigation.
func (c CPUSet) GetRemainingList() []Thread {
threads := make([]Thread, 0, len(c))
for _, core := range c {
// If we're vulnerable, take only one thread from the pair.
if core.isVulnerable {
threads = append(threads, core.threads[0])
continue
}
// Otherwise don't shutdown anything.
threads = append(threads, core.threads...)
}
return threads
}
// GetShutdownList returns the list of threads that will be shutdown on
// mitigation.
func (c CPUSet) GetShutdownList() []Thread {
threads := make([]Thread, 0)
for _, core := range c {
// Only if we're vulnerable do shutdown anything. In this case,
// shutdown all but the first entry.
if core.isVulnerable && len(core.threads) > 1 {
threads = append(threads, core.threads[1:]...)
}
}
return threads
}
// ThreadGroup represents Hyperthread pairs on the same physical/core ID.
type ThreadGroup struct {
threads []Thread
isVulnerable bool
}
// String implements the String method for threadGroup.
func (c ThreadGroup) String() string {
ret := fmt.Sprintf("ThreadGroup:\nIsVulnerable: %t\n", c.isVulnerable)
for _, processor := range c.threads {
ret += fmt.Sprintf("%s\n", processor)
}
return ret
}
// getThreads returns threads structs from reading /proc/cpuinfo.
func getThreads(data string) ([]Thread, error) {
// Each processor entry should start with the
// processor key. Find the beginings of each.
r := buildRegex(processorKey, `\d+`)
indices := r.FindAllStringIndex(data, -1)
if len(indices) < 1 {
return nil, fmt.Errorf("no cpus found for: %q", data)
}
// Add the ending index for last entry.
indices = append(indices, []int{len(data), -1})
// Valid cpus are now defined by strings in between
// indexes (e.g. data[index[i], index[i+1]]).
// There should be len(indicies) - 1 CPUs
// since the last index is the end of the string.
cpus := make([]Thread, 0, len(indices))
// Find each string that represents a CPU. These begin "processor".
for i := 1; i < len(indices); i++ {
start := indices[i-1][0]
end := indices[i][0]
// Parse the CPU entry, which should be between start/end.
c, err := newThread(data[start:end])
if err != nil {
return nil, err
}
cpus = append(cpus, c)
}
return cpus, nil
}
// GetThreadsFromPossible makes threads from data read from /sys/devices/system/cpu/possible.
func GetThreadsFromPossible(data []byte) ([]Thread, error) {
possibleRegex := regexp.MustCompile(`(?m)^(\d+)(-(\d+))?$`)
matches := possibleRegex.FindStringSubmatch(string(data))
if len(matches) != 4 {
return nil, fmt.Errorf("mismatch regex from possible: %q", string(data))
}
// If matches[3] is empty, we only have one cpu entry.
if matches[3] == "" {
matches[3] = matches[1]
}
begin, err := strconv.ParseInt(matches[1], 10, 64)
if err != nil {
return nil, fmt.Errorf("failed to parse begin: %v", err)
}
end, err := strconv.ParseInt(matches[3], 10, 64)
if err != nil {
return nil, fmt.Errorf("failed to parse end: %v", err)
}
if begin > end || begin < 0 || end < 0 {
return nil, fmt.Errorf("invalid cpu bounds from possible: begin: %d end: %d", begin, end)
}
ret := make([]Thread, 0, end-begin)
for i := begin; i <= end; i++ {
ret = append(ret, Thread{
processorNumber: i,
id: threadID{
physicalID: 0, // we don't care about id for enable ops.
coreID: 0,
},
})
}
return ret, nil
}
// threadID for each thread is defined by the physical and
// core IDs. If equal, two threads are Hyperthread pairs.
type threadID struct {
physicalID int64
coreID int64
}
// Thread represents pertinent info about a single hyperthread in a pair.
type Thread struct {
processorNumber int64 // the processor number of this CPU.
vendorID string // the vendorID of CPU (e.g. AuthenticAMD).
cpuFamily int64 // CPU family number (e.g. 6 for CascadeLake/Skylake).
model int64 // CPU model number (e.g. 85 for CascadeLake/Skylake).
id threadID // id for this thread
bugs map[string]struct{} // map of vulnerabilities parsed from the 'bugs' field.
}
// newThread parses a CPU from a single cpu entry from /proc/cpuinfo.
func newThread(data string) (Thread, error) {
empty := Thread{}
processor, err := parseProcessor(data)
if err != nil {
return empty, err
}
vendorID, err := parseVendorID(data)
if err != nil {
return empty, err
}
cpuFamily, err := parseCPUFamily(data)
if err != nil {
return empty, err
}
model, err := parseModel(data)
if err != nil {
return empty, err
}
physicalID, err := parsePhysicalID(data)
if err != nil {
return empty, err
}
coreID, err := parseCoreID(data)
if err != nil {
return empty, err
}
bugs, err := parseBugs(data)
if err != nil {
return empty, err
}
return Thread{
processorNumber: processor,
vendorID: vendorID,
cpuFamily: cpuFamily,
model: model,
id: threadID{
physicalID: physicalID,
coreID: coreID,
},
bugs: bugs,
}, nil
}
// String implements the String method for thread.
func (t Thread) String() string {
template := `CPU: %d
CPU ID: %+v
Vendor: %s
Family/Model: %d/%d
Bugs: %s
`
bugs := make([]string, 0)
for bug := range t.bugs {
bugs = append(bugs, bug)
}
return fmt.Sprintf(template, t.processorNumber, t.id, t.vendorID, t.cpuFamily, t.model, strings.Join(bugs, ","))
}
// Enable turns on the CPU by writing 1 to /sys/devices/cpu/cpu{N}/online.
func (t Thread) Enable() error {
// Linux ensures that "cpu0" is always online.
if t.processorNumber == 0 {
return nil
}
cpuPath := fmt.Sprintf(cpuOnlineTemplate, t.processorNumber)
f, err := os.OpenFile(cpuPath, os.O_WRONLY|os.O_CREATE, 0644)
if err != nil {
return fmt.Errorf("failed to open file %s: %v", cpuPath, err)
}
if _, err = f.Write([]byte{'1'}); err != nil {
return fmt.Errorf("failed to write '1' to %s: %v", cpuPath, err)
}
return nil
}
// Disable turns off the CPU by writing 0 to /sys/devices/cpu/cpu{N}/online.
func (t Thread) Disable() error {
// The core labeled "cpu0" can never be taken offline via this method.
// Linux will return EPERM if the user even creates a file at the /sys
// path above.
if t.processorNumber == 0 {
return fmt.Errorf("invalid shutdown operation: cpu0 cannot be disabled")
}
cpuPath := fmt.Sprintf(cpuOnlineTemplate, t.processorNumber)
return ioutil.WriteFile(cpuPath, []byte{'0'}, 0644)
}
// IsVulnerable checks if a CPU is vulnerable to mds.
func (t Thread) IsVulnerable() bool {
_, ok := t.bugs[mds]
return ok
}
// isActive checks if a CPU is active from /sys/devices/system/cpu/cpu{N}/online
// If the file does not exist (ioutil returns in error), we assume the CPU is on.
func (t Thread) isActive() bool {
cpuPath := fmt.Sprintf(cpuOnlineTemplate, t.processorNumber)
data, err := ioutil.ReadFile(cpuPath)
if err != nil {
return true
}
return len(data) > 0 && data[0] != '0'
}
// SimilarTo checks family/model/bugs fields for equality of two
// processors.
func (t Thread) SimilarTo(other Thread) bool {
if t.vendorID != other.vendorID {
return false
}
if other.cpuFamily != t.cpuFamily {
return false
}
if other.model != t.model {
return false
}
if len(other.bugs) != len(t.bugs) {
return false
}
for bug := range t.bugs {
if _, ok := other.bugs[bug]; !ok {
return false
}
}
return true
}
// parseProcessor grabs the processor field from /proc/cpuinfo output.
func parseProcessor(data string) (int64, error) {
return parseIntegerResult(data, processorKey)
}
// parseVendorID grabs the vendor_id field from /proc/cpuinfo output.
func parseVendorID(data string) (string, error) {
return parseRegex(data, vendorIDKey, `[\w\d]+`)
}
// parseCPUFamily grabs the cpu family field from /proc/cpuinfo output.
func parseCPUFamily(data string) (int64, error) {
return parseIntegerResult(data, cpuFamilyKey)
}
// parseModel grabs the model field from /proc/cpuinfo output.
func parseModel(data string) (int64, error) {
return parseIntegerResult(data, modelKey)
}
// parsePhysicalID parses the physical id field.
func parsePhysicalID(data string) (int64, error) {
return parseIntegerResult(data, physicalIDKey)
}
// parseCoreID parses the core id field.
func parseCoreID(data string) (int64, error) {
return parseIntegerResult(data, coreIDKey)
}
// parseBugs grabs the bugs field from /proc/cpuinfo output.
func parseBugs(data string) (map[string]struct{}, error) {
result, err := parseRegex(data, bugsKey, `[\d\w\s]*`)
if err != nil {
return nil, err
}
bugs := strings.Split(result, " ")
ret := make(map[string]struct{}, len(bugs))
for _, bug := range bugs {
ret[bug] = struct{}{}
}
return ret, nil
}
// parseIntegerResult parses fields expecting an integer.
func parseIntegerResult(data, key string) (int64, error) {
result, err := parseRegex(data, key, `\d+`)
if err != nil {
return 0, err
}
return strconv.ParseInt(result, 0, 64)
}
// buildRegex builds a regex for parsing each CPU field.
func buildRegex(key, match string) *regexp.Regexp {
reg := fmt.Sprintf(`(?m)^%s\s*:\s*(.*)$`, key)
return regexp.MustCompile(reg)
}
// parseRegex parses data with key inserted into a standard regex template.
func parseRegex(data, key, match string) (string, error) {
r := buildRegex(key, match)
matches := r.FindStringSubmatch(data)
if len(matches) < 2 {
return "", fmt.Errorf("failed to match key %q: %q", key, data)
}
return matches[1], nil
}