re PR libstdc++/29426 (static __recursive_mutex init vs __GTHREAD_RECURSIVE_MUTEX_INIT_FUNCTION)

2006-10-11  Benjamin Kosnik  <bkoz@redhat.com>

	PR libstdc++/29426
	* libsupc++/guard.cc (get_static_mutex): New. 
	(mutex_wrapper::mutex_wrapper): Use it to get properly initialized
	recursive mutex without ordering issues.

	* src/locale_init.cc (__get_locale_mutex): No need to
	uglify. Change to get_locale_mutex.

From-SVN: r117643
This commit is contained in:
Benjamin Kosnik 2006-10-11 20:18:36 +00:00 committed by Benjamin Kosnik
parent 7a1af5508c
commit e64690af46
3 changed files with 44 additions and 22 deletions

View File

@ -1,3 +1,13 @@
2006-10-11 Benjamin Kosnik <bkoz@redhat.com>
PR libstdc++/29426
* libsupc++/guard.cc (get_static_mutex): New.
(mutex_wrapper::mutex_wrapper): Use it to get properly initialized
recursive mutex without ordering issues.
* src/locale_init.cc (__get_locale_mutex): No need to
uglify. Change to get_locale_mutex.
2006-10-11 Paolo Carlini <pcarlini@suse.de> 2006-10-11 Paolo Carlini <pcarlini@suse.de>
* testsuite/22_locale/num_put/put/char/11.cc: New. * testsuite/22_locale/num_put/put/char/11.cc: New.

View File

@ -32,6 +32,7 @@
#include <bits/c++config.h> #include <bits/c++config.h>
#include <cxxabi.h> #include <cxxabi.h>
#include <exception> #include <exception>
#include <new>
#include <ext/atomicity.h> #include <ext/atomicity.h>
#include <ext/concurrence.h> #include <ext/concurrence.h>
@ -43,7 +44,22 @@
namespace namespace
{ {
// A single mutex controlling all static initializations. // A single mutex controlling all static initializations.
__gnu_cxx::__recursive_mutex static_mutex; static __gnu_cxx::__recursive_mutex* static_mutex;
typedef char fake_recursive_mutex[sizeof(__gnu_cxx::__recursive_mutex)]
__attribute__ ((aligned(__alignof__(__gnu_cxx::__recursive_mutex))));
fake_recursive_mutex fake_mutex;
static void init()
{ static_mutex = new (&fake_mutex) __gnu_cxx::__recursive_mutex(); }
__gnu_cxx::__recursive_mutex&
get_static_mutex()
{
static __gthread_once_t once = __GTHREAD_ONCE_INIT;
__gthread_once(&once, init);
return *static_mutex;
}
} }
#ifndef _GLIBCXX_GUARD_TEST_AND_ACQUIRE #ifndef _GLIBCXX_GUARD_TEST_AND_ACQUIRE
@ -98,18 +114,14 @@ namespace __cxxabiv1
{ {
static inline int static inline int
recursion_push (__guard* g) recursion_push (__guard* g)
{ { return ((char *)g)[1]++; }
return ((char *)g)[1]++;
}
static inline void static inline void
recursion_pop (__guard* g) recursion_pop (__guard* g)
{ { --((char *)g)[1]; }
--((char *)g)[1];
}
static int static int
acquire_1 (__guard *g) acquire (__guard *g)
{ {
if (_GLIBCXX_GUARD_TEST (g)) if (_GLIBCXX_GUARD_TEST (g))
return 0; return 0;
@ -143,17 +155,17 @@ namespace __cxxabiv1
{ {
bool unlock; bool unlock;
mutex_wrapper() : unlock(true) mutex_wrapper() : unlock(true)
{ { get_static_mutex().lock(); }
static_mutex.lock();
}
~mutex_wrapper() ~mutex_wrapper()
{ {
if (unlock) if (unlock)
static_mutex.unlock(); static_mutex->unlock();
} }
} mw; };
if (acquire_1 (g)) mutex_wrapper mw;
if (acquire (g))
{ {
mw.unlock = false; mw.unlock = false;
return 1; return 1;
@ -163,7 +175,7 @@ namespace __cxxabiv1
} }
#endif #endif
return acquire_1 (g); return acquire (g);
} }
extern "C" extern "C"
@ -172,7 +184,7 @@ namespace __cxxabiv1
recursion_pop (g); recursion_pop (g);
#ifdef __GTHREADS #ifdef __GTHREADS
if (__gthread_active_p ()) if (__gthread_active_p ())
static_mutex.unlock(); static_mutex->unlock();
#endif #endif
} }
@ -183,7 +195,7 @@ namespace __cxxabiv1
_GLIBCXX_GUARD_SET_AND_RELEASE (g); _GLIBCXX_GUARD_SET_AND_RELEASE (g);
#ifdef __GTHREADS #ifdef __GTHREADS
if (__gthread_active_p ()) if (__gthread_active_p ())
static_mutex.unlock(); static_mutex->unlock();
#endif #endif
} }
} }

View File

@ -37,7 +37,7 @@
namespace namespace
{ {
__gnu_cxx::__mutex& __gnu_cxx::__mutex&
__get_locale_mutex() get_locale_mutex()
{ {
static __gnu_cxx::__mutex locale_mutex; static __gnu_cxx::__mutex locale_mutex;
return locale_mutex; return locale_mutex;
@ -212,7 +212,7 @@ _GLIBCXX_BEGIN_NAMESPACE(std)
locale::locale() throw() : _M_impl(0) locale::locale() throw() : _M_impl(0)
{ {
_S_initialize(); _S_initialize();
__gnu_cxx::__scoped_lock sentry(__get_locale_mutex()); __gnu_cxx::__scoped_lock sentry(get_locale_mutex());
_S_global->_M_add_reference(); _S_global->_M_add_reference();
_M_impl = _S_global; _M_impl = _S_global;
} }
@ -223,7 +223,7 @@ _GLIBCXX_BEGIN_NAMESPACE(std)
_S_initialize(); _S_initialize();
_Impl* __old; _Impl* __old;
{ {
__gnu_cxx::__scoped_lock sentry(__get_locale_mutex()); __gnu_cxx::__scoped_lock sentry(get_locale_mutex());
__old = _S_global; __old = _S_global;
__other._M_impl->_M_add_reference(); __other._M_impl->_M_add_reference();
_S_global = __other._M_impl; _S_global = __other._M_impl;