|  | // Copyright 2020 The Monogon Project Authors. | 
|  | // | 
|  | // SPDX-License-Identifier: Apache-2.0 | 
|  | // | 
|  | // 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. | 
|  |  | 
|  | // ktestinit is an init designed to run inside a lightweight VM for running | 
|  | // tests in there.  It performs basic platform initialization like mounting | 
|  | // kernel filesystems and launches the test executable at /tester, passes the | 
|  | // exit code back out over the control socket to ktest and then terminates the | 
|  | // default VM kernel. | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "errors" | 
|  | "fmt" | 
|  | "os" | 
|  | "os/exec" | 
|  |  | 
|  | "golang.org/x/sys/unix" | 
|  | ) | 
|  |  | 
|  | func mountInit() error { | 
|  | for _, el := range []struct { | 
|  | dir   string | 
|  | fs    string | 
|  | flags uintptr | 
|  | }{ | 
|  | {"/sys", "sysfs", unix.MS_NOEXEC | unix.MS_NOSUID | unix.MS_NODEV}, | 
|  | {"/proc", "proc", unix.MS_NOEXEC | unix.MS_NOSUID | unix.MS_NODEV}, | 
|  | {"/dev", "devtmpfs", unix.MS_NOEXEC | unix.MS_NOSUID}, | 
|  | {"/dev/pts", "devpts", unix.MS_NOEXEC | unix.MS_NOSUID}, | 
|  | {"/tmp", "tmpfs", 0}, | 
|  | } { | 
|  | if err := os.Mkdir(el.dir, 0755); err != nil && !os.IsExist(err) { | 
|  | return fmt.Errorf("could not make %s: %w", el.dir, err) | 
|  | } | 
|  | if err := unix.Mount(el.fs, el.dir, el.fs, el.flags, ""); err != nil { | 
|  | return fmt.Errorf("could not mount %s on %s: %w", el.fs, el.dir, err) | 
|  | } | 
|  | } | 
|  | return nil | 
|  | } | 
|  |  | 
|  | func main() { | 
|  | if err := mountInit(); err != nil { | 
|  | panic(err) | 
|  | } | 
|  |  | 
|  | // First virtual serial is always stdout, second is control | 
|  | ioConn, err := os.OpenFile("/dev/vport1p1", os.O_RDWR, 0) | 
|  | if err != nil { | 
|  | fmt.Printf("Failed to open communication device: %v\n", err) | 
|  | return | 
|  | } | 
|  | cmd := exec.Command("/tester", "-test.v") | 
|  | cmd.Stderr = os.Stderr | 
|  | cmd.Stdout = os.Stdout | 
|  | cmd.Env = append(cmd.Env, "IN_KTEST=true") | 
|  | if err := cmd.Run(); err != nil { | 
|  | var exerr *exec.ExitError | 
|  | if errors.As(err, &exerr) { | 
|  | if _, err := ioConn.Write([]byte{uint8(exerr.ExitCode())}); err != nil { | 
|  | panic(err) | 
|  | } | 
|  | } else if err != nil { | 
|  | fmt.Printf("Failed to execute tests (tests didn't run): %v", err) | 
|  | } | 
|  | } else { | 
|  | ioConn.Write([]byte{0}) | 
|  | } | 
|  | ioConn.Close() | 
|  |  | 
|  | unix.Reboot(unix.LINUX_REBOOT_CMD_RESTART) | 
|  | } |