fiss

Friedel's Initialization and Service Supervision
Log | Files | Refs | LICENSE

commit ddbc46150b2e74b3a74ff64df42617d694733583
parent ddee3e83c8f682a0717a716b746caa1f26a4bcff
Author: Friedel Schon <[email protected]>
Date:   Mon, 24 Apr 2023 11:03:28 +0200

adding sigremap from dump-init

Diffstat:
Asrc/exec/sigremap.c | 313+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 313 insertions(+), 0 deletions(-)

diff --git a/src/exec/sigremap.c b/src/exec/sigremap.c @@ -0,0 +1,313 @@ +/* + * sigremap is a simple wrapper program designed to run as PID 1 and pass + * signals to its children. + * + * Usage: + * ./sigremap python -c 'while True: pass' + * + * To get debug output on stderr, run with '-v'. + */ + +#include "signame.h" + +#include <assert.h> +#include <errno.h> +#include <getopt.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <sys/ioctl.h> +#include <sys/wait.h> +#include <unistd.h> + +#define PRINTERR(...) \ + fprintf(stderr, "[sigremap] " __VA_ARGS__) + +#define DEBUG(...) \ + do { \ + if (debug) { \ + PRINTERR(__VA_ARGS__); \ + } \ + } while (0) + +#define set_signal_undefined(old, new) \ + if (signal_remap[old] == -1) \ + signal_remap[old] = new; + + +// Signals we care about are numbered from 1 to 31, inclusive. +// (32 and above are real-time signals.) +// TODO: this is likely not portable outside of Linux, or on strange architectures +#define MAXSIG 31 + +// Indices are one-indexed (signal 1 is at index 1). Index zero is unused. +// User-specified signal rewriting. +int signal_remap[MAXSIG + 1] = { [0 ... MAXSIG] = -1 }; +// One-time ignores due to TTY quirks. 0 = no skip, 1 = skip the next-received signal. +char signal_temporary_ignores[MAXSIG + 1] = { [0 ... MAXSIG] = 0 }; + +pid_t child_pid = -1; +char debug = 0; +char use_setsid = 1; + +int translate_signal(int signum) { + if (signum <= 0 || signum > MAXSIG) { + return signum; + } else { + int translated = signal_remap[signum]; + if (translated == -1) { + return signum; + } else { + DEBUG("Translating signal %d to %d.\n", signum, translated); + return translated; + } + } +} + +void forward_signal(int signum) { + signum = translate_signal(signum); + if (signum != 0) { + kill(use_setsid ? -child_pid : child_pid, signum); + DEBUG("Forwarded signal %d to children.\n", signum); + } else { + DEBUG("Not forwarding signal %d to children (ignored).\n", signum); + } +} + +/* + * The sigremap signal handler. + * + * The main job of this signal handler is to forward signals along to our child + * process(es). In setsid mode, this means signaling the entire process group + * rooted at our child. In non-setsid mode, this is just signaling the primary + * child. + * + * In most cases, simply proxying the received signal is sufficient. If we + * receive a job control signal, however, we should not only forward it, but + * also sleep sigremap itself. + * + * This allows users to run foreground processes using sigremap and to + * control them using normal shell job control features (e.g. Ctrl-Z to + * generate a SIGTSTP and suspend the process). + * + * The libc manual is useful: + * https://www.gnu.org/software/libc/manual/html_node/Job-Control-Signals.html + * + */ +void handle_signal(int signum) { + DEBUG("Received signal %d.\n", signum); + + if (signal_temporary_ignores[signum] == 1) { + DEBUG("Ignoring tty hand-off signal %d.\n", signum); + signal_temporary_ignores[signum] = 0; + } else if (signum == SIGCHLD) { + int status, exit_status; + pid_t killed_pid; + while ((killed_pid = waitpid(-1, &status, WNOHANG)) > 0) { + if (WIFEXITED(status)) { + exit_status = WEXITSTATUS(status); + DEBUG("A child with PID %d exited with exit status %d.\n", killed_pid, exit_status); + } else { + assert(WIFSIGNALED(status)); + exit_status = 128 + WTERMSIG(status); + DEBUG("A child with PID %d was terminated by signal %d.\n", killed_pid, exit_status - 128); + } + + if (killed_pid == child_pid) { + forward_signal(SIGTERM); // send SIGTERM to any remaining children + DEBUG("Child exited with status %d. Goodbye.\n", exit_status); + exit(exit_status); + } + } + } else { + forward_signal(signum); + if (signum == SIGTSTP || signum == SIGTTOU || signum == SIGTTIN) { + DEBUG("Suspending self due to TTY signal.\n"); + kill(getpid(), SIGSTOP); + } + } +} + +void print_help(char* argv[]) { + fprintf(stderr, + "Usage: %s [option] [old-signal=new-signal] command [[arg] ...]\n" + "\n" + "sigremap is a simple process supervisor that forwards signals to children.\n" + "It is designed to run as PID1 in minimal container environments.\n" + "\n" + "Optional arguments:\n" + " -s, --single Run in single-child mode.\n" + " In this mode, signals are only proxied to the\n" + " direct child and not any of its descendants.\n" + " -r, --remap s:r remap received signal s to new signal r before proxying.\n" + " To ignore (not proxy) a signal, remap it to 0.\n" + " This option can be specified multiple times.\n" + " -v, --verbose Print debugging information to stderr.\n" + " -h, --help Print this help message and exit.\n" + " -V, --version Print the current version and exit.\n" + "\n" + "Full help is available online at https://github.com/Yelp/sigremap\n", + argv[0]); +} + + +char** parse_command(int argc, char* argv[]) { + int opt; + struct option long_options[] = { + { "help", no_argument, NULL, 'h' }, + { "single", no_argument, NULL, 's' }, + { "verbose", no_argument, NULL, 'v' }, + { "version", no_argument, NULL, 'V' }, + { NULL, 0, NULL, 0 }, + }; + while ((opt = getopt_long(argc, argv, "+hvVs", long_options, NULL)) != -1) { + switch (opt) { + case 'h': + print_help(argv); + exit(0); + case 'v': + debug = 1; + break; + case 'V': + // fprintf(stderr, "sigremap v%.*s", VERSION_len, VERSION); + exit(0); + case 'c': + use_setsid = 0; + break; + default: + exit(1); + } + } + + argc -= optind, argv += optind; + + while (argc > 0) { + char *old, *new; + if ((new = strchr(argv[0], '=')) == NULL) + break; + + old = argv[0]; + *new = '\0'; + new ++; + + int oldsig, newsig; + + if ((oldsig = signame(old)) == -1) { + fprintf(stderr, "error: invalid old signal '%s'\n", old); + exit(1); + } + if ((newsig = signame(new)) == -1) { + fprintf(stderr, "error: invalid new signal '%s'\n", new); + exit(1); + } + signal_remap[oldsig] = newsig; + + argc--, argv++; + } + + if (argc < 1) { + fprintf( + stderr, + "Usage: %s [option] program [args]\n" + "Try %s --help for full usage.\n", + argv[0], argv[0]); + exit(1); + } + + if (use_setsid) { + set_signal_undefined(SIGTSTP, SIGSTOP); + set_signal_undefined(SIGTSTP, SIGTTOU); + set_signal_undefined(SIGTSTP, SIGTTIN); + } + + return &argv[optind]; +} + +// A dummy signal handler used for signals we care about. +// On the FreeBSD kernel, ignored signals cannot be waited on by `sigwait` (but +// they can be on Linux). We must provide a dummy handler. +// https://lists.freebsd.org/pipermail/freebsd-ports/2009-October/057340.html +void dummy(int signum) { + (void) signum; +} + +int main(int argc, char* argv[]) { + char** cmd = parse_command(argc, argv); + sigset_t all_signals; + sigfillset(&all_signals); + sigprocmask(SIG_BLOCK, &all_signals, NULL); + + for (int i = 1; i <= MAXSIG; i++) { + signal(i, dummy); + } + + /* + * Detach sigremap from controlling tty, so that the child's session can + * attach to it instead. + * + * We want the child to be able to be the session leader of the TTY so that + * it can do normal job control. + */ + if (use_setsid) { + if (ioctl(STDIN_FILENO, TIOCNOTTY) == -1) { + DEBUG( + "Unable to detach from controlling tty (errno=%d %s).\n", + errno, + strerror(errno)); + } else { + /* + * When the session leader detaches from its controlling tty via + * TIOCNOTTY, the kernel sends SIGHUP and SIGCONT to the process + * group. We need to be careful not to forward these on to the + * sigremap child so that it doesn't receive a SIGHUP and + * terminate itself (#136). + */ + if (getsid(0) == getpid()) { + DEBUG("Detached from controlling tty, ignoring the first SIGHUP and SIGCONT we receive.\n"); + signal_temporary_ignores[SIGHUP] = 1; + signal_temporary_ignores[SIGCONT] = 1; + } else { + DEBUG("Detached from controlling tty, but was not session leader.\n"); + } + } + } + + child_pid = fork(); + if (child_pid < 0) { + PRINTERR("Unable to fork. Exiting.\n"); + return 1; + } else if (child_pid == 0) { + /* child */ + sigprocmask(SIG_UNBLOCK, &all_signals, NULL); + if (use_setsid) { + if (setsid() == -1) { + PRINTERR( + "Unable to setsid (errno=%d %s). Exiting.\n", + errno, + strerror(errno)); + exit(1); + } + + if (ioctl(STDIN_FILENO, TIOCSCTTY, 0) == -1) { + DEBUG( + "Unable to attach to controlling tty (errno=%d %s).\n", + errno, + strerror(errno)); + } + DEBUG("setsid complete.\n"); + } + execvp(cmd[0], &cmd[0]); + + // if this point is reached, exec failed, so we should exit nonzero + PRINTERR("%s: %s\n", cmd[0], strerror(errno)); + return 2; + } else { + /* parent */ + DEBUG("Child spawned with PID %d.\n", child_pid); + for (;;) { + int signum; + sigwait(&all_signals, &signum); + handle_signal(signum); + } + } +}