Skip to content

Commit

Permalink
Add basic kernel tuning functionality
Browse files Browse the repository at this point in the history
Checks /etc/pivot/kernel-args for argument changes. This early
version only supports bare arguments that are in the whitelist.

See conversation at openshift/machine-config-operator#576

Signed-off-by: Steve Milner <[email protected]>
  • Loading branch information
ashcrow committed Mar 26, 2019
1 parent 6c181a8 commit cfada35
Show file tree
Hide file tree
Showing 3 changed files with 177 additions and 1 deletion.
113 changes: 112 additions & 1 deletion cmd/root.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package cmd

import (
"bufio"
"encoding/json"
"flag"
"fmt"
Expand Down Expand Up @@ -31,9 +32,15 @@ const (
etcPivotFile = "/etc/pivot/image-pullspec"
runPivotRebootFile = "/run/pivot/reboot-needed"
// Pull secret. Written by the machine-config-operator
kubeletAuthFile = "/var/lib/kubelet/config.json"
kubeletAuthFile = "/var/lib/kubelet/config.json"
// File containing kernel arg changes for tuning
kernelTuningFile = "/etc/pivot/kernel-args"
)

// TODO: fill out the whitelist
// tuneableArgsWhitelist contains allowed keys for tunable arguments
var tuneableArgsWhitelist = []string{"nosmt"}

// RootCmd houses the cobra config for the main command
var RootCmd = &cobra.Command{
Use: "pivot [FLAGS] [IMAGE_PULLSPEC]",
Expand All @@ -51,6 +58,93 @@ func init() {
pflag.CommandLine.AddGoFlagSet(flag.CommandLine)
}

// isArgTuneable returns if the argument provided is allowed to be modified
func isArgTunable(arg string) bool {
for _, key := range tuneableArgsWhitelist {
if arg == key {
return true
}
}
return false
}

// parseTuningFile parses the kernel argument tuning file
func parseTuningFile(tuningFilePath string) (addArguments []types.TuneArgument, deleteArguments []types.TuneArgument, err error) {
if tuningFilePath == "" {
tuningFilePath = kernelTuningFile
}
// Return fast if the file does not exist
if _, err := os.Stat(tuningFilePath); os.IsNotExist(err) {
glog.V(2).Infof("no kernel tuning needed as %s does not exist", tuningFilePath)
// This isn't an error. Return out.
return addArguments, deleteArguments, err
}
// Read and parse the file
file, err := os.Open(tuningFilePath)
if err != nil {
// If we have an issue reading return an error
glog.Infof("Unable to open %s for reading: %v", tuningFilePath, err)
return addArguments, deleteArguments, err
}

// Parse the tuning lines
scanner := bufio.NewScanner(file)
for scanner.Scan() {
line := scanner.Text()
if strings.HasPrefix(line, "ADD") {
// NOTE: Today only specific bare kernel arguments are allowed so
// there is not a need to split on =.
key := line[4:]
if isArgTunable(key) {
addArguments = append(addArguments, types.TuneArgument{Key: key, Bare: true})
} else {
glog.Infof("%s not a whitelisted kernel argument", key)
}
} else if strings.HasPrefix(line, "DELETE") {
// NOTE: Today only specific bare kernel arguments are allowed so
// there is not a need to split on =.
key := line[7:]
if isArgTunable(key) {
deleteArguments = append(deleteArguments, types.TuneArgument{Key: key, Bare: true})
} else {
glog.Infof("%s not a whitelisted kernel argument", key)
}
} else {
glog.V(2).Infof(`skipping malformed line in %s: "%s\n"`, tuningFilePath, line)
}
}
return addArguments, deleteArguments, nil
}

// updateTuningArgs executes additions and removals of kernel tuning arguments
func updateTuningArgs(tuningFilePath string) (changed bool, err error) {
changed = false
additions, deletions, err := parseTuningFile(tuningFilePath)
if err != nil {
return changed, err
}

// Execute additions
for _, toAdd := range additions {
if toAdd.Bare {
changed = true
utils.Run("rpm-ostree", "kargs", fmt.Sprintf("--delete=%s=%s", toAdd.Key, toAdd.Key))
} else {
// TODO: currently not supported
}
}
// Execute deletions
for _, toDelete := range deletions {
if toDelete.Bare {
changed = true
utils.Run("rpm-ostree", "kargs", fmt.Sprintf("--append=%s=%s", toDelete.Key, toDelete.Key))
} else {
// TODO: currently not supported
}
}
return changed, nil
}

// podmanRemove kills and removes a container
func podmanRemove(cid string) {
utils.RunIgnoreErr("podman", "kill", cid)
Expand Down Expand Up @@ -241,6 +335,23 @@ func Execute(cmd *cobra.Command, args []string) {
utils.RunIgnoreErr("podman", "rmi", imgid)
}

// Check to see if we need to tune kernel arguments
tuningChanged, err := updateTuningArgs(kernelTuningFile)
if err != nil {
glog.Infof("unable to parse tuning file %s: %s", kernelTuningFile, err)
}
// If tuning changes but the oscontainer didn't we still denote we changed
// for the reboot
if tuningChanged {
changed = true
// Remove the as it's been processed
err = os.Remove(kernelTuningFile)
if err != nil {
glog.Infof(`Unable to remove kernel tuning file %s: "%s"`, kernelTuningFile, err)
}

}

if !changed {
glog.Info("Already at target pivot; exiting...")
if exit_77 {
Expand Down
57 changes: 57 additions & 0 deletions cmd/root_test.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
package cmd

import (
"io/ioutil"
"os"
"testing"
)

Expand Down Expand Up @@ -30,3 +32,58 @@ func TestCompareOSImageURL(t *testing.T) {
t.Fatalf("Expected err")
}
}

// writeTestTuningFile writes out a file to use in the test
func writeTestTuningFile(content []byte) (filePath string, err error) {
tmpfile, err := ioutil.TempFile("", "testParseTuningFile")
if err != nil {
return "", err
}
filePath = tmpfile.Name()
if _, err := tmpfile.Write(content); err != nil {
return filePath, err
}
if err := tmpfile.Close(); err != nil {
return filePath, err
}
return filePath, nil
}

func TestParseTuningFile(t *testing.T) {
// Test with addition/deletion and verify white list
testFilePath, err := writeTestTuningFile([]byte("ADD nosmt\nADD aaaa\nDELETE nosmt\nDELETE nope"))
defer os.Remove(testFilePath)
if err != nil {
t.Fatalf("unable to write test file %s: %s", testFilePath, err)
}
add, delete, err := parseTuningFile(testFilePath)
if err != nil {
t.Fatalf(`Expected no error, got %s`, err)
}
if len(add) != 1 {
t.Fatalf("Expected 1 addition, got %v", len(add))
}

if len(delete) != 1 {
t.Fatalf("Expected 1 deletion, got %v", len(add))
}

// Test with no changes
testFilePath, err = writeTestTuningFile([]byte(""))
defer os.Remove(testFilePath)
if err != nil {
t.Fatalf("unable to write test file %s: %s", testFilePath, err)
}
add, delete, err = parseTuningFile(testFilePath)
if err != nil {
t.Fatalf(`Expected no error, got %s`, err)
}

if len(add) != 0 {
t.Fatalf("Expected 0 addition, got %v", len(add))
}

if len(delete) != 0 {
t.Fatalf("Expected 0 deletion, got %v", len(add))
}
}
8 changes: 8 additions & 0 deletions types/tuneargument.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
package types

// TuneArgument represents a single tuning argument
type TuneArgument struct {
Key string `json:"key"` // The name of the argument (or argument itself if Bare)
Value string `json:"value"` // The value of the argument
Bare bool `json:"bare"` // If the kernel argument is a bare argument (no value expected)
}

0 comments on commit cfada35

Please sign in to comment.