| /* Implementation of sigwait function from POSIX.1c. | 
 |    Copyright (C) 1996-2016 Free Software Foundation, Inc. | 
 |    This file is part of the GNU C Library. | 
 |    Contributed by Ulrich Drepper <drepper@cygnus.com>, 1996. | 
 |  | 
 |    The GNU C Library is free software; you can redistribute it and/or | 
 |    modify it under the terms of the GNU Lesser General Public | 
 |    License as published by the Free Software Foundation; either | 
 |    version 2.1 of the License, or (at your option) any later version. | 
 |  | 
 |    The GNU C Library is distributed in the hope that it will be useful, | 
 |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
 |    Lesser General Public License for more details. | 
 |  | 
 |    You should have received a copy of the GNU Lesser General Public | 
 |    License along with the GNU C Library; if not, see | 
 |    <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include <errno.h> | 
 | #include <signal.h> | 
 | #include <stddef.h>		/* For NULL.  */ | 
 | #include <sysdep-cancel.h> | 
 |  | 
 | /* This is our dummy signal handler we use here.  */ | 
 | static void ignore_signal (int sig); | 
 |  | 
 | /* Place where to remember which signal we got.  Please note that this | 
 |    implementation cannot be used for the threaded libc.  The | 
 |    libpthread must provide an own version.  */ | 
 | static int was_sig; | 
 |  | 
 |  | 
 | static int | 
 | do_sigwait (const sigset_t *set, int *sig) | 
 | { | 
 |   sigset_t tmp_mask; | 
 |   struct sigaction saved[NSIG]; | 
 |   struct sigaction action; | 
 |   int save_errno; | 
 |   int this; | 
 |  | 
 |   /* Prepare set.  */ | 
 |   __sigfillset (&tmp_mask); | 
 |  | 
 |   /* Unblock all signals in the SET and register our nice handler.  */ | 
 |   action.sa_handler = ignore_signal; | 
 |   action.sa_flags = 0; | 
 |   __sigfillset (&action.sa_mask);	/* Block all signals for handler.  */ | 
 |  | 
 |   /* Make sure we recognize error conditions by setting WAS_SIG to a | 
 |      value which does not describe a legal signal number.  */ | 
 |   was_sig = -1; | 
 |  | 
 |   for (this = 1; this < NSIG; ++this) | 
 |     if (__sigismember (set, this)) | 
 |       { | 
 | 	/* Unblock this signal.  */ | 
 | 	__sigdelset (&tmp_mask, this); | 
 |  | 
 | 	/* Register temporary action handler.  */ | 
 | 	if (__sigaction (this, &action, &saved[this]) != 0) | 
 | 	  goto restore_handler; | 
 |       } | 
 |  | 
 |   /* Now we can wait for signals.  */ | 
 |   __sigsuspend (&tmp_mask); | 
 |  | 
 |  restore_handler: | 
 |   save_errno = errno; | 
 |  | 
 |   while (--this >= 1) | 
 |     if (__sigismember (set, this)) | 
 |       /* We ignore errors here since we must restore all handlers.  */ | 
 |       __sigaction (this, &saved[this], NULL); | 
 |  | 
 |   __set_errno (save_errno); | 
 |  | 
 |   /* Store the result and return.  */ | 
 |   *sig = was_sig; | 
 |   return was_sig == -1 ? -1 : 0; | 
 | } | 
 |  | 
 |  | 
 | int | 
 | __sigwait (const sigset_t *set, int *sig) | 
 | { | 
 |   if (SINGLE_THREAD_P) | 
 |     return do_sigwait (set, sig); | 
 |  | 
 |   int oldtype = LIBC_CANCEL_ASYNC (); | 
 |  | 
 |   int result = do_sigwait (set, sig); | 
 |  | 
 |   LIBC_CANCEL_RESET (oldtype); | 
 |  | 
 |   return result; | 
 | } | 
 | libc_hidden_def (__sigwait) | 
 | weak_alias (__sigwait, sigwait) | 
 |  | 
 |  | 
 | static void | 
 | ignore_signal (int sig) | 
 | { | 
 |   /* Remember the signal.  */ | 
 |   was_sig = sig; | 
 | } |