Fix Bug#24962
[emacs.git] / lib / pthread_sigmask.c
blobf67e442bf3d4536536e4ddfeba4429c2d5b8b4cd
1 /* POSIX compatible signal blocking for threads.
2 Copyright (C) 2011-2016 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/>. */
17 #include <config.h>
19 /* Specification. */
20 #include <signal.h>
22 #include <errno.h>
23 #include <stddef.h>
25 #if PTHREAD_SIGMASK_INEFFECTIVE
26 # include <string.h>
27 #endif
29 #if PTHREAD_SIGMASK_UNBLOCK_BUG
30 # include <unistd.h>
31 #endif
33 int
34 pthread_sigmask (int how, const sigset_t *new_mask, sigset_t *old_mask)
35 #undef pthread_sigmask
37 #if HAVE_PTHREAD_SIGMASK
38 int ret;
40 # if PTHREAD_SIGMASK_INEFFECTIVE
41 sigset_t omask, omask_copy;
42 sigset_t *old_mask_ptr = &omask;
43 sigemptyset (&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);
48 # else
49 sigset_t *old_mask_ptr = old_mask;
50 # endif
52 ret = pthread_sigmask (how, new_mask, old_mask_ptr);
54 # if PTHREAD_SIGMASK_INEFFECTIVE
55 if (ret == 0)
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);
69 if (old_mask)
70 memcpy (old_mask, &omask, sizeof omask);
72 # endif
73 # if PTHREAD_SIGMASK_FAILS_WITH_ERRNO
74 if (ret == -1)
75 return errno;
76 # endif
77 # if PTHREAD_SIGMASK_UNBLOCK_BUG
78 if (ret == 0
79 && new_mask != NULL
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. */
84 usleep (1);
86 # endif
87 return ret;
88 #else
89 int ret = sigprocmask (how, new_mask, old_mask);
90 return (ret < 0 ? errno : 0);
91 #endif