// Copyright 2018 syzkaller project authors. All rights reserved.
// Use of this source code is governed by Apache 2 LICENSE that can be found in the LICENSE file.
package build
import (
"fmt"
"os"
"path/filepath"
"runtime"
"strconv"
"time"
"github.com/google/syzkaller/pkg/osutil"
)
type akaros struct{}
func (ctx akaros) build(targetArch, vmType, kernelDir, outputDir, compiler, userspaceDir,
cmdlineFile, sysctlFile string, config []byte) error {
configFile := filepath.Join(kernelDir, ".config")
if err := osutil.WriteFile(configFile, config); err != nil {
return fmt.Errorf("failed to write config file: %v", err)
}
if err := osutil.SandboxChown(configFile); err != nil {
return err
}
sshkey := filepath.Join(kernelDir, "key")
sshkeyPub := sshkey + ".pub"
os.Remove(sshkey)
os.Remove(sshkeyPub)
if _, err := osutil.RunCmd(10*time.Minute, "", "ssh-keygen", "-t", "rsa", "-b", "2048",
"-N", "", "-C", "", "-f", sshkey); err != nil {
return err
}
if err := osutil.SandboxChown(sshkeyPub); err != nil {
return err
}
if err := ctx.make(kernelDir, "", "olddefconfig", "ARCH=x86"); err != nil {
return err
}
if err := ctx.make(kernelDir, "", "xcc"); err != nil {
return err
}
if err := ctx.make(kernelDir, "tools/dev-libs/elfutils", "install"); err != nil {
return err
}
if err := ctx.make(kernelDir, "", "apps-install"); err != nil {
return err
}
if err := ctx.make(kernelDir, "", "fill-kfs"); err != nil {
return err
}
targetKey := filepath.Join(kernelDir, "kern", "kfs", ".ssh", "authorized_keys")
if err := os.Rename(sshkeyPub, targetKey); err != nil {
return err
}
const init = `#!/bin/bash
/ifconfig
dropbear -F 2>db_out &
bash
`
initFile := filepath.Join(kernelDir, "kern", "kfs", "init.sh")
if err := osutil.WriteFile(initFile, []byte(init)); err != nil {
return fmt.Errorf("failed to write init script: %v", err)
}
if err := osutil.SandboxChown(initFile); err != nil {
return err
}
if err := os.Chmod(initFile, 0770); err != nil {
return err
}
if err := ctx.cmd(kernelDir, "dropbear", "./CONFIGURE_AKAROS"); err != nil {
return err
}
if err := ctx.make(kernelDir, "dropbear/build"); err != nil {
return err
}
if err := ctx.make(kernelDir, "dropbear/build", "install"); err != nil {
return err
}
if err := ctx.make(kernelDir, ""); err != nil {
return err
}
if err := osutil.WriteFile(filepath.Join(outputDir, "image"), nil); err != nil {
return fmt.Errorf("failed to write image file: %v", err)
}
for src, dst := range map[string]string{
".config": "kernel.config",
"key": "key",
"obj/kern/akaros-kernel": "kernel",
"obj/kern/akaros-kernel-64b": "obj/akaros-kernel-64b",
} {
fullSrc := filepath.Join(kernelDir, filepath.FromSlash(src))
fullDst := filepath.Join(outputDir, filepath.FromSlash(dst))
if err := osutil.CopyFile(fullSrc, fullDst); err != nil {
return fmt.Errorf("faied to copy %v: %v", src, err)
}
}
return nil
}
func (ctx akaros) clean(kernelDir string) error {
// Note: this does not clean toolchain and elfutils.
return ctx.make(kernelDir, "", "realclean")
}
func (ctx akaros) make(kernelDir, runDir string, args ...string) error {
args = append([]string{"-j", strconv.Itoa(runtime.NumCPU())}, args...)
return ctx.cmd(kernelDir, runDir, "make", args...)
}
func (ctx akaros) cmd(kernelDir, runDir string, bin string, args ...string) error {
cmd := osutil.Command(bin, args...)
if err := osutil.Sandbox(cmd, true, false); err != nil {
return err
}
cmd.Dir = kernelDir
if runDir != "" {
cmd.Dir = filepath.Join(kernelDir, filepath.FromSlash(runDir))
}
cmd.Env = append([]string{}, os.Environ()...)
cmd.Env = append(cmd.Env, []string{
"MAKE_JOBS=" + strconv.Itoa(runtime.NumCPU()),
"AKAROS_ROOT=" + kernelDir,
"AKAROS_XCC_ROOT=" + filepath.Join(kernelDir, "toolchain", "x86_64-ucb-akaros-gcc"),
"X86_64_INSTDIR=" + filepath.Join(kernelDir, "toolchain", "x86_64-ucb-akaros-gcc"),
"PATH=" + filepath.Join(kernelDir, "toolchain", "x86_64-ucb-akaros-gcc", "bin") +
string(filepath.ListSeparator) + os.Getenv("PATH"),
}...)
_, err := osutil.Run(time.Hour, cmd)
return err
}