* calendar/todos.el: Further comment revision.
[emacs.git] / m4 / pthread_sigmask.m4
blob884edbdb5bb685e6ab7f2fe8192922076aaa4238
1 # pthread_sigmask.m4 serial 13
2 dnl Copyright (C) 2011-2012 Free Software Foundation, Inc.
3 dnl This file is free software; the Free Software Foundation
4 dnl gives unlimited permission to copy and/or distribute it,
5 dnl with or without modifications, as long as this notice is preserved.
7 AC_DEFUN([gl_FUNC_PTHREAD_SIGMASK],
9   AC_REQUIRE([gl_SIGNAL_H_DEFAULTS])
11   AC_CHECK_FUNCS_ONCE([pthread_sigmask])
12   LIB_PTHREAD_SIGMASK=
14   dnl Test whether the gnulib module 'threadlib' is in use.
15   dnl Some packages like Emacs use --avoid=threadlib.
16   dnl Write the symbol in such a way that it does not cause 'aclocal' to pick
17   dnl the threadlib.m4 file that is installed in $PREFIX/share/aclocal/.
18   m4_ifdef([gl_][THREADLIB], [
19     AC_REQUIRE([gl_][THREADLIB])
21     if test "$gl_threads_api" = posix; then
22       if test $ac_cv_func_pthread_sigmask = yes; then
23         dnl pthread_sigmask is available without -lpthread.
24         :
25       else
26         if test -n "$LIBMULTITHREAD"; then
27           AC_CACHE_CHECK([for pthread_sigmask in $LIBMULTITHREAD],
28             [gl_cv_func_pthread_sigmask_in_LIBMULTITHREAD],
29             [gl_save_LIBS="$LIBS"
30              LIBS="$LIBS $LIBMULTITHREAD"
31              AC_LINK_IFELSE(
32                [AC_LANG_PROGRAM(
33                   [[#include <pthread.h>
34                     #include <signal.h>
35                   ]],
36                   [[return pthread_sigmask (0, (sigset_t *) 0, (sigset_t *) 0);]])
37                ],
38                [gl_cv_func_pthread_sigmask_in_LIBMULTITHREAD=yes],
39                [gl_cv_func_pthread_sigmask_in_LIBMULTITHREAD=no])
40              LIBS="$gl_save_LIBS"
41             ])
42           if test $gl_cv_func_pthread_sigmask_in_LIBMULTITHREAD = yes; then
43             dnl pthread_sigmask is available with -lpthread.
44             LIB_PTHREAD_SIGMASK="$LIBMULTITHREAD"
45           else
46             dnl pthread_sigmask is not available at all.
47             HAVE_PTHREAD_SIGMASK=0
48           fi
49         else
50           dnl pthread_sigmask is not available at all.
51           HAVE_PTHREAD_SIGMASK=0
52         fi
53       fi
54     else
55       dnl pthread_sigmask may exist but does not interoperate with the chosen
56       dnl multithreading facility.
57       dnl If "$gl_threads_api" = pth, we could use the function pth_sigmask,
58       dnl but it is equivalent to sigprocmask, so we choose to emulate
59       dnl pthread_sigmask with sigprocmask also in this case. This yields fewer
60       dnl link dependencies.
61       if test $ac_cv_func_pthread_sigmask = yes; then
62         REPLACE_PTHREAD_SIGMASK=1
63       else
64         HAVE_PTHREAD_SIGMASK=0
65       fi
66     fi
67   ], [
68     dnl The module 'threadlib' is not in use, due to --avoid=threadlib being
69     dnl specified.
70     dnl The package either has prepared CPPFLAGS and LIBS for use of POSIX:2008
71     dnl threads, or wants to build single-threaded programs.
72     if test $ac_cv_func_pthread_sigmask = yes; then
73       dnl pthread_sigmask exists and does not require extra libraries.
74       dnl Assume that it is declared.
75       :
76     else
77       dnl pthread_sigmask either does not exist or needs extra libraries.
78       HAVE_PTHREAD_SIGMASK=0
79       dnl Define the symbol rpl_pthread_sigmask, not pthread_sigmask,
80       dnl so as to not accidentally override the system's pthread_sigmask
81       dnl symbol from libpthread. This is necessary on IRIX 6.5.
82       REPLACE_PTHREAD_SIGMASK=1
83     fi
84   ])
86   AC_SUBST([LIB_PTHREAD_SIGMASK])
87   dnl We don't need a variable LTLIB_PTHREAD_SIGMASK, because when
88   dnl "$gl_threads_api" = posix, $LTLIBMULTITHREAD and $LIBMULTITHREAD are the
89   dnl same: either both empty or both "-lpthread".
91   dnl Now test for some bugs in the system function.
92   if test $HAVE_PTHREAD_SIGMASK = 1; then
93     AC_REQUIRE([AC_PROG_CC])
94     AC_REQUIRE([AC_CANONICAL_HOST]) dnl for cross-compiles
96     dnl On FreeBSD 6.4, HP-UX 11.31, Solaris 9, in programs that are not linked
97     dnl with -lpthread, the pthread_sigmask() function always returns 0 and has
98     dnl no effect.
99     if test -z "$LIB_PTHREAD_SIGMASK"; then
100       AC_CACHE_CHECK([whether pthread_sigmask works without -lpthread],
101         [gl_cv_func_pthread_sigmask_in_libc_works],
102         [
103           AC_RUN_IFELSE(
104             [AC_LANG_SOURCE([[
105 #include <pthread.h>
106 #include <signal.h>
107 #include <stddef.h>
108 int main ()
110   sigset_t set;
111   sigemptyset (&set);
112   return pthread_sigmask (1729, &set, NULL) != 0;
113 }]])],
114             [gl_cv_func_pthread_sigmask_in_libc_works=no],
115             [gl_cv_func_pthread_sigmask_in_libc_works=yes],
116             [
117 changequote(,)dnl
118              case "$host_os" in
119                freebsd* | hpux* | solaris | solaris2.[2-9]*)
120                  gl_cv_func_pthread_sigmask_in_libc_works="guessing no";;
121                *)
122                  gl_cv_func_pthread_sigmask_in_libc_works="guessing yes";;
123              esac
124 changequote([,])dnl
125             ])
126         ])
127       case "$gl_cv_func_pthread_sigmask_in_libc_works" in
128         *no)
129           REPLACE_PTHREAD_SIGMASK=1
130           AC_DEFINE([PTHREAD_SIGMASK_INEFFECTIVE], [1],
131             [Define to 1 if pthread_sigmask() may returns 0 and have no effect.])
132           ;;
133       esac
134     fi
136     dnl On Cygwin 1.7.5, the pthread_sigmask() has a wrong return value
137     dnl convention: Upon failure, it returns -1 and sets errno.
138     AC_CACHE_CHECK([whether pthread_sigmask returns error numbers],
139       [gl_cv_func_pthread_sigmask_return_works],
140       [
141         gl_save_LIBS="$LIBS"
142         LIBS="$LIBS $LIB_PTHREAD_SIGMASK"
143         AC_RUN_IFELSE(
144           [AC_LANG_SOURCE([[
145 #include <pthread.h>
146 #include <signal.h>
147 #include <stddef.h>
148 int main ()
150   sigset_t set;
151   sigemptyset (&set);
152   if (pthread_sigmask (1729, &set, NULL) == -1)
153     return 1;
154   return 0;
155 }]])],
156           [gl_cv_func_pthread_sigmask_return_works=yes],
157           [gl_cv_func_pthread_sigmask_return_works=no],
158           [case "$host_os" in
159              cygwin*)
160                gl_cv_func_pthread_sigmask_return_works="guessing no";;
161              *)
162                gl_cv_func_pthread_sigmask_return_works="guessing yes";;
163            esac
164           ])
165         LIBS="$gl_save_LIBS"
166       ])
167     case "$gl_cv_func_pthread_sigmask_return_works" in
168       *no)
169         REPLACE_PTHREAD_SIGMASK=1
170         AC_DEFINE([PTHREAD_SIGMASK_FAILS_WITH_ERRNO], [1],
171           [Define to 1 if pthread_sigmask(), when it fails, returns -1 and sets errno.])
172         ;;
173     esac
175     dnl On IRIX 6.5, in a single-threaded program, pending signals are not
176     dnl immediately delivered when they are unblocked through pthread_sigmask,
177     dnl only a little while later.
178     AC_CACHE_CHECK([whether pthread_sigmask unblocks signals correctly],
179       [gl_cv_func_pthread_sigmask_unblock_works],
180       [
181         case "$host_os" in
182           irix*)
183             gl_cv_func_pthread_sigmask_unblock_works="guessing no";;
184           *)
185             gl_cv_func_pthread_sigmask_unblock_works="guessing yes";;
186         esac
187         dnl Here we link against $LIBMULTITHREAD, not only $LIB_PTHREAD_SIGMASK,
188         dnl otherwise we get a false positive on those platforms where
189         dnl $gl_cv_func_pthread_sigmask_in_libc_works is "no".
190         gl_save_LIBS="$LIBS"
191         LIBS="$LIBS $LIBMULTITHREAD"
192         AC_RUN_IFELSE(
193           [AC_LANG_SOURCE([[
194 #include <pthread.h>
195 #include <signal.h>
196 #include <stdio.h>
197 #include <stdlib.h>
198 #include <unistd.h>
199 static volatile int sigint_occurred;
200 static void
201 sigint_handler (int sig)
203   sigint_occurred++;
205 int main ()
207   sigset_t set;
208   int pid = getpid ();
209   char command[80];
210   signal (SIGINT, sigint_handler);
211   sigemptyset (&set);
212   sigaddset (&set, SIGINT);
213   if (!(pthread_sigmask (SIG_BLOCK, &set, NULL) == 0))
214     return 1;
215   sprintf (command, "sh -c 'sleep 1; kill -%d %d' &", SIGINT, pid);
216   if (!(system (command) == 0))
217     return 2;
218   sleep (2);
219   if (!(sigint_occurred == 0))
220     return 3;
221   if (!(pthread_sigmask (SIG_UNBLOCK, &set, NULL) == 0))
222     return 4;
223   if (!(sigint_occurred == 1)) /* This fails on IRIX.  */
224     return 5;
225   return 0;
226 }]])],
227           [:],
228           [gl_cv_func_pthread_sigmask_unblock_works=no],
229           [:])
230         LIBS="$gl_save_LIBS"
231       ])
232     case "$gl_cv_func_pthread_sigmask_unblock_works" in
233       *no)
234         REPLACE_PTHREAD_SIGMASK=1
235         AC_DEFINE([PTHREAD_SIGMASK_UNBLOCK_BUG], [1],
236           [Define to 1 if pthread_sigmask() unblocks signals incorrectly.])
237         ;;
238     esac
239   fi
242 # Prerequisite of lib/pthread_sigmask.c.
243 AC_DEFUN([gl_PREREQ_PTHREAD_SIGMASK],
245   if test $HAVE_PTHREAD_SIGMASK = 1; then
246     AC_DEFINE([HAVE_PTHREAD_SIGMASK], [1],
247       [Define to 1 if the pthread_sigmask function can be used (despite bugs).])
248   fi