diff options
author | bkoz <bkoz@138bc75d-0d04-0410-961f-82ee72b054a4> | 2005-11-22 06:54:08 +0000 |
---|---|---|
committer | bkoz <bkoz@138bc75d-0d04-0410-961f-82ee72b054a4> | 2005-11-22 06:54:08 +0000 |
commit | 6b2678bbc932d84edcc5ae30b46362fbf6f2fc31 (patch) | |
tree | 3b23806d48c5499ec0ea11b68e5bc9006c2387e6 /libstdc++-v3/libsupc++ | |
parent | e284b8fbf0882a7734b8f93f351cdb6bb57338e7 (diff) | |
download | ppe42-gcc-6b2678bbc932d84edcc5ae30b46362fbf6f2fc31.tar.gz ppe42-gcc-6b2678bbc932d84edcc5ae30b46362fbf6f2fc31.zip |
2005-11-21 Benjamin Kosnik <bkoz@redhat.com>
Ulrich Drepper <drepper@redhat.com>
PR libstdc++/23591
* scripts/create_testsuite_files: Support for "C" test files.
* testsuite/lib/libstdc++.exp: Same.
* testsuite/libstdc++-dg/normal.exp: Same.
* testsuite/ext/mt_allocator/22309_thread.cc: Update names.
* testsuite/19_diagnostics/23591_thread-1.c: New.
* testsuite/testsuite_shared.cc: Add tests, rename existing functions.
* libsupc++/eh_globals.cc: Make global thread local if possible.
* configure.ac: Use GCC_CHECK_TLS.
* acinclude.m4: Include tls.m4.
* configure: Regenerate.
* config.h.in: Same.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@107350 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'libstdc++-v3/libsupc++')
-rw-r--r-- | libstdc++-v3/libsupc++/Makefile.in | 5 | ||||
-rw-r--r-- | libstdc++-v3/libsupc++/eh_globals.cc | 150 |
2 files changed, 91 insertions, 64 deletions
diff --git a/libstdc++-v3/libsupc++/Makefile.in b/libstdc++-v3/libsupc++/Makefile.in index 016bdb3dba9..1d5ee5695b4 100644 --- a/libstdc++-v3/libsupc++/Makefile.in +++ b/libstdc++-v3/libsupc++/Makefile.in @@ -42,11 +42,12 @@ DIST_COMMON = $(glibcxxinstall_HEADERS) $(srcdir)/Makefile.am \ $(srcdir)/Makefile.in $(top_srcdir)/fragment.am subdir = libsupc++ ACLOCAL_M4 = $(top_srcdir)/aclocal.m4 -am__aclocal_m4_deps = $(top_srcdir)/../config/lead-dot.m4 \ +am__aclocal_m4_deps = $(top_srcdir)/../config/enable.m4 \ + $(top_srcdir)/../config/lead-dot.m4 \ $(top_srcdir)/../config/no-executables.m4 \ $(top_srcdir)/../libtool.m4 $(top_srcdir)/crossconfig.m4 \ $(top_srcdir)/linkage.m4 $(top_srcdir)/acinclude.m4 \ - $(top_srcdir)/configure.ac + $(top_srcdir)/../config/tls.m4 $(top_srcdir)/configure.ac am__configure_deps = $(am__aclocal_m4_deps) $(CONFIGURE_DEPENDENCIES) \ $(ACLOCAL_M4) CONFIG_HEADER = $(top_builddir)/config.h diff --git a/libstdc++-v3/libsupc++/eh_globals.cc b/libstdc++-v3/libsupc++/eh_globals.cc index 00465dedf93..9b7e916716d 100644 --- a/libstdc++-v3/libsupc++/eh_globals.cc +++ b/libstdc++-v3/libsupc++/eh_globals.cc @@ -1,5 +1,5 @@ // -*- C++ -*- Manage the thread-local exception globals. -// Copyright (C) 2001, 2004 Free Software Foundation, Inc. +// Copyright (C) 2001, 2004, 2005 Free Software Foundation, Inc. // // This file is part of GCC. // @@ -27,101 +27,127 @@ // invalidate any other reasons why the executable file might be covered by // the GNU General Public License. - +#include <bits/c++config.h> #include <exception> #include <cstdlib> +#include "cxxabi.h" #include "unwind-cxx.h" -#include "bits/c++config.h" #include "bits/gthr.h" using namespace __cxxabiv1; +#if _GLIBCXX_HAVE_TLS + +namespace __gnu_internal +{ + using namespace abi; + using namespace std; + + __cxa_eh_globals* + get_global() throw() + { + static __thread __cxa_eh_globals global; + return &global; + } +} + +extern "C" __cxa_eh_globals* +__cxxabiv1::__cxa_get_globals_fast() throw() +{ return __gnu_internal::get_global(); } + +extern "C" __cxa_eh_globals* +__cxxabiv1::__cxa_get_globals() throw() +{ return __gnu_internal::get_global(); } + + +#else // Single-threaded fallback buffer. -static __cxa_eh_globals globals_static; +static __cxa_eh_globals eh_globals; #if __GTHREADS -static __gthread_key_t globals_key; -static int use_thread_key = -1; static void -get_globals_dtor (void *ptr) +eh_globals_dtor(void* ptr) { if (ptr) { - __cxa_exception *exn, *next; - exn = ((__cxa_eh_globals *) ptr)->caughtExceptions; + __cxa_eh_globals* g = reinterpret_cast<__cxa_eh_globals*>(ptr); + __cxa_exception* exn = g->caughtExceptions; + __cxa_exception* next; while (exn) { next = exn->nextException; - _Unwind_DeleteException (&exn->unwindHeader); + _Unwind_DeleteException(&exn->unwindHeader); exn = next; } - std::free (ptr); + std::free(ptr); } } -static void -get_globals_init () +struct __eh_globals_init { - use_thread_key = - (__gthread_key_create (&globals_key, get_globals_dtor) == 0); -} - -static void -get_globals_init_once () + __gthread_key_t _M_key; + bool _M_init; + + __eh_globals_init() : _M_init(false) + { + if (__gthread_active_p()) + _M_init = __gthread_key_create(&_M_key, eh_globals_dtor) == 0; + } + + ~__eh_globals_init() + { + if (_M_init) + __gthread_key_delete(_M_key); + } +}; + +static __eh_globals_init init; + +extern "C" __cxa_eh_globals* +__cxxabiv1::__cxa_get_globals_fast() throw() { - static __gthread_once_t once = __GTHREAD_ONCE_INIT; - if (__gthread_once (&once, get_globals_init) != 0 - || use_thread_key < 0) - use_thread_key = 0; + __cxa_eh_globals* g; + if (init._M_init) + g = static_cast<__cxa_eh_globals*>(__gthread_getspecific(init._M_key)); + else + g = &eh_globals; + return g; } -#endif -extern "C" __cxa_eh_globals * -__cxxabiv1::__cxa_get_globals_fast () throw() +extern "C" __cxa_eh_globals* +__cxxabiv1::__cxa_get_globals() throw() { -#if __GTHREADS - if (use_thread_key) - return (__cxa_eh_globals *) __gthread_getspecific (globals_key); + __cxa_eh_globals* g; + if (init._M_init) + { + g = static_cast<__cxa_eh_globals*>(__gthread_getspecific(init._M_key)); + if (!g) + { + void* v = std::malloc(sizeof(__cxa_eh_globals)); + if (v == 0 || __gthread_setspecific(init._M_key, v) != 0) + std::terminate(); + g = static_cast<__cxa_eh_globals*>(v); + g->caughtExceptions = 0; + g->uncaughtExceptions = 0; + } + } else - return &globals_static; -#else - return &globals_static; -#endif + g = &eh_globals; + return g; } -extern "C" __cxa_eh_globals * -__cxxabiv1::__cxa_get_globals () throw() -{ -#if __GTHREADS - __cxa_eh_globals *g; - - if (use_thread_key == 0) - return &globals_static; +#else - if (use_thread_key < 0) - { - get_globals_init_once (); +extern "C" __cxa_eh_globals* +__cxxabiv1::__cxa_get_globals_fast() throw() +{ return &eh_globals; } - // Make sure use_thread_key got initialized. - if (use_thread_key == 0) - return &globals_static; - } +extern "C" __cxa_eh_globals* +__cxxabiv1::__cxa_get_globals() throw() +{ return &eh_globals; } - g = (__cxa_eh_globals *) __gthread_getspecific (globals_key); - if (! g) - { - if ((g = (__cxa_eh_globals *) - std::malloc (sizeof (__cxa_eh_globals))) == 0 - || __gthread_setspecific (globals_key, (void *) g) != 0) - std::terminate (); - g->caughtExceptions = 0; - g->uncaughtExceptions = 0; - } +#endif - return g; -#else - return &globals_static; #endif -} |