2019-02-11 05:05:43 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2020-02-28 00:30:10 +00:00
|
|
|
"io"
|
2019-02-11 05:05:43 +00:00
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
2019-03-07 00:35:11 +00:00
|
|
|
"runtime"
|
2019-02-11 05:05:43 +00:00
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
|
2020-11-19 20:22:17 +00:00
|
|
|
"golang.org/x/term"
|
2019-02-11 05:05:43 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// VERSION is the current version of init
|
2022-02-04 18:59:42 +00:00
|
|
|
VERSION = "Direktil init v2.0"
|
2019-02-11 05:05:43 +00:00
|
|
|
|
|
|
|
rootMountFlags = 0
|
|
|
|
bootMountFlags = syscall.MS_NOEXEC | syscall.MS_NODEV | syscall.MS_NOSUID | syscall.MS_RDONLY
|
|
|
|
layerMountFlags = syscall.MS_RDONLY
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
bootVersion string
|
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
2019-03-07 00:35:11 +00:00
|
|
|
runtime.LockOSThread()
|
|
|
|
|
2019-02-11 05:05:43 +00:00
|
|
|
log.Print("Welcome to ", VERSION)
|
|
|
|
|
|
|
|
// essential mounts
|
|
|
|
mount("none", "/proc", "proc", 0, "")
|
|
|
|
mount("none", "/sys", "sysfs", 0, "")
|
|
|
|
mount("none", "/dev", "devtmpfs", 0, "")
|
|
|
|
|
|
|
|
// get the "boot version"
|
|
|
|
bootVersion = param("version", "current")
|
|
|
|
log.Printf("booting system %q", bootVersion)
|
|
|
|
|
2022-02-04 18:59:42 +00:00
|
|
|
_, err := os.Stat("/config.yaml")
|
2019-02-11 05:05:43 +00:00
|
|
|
if err != nil {
|
2022-02-04 18:59:42 +00:00
|
|
|
if os.IsNotExist(err) {
|
|
|
|
bootV1()
|
|
|
|
return
|
2019-02-11 05:05:43 +00:00
|
|
|
}
|
2022-02-04 18:59:42 +00:00
|
|
|
fatal("stat failed: ", err)
|
2019-02-11 05:05:43 +00:00
|
|
|
}
|
|
|
|
|
2022-02-04 18:59:42 +00:00
|
|
|
bootV2()
|
2019-02-11 05:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func layerPath(name string) string {
|
|
|
|
return fmt.Sprintf("/boot/%s/layers/%s.fs", bootVersion, name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func fatal(v ...interface{}) {
|
|
|
|
log.Print("*** FATAL ***")
|
|
|
|
log.Print(v...)
|
2020-11-19 20:22:17 +00:00
|
|
|
die()
|
2019-02-11 05:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func fatalf(pattern string, v ...interface{}) {
|
|
|
|
log.Print("*** FATAL ***")
|
|
|
|
log.Printf(pattern, v...)
|
2020-11-19 20:22:17 +00:00
|
|
|
die()
|
|
|
|
}
|
|
|
|
|
|
|
|
func die() {
|
2022-02-04 18:59:42 +00:00
|
|
|
fmt.Println("\nwill reboot in 1 minute; press r to reboot now, o to power off, s to get a shell")
|
2020-11-19 20:22:17 +00:00
|
|
|
|
|
|
|
deadline := time.Now().Add(time.Minute)
|
|
|
|
|
|
|
|
term.MakeRaw(int(os.Stdin.Fd())) // disable line buffering
|
|
|
|
os.Stdin.SetReadDeadline(deadline)
|
|
|
|
|
|
|
|
b := []byte{0}
|
|
|
|
for {
|
|
|
|
_, err := os.Stdin.Read(b)
|
|
|
|
if err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2022-02-04 18:59:42 +00:00
|
|
|
fmt.Println(string(b))
|
|
|
|
|
2020-11-19 20:22:17 +00:00
|
|
|
switch b[0] {
|
|
|
|
case 'o':
|
|
|
|
syscall.Reboot(syscall.LINUX_REBOOT_CMD_POWER_OFF)
|
|
|
|
case 'r':
|
|
|
|
syscall.Reboot(syscall.LINUX_REBOOT_CMD_RESTART)
|
2022-02-04 18:59:42 +00:00
|
|
|
case 's':
|
|
|
|
err = syscall.Exec("/bin/ash", []string{"/bin/ash"}, os.Environ())
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("failed to start the shell:", err)
|
|
|
|
}
|
2020-11-19 20:22:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
syscall.Reboot(syscall.LINUX_REBOOT_CMD_RESTART)
|
2019-02-11 05:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func losetup(dev, file string) {
|
|
|
|
run("/sbin/losetup", dev, file)
|
|
|
|
}
|
|
|
|
|
|
|
|
func run(cmd string, args ...string) {
|
|
|
|
if output, err := exec.Command(cmd, args...).CombinedOutput(); err != nil {
|
|
|
|
fatalf("command %s %q failed: %v\n%s", cmd, args, err, string(output))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func mkdir(dir string, mode os.FileMode) {
|
|
|
|
if err := os.MkdirAll(dir, mode); err != nil {
|
|
|
|
fatalf("mkdir %q failed: %v", dir, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func mount(source, target, fstype string, flags uintptr, data string) {
|
|
|
|
if _, err := os.Stat(target); os.IsNotExist(err) {
|
|
|
|
mkdir(target, 0755)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := syscall.Mount(source, target, fstype, flags, data); err != nil {
|
|
|
|
fatalf("mount %q %q -t %q -o %q failed: %v", source, target, fstype, data, err)
|
|
|
|
}
|
|
|
|
log.Printf("mounted %q", target)
|
|
|
|
}
|
2020-02-28 00:30:10 +00:00
|
|
|
|
|
|
|
func cp(srcPath, dstPath string) {
|
|
|
|
var err error
|
|
|
|
defer func() {
|
|
|
|
if err != nil {
|
|
|
|
fatalf("cp %s %s failed: %v", srcPath, dstPath, err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
src, err := os.Open(srcPath)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
defer src.Close()
|
|
|
|
|
|
|
|
dst, err := os.Create(dstPath)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
defer dst.Close()
|
|
|
|
|
|
|
|
_, err = io.Copy(dst, src)
|
|
|
|
}
|