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}