Update.
[glibc.git] / sysdeps / unix / sysv / linux / sigtimedwait.c
blobadeadc1587cbdd1eb5cb88166208e18d4490e638
1 /* Copyright (C) 1997,1998,2000,2002,2003,2004 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, write to the Free
16 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
17 02111-1307 USA. */
19 #include <errno.h>
20 #include <signal.h>
21 #include <string.h>
23 #include <sysdep-cancel.h>
24 #include <sys/syscall.h>
25 #include <bp-checks.h>
27 #ifdef __NR_rt_sigtimedwait
29 static int
30 do_sigtimedwait (const sigset_t *set, siginfo_t *info,
31 const struct timespec *timeout)
33 #ifdef SIGCANCEL
34 sigset_t tmpset;
35 if (set != NULL
36 && (__builtin_expect (__sigismember (set, SIGCANCEL), 0)
37 # ifdef SIGSETXID
38 || __builtin_expect (__sigismember (set, SIGSETXID), 0)
39 # endif
42 /* Create a temporary mask without the bit for SIGCANCEL set. */
43 // We are not copying more than we have to.
44 memcpy (&tmpset, set, _NSIG / 8);
45 __sigdelset (&tmpset, SIGCANCEL);
46 # ifdef SIGSETXID
47 __sigdelset (&tmpset, SIGSETXID);
48 # endif
49 set = &tmpset;
51 #endif
53 /* XXX The size argument hopefully will have to be changed to the
54 real size of the user-level sigset_t. */
55 int result = INLINE_SYSCALL (rt_sigtimedwait, 4, CHECK_SIGSET (set),
56 CHECK_1 (info), timeout, _NSIG / 8);
58 /* The kernel generates a SI_TKILL code in si_code in case tkill is
59 used. tkill is transparently used in raise(). Since having
60 SI_TKILL as a code is useful in general we fold the results
61 here. */
62 if (result != -1 && info != NULL && info->si_code == SI_TKILL)
63 info->si_code = SI_USER;
65 return result;
69 /* Return any pending signal or wait for one for the given time. */
70 int
71 __sigtimedwait (set, info, timeout)
72 const sigset_t *set;
73 siginfo_t *info;
74 const struct timespec *timeout;
76 if (SINGLE_THREAD_P)
77 return do_sigtimedwait (set, info, timeout);
79 int oldtype = LIBC_CANCEL_ASYNC ();
81 /* XXX The size argument hopefully will have to be changed to the
82 real size of the user-level sigset_t. */
83 int result = do_sigtimedwait (set, info, timeout);
85 LIBC_CANCEL_RESET (oldtype);
87 return result;
89 libc_hidden_def (__sigtimedwait)
90 weak_alias (__sigtimedwait, sigtimedwait)
91 #else
92 # include <sysdeps/generic/sigtimedwait.c>
93 #endif