unix/fiss

src/sigremap/sigremap.c in master
Repositories | Summary | Log | Files | LICENSE

sigremap.c (8264B) download


  1/*  Copyright (c) 2015 Yelp, Inc.
  2    With modification 2023 Friedel Schon
  3
  4    Permission is hereby granted, free of charge, to any person obtaining a copy
  5    of this software and associated documentation files (the "Software"), to deal
  6    in the Software without restriction, including without limitation the rights
  7    to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8    copies of the Software, and to permit persons to whom the Software is
  9    furnished to do so, subject to the following conditions:
 10
 11    The above copyright notice and this permission notice shall be included in
 12    all copies or substantial portions of the Software.
 13
 14    THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 15    IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 16    FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 17    AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 18    LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 19    OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 20    THE SOFTWARE.
 21    */
 22
 23
 24#include "message.h"
 25#include "signame.h"
 26#include "util.h"
 27
 28#include <assert.h>
 29#include <errno.h>
 30#include <getopt.h>
 31#include <stdbool.h>
 32#include <stdio.h>
 33#include <stdlib.h>
 34#include <string.h>
 35#include <sys/ioctl.h>
 36#include <sys/wait.h>
 37#include <unistd.h>
 38
 39
 40const char* current_prog(void) {
 41	return "sigremap";
 42}
 43
 44#define DEBUG(...)                  \
 45	do {                            \
 46		if (debug)                  \
 47			fprint(1, __VA_ARGS__); \
 48	} while (0)
 49
 50#define set_signal_undefined(old, new) \
 51	if (signal_remap[old] == -1)       \
 52		signal_remap[old] = new;
 53
 54
 55// Signals we care about are numbered from 1 to 31, inclusive.
 56// (32 and above are real-time signals.)
 57// TODO: this is likely not portable outside of Linux, or on strange architectures
 58#define MAXSIG 31
 59
 60// Indices are one-indexed (signal 1 is at index 1). Index zero is unused.
 61// User-specified signal rewriting.
 62static int signal_remap[MAXSIG + 1];
 63// One-time ignores due to TTY quirks. 0 = no skip, 1 = skip the next-received signal.
 64static bool signal_temporary_ignores[MAXSIG + 1];
 65
 66static int  child_pid  = -1;
 67static bool debug      = false;
 68static bool use_setsid = true;
 69
 70
 71/*
 72 * The sigremap signal handler.
 73 *
 74 * The main job of this signal handler is to forward signals along to our child
 75 * process(es). In setsid mode, this means signaling the entire process group
 76 * rooted at our child. In non-setsid mode, this is just signaling the primary
 77 * child.
 78 *
 79 * In most cases, simply proxying the received signal is sufficient. If we
 80 * receive a job control signal, however, we should not only forward it, but
 81 * also sleep sigremap itself.
 82 *
 83 * This allows users to run foreground processes using sigremap and to
 84 * control them using normal shell job control features (e.g. Ctrl-Z to
 85 * generate a SIGTSTP and suspend the process).
 86 *
 87 * The libc manual is useful:
 88 * https://www.gnu.org/software/libc/manual/html_node/Job-Control-Signals.html
 89 *
 90 */
 91static void handle_signal(int signum) {
 92	DEBUG("Received signal %d.\n", signum);
 93
 94	if (signal_temporary_ignores[signum] == 1) {
 95		DEBUG("Ignoring tty hand-off signal %d.\n", signum);
 96		signal_temporary_ignores[signum] = 0;
 97	} else if (signum == SIGCHLD) {
 98		int status, exit_status;
 99		int killed_pid;
100		while ((killed_pid = waitpid(-1, &status, WNOHANG)) > 0) {
101			if (WIFEXITED(status)) {
102				exit_status = WEXITSTATUS(status);
103				DEBUG("A child with PID %d exited with exit status %d.\n", killed_pid, exit_status);
104			} else {
105				assert(WIFSIGNALED(status));
106				exit_status = 128 + WTERMSIG(status);
107				DEBUG("A child with PID %d was terminated by signal %d.\n", killed_pid, exit_status - 128);
108			}
109
110			if (killed_pid == child_pid) {
111				kill(use_setsid ? -child_pid : child_pid, SIGTERM);    // send SIGTERM to any remaining children
112				DEBUG("Child exited with status %d. Goodbye.\n", exit_status);
113				exit(exit_status);
114			}
115		}
116	} else {
117		if (signum <= MAXSIG && signal_remap[signum] != -1) {
118			DEBUG("Translating signal %d to %d.\n", signum, signal_remap[signum]);
119			signum = signal_remap[signum];
120		}
121
122		kill(use_setsid ? -child_pid : child_pid, signum);
123		DEBUG("Forwarded signal %d to children.\n", signum);
124
125		if (signum == SIGTSTP || signum == SIGTTOU || signum == SIGTTIN) {
126			DEBUG("Suspending self due to TTY signal.\n");
127			kill(getpid(), SIGSTOP);
128		}
129	}
130}
131
132static char** parse_command(int argc, char* argv[]) {
133	int           opt;
134	struct option long_options[] = {
135		{ "single", no_argument, NULL, 's' },
136		{ "verbose", no_argument, NULL, 'v' },
137		{ "version", no_argument, NULL, 'V' },
138		{ NULL, 0, NULL, 0 },
139	};
140	char *old, *new;
141	int   oldsig, newsig;
142
143	while ((opt = getopt_long(argc, argv, "+:hvVs", long_options, NULL)) != -1) {
144		switch (opt) {
145			case 'v':
146				debug = true;
147				break;
148			case 'V':
149				print_version_exit();
150			case 'c':
151				use_setsid = false;
152				break;
153			default:
154				print_usage_exit(PROG_SIGREMAP, 1);
155		}
156	}
157
158	argc -= optind, argv += optind;
159
160	while (argc > 0) {
161		if ((new = strchr(argv[0], '=')) == NULL)
162			break;
163
164		old  = argv[0];
165		*new = '\0';
166		new ++;
167
168		if ((oldsig = signame(old)) == -1) {
169			fprint(1, "error: invalid old signal '%s'\n", old);
170			exit(1);
171		}
172		if ((newsig = signame(new)) == -1) {
173			fprint(1, "error: invalid new signal '%s'\n", new);
174			exit(1);
175		}
176		signal_remap[oldsig] = newsig;
177
178		argc--, argv++;
179	}
180
181	if (argc < 1) {
182		print_usage_exit(PROG_SIGREMAP, 1);
183	}
184
185	if (use_setsid) {
186		set_signal_undefined(SIGTSTP, SIGSTOP);
187		set_signal_undefined(SIGTSTP, SIGTTOU);
188		set_signal_undefined(SIGTSTP, SIGTTIN);
189	}
190
191	return &argv[optind];
192}
193
194// A dummy signal handler used for signals we care about.
195// On the FreeBSD kernel, ignored signals cannot be waited on by `sigwait` (but
196// they can be on Linux). We must provide a dummy handler.
197// https://lists.freebsd.org/pipermail/freebsd-ports/2009-October/057340.html
198static void dummy(int signum) {
199	(void) signum;
200}
201
202int main(int argc, char* argv[]) {
203	char**   cmd = parse_command(argc, argv);
204	sigset_t all_signals;
205	int      signum;
206
207	sigfillset(&all_signals);
208	sigprocmask(SIG_BLOCK, &all_signals, NULL);
209
210	for (int i = 1; i <= MAXSIG; i++) {
211		signal_remap[i]             = -1;
212		signal_temporary_ignores[i] = false;
213
214		signal(i, dummy);
215	}
216
217	/*
218	 * Detach sigremap from controlling tty, so that the child's session can
219	 * attach to it instead.
220	 *
221	 * We want the child to be able to be the session leader of the TTY so that
222	 * it can do normal job control.
223	 */
224	if (use_setsid) {
225		if (ioctl(STDIN_FILENO, TIOCNOTTY) == -1) {
226			DEBUG(
227			    "Unable to detach from controlling tty (errno=%d %s).\n",
228			    errno,
229			    strerror(errno));
230		} else {
231			/*
232			 * When the session leader detaches from its controlling tty via
233			 * TIOCNOTTY, the kernel sends SIGHUP and SIGCONT to the process
234			 * group. We need to be careful not to forward these on to the
235			 * sigremap child so that it doesn't receive a SIGHUP and
236			 * terminate itself (#136).
237			 */
238			if (getsid(0) == getpid()) {
239				DEBUG("Detached from controlling tty, ignoring the first SIGHUP and SIGCONT we receive.\n");
240				signal_temporary_ignores[SIGHUP]  = 1;
241				signal_temporary_ignores[SIGCONT] = 1;
242			} else {
243				DEBUG("Detached from controlling tty, but was not session leader.\n");
244			}
245		}
246	}
247
248	child_pid = fork();
249	if (child_pid < 0) {
250		fprint(1, "error: unable to fork: %r\n");
251		return 1;
252	} else if (child_pid == 0) {
253		/* child */
254		sigprocmask(SIG_UNBLOCK, &all_signals, NULL);
255		if (use_setsid) {
256			if (setsid() == -1) {
257				fprint(1, "error: unable to setsid: %r\n");
258				exit(1);
259			}
260
261			if (ioctl(STDIN_FILENO, TIOCSCTTY, 0) == -1) {
262				DEBUG(
263				    "Unable to attach to controlling tty (errno=%d %s).\n",
264				    errno,
265				    strerror(errno));
266			}
267			DEBUG("setsid complete.\n");
268		}
269		execvp(cmd[0], cmd);
270
271		// if this point is reached, exec failed, so we should exit nonzero
272		print_errno("error: unable to execute %s: %s\n", cmd[0]);
273		_exit(2);
274	}
275
276	/* parent */
277	DEBUG("Child spawned with PID %d.\n", child_pid);
278	for (;;) {
279		sigwait(&all_signals, &signum);
280		handle_signal(signum);
281	}
282}