Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 1 | /* sig.c - interface for shell signal handlers and signal initialization. */ |
| 2 | |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 3 | /* Copyright (C) 1994-2013 Free Software Foundation, Inc. |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 4 | |
| 5 | This file is part of GNU Bash, the Bourne Again SHell. |
| 6 | |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 7 | Bash is free software: you can redistribute it and/or modify |
| 8 | it under the terms of the GNU General Public License as published by |
| 9 | the Free Software Foundation, either version 3 of the License, or |
| 10 | (at your option) any later version. |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 11 | |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 12 | Bash is distributed in the hope that it will be useful, |
| 13 | but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 15 | GNU General Public License for more details. |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 16 | |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 17 | You should have received a copy of the GNU General Public License |
| 18 | along with Bash. If not, see <http://www.gnu.org/licenses/>. |
| 19 | */ |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 20 | |
| 21 | #include "config.h" |
| 22 | |
| 23 | #include "bashtypes.h" |
| 24 | |
| 25 | #if defined (HAVE_UNISTD_H) |
Jari Aalto | cce855b | 1998-04-17 19:52:44 +0000 | [diff] [blame] | 26 | # ifdef _MINIX |
| 27 | # include <sys/types.h> |
| 28 | # endif |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 29 | # include <unistd.h> |
| 30 | #endif |
| 31 | |
| 32 | #include <stdio.h> |
| 33 | #include <signal.h> |
| 34 | |
Jari Aalto | b80f644 | 2004-07-27 13:29:18 +0000 | [diff] [blame] | 35 | #include "bashintl.h" |
| 36 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 37 | #include "shell.h" |
| 38 | #if defined (JOB_CONTROL) |
| 39 | #include "jobs.h" |
| 40 | #endif /* JOB_CONTROL */ |
| 41 | #include "siglist.h" |
| 42 | #include "sig.h" |
| 43 | #include "trap.h" |
| 44 | |
| 45 | #include "builtins/common.h" |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 46 | #include "builtins/builtext.h" |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 47 | |
| 48 | #if defined (READLINE) |
| 49 | # include "bashline.h" |
Chet Ramey | d5d0096 | 2011-11-22 20:00:47 -0500 | [diff] [blame] | 50 | # include <readline/readline.h> |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 51 | #endif |
| 52 | |
| 53 | #if defined (HISTORY) |
| 54 | # include "bashhist.h" |
| 55 | #endif |
| 56 | |
| 57 | extern int last_command_exit_value; |
Jari Aalto | b80f644 | 2004-07-27 13:29:18 +0000 | [diff] [blame] | 58 | extern int last_command_exit_signal; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 59 | extern int return_catch_flag; |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 60 | extern int loop_level, continuing, breaking, funcnest; |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 61 | extern int executing_list; |
| 62 | extern int comsub_ignore_return; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 63 | extern int parse_and_execute_level, shell_initialized; |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 64 | #if defined (HISTORY) |
| 65 | extern int history_lines_this_session; |
| 66 | #endif |
Chet Ramey | d5d0096 | 2011-11-22 20:00:47 -0500 | [diff] [blame] | 67 | extern int no_line_editing; |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 68 | extern int wait_signal_received; |
| 69 | extern sh_builtin_func_t *this_shell_builtin; |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 70 | |
| 71 | extern void initialize_siglist (); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 72 | |
| 73 | /* Non-zero after SIGINT. */ |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 74 | volatile sig_atomic_t interrupt_state = 0; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 75 | |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 76 | /* Non-zero after SIGWINCH */ |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 77 | volatile sig_atomic_t sigwinch_received = 0; |
| 78 | |
| 79 | /* Non-zero after SIGTERM */ |
| 80 | volatile sig_atomic_t sigterm_received = 0; |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 81 | |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 82 | /* Set to the value of any terminating signal received. */ |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 83 | volatile sig_atomic_t terminating_signal = 0; |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 84 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 85 | /* The environment at the top-level R-E loop. We use this in |
| 86 | the case of error return. */ |
| 87 | procenv_t top_level; |
| 88 | |
| 89 | #if defined (JOB_CONTROL) || defined (HAVE_POSIX_SIGNALS) |
| 90 | /* The signal masks that this shell runs with. */ |
| 91 | sigset_t top_level_mask; |
| 92 | #endif /* JOB_CONTROL */ |
| 93 | |
| 94 | /* When non-zero, we throw_to_top_level (). */ |
| 95 | int interrupt_immediately = 0; |
| 96 | |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 97 | /* When non-zero, we call the terminating signal handler immediately. */ |
| 98 | int terminate_immediately = 0; |
| 99 | |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 100 | #if defined (SIGWINCH) |
| 101 | static SigHandler *old_winch = (SigHandler *)SIG_DFL; |
| 102 | #endif |
| 103 | |
Jari Aalto | f73dda0 | 2001-11-13 17:56:06 +0000 | [diff] [blame] | 104 | static void initialize_shell_signals __P((void)); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 105 | |
| 106 | void |
Jari Aalto | 7117c2d | 2002-07-17 14:10:11 +0000 | [diff] [blame] | 107 | initialize_signals (reinit) |
| 108 | int reinit; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 109 | { |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 110 | initialize_shell_signals (); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 111 | initialize_job_signals (); |
Jari Aalto | e8ce775 | 1997-09-22 20:22:27 +0000 | [diff] [blame] | 112 | #if !defined (HAVE_SYS_SIGLIST) && !defined (HAVE_UNDER_SYS_SIGLIST) && !defined (HAVE_STRSIGNAL) |
Jari Aalto | 7117c2d | 2002-07-17 14:10:11 +0000 | [diff] [blame] | 113 | if (reinit == 0) |
| 114 | initialize_siglist (); |
Jari Aalto | e8ce775 | 1997-09-22 20:22:27 +0000 | [diff] [blame] | 115 | #endif /* !HAVE_SYS_SIGLIST && !HAVE_UNDER_SYS_SIGLIST && !HAVE_STRSIGNAL */ |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 116 | } |
| 117 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 118 | /* A structure describing a signal that terminates the shell if not |
| 119 | caught. The orig_handler member is present so children can reset |
| 120 | these signals back to their original handlers. */ |
| 121 | struct termsig { |
| 122 | int signum; |
| 123 | SigHandler *orig_handler; |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 124 | int orig_flags; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 125 | }; |
| 126 | |
| 127 | #define NULL_HANDLER (SigHandler *)SIG_DFL |
| 128 | |
| 129 | /* The list of signals that would terminate the shell if not caught. |
| 130 | We catch them, but just so that we can write the history file, |
| 131 | and so forth. */ |
| 132 | static struct termsig terminating_signals[] = { |
| 133 | #ifdef SIGHUP |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 134 | { SIGHUP, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 135 | #endif |
| 136 | |
| 137 | #ifdef SIGINT |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 138 | { SIGINT, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 139 | #endif |
| 140 | |
| 141 | #ifdef SIGILL |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 142 | { SIGILL, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 143 | #endif |
| 144 | |
| 145 | #ifdef SIGTRAP |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 146 | { SIGTRAP, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 147 | #endif |
| 148 | |
| 149 | #ifdef SIGIOT |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 150 | { SIGIOT, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 151 | #endif |
| 152 | |
| 153 | #ifdef SIGDANGER |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 154 | { SIGDANGER, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 155 | #endif |
| 156 | |
| 157 | #ifdef SIGEMT |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 158 | { SIGEMT, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 159 | #endif |
| 160 | |
| 161 | #ifdef SIGFPE |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 162 | { SIGFPE, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 163 | #endif |
| 164 | |
| 165 | #ifdef SIGBUS |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 166 | { SIGBUS, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 167 | #endif |
| 168 | |
| 169 | #ifdef SIGSEGV |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 170 | { SIGSEGV, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 171 | #endif |
| 172 | |
| 173 | #ifdef SIGSYS |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 174 | { SIGSYS, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 175 | #endif |
| 176 | |
| 177 | #ifdef SIGPIPE |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 178 | { SIGPIPE, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 179 | #endif |
| 180 | |
| 181 | #ifdef SIGALRM |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 182 | { SIGALRM, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 183 | #endif |
| 184 | |
| 185 | #ifdef SIGTERM |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 186 | { SIGTERM, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 187 | #endif |
| 188 | |
| 189 | #ifdef SIGXCPU |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 190 | { SIGXCPU, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 191 | #endif |
| 192 | |
| 193 | #ifdef SIGXFSZ |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 194 | { SIGXFSZ, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 195 | #endif |
| 196 | |
| 197 | #ifdef SIGVTALRM |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 198 | { SIGVTALRM, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 199 | #endif |
| 200 | |
Jari Aalto | bb70624 | 2000-03-17 21:46:59 +0000 | [diff] [blame] | 201 | #if 0 |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 202 | #ifdef SIGPROF |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 203 | { SIGPROF, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 204 | #endif |
Jari Aalto | bb70624 | 2000-03-17 21:46:59 +0000 | [diff] [blame] | 205 | #endif |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 206 | |
| 207 | #ifdef SIGLOST |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 208 | { SIGLOST, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 209 | #endif |
| 210 | |
| 211 | #ifdef SIGUSR1 |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 212 | { SIGUSR1, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 213 | #endif |
| 214 | |
| 215 | #ifdef SIGUSR2 |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 216 | { SIGUSR2, NULL_HANDLER, 0 }, |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 217 | #endif |
| 218 | }; |
| 219 | |
| 220 | #define TERMSIGS_LENGTH (sizeof (terminating_signals) / sizeof (struct termsig)) |
| 221 | |
| 222 | #define XSIG(x) (terminating_signals[x].signum) |
| 223 | #define XHANDLER(x) (terminating_signals[x].orig_handler) |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 224 | #define XSAFLAGS(x) (terminating_signals[x].orig_flags) |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 225 | |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 226 | static int termsigs_initialized = 0; |
| 227 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 228 | /* Initialize signals that will terminate the shell to do some |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 229 | unwind protection. For non-interactive shells, we only call |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 230 | this when a trap is defined for EXIT (0) or when trap is run |
| 231 | to display signal dispositions. */ |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 232 | void |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 233 | initialize_terminating_signals () |
| 234 | { |
| 235 | register int i; |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 236 | #if defined (HAVE_POSIX_SIGNALS) |
| 237 | struct sigaction act, oact; |
| 238 | #endif |
| 239 | |
| 240 | if (termsigs_initialized) |
| 241 | return; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 242 | |
| 243 | /* The following code is to avoid an expensive call to |
| 244 | set_signal_handler () for each terminating_signals. Fortunately, |
| 245 | this is possible in Posix. Unfortunately, we have to call signal () |
| 246 | on non-Posix systems for each signal in terminating_signals. */ |
| 247 | #if defined (HAVE_POSIX_SIGNALS) |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 248 | act.sa_handler = termsig_sighandler; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 249 | act.sa_flags = 0; |
| 250 | sigemptyset (&act.sa_mask); |
| 251 | sigemptyset (&oact.sa_mask); |
| 252 | for (i = 0; i < TERMSIGS_LENGTH; i++) |
| 253 | sigaddset (&act.sa_mask, XSIG (i)); |
| 254 | for (i = 0; i < TERMSIGS_LENGTH; i++) |
| 255 | { |
Jari Aalto | 7117c2d | 2002-07-17 14:10:11 +0000 | [diff] [blame] | 256 | /* If we've already trapped it, don't do anything. */ |
| 257 | if (signal_is_trapped (XSIG (i))) |
| 258 | continue; |
| 259 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 260 | sigaction (XSIG (i), &act, &oact); |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 261 | XHANDLER(i) = oact.sa_handler; |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 262 | XSAFLAGS(i) = oact.sa_flags; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 263 | /* Don't do anything with signals that are ignored at shell entry |
| 264 | if the shell is not interactive. */ |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 265 | /* XXX - should we do this for interactive shells, too? */ |
| 266 | if (interactive_shell == 0 && XHANDLER (i) == SIG_IGN) |
Jari Aalto | 28ef6c3 | 2001-04-06 19:14:31 +0000 | [diff] [blame] | 267 | { |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 268 | sigaction (XSIG (i), &oact, &act); |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 269 | set_signal_hard_ignored (XSIG (i)); |
Jari Aalto | 28ef6c3 | 2001-04-06 19:14:31 +0000 | [diff] [blame] | 270 | } |
Jari Aalto | cce855b | 1998-04-17 19:52:44 +0000 | [diff] [blame] | 271 | #if defined (SIGPROF) && !defined (_MINIX) |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 272 | if (XSIG (i) == SIGPROF && XHANDLER (i) != SIG_DFL && XHANDLER (i) != SIG_IGN) |
Jari Aalto | 28ef6c3 | 2001-04-06 19:14:31 +0000 | [diff] [blame] | 273 | sigaction (XSIG (i), &oact, (struct sigaction *)NULL); |
Jari Aalto | cce855b | 1998-04-17 19:52:44 +0000 | [diff] [blame] | 274 | #endif /* SIGPROF && !_MINIX */ |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 275 | } |
| 276 | |
| 277 | #else /* !HAVE_POSIX_SIGNALS */ |
| 278 | |
| 279 | for (i = 0; i < TERMSIGS_LENGTH; i++) |
| 280 | { |
Jari Aalto | 7117c2d | 2002-07-17 14:10:11 +0000 | [diff] [blame] | 281 | /* If we've already trapped it, don't do anything. */ |
| 282 | if (signal_is_trapped (XSIG (i))) |
| 283 | continue; |
| 284 | |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 285 | XHANDLER(i) = signal (XSIG (i), termsig_sighandler); |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 286 | XSAFLAGS(i) = 0; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 287 | /* Don't do anything with signals that are ignored at shell entry |
| 288 | if the shell is not interactive. */ |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 289 | /* XXX - should we do this for interactive shells, too? */ |
| 290 | if (interactive_shell == 0 && XHANDLER (i) == SIG_IGN) |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 291 | { |
Jari Aalto | 28ef6c3 | 2001-04-06 19:14:31 +0000 | [diff] [blame] | 292 | signal (XSIG (i), SIG_IGN); |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 293 | set_signal_hard_ignored (XSIG (i)); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 294 | } |
Jari Aalto | cce855b | 1998-04-17 19:52:44 +0000 | [diff] [blame] | 295 | #ifdef SIGPROF |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 296 | if (XSIG (i) == SIGPROF && XHANDLER (i) != SIG_DFL && XHANDLER (i) != SIG_IGN) |
Jari Aalto | 28ef6c3 | 2001-04-06 19:14:31 +0000 | [diff] [blame] | 297 | signal (XSIG (i), XHANDLER (i)); |
Jari Aalto | cce855b | 1998-04-17 19:52:44 +0000 | [diff] [blame] | 298 | #endif |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 299 | } |
| 300 | |
| 301 | #endif /* !HAVE_POSIX_SIGNALS */ |
| 302 | |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 303 | termsigs_initialized = 1; |
| 304 | } |
| 305 | |
| 306 | static void |
| 307 | initialize_shell_signals () |
| 308 | { |
| 309 | if (interactive) |
| 310 | initialize_terminating_signals (); |
| 311 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 312 | #if defined (JOB_CONTROL) || defined (HAVE_POSIX_SIGNALS) |
| 313 | /* All shells use the signal mask they inherit, and pass it along |
| 314 | to child processes. Children will never block SIGCHLD, though. */ |
| 315 | sigemptyset (&top_level_mask); |
| 316 | sigprocmask (SIG_BLOCK, (sigset_t *)NULL, &top_level_mask); |
Jari Aalto | bb70624 | 2000-03-17 21:46:59 +0000 | [diff] [blame] | 317 | # if defined (SIGCHLD) |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 318 | sigdelset (&top_level_mask, SIGCHLD); |
Jari Aalto | bb70624 | 2000-03-17 21:46:59 +0000 | [diff] [blame] | 319 | # endif |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 320 | #endif /* JOB_CONTROL || HAVE_POSIX_SIGNALS */ |
| 321 | |
| 322 | /* And, some signals that are specifically ignored by the shell. */ |
| 323 | set_signal_handler (SIGQUIT, SIG_IGN); |
| 324 | |
| 325 | if (interactive) |
| 326 | { |
| 327 | set_signal_handler (SIGINT, sigint_sighandler); |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 328 | get_original_signal (SIGTERM); |
| 329 | if (signal_is_hard_ignored (SIGTERM) == 0) |
| 330 | set_signal_handler (SIGTERM, sigterm_sighandler); |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 331 | set_sigwinch_handler (); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 332 | } |
| 333 | } |
| 334 | |
| 335 | void |
| 336 | reset_terminating_signals () |
| 337 | { |
| 338 | register int i; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 339 | #if defined (HAVE_POSIX_SIGNALS) |
| 340 | struct sigaction act; |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 341 | #endif |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 342 | |
Jari Aalto | d166f04 | 1997-06-05 14:59:13 +0000 | [diff] [blame] | 343 | if (termsigs_initialized == 0) |
| 344 | return; |
| 345 | |
| 346 | #if defined (HAVE_POSIX_SIGNALS) |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 347 | act.sa_flags = 0; |
| 348 | sigemptyset (&act.sa_mask); |
| 349 | for (i = 0; i < TERMSIGS_LENGTH; i++) |
| 350 | { |
| 351 | /* Skip a signal if it's trapped or handled specially, because the |
| 352 | trap code will restore the correct value. */ |
| 353 | if (signal_is_trapped (XSIG (i)) || signal_is_special (XSIG (i))) |
| 354 | continue; |
| 355 | |
| 356 | act.sa_handler = XHANDLER (i); |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 357 | act.sa_flags = XSAFLAGS (i); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 358 | sigaction (XSIG (i), &act, (struct sigaction *) NULL); |
| 359 | } |
| 360 | #else /* !HAVE_POSIX_SIGNALS */ |
| 361 | for (i = 0; i < TERMSIGS_LENGTH; i++) |
| 362 | { |
| 363 | if (signal_is_trapped (XSIG (i)) || signal_is_special (XSIG (i))) |
| 364 | continue; |
| 365 | |
| 366 | signal (XSIG (i), XHANDLER (i)); |
| 367 | } |
| 368 | #endif /* !HAVE_POSIX_SIGNALS */ |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 369 | |
| 370 | termsigs_initialized = 0; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 371 | } |
| 372 | #undef XSIG |
| 373 | #undef XHANDLER |
| 374 | |
Jari Aalto | f1be666 | 2008-11-18 13:15:12 +0000 | [diff] [blame] | 375 | /* Run some of the cleanups that should be performed when we run |
| 376 | jump_to_top_level from a builtin command context. XXX - might want to |
| 377 | also call reset_parser here. */ |
| 378 | void |
| 379 | top_level_cleanup () |
| 380 | { |
| 381 | /* Clean up string parser environment. */ |
| 382 | while (parse_and_execute_level) |
| 383 | parse_and_execute_cleanup (); |
| 384 | |
| 385 | #if defined (PROCESS_SUBSTITUTION) |
| 386 | unlink_fifo_list (); |
| 387 | #endif /* PROCESS_SUBSTITUTION */ |
| 388 | |
| 389 | run_unwind_protects (); |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 390 | loop_level = continuing = breaking = funcnest = 0; |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 391 | executing_list = comsub_ignore_return = return_catch_flag = 0; |
Jari Aalto | f1be666 | 2008-11-18 13:15:12 +0000 | [diff] [blame] | 392 | } |
| 393 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 394 | /* What to do when we've been interrupted, and it is safe to handle it. */ |
| 395 | void |
| 396 | throw_to_top_level () |
| 397 | { |
| 398 | int print_newline = 0; |
| 399 | |
| 400 | if (interrupt_state) |
| 401 | { |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 402 | if (last_command_exit_value < 128) |
| 403 | last_command_exit_value = 128 + SIGINT; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 404 | print_newline = 1; |
| 405 | DELINTERRUPT; |
| 406 | } |
| 407 | |
| 408 | if (interrupt_state) |
| 409 | return; |
| 410 | |
Jari Aalto | b80f644 | 2004-07-27 13:29:18 +0000 | [diff] [blame] | 411 | last_command_exit_signal = (last_command_exit_value > 128) ? |
| 412 | (last_command_exit_value - 128) : 0; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 413 | last_command_exit_value |= 128; |
| 414 | |
| 415 | /* Run any traps set on SIGINT. */ |
| 416 | run_interrupt_trap (); |
| 417 | |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 418 | /* Clean up string parser environment. */ |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 419 | while (parse_and_execute_level) |
| 420 | parse_and_execute_cleanup (); |
| 421 | |
| 422 | #if defined (JOB_CONTROL) |
Jari Aalto | 28ef6c3 | 2001-04-06 19:14:31 +0000 | [diff] [blame] | 423 | give_terminal_to (shell_pgrp, 0); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 424 | #endif /* JOB_CONTROL */ |
| 425 | |
| 426 | #if defined (JOB_CONTROL) || defined (HAVE_POSIX_SIGNALS) |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 427 | /* This needs to stay because jobs.c:make_child() uses it without resetting |
| 428 | the signal mask. */ |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 429 | sigprocmask (SIG_SETMASK, &top_level_mask, (sigset_t *)NULL); |
| 430 | #endif |
| 431 | |
| 432 | reset_parser (); |
| 433 | |
| 434 | #if defined (READLINE) |
| 435 | if (interactive) |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 436 | bashline_reset (); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 437 | #endif /* READLINE */ |
| 438 | |
| 439 | #if defined (PROCESS_SUBSTITUTION) |
| 440 | unlink_fifo_list (); |
| 441 | #endif /* PROCESS_SUBSTITUTION */ |
| 442 | |
| 443 | run_unwind_protects (); |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 444 | loop_level = continuing = breaking = funcnest = 0; |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 445 | executing_list = comsub_ignore_return = return_catch_flag = 0; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 446 | |
| 447 | if (interactive && print_newline) |
| 448 | { |
| 449 | fflush (stdout); |
| 450 | fprintf (stderr, "\n"); |
| 451 | fflush (stderr); |
| 452 | } |
| 453 | |
| 454 | /* An interrupted `wait' command in a script does not exit the script. */ |
| 455 | if (interactive || (interactive_shell && !shell_initialized) || |
| 456 | (print_newline && signal_is_trapped (SIGINT))) |
| 457 | jump_to_top_level (DISCARD); |
| 458 | else |
| 459 | jump_to_top_level (EXITPROG); |
| 460 | } |
| 461 | |
| 462 | /* This is just here to isolate the longjmp calls. */ |
| 463 | void |
| 464 | jump_to_top_level (value) |
| 465 | int value; |
| 466 | { |
| 467 | longjmp (top_level, value); |
| 468 | } |
| 469 | |
| 470 | sighandler |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 471 | termsig_sighandler (sig) |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 472 | int sig; |
| 473 | { |
Chet Ramey | 89a9286 | 2011-11-21 20:49:12 -0500 | [diff] [blame] | 474 | /* If we get called twice with the same signal before handling it, |
| 475 | terminate right away. */ |
| 476 | if ( |
| 477 | #ifdef SIGHUP |
| 478 | sig != SIGHUP && |
| 479 | #endif |
| 480 | #ifdef SIGINT |
| 481 | sig != SIGINT && |
| 482 | #endif |
| 483 | #ifdef SIGDANGER |
| 484 | sig != SIGDANGER && |
| 485 | #endif |
| 486 | #ifdef SIGPIPE |
| 487 | sig != SIGPIPE && |
| 488 | #endif |
| 489 | #ifdef SIGALRM |
| 490 | sig != SIGALRM && |
| 491 | #endif |
| 492 | #ifdef SIGTERM |
| 493 | sig != SIGTERM && |
| 494 | #endif |
| 495 | #ifdef SIGXCPU |
| 496 | sig != SIGXCPU && |
| 497 | #endif |
| 498 | #ifdef SIGXFSZ |
| 499 | sig != SIGXFSZ && |
| 500 | #endif |
| 501 | #ifdef SIGVTALRM |
| 502 | sig != SIGVTALRM && |
| 503 | #endif |
| 504 | #ifdef SIGLOST |
| 505 | sig != SIGLOST && |
| 506 | #endif |
| 507 | #ifdef SIGUSR1 |
| 508 | sig != SIGUSR1 && |
| 509 | #endif |
| 510 | #ifdef SIGUSR2 |
| 511 | sig != SIGUSR2 && |
| 512 | #endif |
| 513 | sig == terminating_signal) |
| 514 | terminate_immediately = 1; |
| 515 | |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 516 | terminating_signal = sig; |
| 517 | |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 518 | /* XXX - should this also trigger when interrupt_immediately is set? */ |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 519 | if (terminate_immediately) |
| 520 | { |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 521 | #if defined (HISTORY) |
| 522 | /* XXX - will inhibit history file being written */ |
Chet Ramey | d5d0096 | 2011-11-22 20:00:47 -0500 | [diff] [blame] | 523 | # if defined (READLINE) |
| 524 | if (interactive_shell == 0 || interactive == 0 || (sig != SIGHUP && sig != SIGTERM) || no_line_editing || (RL_ISSTATE (RL_STATE_READCMD) == 0)) |
| 525 | # endif |
| 526 | history_lines_this_session = 0; |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 527 | #endif |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 528 | terminate_immediately = 0; |
| 529 | termsig_handler (sig); |
| 530 | } |
| 531 | |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 532 | #if defined (READLINE) |
| 533 | /* Set the event hook so readline will call it after the signal handlers |
| 534 | finish executing, so if this interrupted character input we can get |
Chet Ramey | bbc8b48 | 2015-01-15 10:21:08 -0500 | [diff] [blame] | 535 | quick response. If readline is active or has modified the terminal we |
| 536 | need to set this no matter what the signal is, though the check for |
| 537 | RL_STATE_TERMPREPPED is possibly redundant. */ |
| 538 | if (RL_ISSTATE (RL_STATE_SIGHANDLER) || RL_ISSTATE (RL_STATE_TERMPREPPED)) |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 539 | bashline_set_event_hook (); |
| 540 | #endif |
| 541 | |
Jari Aalto | 0628567 | 2006-10-10 14:15:34 +0000 | [diff] [blame] | 542 | SIGRETURN (0); |
| 543 | } |
| 544 | |
| 545 | void |
| 546 | termsig_handler (sig) |
| 547 | int sig; |
| 548 | { |
| 549 | static int handling_termsig = 0; |
| 550 | |
| 551 | /* Simple semaphore to keep this function from being executed multiple |
| 552 | times. Since we no longer are running as a signal handler, we don't |
| 553 | block multiple occurrences of the terminating signals while running. */ |
| 554 | if (handling_termsig) |
| 555 | return; |
| 556 | handling_termsig = 1; |
| 557 | terminating_signal = 0; /* keep macro from re-testing true. */ |
| 558 | |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 559 | /* I don't believe this condition ever tests true. */ |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 560 | if (sig == SIGINT && signal_is_trapped (SIGINT)) |
| 561 | run_interrupt_trap (); |
| 562 | |
| 563 | #if defined (HISTORY) |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 564 | /* If we don't do something like this, the history will not be saved when |
| 565 | an interactive shell is running in a terminal window that gets closed |
| 566 | with the `close' button. We can't test for RL_STATE_READCMD because |
| 567 | readline no longer handles SIGTERM synchronously. */ |
| 568 | if (interactive_shell && interactive && (sig == SIGHUP || sig == SIGTERM) && remember_on_history) |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 569 | maybe_save_shell_history (); |
| 570 | #endif /* HISTORY */ |
| 571 | |
| 572 | #if defined (JOB_CONTROL) |
Chet Ramey | 0001803 | 2011-11-21 20:51:19 -0500 | [diff] [blame] | 573 | if (sig == SIGHUP && (interactive || (subshell_environment & (SUBSHELL_COMSUB|SUBSHELL_PROCSUB)))) |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 574 | hangup_all_jobs (); |
| 575 | end_job_control (); |
| 576 | #endif /* JOB_CONTROL */ |
| 577 | |
| 578 | #if defined (PROCESS_SUBSTITUTION) |
| 579 | unlink_fifo_list (); |
| 580 | #endif /* PROCESS_SUBSTITUTION */ |
| 581 | |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 582 | /* Reset execution context */ |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 583 | loop_level = continuing = breaking = funcnest = 0; |
Jari Aalto | 3185942 | 2009-01-12 13:36:28 +0000 | [diff] [blame] | 584 | executing_list = comsub_ignore_return = return_catch_flag = 0; |
| 585 | |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 586 | run_exit_trap (); /* XXX - run exit trap possibly in signal context? */ |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 587 | set_signal_handler (sig, SIG_DFL); |
| 588 | kill (getpid (), sig); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 589 | } |
| 590 | |
| 591 | /* What we really do when SIGINT occurs. */ |
| 592 | sighandler |
| 593 | sigint_sighandler (sig) |
| 594 | int sig; |
| 595 | { |
| 596 | #if defined (MUST_REINSTALL_SIGHANDLERS) |
| 597 | signal (sig, sigint_sighandler); |
| 598 | #endif |
| 599 | |
| 600 | /* interrupt_state needs to be set for the stack of interrupts to work |
| 601 | right. Should it be set unconditionally? */ |
| 602 | if (interrupt_state == 0) |
| 603 | ADDINTERRUPT; |
| 604 | |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 605 | /* We will get here in interactive shells with job control active; allow |
| 606 | an interactive wait to be interrupted. */ |
| 607 | if (this_shell_builtin && this_shell_builtin == wait_builtin) |
| 608 | { |
| 609 | last_command_exit_value = 128 + sig; |
| 610 | wait_signal_received = sig; |
| 611 | SIGRETURN (0); |
| 612 | } |
| 613 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 614 | if (interrupt_immediately) |
| 615 | { |
| 616 | interrupt_immediately = 0; |
Chet Ramey | 0001803 | 2011-11-21 20:51:19 -0500 | [diff] [blame] | 617 | last_command_exit_value = 128 + sig; |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 618 | throw_to_top_level (); |
| 619 | } |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 620 | #if defined (READLINE) |
| 621 | /* Set the event hook so readline will call it after the signal handlers |
| 622 | finish executing, so if this interrupted character input we can get |
| 623 | quick response. */ |
| 624 | else if (RL_ISSTATE (RL_STATE_SIGHANDLER)) |
| 625 | bashline_set_event_hook (); |
| 626 | #endif |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 627 | |
| 628 | SIGRETURN (0); |
| 629 | } |
| 630 | |
Jari Aalto | 95732b4 | 2005-12-07 14:08:12 +0000 | [diff] [blame] | 631 | #if defined (SIGWINCH) |
| 632 | sighandler |
| 633 | sigwinch_sighandler (sig) |
| 634 | int sig; |
| 635 | { |
| 636 | #if defined (MUST_REINSTALL_SIGHANDLERS) |
| 637 | set_signal_handler (SIGWINCH, sigwinch_sighandler); |
| 638 | #endif /* MUST_REINSTALL_SIGHANDLERS */ |
| 639 | sigwinch_received = 1; |
| 640 | SIGRETURN (0); |
| 641 | } |
| 642 | #endif /* SIGWINCH */ |
| 643 | |
| 644 | void |
| 645 | set_sigwinch_handler () |
| 646 | { |
| 647 | #if defined (SIGWINCH) |
| 648 | old_winch = set_signal_handler (SIGWINCH, sigwinch_sighandler); |
| 649 | #endif |
| 650 | } |
| 651 | |
| 652 | void |
| 653 | unset_sigwinch_handler () |
| 654 | { |
| 655 | #if defined (SIGWINCH) |
| 656 | set_signal_handler (SIGWINCH, old_winch); |
| 657 | #endif |
| 658 | } |
| 659 | |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 660 | sighandler |
| 661 | sigterm_sighandler (sig) |
| 662 | int sig; |
| 663 | { |
| 664 | sigterm_received = 1; /* XXX - counter? */ |
| 665 | SIGRETURN (0); |
| 666 | } |
| 667 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 668 | /* Signal functions used by the rest of the code. */ |
| 669 | #if !defined (HAVE_POSIX_SIGNALS) |
| 670 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 671 | /* Perform OPERATION on NEWSET, perhaps leaving information in OLDSET. */ |
| 672 | sigprocmask (operation, newset, oldset) |
| 673 | int operation, *newset, *oldset; |
| 674 | { |
| 675 | int old, new; |
| 676 | |
| 677 | if (newset) |
| 678 | new = *newset; |
| 679 | else |
| 680 | new = 0; |
| 681 | |
| 682 | switch (operation) |
| 683 | { |
| 684 | case SIG_BLOCK: |
| 685 | old = sigblock (new); |
| 686 | break; |
| 687 | |
| 688 | case SIG_SETMASK: |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 689 | old = sigsetmask (new); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 690 | break; |
| 691 | |
| 692 | default: |
Jari Aalto | b80f644 | 2004-07-27 13:29:18 +0000 | [diff] [blame] | 693 | internal_error (_("sigprocmask: %d: invalid operation"), operation); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 694 | } |
| 695 | |
| 696 | if (oldset) |
| 697 | *oldset = old; |
| 698 | } |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 699 | |
| 700 | #else |
| 701 | |
| 702 | #if !defined (SA_INTERRUPT) |
| 703 | # define SA_INTERRUPT 0 |
| 704 | #endif |
| 705 | |
| 706 | #if !defined (SA_RESTART) |
| 707 | # define SA_RESTART 0 |
| 708 | #endif |
| 709 | |
| 710 | SigHandler * |
| 711 | set_signal_handler (sig, handler) |
| 712 | int sig; |
| 713 | SigHandler *handler; |
| 714 | { |
| 715 | struct sigaction act, oact; |
| 716 | |
| 717 | act.sa_handler = handler; |
| 718 | act.sa_flags = 0; |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 719 | |
| 720 | /* XXX - bash-4.2 */ |
| 721 | /* We don't want a child death to interrupt interruptible system calls, even |
| 722 | if we take the time to reap children */ |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 723 | #if defined (SIGCHLD) |
Chet Ramey | 30d188c | 2011-11-21 20:57:16 -0500 | [diff] [blame] | 724 | if (sig == SIGCHLD) |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 725 | act.sa_flags |= SA_RESTART; /* XXX */ |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 726 | #endif |
| 727 | /* If we're installing a SIGTERM handler for interactive shells, we want |
| 728 | it to be as close to SIG_IGN as possible. */ |
| 729 | if (sig == SIGTERM && handler == sigterm_sighandler) |
| 730 | act.sa_flags |= SA_RESTART; /* XXX */ |
Chet Ramey | 495aee4 | 2011-11-22 19:11:26 -0500 | [diff] [blame] | 731 | |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 732 | sigemptyset (&act.sa_mask); |
| 733 | sigemptyset (&oact.sa_mask); |
Chet Ramey | ac50fba | 2014-02-26 09:36:43 -0500 | [diff] [blame] | 734 | if (sigaction (sig, &act, &oact) == 0) |
| 735 | return (oact.sa_handler); |
| 736 | else |
| 737 | return (SIG_DFL); |
Jari Aalto | ccc6cda | 1996-12-23 17:02:34 +0000 | [diff] [blame] | 738 | } |
| 739 | #endif /* HAVE_POSIX_SIGNALS */ |