|  | // 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. | 
|  |  | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "fmt" | 
|  | "os" | 
|  | "os/signal" | 
|  | "runtime/debug" | 
|  | "git.monogon.dev/source/smalltown.git/internal/network" | 
|  | "git.monogon.dev/source/smalltown.git/internal/node" | 
|  | "git.monogon.dev/source/smalltown.git/pkg/tpm" | 
|  |  | 
|  | "go.uber.org/zap" | 
|  | "golang.org/x/sys/unix" | 
|  | ) | 
|  |  | 
|  | func main() { | 
|  | defer func() { | 
|  | if r := recover(); r != nil { | 
|  | fmt.Println("Init panicked:", r) | 
|  | debug.PrintStack() | 
|  | } | 
|  | unix.Sync() | 
|  | // TODO: Switch this to Reboot when init panics are less likely | 
|  | unix.Reboot(unix.LINUX_REBOOT_CMD_POWER_OFF) | 
|  | }() | 
|  | logger, err := zap.NewDevelopment() | 
|  | if err != nil { | 
|  | panic(err) | 
|  | } | 
|  | logger.Info("Starting Smalltown Init") | 
|  |  | 
|  | // Set up bare minimum mounts | 
|  | if err := os.Mkdir("/sys", 0755); err != nil { | 
|  | panic(err) | 
|  | } | 
|  | if err := unix.Mount("sysfs", "/sys", "sysfs", unix.MS_NOEXEC|unix.MS_NOSUID|unix.MS_NODEV, ""); err != nil { | 
|  | panic(err) | 
|  | } | 
|  |  | 
|  | if err := os.Mkdir("/proc", 0755); err != nil { | 
|  | panic(err) | 
|  | } | 
|  | if err := unix.Mount("procfs", "/proc", "proc", unix.MS_NOEXEC|unix.MS_NOSUID|unix.MS_NODEV, ""); err != nil { | 
|  | panic(err) | 
|  | } | 
|  |  | 
|  | signalChannel := make(chan os.Signal, 2) | 
|  | signal.Notify(signalChannel) | 
|  |  | 
|  | if err := tpm.Initialize(logger.With(zap.String("component", "tpm"))); err != nil { | 
|  | logger.Panic("Failed to initialize TPM 2.0", zap.Error(err)) | 
|  | } | 
|  |  | 
|  | networkSvc, err := network.NewNetworkService(network.Config{}, logger.With(zap.String("component", "network"))) | 
|  | if err != nil { | 
|  | panic(err) | 
|  | } | 
|  | networkSvc.Start() | 
|  |  | 
|  | nodeInstance, err := node.NewSmalltownNode(logger, 7833, 7834) | 
|  | if err != nil { | 
|  | panic(err) | 
|  | } | 
|  |  | 
|  | err = nodeInstance.Start() | 
|  | if err != nil { | 
|  | panic(err) | 
|  | } | 
|  |  | 
|  | // We're PID1, so orphaned processes get reparented to us to clean up | 
|  | for { | 
|  | sig := <-signalChannel | 
|  | switch sig { | 
|  | case unix.SIGCHLD: | 
|  | var status unix.WaitStatus | 
|  | var rusage unix.Rusage | 
|  | for { | 
|  | res, err := unix.Wait4(-1, &status, unix.WNOHANG, &rusage) | 
|  | if err != nil && err != unix.ECHILD { | 
|  | logger.Error("Failed to wait on orphaned child", zap.Error(err)) | 
|  | break | 
|  | } | 
|  | if res <= 0 { | 
|  | break | 
|  | } | 
|  | } | 
|  | // TODO(lorenz): We can probably get more than just SIGCHLD as init, but I can't think | 
|  | // of any others right now, just log them in case we hit any of them. | 
|  | default: | 
|  | logger.Warn("Got unexpected signal", zap.String("signal", sig.String())) | 
|  | } | 
|  | } | 
|  | } |