annotate lib/sigaction.c @ 17363:5a51fb7777a9

sys_select, sys_time: port 2013-01-30 Solaris 2.6 fix to Cygwin Problem reported by Marco Atzeri in <http://lists.gnu.org/archive/html/bug-gnulib/2013-03/msg00000.html>. * lib/sys_select.in.h [HAVE_SYS_SELECT_H && _CYGWIN_SYS_TIME_H]: Simply delegate to the system <sys/select.h> in this case too. Also, pay attention to _GL_SYS_SELECT_H_REDIRECT_FROM_SYS_TIME_H only if OSF/1, since otherwise Cygwin breaks, and it doesn't seem to be needed on Solaris either. * lib/sys_time.in.h [_CYGWIN_SYS_TIME_H]: Simply delgate to the system <sys/time.h> in this case.
author Paul Eggert <eggert@cs.ucla.edu>
date Tue, 19 Mar 2013 09:08:47 -0700
parents e542fd46ad6f
children 344018b6e5d7
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
10229
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
1 /* POSIX compatible signal blocking.
17249
e542fd46ad6f maint: update all copyright year number ranges
Eric Blake <eblake@redhat.com>
parents: 16242
diff changeset
2 Copyright (C) 2008-2013 Free Software Foundation, Inc.
10229
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
3 Written by Eric Blake <ebb9@byu.net>, 2008.
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
4
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
5 This program is free software: you can redistribute it and/or modify
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
6 it under the terms of the GNU General Public License as published by
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
7 the Free Software Foundation; either version 3 of the License, or
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
8 (at your option) any later version.
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
9
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
10 This program is distributed in the hope that it will be useful,
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
13 GNU General Public License for more details.
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
14
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
15 You should have received a copy of the GNU General Public License
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
16 along with this program. If not, see <http://www.gnu.org/licenses/>. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
17
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
18 #include <config.h>
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
19
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
20 /* Specification. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
21 #include <signal.h>
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
22
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
23 #include <errno.h>
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
24 #include <stdint.h>
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
25 #include <stdlib.h>
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
26
16242
59c686e5b2df Talk about "native Windows API", not "Woe32".
Bruno Haible <bruno@clisp.org>
parents: 16214
diff changeset
27 /* This implementation of sigaction is tailored to native Windows behavior:
10229
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
28 signal() has SysV semantics (ie. the handler is uninstalled before
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
29 it is invoked). This is an inherent data race if an asynchronous
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
30 signal is sent twice in a row before we can reinstall our handler,
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
31 but there's nothing we can do about it. Meanwhile, sigprocmask()
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
32 is not present, and while we can use the gnulib replacement to
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
33 provide critical sections, it too suffers from potential data races
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
34 in the face of an ill-timed asynchronous signal. And we compound
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
35 the situation by reading static storage in a signal handler, which
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
36 POSIX warns is not generically async-signal-safe. Oh well.
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
37
10236
d7f97416cbe1 Define SA_RESTART also for mingw.
Bruno Haible <bruno@clisp.org>
parents: 10229
diff changeset
38 Additionally:
d7f97416cbe1 Define SA_RESTART also for mingw.
Bruno Haible <bruno@clisp.org>
parents: 10229
diff changeset
39 - We don't implement SA_NOCLDSTOP or SA_NOCLDWAIT, because SIGCHLD
d7f97416cbe1 Define SA_RESTART also for mingw.
Bruno Haible <bruno@clisp.org>
parents: 10229
diff changeset
40 is not defined.
d7f97416cbe1 Define SA_RESTART also for mingw.
Bruno Haible <bruno@clisp.org>
parents: 10229
diff changeset
41 - We don't implement SA_ONSTACK, because sigaltstack() is not present.
16214
ec738d6aeef5 Talk about "native Windows API", not "Win32".
Bruno Haible <bruno@clisp.org>
parents: 16201
diff changeset
42 - We ignore SA_RESTART, because blocking native Windows API calls are
ec738d6aeef5 Talk about "native Windows API", not "Win32".
Bruno Haible <bruno@clisp.org>
parents: 16201
diff changeset
43 not interrupted anyway when an asynchronous signal occurs, and the
ec738d6aeef5 Talk about "native Windows API", not "Win32".
Bruno Haible <bruno@clisp.org>
parents: 16201
diff changeset
44 MSVCRT runtime never sets errno to EINTR.
10236
d7f97416cbe1 Define SA_RESTART also for mingw.
Bruno Haible <bruno@clisp.org>
parents: 10229
diff changeset
45 - We don't implement SA_SIGINFO because it is impossible to do so
d7f97416cbe1 Define SA_RESTART also for mingw.
Bruno Haible <bruno@clisp.org>
parents: 10229
diff changeset
46 portably.
10229
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
47
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
48 POSIX states that an application should not mix signal() and
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
49 sigaction(). We support the use of signal() within the gnulib
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
50 sigprocmask() substitute, but all other application code linked
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
51 with this module should stick with only sigaction(). */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
52
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
53 /* Check some of our assumptions. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
54 #if defined SIGCHLD || defined HAVE_SIGALTSTACK || defined HAVE_SIGINTERRUPT
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
55 # error "Revisit the assumptions made in the sigaction module"
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
56 #endif
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
57
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
58 /* Out-of-range substitutes make a good fallback for uncatchable
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
59 signals. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
60 #ifndef SIGKILL
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
61 # define SIGKILL (-1)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
62 #endif
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
63 #ifndef SIGSTOP
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
64 # define SIGSTOP (-1)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
65 #endif
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
66
10689
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
67 /* On native Windows, as of 2008, the signal SIGABRT_COMPAT is an alias
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
68 for the signal SIGABRT. Only one signal handler is stored for both
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
69 SIGABRT and SIGABRT_COMPAT. SIGABRT_COMPAT is not a signal of its own. */
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
70 #if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
71 # undef SIGABRT_COMPAT
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
72 # define SIGABRT_COMPAT 6
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
73 #endif
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
74
10229
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
75 /* A signal handler. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
76 typedef void (*handler_t) (int signal);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
77
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
78 /* Set of current actions. If sa_handler for an entry is NULL, then
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
79 that signal is not currently handled by the sigaction handler. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
80 static struct sigaction volatile action_array[NSIG] /* = 0 */;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
81
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
82 /* Signal handler that is installed for signals. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
83 static void
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
84 sigaction_handler (int sig)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
85 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
86 handler_t handler;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
87 sigset_t mask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
88 sigset_t oldmask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
89 int saved_errno = errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
90 if (sig < 0 || NSIG <= sig || !action_array[sig].sa_handler)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
91 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
92 /* Unexpected situation; be careful to avoid recursive abort. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
93 if (sig == SIGABRT)
12421
e8d2c6fc33ad Use spaces for indentation, not tabs.
Bruno Haible <bruno@clisp.org>
parents: 10689
diff changeset
94 signal (SIGABRT, SIG_DFL);
10229
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
95 abort ();
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
96 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
97
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
98 /* Reinstall the signal handler when required; otherwise update the
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
99 bookkeeping so that the user's handler may call sigaction and get
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
100 accurate results. We know the signal isn't currently blocked, or
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
101 we wouldn't be in its handler, therefore we know that we are not
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
102 interrupting a sigaction() call. There is a race where any
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
103 asynchronous instance of the same signal occurring before we
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
104 reinstall the handler will trigger the default handler; oh
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
105 well. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
106 handler = action_array[sig].sa_handler;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
107 if ((action_array[sig].sa_flags & SA_RESETHAND) == 0)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
108 signal (sig, sigaction_handler);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
109 else
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
110 action_array[sig].sa_handler = NULL;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
111
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
112 /* Block appropriate signals. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
113 mask = action_array[sig].sa_mask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
114 if ((action_array[sig].sa_flags & SA_NODEFER) == 0)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
115 sigaddset (&mask, sig);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
116 sigprocmask (SIG_BLOCK, &mask, &oldmask);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
117
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
118 /* Invoke the user's handler, then restore prior mask. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
119 errno = saved_errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
120 handler (sig);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
121 saved_errno = errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
122 sigprocmask (SIG_SETMASK, &oldmask, NULL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
123 errno = saved_errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
124 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
125
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
126 /* Change and/or query the action that will be taken on delivery of
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
127 signal SIG. If not NULL, ACT describes the new behavior. If not
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
128 NULL, OACT is set to the prior behavior. Return 0 on success, or
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
129 set errno and return -1 on failure. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
130 int
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
131 sigaction (int sig, const struct sigaction *restrict act,
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
132 struct sigaction *restrict oact)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
133 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
134 sigset_t mask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
135 sigset_t oldmask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
136 int saved_errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
137
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
138 if (sig < 0 || NSIG <= sig || sig == SIGKILL || sig == SIGSTOP
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
139 || (act && act->sa_handler == SIG_ERR))
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
140 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
141 errno = EINVAL;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
142 return -1;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
143 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
144
14380
5cee7a3cd6ca maint: adjust cpp indentation for my modules, as well
Eric Blake <eblake@redhat.com>
parents: 14079
diff changeset
145 #ifdef SIGABRT_COMPAT
10689
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
146 if (sig == SIGABRT_COMPAT)
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
147 sig = SIGABRT;
14380
5cee7a3cd6ca maint: adjust cpp indentation for my modules, as well
Eric Blake <eblake@redhat.com>
parents: 14079
diff changeset
148 #endif
10689
5fbf24129e48 Take into account the role of SIGABRT_COMPAT on Windows 2008.
Bruno Haible <bruno@clisp.org>
parents: 10236
diff changeset
149
10229
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
150 /* POSIX requires sigaction() to be async-signal-safe. In other
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
151 words, if an asynchronous signal can occur while we are anywhere
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
152 inside this function, the user's handler could then call
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
153 sigaction() recursively and expect consistent results. We meet
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
154 this rule by using sigprocmask to block all signals before
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
155 modifying any data structure that could be read from a signal
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
156 handler; this works since we know that the gnulib sigprocmask
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
157 replacement does not try to use sigaction() from its handler. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
158 if (!act && !oact)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
159 return 0;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
160 sigfillset (&mask);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
161 sigprocmask (SIG_BLOCK, &mask, &oldmask);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
162 if (oact)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
163 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
164 if (action_array[sig].sa_handler)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
165 *oact = action_array[sig];
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
166 else
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
167 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
168 /* Safe to change the handler at will here, since all
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
169 signals are currently blocked. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
170 oact->sa_handler = signal (sig, SIG_DFL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
171 if (oact->sa_handler == SIG_ERR)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
172 goto failure;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
173 signal (sig, oact->sa_handler);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
174 oact->sa_flags = SA_RESETHAND | SA_NODEFER;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
175 sigemptyset (&oact->sa_mask);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
176 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
177 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
178
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
179 if (act)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
180 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
181 /* Safe to install the handler before updating action_array,
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
182 since all signals are currently blocked. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
183 if (act->sa_handler == SIG_DFL || act->sa_handler == SIG_IGN)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
184 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
185 if (signal (sig, act->sa_handler) == SIG_ERR)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
186 goto failure;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
187 action_array[sig].sa_handler = NULL;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
188 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
189 else
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
190 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
191 if (signal (sig, sigaction_handler) == SIG_ERR)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
192 goto failure;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
193 action_array[sig] = *act;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
194 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
195 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
196 sigprocmask (SIG_SETMASK, &oldmask, NULL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
197 return 0;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
198
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
199 failure:
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
200 saved_errno = errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
201 sigprocmask (SIG_SETMASK, &oldmask, NULL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
202 errno = saved_errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
203 return -1;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
204 }