annotate lib/sigaction.c @ 10229:29502a2dd08a

New module sigaction, for mingw. * modules/sigaction: New module... * modules/sigaction-tests: ...and its test. * m4/sigaction.m4: New file. * lib/sigaction.c: Likewise. * tests/test-sigaction.c: Likewise. * m4/signal_h.m4 (gl_SIGNAL_H_DEFAULTS): Add sigaction variables. * modules/signal (Makefile.am): Likewise. * lib/signal.in.h (!@HAVE_SIGACTION@): Define replacements when needed. * doc/posix-headers/signal.texi (signal.h): Mention provided types. * doc/posix-functions/siginterrupt.texi (siginterrupt): Mention that sigaction is preferable. * doc/posix-functions/sigaction.texi (sigaction): Mention new module. * MODULES.html.sh (Support for systems lacking POSIX:2001): Add sigaction. Signed-off-by: Eric Blake <ebb9@byu.net>
author Eric Blake <ebb9@byu.net>
date Sat, 21 Jun 2008 14:32:55 -0600
parents
children d7f97416cbe1
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.
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
2 Copyright (C) 2008 Free Software Foundation, Inc.
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
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
27 /* This implementation of sigaction is tailored to Woe32 behavior:
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
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
38 Additionally, SIGCHLD is not defined, so we don't implement
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
39 SA_NOCLDSTOP or SA_NOCLDWAIT; sigaltstack() is not present, so we
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
40 don't implement SA_ONSTACK; and siginterrupt() is not present, so
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
41 we don't implement SA_RESTART. Supporting SA_SIGINFO is impossible
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
42 to do portably.
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
43
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
44 POSIX states that an application should not mix signal() and
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
45 sigaction(). We support the use of signal() within the gnulib
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
46 sigprocmask() substitute, but all other application code linked
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
47 with this module should stick with only sigaction(). */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
48
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
49 /* Check some of our assumptions. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
50 #if defined SIGCHLD || defined HAVE_SIGALTSTACK || defined HAVE_SIGINTERRUPT
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
51 # error "Revisit the assumptions made in the sigaction module"
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
52 #endif
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
53
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
54 /* Out-of-range substitutes make a good fallback for uncatchable
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
55 signals. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
56 #ifndef SIGKILL
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
57 # define SIGKILL (-1)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
58 #endif
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
59 #ifndef SIGSTOP
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
60 # define SIGSTOP (-1)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
61 #endif
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
62
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
63 /* A signal handler. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
64 typedef void (*handler_t) (int signal);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
65
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
66 /* 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
67 that signal is not currently handled by the sigaction handler. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
68 static struct sigaction volatile action_array[NSIG] /* = 0 */;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
69
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
70 /* Signal handler that is installed for signals. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
71 static void
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
72 sigaction_handler (int sig)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
73 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
74 handler_t handler;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
75 sigset_t mask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
76 sigset_t oldmask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
77 int saved_errno = errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
78 if (sig < 0 || NSIG <= sig || !action_array[sig].sa_handler)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
79 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
80 /* Unexpected situation; be careful to avoid recursive abort. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
81 if (sig == SIGABRT)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
82 signal (SIGABRT, SIG_DFL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
83 abort ();
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
84 }
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 /* Reinstall the signal handler when required; otherwise update the
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
87 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
88 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
89 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
90 interrupting a sigaction() call. There is a race where any
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
91 asynchronous instance of the same signal occurring before we
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
92 reinstall the handler will trigger the default handler; oh
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
93 well. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
94 handler = action_array[sig].sa_handler;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
95 if ((action_array[sig].sa_flags & SA_RESETHAND) == 0)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
96 signal (sig, sigaction_handler);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
97 else
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
98 action_array[sig].sa_handler = NULL;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
99
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
100 /* Block appropriate signals. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
101 mask = action_array[sig].sa_mask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
102 if ((action_array[sig].sa_flags & SA_NODEFER) == 0)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
103 sigaddset (&mask, sig);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
104 sigprocmask (SIG_BLOCK, &mask, &oldmask);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
105
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
106 /* Invoke the user's handler, then restore prior mask. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
107 errno = saved_errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
108 handler (sig);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
109 saved_errno = errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
110 sigprocmask (SIG_SETMASK, &oldmask, NULL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
111 errno = saved_errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
112 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
113
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
114 /* 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
115 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
116 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
117 set errno and return -1 on failure. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
118 int
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
119 sigaction (int sig, const struct sigaction *restrict act,
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
120 struct sigaction *restrict oact)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
121 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
122 sigset_t mask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
123 sigset_t oldmask;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
124 int saved_errno;
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 if (sig < 0 || NSIG <= sig || sig == SIGKILL || sig == SIGSTOP
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
127 || (act && act->sa_handler == SIG_ERR))
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
128 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
129 errno = EINVAL;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
130 return -1;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
131 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
132
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
133 /* POSIX requires sigaction() to be async-signal-safe. In other
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
134 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
135 inside this function, the user's handler could then call
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
136 sigaction() recursively and expect consistent results. We meet
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
137 this rule by using sigprocmask to block all signals before
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
138 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
139 handler; this works since we know that the gnulib sigprocmask
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
140 replacement does not try to use sigaction() from its handler. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
141 if (!act && !oact)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
142 return 0;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
143 sigfillset (&mask);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
144 sigprocmask (SIG_BLOCK, &mask, &oldmask);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
145 if (oact)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
146 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
147 if (action_array[sig].sa_handler)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
148 *oact = action_array[sig];
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
149 else
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
150 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
151 /* Safe to change the handler at will here, since all
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
152 signals are currently blocked. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
153 oact->sa_handler = signal (sig, SIG_DFL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
154 if (oact->sa_handler == SIG_ERR)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
155 goto failure;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
156 signal (sig, oact->sa_handler);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
157 oact->sa_flags = SA_RESETHAND | SA_NODEFER;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
158 sigemptyset (&oact->sa_mask);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
159 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
160 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
161
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
162 if (act)
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 /* Safe to install the handler before updating action_array,
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
165 since all signals are currently blocked. */
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
166 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
167 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
168 if (signal (sig, act->sa_handler) == SIG_ERR)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
169 goto failure;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
170 action_array[sig].sa_handler = NULL;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
171 }
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
172 else
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
173 {
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
174 if (signal (sig, sigaction_handler) == SIG_ERR)
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
175 goto failure;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
176 action_array[sig] = *act;
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 sigprocmask (SIG_SETMASK, &oldmask, NULL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
180 return 0;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
181
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
182 failure:
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
183 saved_errno = errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
184 sigprocmask (SIG_SETMASK, &oldmask, NULL);
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
185 errno = saved_errno;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
186 return -1;
29502a2dd08a New module sigaction, for mingw.
Eric Blake <ebb9@byu.net>
parents:
diff changeset
187 }