| 1 |
722 |
jeremybenn |
dnl Check whether the target supports TLS.
|
| 2 |
|
|
AC_DEFUN([GCC_CHECK_TLS], [
|
| 3 |
|
|
AC_REQUIRE([AC_CANONICAL_HOST])
|
| 4 |
|
|
GCC_ENABLE(tls, yes, [], [Use thread-local storage])
|
| 5 |
|
|
AC_CACHE_CHECK([whether the target supports thread-local storage],
|
| 6 |
|
|
gcc_cv_have_tls, [
|
| 7 |
|
|
AC_RUN_IFELSE([__thread int a; int b; int main() { return a = b; }],
|
| 8 |
|
|
[dnl If the test case passed with dynamic linking, try again with
|
| 9 |
|
|
dnl static linking, but only if static linking is supported (not
|
| 10 |
|
|
dnl on Solaris 10). This fails with some older Red Hat releases.
|
| 11 |
|
|
chktls_save_LDFLAGS="$LDFLAGS"
|
| 12 |
|
|
LDFLAGS="-static $LDFLAGS"
|
| 13 |
|
|
AC_LINK_IFELSE([int main() { return 0; }],
|
| 14 |
|
|
[AC_RUN_IFELSE([__thread int a; int b; int main() { return a = b; }],
|
| 15 |
|
|
[gcc_cv_have_tls=yes], [gcc_cv_have_tls=no],[])],
|
| 16 |
|
|
[gcc_cv_have_tls=yes])
|
| 17 |
|
|
LDFLAGS="$chktls_save_LDFLAGS"
|
| 18 |
|
|
if test $gcc_cv_have_tls = yes; then
|
| 19 |
|
|
dnl So far, the binutils and the compiler support TLS.
|
| 20 |
|
|
dnl Also check whether the libc supports TLS, i.e. whether a variable
|
| 21 |
|
|
dnl with __thread linkage has a different address in different threads.
|
| 22 |
|
|
dnl First, find the thread_CFLAGS necessary for linking a program that
|
| 23 |
|
|
dnl calls pthread_create.
|
| 24 |
|
|
chktls_save_CFLAGS="$CFLAGS"
|
| 25 |
|
|
thread_CFLAGS=failed
|
| 26 |
|
|
for flag in '' '-pthread' '-lpthread'; do
|
| 27 |
|
|
CFLAGS="$flag $chktls_save_CFLAGS"
|
| 28 |
|
|
AC_LINK_IFELSE(
|
| 29 |
|
|
[AC_LANG_PROGRAM(
|
| 30 |
|
|
[#include <pthread.h>
|
| 31 |
|
|
void *g(void *d) { return NULL; }],
|
| 32 |
|
|
[pthread_t t; pthread_create(&t,NULL,g,NULL);])],
|
| 33 |
|
|
[thread_CFLAGS="$flag"])
|
| 34 |
|
|
if test "X$thread_CFLAGS" != Xfailed; then
|
| 35 |
|
|
break
|
| 36 |
|
|
fi
|
| 37 |
|
|
done
|
| 38 |
|
|
CFLAGS="$chktls_save_CFLAGS"
|
| 39 |
|
|
if test "X$thread_CFLAGS" != Xfailed; then
|
| 40 |
|
|
CFLAGS="$thread_CFLAGS $chktls_save_CFLAGS"
|
| 41 |
|
|
dnl Test for an old glibc bug that violated the __thread property.
|
| 42 |
|
|
dnl Use volatile to ensure the compiler won't optimize away pointer
|
| 43 |
|
|
dnl accesses it might otherwise assume to be redundant, or reorder
|
| 44 |
|
|
dnl them and reuse storage, which might lead to them pointing to
|
| 45 |
|
|
dnl the same location.
|
| 46 |
|
|
AC_RUN_IFELSE(
|
| 47 |
|
|
[AC_LANG_PROGRAM(
|
| 48 |
|
|
[#include <pthread.h>
|
| 49 |
|
|
__thread int a;
|
| 50 |
|
|
static int *volatile a_in_other_thread;
|
| 51 |
|
|
static void *
|
| 52 |
|
|
thread_func (void *arg)
|
| 53 |
|
|
{
|
| 54 |
|
|
a_in_other_thread = &a;
|
| 55 |
|
|
return (void *)0;
|
| 56 |
|
|
}],
|
| 57 |
|
|
[pthread_t thread;
|
| 58 |
|
|
void *thread_retval;
|
| 59 |
|
|
int *volatile a_in_main_thread;
|
| 60 |
|
|
a_in_main_thread = &a;
|
| 61 |
|
|
if (pthread_create (&thread, (pthread_attr_t *)0,
|
| 62 |
|
|
thread_func, (void *)0))
|
| 63 |
|
|
return 0;
|
| 64 |
|
|
if (pthread_join (thread, &thread_retval))
|
| 65 |
|
|
return 0;
|
| 66 |
|
|
return (a_in_other_thread == a_in_main_thread);])],
|
| 67 |
|
|
[gcc_cv_have_tls=yes], [gcc_cv_have_tls=no], [])
|
| 68 |
|
|
CFLAGS="$chktls_save_CFLAGS"
|
| 69 |
|
|
fi
|
| 70 |
|
|
fi],
|
| 71 |
|
|
[gcc_cv_have_tls=no],
|
| 72 |
|
|
[dnl This is the cross-compiling case. Assume libc supports TLS if the
|
| 73 |
|
|
dnl binutils and the compiler do.
|
| 74 |
|
|
AC_LINK_IFELSE([__thread int a; int b; int main() { return a = b; }],
|
| 75 |
|
|
[chktls_save_LDFLAGS="$LDFLAGS"
|
| 76 |
|
|
dnl Shared library options may depend on the host; this check
|
| 77 |
|
|
dnl is only known to be needed for GNU/Linux.
|
| 78 |
|
|
case $host in
|
| 79 |
|
|
*-*-linux*)
|
| 80 |
|
|
LDFLAGS="-shared -Wl,--no-undefined $LDFLAGS"
|
| 81 |
|
|
;;
|
| 82 |
|
|
esac
|
| 83 |
|
|
chktls_save_CFLAGS="$CFLAGS"
|
| 84 |
|
|
CFLAGS="-fPIC $CFLAGS"
|
| 85 |
|
|
dnl If -shared works, test if TLS works in a shared library.
|
| 86 |
|
|
AC_LINK_IFELSE([int f() { return 0; }],
|
| 87 |
|
|
[AC_LINK_IFELSE([__thread int a; int b; int f() { return a = b; }],
|
| 88 |
|
|
[gcc_cv_have_tls=yes],
|
| 89 |
|
|
[gcc_cv_have_tls=no])],
|
| 90 |
|
|
[gcc_cv_have_tls=yes])
|
| 91 |
|
|
CFLAGS="$chktls_save_CFLAGS"
|
| 92 |
|
|
LDFLAGS="$chktls_save_LDFLAGS"], [gcc_cv_have_tls=no])
|
| 93 |
|
|
]
|
| 94 |
|
|
)])
|
| 95 |
|
|
if test "$enable_tls $gcc_cv_have_tls" = "yes yes"; then
|
| 96 |
|
|
AC_DEFINE(HAVE_TLS, 1,
|
| 97 |
|
|
[Define to 1 if the target supports thread-local storage.])
|
| 98 |
|
|
fi])
|
| 99 |
|
|
|
| 100 |
|
|
dnl Check whether the target assembler supports TLS.
|
| 101 |
|
|
AC_DEFUN([GCC_CHECK_CC_TLS], [
|
| 102 |
|
|
GCC_ENABLE(tls, yes, [], [Use thread-local storage])
|
| 103 |
|
|
AC_CACHE_CHECK([whether the target assembler supports thread-local storage],
|
| 104 |
|
|
gcc_cv_have_cc_tls, [
|
| 105 |
|
|
AC_COMPILE_IFELSE([__thread int a; int b; int main() { return a = b; }],
|
| 106 |
|
|
[gcc_cv_have_cc_tls=yes], [gcc_cv_have_cc_tls=no])]
|
| 107 |
|
|
)])
|
| 108 |
|
|
if test "$enable_tls $gcc_cv_have_cc_tls" = "yes yes"; then
|
| 109 |
|
|
AC_DEFINE(HAVE_CC_TLS, 1,
|
| 110 |
|
|
[Define to 1 if the target assembler supports thread-local storage.])
|
| 111 |
|
|
fi])
|
| 112 |
|
|
|
| 113 |
|
|
dnl Check whether TLS is emulated.
|
| 114 |
|
|
AC_DEFUN([GCC_CHECK_EMUTLS], [
|
| 115 |
|
|
AC_CACHE_CHECK([whether the thread-local storage support is from emutls],
|
| 116 |
|
|
gcc_cv_use_emutls, [
|
| 117 |
|
|
gcc_cv_use_emutls=no
|
| 118 |
|
|
echo '__thread int a; int b; int main() { return a = b; }' > conftest.c
|
| 119 |
|
|
if AC_TRY_COMMAND(${CC-cc} -Werror -S -o conftest.s conftest.c 1>&AS_MESSAGE_LOG_FD); then
|
| 120 |
|
|
if grep __emutls_get_address conftest.s > /dev/null; then
|
| 121 |
|
|
gcc_cv_use_emutls=yes
|
| 122 |
|
|
fi
|
| 123 |
|
|
fi
|
| 124 |
|
|
rm -f conftest.*
|
| 125 |
|
|
])
|
| 126 |
|
|
if test "$gcc_cv_use_emutls" = "yes" ; then
|
| 127 |
|
|
AC_DEFINE(USE_EMUTLS, 1,
|
| 128 |
|
|
[Define to 1 if the target use emutls for thread-local storage.])
|
| 129 |
|
|
fi])
|