1 /* POSIX compatible signal blocking for threads.
2 Copyright (C) 2011-2012 Free Software Foundation, Inc.
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 3 of the License, or
7 (at your option) any later version.
9 This program 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
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <http://www.gnu.org/licenses/>. */
25 #if PTHREAD_SIGMASK_INEFFECTIVE
29 #if PTHREAD_SIGMASK_UNBLOCK_BUG
34 pthread_sigmask (int how
, const sigset_t
*new_mask
, sigset_t
*old_mask
)
35 #undef pthread_sigmask
37 #if HAVE_PTHREAD_SIGMASK
40 # if PTHREAD_SIGMASK_INEFFECTIVE
41 sigset_t omask
, omask_copy
;
42 sigset_t
*old_mask_ptr
= &omask
;
44 /* Add a signal unlikely to be blocked, so that OMASK_COPY
45 is unlikely to match the actual mask. */
46 sigaddset (&omask
, SIGILL
);
47 memcpy (&omask_copy
, &omask
, sizeof omask
);
49 sigset_t
*old_mask_ptr
= old_mask
;
52 ret
= pthread_sigmask (how
, new_mask
, old_mask_ptr
);
54 # if PTHREAD_SIGMASK_INEFFECTIVE
57 /* Detect whether pthread_sigmask is currently ineffective.
58 Don't cache the information: libpthread.so could be dynamically
59 loaded after the program started and after pthread_sigmask was
60 called for the first time. */
61 if (memcmp (&omask_copy
, &omask
, sizeof omask
) == 0
62 && pthread_sigmask (1729, &omask_copy
, NULL
) == 0)
64 /* pthread_sigmask is currently ineffective. The program is not
65 linked to -lpthread. So use sigprocmask instead. */
66 return (sigprocmask (how
, new_mask
, old_mask
) < 0 ? errno
: 0);
70 memcpy (old_mask
, &omask
, sizeof omask
);
73 # if PTHREAD_SIGMASK_FAILS_WITH_ERRNO
77 # if PTHREAD_SIGMASK_UNBLOCK_BUG
80 && (how
== SIG_UNBLOCK
|| how
== SIG_SETMASK
))
82 /* Give the OS the opportunity to raise signals that were pending before
83 the pthread_sigmask call and have now been unblocked. */
89 int ret
= sigprocmask (how
, new_mask
, old_mask
);
90 return (ret
< 0 ? errno
: 0);