diff options
author | dzwdz | 2022-08-13 17:40:44 +0200 |
---|---|---|
committer | dzwdz | 2022-08-13 17:40:44 +0200 |
commit | d69d7614ea8b9720f934a94249ee8199bbcaf70c (patch) | |
tree | 3c82821e3c9f90148fdf392e5a26a9b5753756e6 | |
parent | 57427191e2cf0f08724b74fba3a35aa41ceda40d (diff) |
user: add /initctl for shutting the system down in a cleaner way
-rw-r--r-- | Makefile | 2 | ||||
-rw-r--r-- | initrd/bin/sh/halt | 2 | ||||
-rw-r--r-- | src/user/app/init/driver/driver.h | 2 | ||||
-rw-r--r-- | src/user/app/init/driver/initctl.c | 40 | ||||
-rw-r--r-- | src/user/app/init/init.c | 26 |
5 files changed, 67 insertions, 5 deletions
@@ -42,7 +42,7 @@ test: all @# the empty echo takes care of that, so the next echos will work just fine @echo > out/qemu.in echo tests > out/qemu.in - echo exit > out/qemu.in + echo halt > out/qemu.in @echo @cat out/qemu.out diff --git a/initrd/bin/sh/halt b/initrd/bin/sh/halt new file mode 100644 index 0000000..563aded --- /dev/null +++ b/initrd/bin/sh/halt @@ -0,0 +1,2 @@ +#!/bin/shell +echo halt > /initctl
\ No newline at end of file diff --git a/src/user/app/init/driver/driver.h b/src/user/app/init/driver/driver.h index c99955a..fc9f51b 100644 --- a/src/user/app/init/driver/driver.h +++ b/src/user/app/init/driver/driver.h @@ -1,5 +1,7 @@ #pragma once +#include <camellia/types.h> +void initctl_drv(handle_t killswitch); void ps2_drv(void); void tmpfs_drv(void); diff --git a/src/user/app/init/driver/initctl.c b/src/user/app/init/driver/initctl.c new file mode 100644 index 0000000..b8fbb9b --- /dev/null +++ b/src/user/app/init/driver/initctl.c @@ -0,0 +1,40 @@ +#include "driver.h" +#include <camellia/syscalls.h> +#include <errno.h> +#include <string.h> +#include <unistd.h> + +void initctl_drv(handle_t killswitch) { + struct fs_wait_response res; + char buf[64]; + const size_t buflen = sizeof buf; + while (!_syscall_fs_wait(buf, buflen, &res)) { + switch (res.op) { + case VFSOP_OPEN: + _syscall_fs_respond(NULL, res.len == 0 ? 0 : -1, 0); + break; + case VFSOP_WRITE: + /* null terminate */ + if (res.len > buflen - 1) + res.len = buflen - 1; + buf[res.len] = '\0'; + /* cut at first whitespace */ + for (size_t i = 0; buf[i]; i++) { + if (isspace(buf[i])) { + buf[i] = '\0'; + break; + } + } + if (!strcmp(buf, "halt")) { + _syscall_write(killswitch, "halt", 4, 0, 0); + exit(1); + } + _syscall_fs_respond(NULL, res.len, 0); + break; + default: + _syscall_fs_respond(NULL, -ENOSYS, 0); + break; + } + } + exit(1); +} diff --git a/src/user/app/init/init.c b/src/user/app/init/init.c index 2ae8048..ee8554f 100644 --- a/src/user/app/init/init.c +++ b/src/user/app/init/init.c @@ -25,6 +25,8 @@ void redirect(const char *exe, const char *out, const char *in) { } int main(void) { + handle_t killswitch_pipe[2]; + freopen("/kdev/com1", "a+", stdout); freopen("/kdev/com1", "a+", stderr); printf("in init (stage 2), main at 0x%x\n", &main); @@ -50,6 +52,16 @@ int main(void) { fs_union(list); } + if (_syscall_pipe(killswitch_pipe, 0) < 0) { + printf("couldn't create the killswitch pipe, quitting...\n"); + return 1; + } + MOUNT_AT("/initctl") { + close(killswitch_pipe[0]); + initctl_drv(killswitch_pipe[1]); + } + close(killswitch_pipe[1]); + if (fork()) { /* used to trigger a kernel bug * 7c96f9c03502e0c60f23f4c550d12a629f3b3daf */ @@ -57,10 +69,16 @@ int main(void) { exit(1); } - redirect("/bin/shell", "/kdev/com1", "/kdev/com1"); - redirect("/bin/shell", "/vtty", "/keyboard"); + if (!fork()) { + // TODO close on exec + close(killswitch_pipe[0]); + redirect("/bin/shell", "/kdev/com1", "/kdev/com1"); + redirect("/bin/shell", "/vtty", "/keyboard"); + _syscall_await(); + printf("init: restarting children not yet implemented\n"); + exit(1); + } - _syscall_await(); - printf("init: quitting\n"); + _syscall_read(killswitch_pipe[0], NULL, 0, 0); return 0; } |