2 * Copyright (c) 1998 Daniel M. Eischen <eischen@vigrid.com>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. All advertising materials mentioning features or use of this software
14 * must display the following acknowledgement:
15 * This product includes software developed by Daniel M. Eischen.
16 * 4. Neither the name of the author nor the names of any co-contributors
17 * may be used to endorse or promote products derived from this software
18 * without specific prior written permission.
20 * THIS SOFTWARE IS PROVIDED BY DANIEL M. EISCHEN AND CONTRIBUTORS ``AS IS''
21 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
24 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32 * $FreeBSD: src/lib/libc_r/test/sigwait_d.c,v 1.1.2.2 2001/06/22 21:44:27 jasone Exp $
33 * $DragonFly: src/lib/libc_r/test/sigwait_d.c,v 1.2 2003/06/17 04:26:48 dillon Exp $
45 #include <pthread_np.h>
48 static int sigcounts
[NSIG
+ 1];
49 static sigset_t wait_mask
;
50 static pthread_mutex_t waiter_mutex
;
61 sigaddset (&mask
, SIGHUP
);
62 sigprocmask (SIG_BLOCK
, &mask
, NULL
);
64 while (sigcounts
[SIGINT
] == 0) {
65 if (sigwait (&wait_mask
, &signo
) != 0) {
67 "Unable to wait for signal, errno %d\n",
72 fprintf (stderr
, "Sigwait caught signal %d\n", signo
);
74 /* Allow the main thread to prevent the sigwait. */
75 pthread_mutex_lock (&waiter_mutex
);
76 pthread_mutex_unlock (&waiter_mutex
);
85 sighandler (int signo
)
87 fprintf (stderr
, " -> Signal handler caught signal %d\n", signo
);
89 if ((signo
>= 0) && (signo
<= NSIG
))
94 send_thread_signal (pthread_t tid
, int signo
)
96 if (pthread_kill (tid
, signo
) != 0) {
97 fprintf (stderr
, "Unable to send thread signal, errno %d.\n",
104 send_process_signal (int signo
)
106 if (kill (getpid (), signo
) != 0) {
107 fprintf (stderr
, "Unable to send process signal, errno %d.\n",
114 int main (int argc
, char *argv
[])
116 pthread_mutexattr_t mattr
;
117 pthread_attr_t pattr
;
120 struct sigaction act
;
122 /* Initialize our signal counts. */
123 memset ((void *) sigcounts
, 0, NSIG
* sizeof (int));
125 /* Setup our wait mask. */
126 sigemptyset (&wait_mask
); /* Default action */
127 sigaddset (&wait_mask
, SIGHUP
); /* terminate */
128 sigaddset (&wait_mask
, SIGINT
); /* terminate */
129 sigaddset (&wait_mask
, SIGQUIT
); /* create core image */
130 sigaddset (&wait_mask
, SIGURG
); /* ignore */
131 sigaddset (&wait_mask
, SIGIO
); /* ignore */
132 sigaddset (&wait_mask
, SIGUSR1
); /* terminate */
134 /* Ignore signals SIGHUP and SIGIO. */
135 sigemptyset (&act
.sa_mask
);
136 sigaddset (&act
.sa_mask
, SIGHUP
);
137 sigaddset (&act
.sa_mask
, SIGIO
);
138 act
.sa_handler
= SIG_IGN
;
140 sigaction (SIGHUP
, &act
, NULL
);
141 sigaction (SIGIO
, &act
, NULL
);
143 /* Install a signal handler for SIGURG */
144 sigemptyset (&act
.sa_mask
);
145 sigaddset (&act
.sa_mask
, SIGURG
);
146 act
.sa_handler
= sighandler
;
147 act
.sa_flags
= SA_RESTART
;
148 sigaction (SIGURG
, &act
, NULL
);
150 /* Install a signal handler for SIGXCPU */
151 sigemptyset (&act
.sa_mask
);
152 sigaddset (&act
.sa_mask
, SIGXCPU
);
153 sigaction (SIGXCPU
, &act
, NULL
);
156 * Initialize the thread attribute.
158 if ((pthread_attr_init (&pattr
) != 0) ||
159 (pthread_attr_setdetachstate (&pattr
,
160 PTHREAD_CREATE_JOINABLE
) != 0)) {
161 fprintf (stderr
, "Unable to initialize thread attributes.\n");
166 * Initialize and create a mutex.
168 if ((pthread_mutexattr_init (&mattr
) != 0) ||
169 (pthread_mutex_init (&waiter_mutex
, &mattr
) != 0)) {
170 fprintf (stderr
, "Unable to create waiter mutex.\n");
175 * Create the sigwaiter thread.
177 if (pthread_create (&tid
, &pattr
, sigwaiter
, NULL
) != 0) {
178 fprintf (stderr
, "Unable to create thread.\n");
181 #if defined(_LIBC_R_)
182 pthread_set_name_np (tid
, "sigwaiter");
186 * Verify that an ignored signal doesn't cause a wakeup.
187 * We don't have a handler installed for SIGIO.
189 send_thread_signal (tid
, SIGIO
);
191 send_process_signal (SIGIO
);
193 if (sigcounts
[SIGIO
] != 0)
195 "FAIL: sigwait wakes up for ignored signal SIGIO.\n");
198 * Verify that a signal with a default action of ignore, for
199 * which we have a signal handler installed, will release a sigwait.
201 send_thread_signal (tid
, SIGURG
);
203 send_process_signal (SIGURG
);
205 if (sigcounts
[SIGURG
] != 2)
206 fprintf (stderr
, "FAIL: sigwait doesn't wake up for SIGURG.\n");
209 * Verify that a signal with a default action that terminates
210 * the process will release a sigwait.
212 send_thread_signal (tid
, SIGUSR1
);
214 send_process_signal (SIGUSR1
);
216 if (sigcounts
[SIGUSR1
] != 2)
218 "FAIL: sigwait doesn't wake up for SIGUSR1.\n");
221 * Verify that if we install a signal handler for a previously
222 * ignored signal, an occurrence of this signal will release
223 * the (already waiting) sigwait.
226 /* Install a signal handler for SIGHUP. */
227 sigemptyset (&act
.sa_mask
);
228 sigaddset (&act
.sa_mask
, SIGHUP
);
229 act
.sa_handler
= sighandler
;
230 act
.sa_flags
= SA_RESTART
;
231 sigaction (SIGHUP
, &act
, NULL
);
233 /* Sending SIGHUP should release the sigwait. */
234 send_process_signal (SIGHUP
);
236 send_thread_signal (tid
, SIGHUP
);
238 if (sigcounts
[SIGHUP
] != 2)
239 fprintf (stderr
, "FAIL: sigwait doesn't wake up for SIGHUP.\n");
242 * Verify that a pending signal in the waiters mask will
243 * cause sigwait to return the pending signal. We do this
244 * by taking the waiters mutex and signaling the waiter to
245 * release him from the sigwait. The waiter will block
246 * on taking the mutex, and we can then send the waiter a
247 * signal which should be added to his pending signals.
248 * The next time the waiter does a sigwait, he should
249 * return with the pending signal.
251 sigcounts
[SIGHUP
] = 0;
252 pthread_mutex_lock (&waiter_mutex
);
253 /* Release the waiter from sigwait. */
254 send_process_signal (SIGHUP
);
256 if (sigcounts
[SIGHUP
] != 1)
257 fprintf (stderr
, "FAIL: sigwait doesn't wake up for SIGHUP.\n");
259 * Add SIGHUP to the process pending signals. Since there is
260 * a signal handler installed for SIGHUP and this signal is
261 * blocked from the waiter thread and unblocked in the main
262 * thread, the signal handler should be called once for SIGHUP.
264 send_process_signal (SIGHUP
);
265 /* Release the waiter thread and allow him to run. */
266 pthread_mutex_unlock (&waiter_mutex
);
268 if (sigcounts
[SIGHUP
] != 2)
270 "FAIL: sigwait doesn't return for pending SIGHUP.\n");
273 * Repeat the above test using pthread_kill and SIGUSR1.
275 sigcounts
[SIGUSR1
] = 0;
276 pthread_mutex_lock (&waiter_mutex
);
277 /* Release the waiter from sigwait. */
278 send_thread_signal (tid
, SIGUSR1
);
280 if (sigcounts
[SIGUSR1
] != 1)
282 "FAIL: sigwait doesn't wake up for SIGUSR1.\n");
283 /* Add SIGUSR1 to the waiters pending signals. */
284 send_thread_signal (tid
, SIGUSR1
);
285 /* Release the waiter thread and allow him to run. */
286 pthread_mutex_unlock (&waiter_mutex
);
288 if (sigcounts
[SIGUSR1
] != 2)
290 "FAIL: sigwait doesn't return for pending SIGUSR1.\n");
293 * Verify that we can still kill the process for a signal
294 * not being waited on by sigwait.
296 send_process_signal (SIGPIPE
);
297 fprintf (stderr
, "FAIL: SIGPIPE did not terminate process.\n");
300 * Wait for the thread to finish.
302 pthread_join (tid
, &exit_status
);