OSDN Git Service

libpthread/nptl: core of the "Native Posix Threading Library" for uClibc
[uclinux-h8/uClibc.git] / libpthread / nptl / sysdeps / unix / sysv / linux / sigwaitinfo.c
1 /* Copyright (C) 1997,1998,2000,2002,2003,2004 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3
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.
8
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.
13
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.  */
18
19 #include <pthreadP.h>
20 #include <errno.h>
21 #include <signal.h>
22 #define __need_NULL
23 #include <stddef.h>
24 #include <string.h>
25
26 #include <sysdep-cancel.h>
27 #include <sys/syscall.h>
28
29 #ifdef __NR_rt_sigtimedwait
30
31 static int
32 do_sigwaitinfo (const sigset_t *set, siginfo_t *info)
33 {
34 #ifdef SIGCANCEL
35   sigset_t tmpset;
36   if (set != NULL
37       && (__builtin_expect (__sigismember (set, SIGCANCEL), 0)
38 # ifdef SIGSETXID
39           || __builtin_expect (__sigismember (set, SIGSETXID), 0)
40 # endif
41           ))
42     {
43       /* Create a temporary mask without the bit for SIGCANCEL set.  */
44       // We are not copying more than we have to.
45       memcpy (&tmpset, set, _NSIG / 8);
46       __sigdelset (&tmpset, SIGCANCEL);
47 # ifdef SIGSETXID
48       __sigdelset (&tmpset, SIGSETXID);
49 # endif
50       set = &tmpset;
51     }
52 #endif
53
54   /* XXX The size argument hopefully will have to be changed to the
55      real size of the user-level sigset_t.  */
56   int result = INLINE_SYSCALL (rt_sigtimedwait, 4, set,
57                                info, NULL, _NSIG / 8);
58
59   /* The kernel generates a SI_TKILL code in si_code in case tkill is
60      used.  tkill is transparently used in raise().  Since having
61      SI_TKILL as a code is useful in general we fold the results
62      here.  */
63   if (result != -1 && info != NULL && info->si_code == SI_TKILL)
64     info->si_code = SI_USER;
65
66   return result;
67 }
68
69
70 /* Return any pending signal or wait for one for the given time.  */
71 int
72 __sigwaitinfo (const sigset_t *set, siginfo_t *info)
73 {
74   if (SINGLE_THREAD_P)
75     return do_sigwaitinfo (set, info);
76
77   int oldtype = LIBC_CANCEL_ASYNC ();
78
79   /* XXX The size argument hopefully will have to be changed to the
80      real size of the user-level sigset_t.  */
81   int result = do_sigwaitinfo (set, info);
82
83   LIBC_CANCEL_RESET (oldtype);
84
85   return result;
86 }
87 weak_alias (__sigwaitinfo, sigwaitinfo)
88 #endif