| 1 | # pthread_sigmask.m4 serial 14 |
| 2 | dnl Copyright (C) 2011-2014 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. |
| 6 | |
| 7 | AC_DEFUN([gl_FUNC_PTHREAD_SIGMASK], |
| 8 | [ |
| 9 | AC_REQUIRE([gl_SIGNAL_H_DEFAULTS]) |
| 10 | |
| 11 | AC_CHECK_FUNCS_ONCE([pthread_sigmask]) |
| 12 | LIB_PTHREAD_SIGMASK= |
| 13 | |
| 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]) |
| 20 | |
| 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 | ]) |
| 85 | |
| 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". |
| 90 | |
| 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 |
| 95 | |
| 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 | case " $LIBS " in |
| 101 | *' -lpthread '*) ;; |
| 102 | *) |
| 103 | AC_CACHE_CHECK([whether pthread_sigmask works without -lpthread], |
| 104 | [gl_cv_func_pthread_sigmask_in_libc_works], |
| 105 | [ |
| 106 | AC_RUN_IFELSE( |
| 107 | [AC_LANG_SOURCE([[ |
| 108 | #include <pthread.h> |
| 109 | #include <signal.h> |
| 110 | #include <stddef.h> |
| 111 | int main () |
| 112 | { |
| 113 | sigset_t set; |
| 114 | sigemptyset (&set); |
| 115 | return pthread_sigmask (1729, &set, NULL) != 0; |
| 116 | }]])], |
| 117 | [gl_cv_func_pthread_sigmask_in_libc_works=no], |
| 118 | [gl_cv_func_pthread_sigmask_in_libc_works=yes], |
| 119 | [ |
| 120 | changequote(,)dnl |
| 121 | case "$host_os" in |
| 122 | freebsd* | hpux* | solaris | solaris2.[2-9]*) |
| 123 | gl_cv_func_pthread_sigmask_in_libc_works="guessing no";; |
| 124 | *) |
| 125 | gl_cv_func_pthread_sigmask_in_libc_works="guessing yes";; |
| 126 | esac |
| 127 | changequote([,])dnl |
| 128 | ]) |
| 129 | ]) |
| 130 | case "$gl_cv_func_pthread_sigmask_in_libc_works" in |
| 131 | *no) |
| 132 | REPLACE_PTHREAD_SIGMASK=1 |
| 133 | AC_DEFINE([PTHREAD_SIGMASK_INEFFECTIVE], [1], |
| 134 | [Define to 1 if pthread_sigmask may return 0 and have no effect.]) |
| 135 | ;; |
| 136 | esac;; |
| 137 | esac |
| 138 | fi |
| 139 | |
| 140 | dnl On Cygwin 1.7.5, the pthread_sigmask() has a wrong return value |
| 141 | dnl convention: Upon failure, it returns -1 and sets errno. |
| 142 | AC_CACHE_CHECK([whether pthread_sigmask returns error numbers], |
| 143 | [gl_cv_func_pthread_sigmask_return_works], |
| 144 | [ |
| 145 | gl_save_LIBS="$LIBS" |
| 146 | LIBS="$LIBS $LIB_PTHREAD_SIGMASK" |
| 147 | AC_RUN_IFELSE( |
| 148 | [AC_LANG_SOURCE([[ |
| 149 | #include <pthread.h> |
| 150 | #include <signal.h> |
| 151 | #include <stddef.h> |
| 152 | int main () |
| 153 | { |
| 154 | sigset_t set; |
| 155 | sigemptyset (&set); |
| 156 | if (pthread_sigmask (1729, &set, NULL) == -1) |
| 157 | return 1; |
| 158 | return 0; |
| 159 | }]])], |
| 160 | [gl_cv_func_pthread_sigmask_return_works=yes], |
| 161 | [gl_cv_func_pthread_sigmask_return_works=no], |
| 162 | [case "$host_os" in |
| 163 | cygwin*) |
| 164 | gl_cv_func_pthread_sigmask_return_works="guessing no";; |
| 165 | *) |
| 166 | gl_cv_func_pthread_sigmask_return_works="guessing yes";; |
| 167 | esac |
| 168 | ]) |
| 169 | LIBS="$gl_save_LIBS" |
| 170 | ]) |
| 171 | case "$gl_cv_func_pthread_sigmask_return_works" in |
| 172 | *no) |
| 173 | REPLACE_PTHREAD_SIGMASK=1 |
| 174 | AC_DEFINE([PTHREAD_SIGMASK_FAILS_WITH_ERRNO], [1], |
| 175 | [Define to 1 if pthread_sigmask(), when it fails, returns -1 and sets errno.]) |
| 176 | ;; |
| 177 | esac |
| 178 | |
| 179 | dnl On IRIX 6.5, in a single-threaded program, pending signals are not |
| 180 | dnl immediately delivered when they are unblocked through pthread_sigmask, |
| 181 | dnl only a little while later. |
| 182 | AC_CACHE_CHECK([whether pthread_sigmask unblocks signals correctly], |
| 183 | [gl_cv_func_pthread_sigmask_unblock_works], |
| 184 | [ |
| 185 | case "$host_os" in |
| 186 | irix*) |
| 187 | gl_cv_func_pthread_sigmask_unblock_works="guessing no";; |
| 188 | *) |
| 189 | gl_cv_func_pthread_sigmask_unblock_works="guessing yes";; |
| 190 | esac |
| 191 | m4_ifdef([gl_][THREADLIB], |
| 192 | [dnl Link against $LIBMULTITHREAD, not only $LIB_PTHREAD_SIGMASK. |
| 193 | dnl Otherwise we get a false positive on those platforms where |
| 194 | dnl $gl_cv_func_pthread_sigmask_in_libc_works is "no". |
| 195 | gl_save_LIBS=$LIBS |
| 196 | LIBS="$LIBS $LIBMULTITHREAD"]) |
| 197 | AC_RUN_IFELSE( |
| 198 | [AC_LANG_SOURCE([[ |
| 199 | #include <pthread.h> |
| 200 | #include <signal.h> |
| 201 | #include <stdio.h> |
| 202 | #include <stdlib.h> |
| 203 | #include <unistd.h> |
| 204 | static volatile int sigint_occurred; |
| 205 | static void |
| 206 | sigint_handler (int sig) |
| 207 | { |
| 208 | sigint_occurred++; |
| 209 | } |
| 210 | int main () |
| 211 | { |
| 212 | sigset_t set; |
| 213 | int pid = getpid (); |
| 214 | char command[80]; |
| 215 | signal (SIGINT, sigint_handler); |
| 216 | sigemptyset (&set); |
| 217 | sigaddset (&set, SIGINT); |
| 218 | if (!(pthread_sigmask (SIG_BLOCK, &set, NULL) == 0)) |
| 219 | return 1; |
| 220 | sprintf (command, "sh -c 'sleep 1; kill -%d %d' &", SIGINT, pid); |
| 221 | if (!(system (command) == 0)) |
| 222 | return 2; |
| 223 | sleep (2); |
| 224 | if (!(sigint_occurred == 0)) |
| 225 | return 3; |
| 226 | if (!(pthread_sigmask (SIG_UNBLOCK, &set, NULL) == 0)) |
| 227 | return 4; |
| 228 | if (!(sigint_occurred == 1)) /* This fails on IRIX. */ |
| 229 | return 5; |
| 230 | return 0; |
| 231 | }]])], |
| 232 | [:], |
| 233 | [gl_cv_func_pthread_sigmask_unblock_works=no], |
| 234 | [:]) |
| 235 | m4_ifdef([gl_][THREADLIB], [LIBS=$gl_save_LIBS]) |
| 236 | ]) |
| 237 | case "$gl_cv_func_pthread_sigmask_unblock_works" in |
| 238 | *no) |
| 239 | REPLACE_PTHREAD_SIGMASK=1 |
| 240 | AC_DEFINE([PTHREAD_SIGMASK_UNBLOCK_BUG], [1], |
| 241 | [Define to 1 if pthread_sigmask() unblocks signals incorrectly.]) |
| 242 | ;; |
| 243 | esac |
| 244 | fi |
| 245 | ]) |
| 246 | |
| 247 | # Prerequisite of lib/pthread_sigmask.c. |
| 248 | AC_DEFUN([gl_PREREQ_PTHREAD_SIGMASK], |
| 249 | [ |
| 250 | if test $HAVE_PTHREAD_SIGMASK = 1; then |
| 251 | AC_DEFINE([HAVE_PTHREAD_SIGMASK], [1], |
| 252 | [Define to 1 if the pthread_sigmask function can be used (despite bugs).]) |
| 253 | fi |
| 254 | ]) |