Skip to content

Commit 8baf838

Browse files
committedJul 7, 2019
Fix PR27658 - Make ~mutex trivial when possible.
Currently std::mutex has a constexpr constructor, but a non-trivial destruction. The constexpr constructor is required to ensure the construction of a mutex with static storage duration happens at compile time, during constant initialization, and not during dynamic initialization. This means that static mutex's are always initialized and can be used safely during dynamic initialization without the "static initialization order fiasco". A trivial destructor is important for similar reasons. If a mutex is used during dynamic initialization it might also be used during program termination. If a static mutex has a non-trivial destructor it will be invoked during termination. This can introduce the "static deinitialization order fiasco". Additionally, function-local statics emit a guard variable around non-trivially destructible types. This results in horrible codegen and adds a runtime cost to every call to that function. non-local static's also result in slightly worse codegen but it's not as big of a problem. Example codegen can be found here: https://goo.gl/3CSzbM Note: This optimization is not safe with every pthread implementation. Some implementations allocate on the first call to pthread_mutex_lock and free the allocation in pthread_mutex_destroy. Also, changing the triviality of the destructor is not an ABI break. At least to the best of my knowledge :-) llvm-svn: 365273
1 parent 9e52c43 commit 8baf838

File tree

6 files changed

+73
-20
lines changed

6 files changed

+73
-20
lines changed
 

‎libcxx/include/__config

+8
Original file line numberDiff line numberDiff line change
@@ -1097,6 +1097,14 @@ _LIBCPP_FUNC_VIS extern "C" void __sanitizer_annotate_contiguous_container(
10971097
_LIBCPP_HAS_NO_THREADS is defined.
10981098
#endif
10991099

1100+
// The Apple, glibc, and Bionic implementation of pthreads implements
1101+
// pthread_mutex_destroy as nop for regular mutexes.
1102+
// TODO(EricWF): Enable this optimization on Apple and Bionic platforms after
1103+
// speaking to their respective stakeholders.
1104+
#if defined(_LIBCPP_HAS_THREAD_API_PTHREAD) && defined(__GLIBC__)
1105+
# define _LIBCPP_HAS_TRIVIAL_MUTEX_DESTRUCTION
1106+
#endif
1107+
11001108
// Systems that use capability-based security (FreeBSD with Capsicum,
11011109
// Nuxi CloudABI) may only provide local filesystem access (using *at()).
11021110
// Functions like open(), rename(), unlink() and stat() should not be

‎libcxx/include/__mutex_base

+9-13
Original file line numberDiff line numberDiff line change
@@ -36,28 +36,24 @@ _LIBCPP_BEGIN_NAMESPACE_STD
3636
# endif
3737
#endif // _LIBCPP_THREAD_SAFETY_ANNOTATION
3838

39+
3940
class _LIBCPP_TYPE_VIS _LIBCPP_THREAD_SAFETY_ANNOTATION(capability("mutex")) mutex
4041
{
41-
#ifndef _LIBCPP_CXX03_LANG
4242
__libcpp_mutex_t __m_ = _LIBCPP_MUTEX_INITIALIZER;
43-
#else
44-
__libcpp_mutex_t __m_;
45-
#endif
4643

4744
public:
4845
_LIBCPP_INLINE_VISIBILITY
49-
#ifndef _LIBCPP_CXX03_LANG
50-
constexpr mutex() = default;
46+
_LIBCPP_CONSTEXPR mutex() = default;
47+
48+
mutex(const mutex&) = delete;
49+
mutex& operator=(const mutex&) = delete;
50+
51+
#if defined(_LIBCPP_HAS_TRIVIAL_MUTEX_DESTRUCTION)
52+
~mutex() = default;
5153
#else
52-
mutex() _NOEXCEPT {__m_ = (__libcpp_mutex_t)_LIBCPP_MUTEX_INITIALIZER;}
53-
#endif
5454
~mutex() _NOEXCEPT;
55+
#endif
5556

56-
private:
57-
mutex(const mutex&);// = delete;
58-
mutex& operator=(const mutex&);// = delete;
59-
60-
public:
6157
void lock() _LIBCPP_THREAD_SAFETY_ANNOTATION(acquire_capability());
6258
bool try_lock() _NOEXCEPT _LIBCPP_THREAD_SAFETY_ANNOTATION(try_acquire_capability(true));
6359
void unlock() _NOEXCEPT _LIBCPP_THREAD_SAFETY_ANNOTATION(release_capability());

‎libcxx/src/CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ set(LIBCXX_SOURCES
2222
locale.cpp
2323
memory.cpp
2424
mutex.cpp
25+
mutex_destructor.cpp
2526
new.cpp
2627
optional.cpp
2728
random.cpp

‎libcxx/src/mutex.cpp

+1-4
Original file line numberDiff line numberDiff line change
@@ -25,10 +25,7 @@ const defer_lock_t defer_lock = {};
2525
const try_to_lock_t try_to_lock = {};
2626
const adopt_lock_t adopt_lock = {};
2727

28-
mutex::~mutex() _NOEXCEPT
29-
{
30-
__libcpp_mutex_destroy(&__m_);
31-
}
28+
// ~mutex is defined elsewhere
3229

3330
void
3431
mutex::lock()

‎libcxx/src/mutex_destructor.cpp

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
//===--------------------- mutex_destructor.cpp ---------------------------===//
2+
//
3+
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4+
// See https://llvm.org/LICENSE.txt for license information.
5+
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6+
//
7+
//===----------------------------------------------------------------------===//
8+
9+
// Define ~mutex.
10+
//
11+
// On some platforms ~mutex has been made trivial and the definition is only
12+
// provided for ABI compatibility.
13+
//
14+
// In order to avoid ODR violations within libc++ itself, we need to ensure
15+
// that *nothing* sees the non-trivial mutex declaration. For this reason
16+
// we re-declare the entire class in this file instead of using
17+
// _LIBCPP_BUILDING_LIBRARY to change the definition in the headers.
18+
19+
#include "__config"
20+
#include "__threading_support"
21+
22+
#if !defined(_LIBCPP_HAS_NO_THREADS)
23+
#if _LIBCPP_ABI_VERSION == 1 || !defined(_LIBCPP_HAS_TRIVIAL_MUTEX_DESTRUCTION)
24+
#define NEEDS_MUTEX_DESTRUCTOR
25+
#endif
26+
#endif
27+
28+
_LIBCPP_BEGIN_NAMESPACE_STD
29+
30+
#ifdef NEEDS_MUTEX_DESTRUCTOR
31+
class _LIBCPP_TYPE_VIS mutex
32+
{
33+
__libcpp_mutex_t __m_ = _LIBCPP_MUTEX_INITIALIZER;
34+
35+
public:
36+
_LIBCPP_ALWAYS_INLINE _LIBCPP_INLINE_VISIBILITY
37+
constexpr mutex() = default;
38+
mutex(const mutex&) = delete;
39+
mutex& operator=(const mutex&) = delete;
40+
~mutex() noexcept;
41+
};
42+
43+
44+
mutex::~mutex() _NOEXCEPT
45+
{
46+
__libcpp_mutex_destroy(&__m_);
47+
}
48+
49+
#endif // !_LIBCPP_HAS_NO_THREADS
50+
_LIBCPP_END_NAMESPACE_STD
51+

‎libcxx/test/std/thread/thread.mutex/thread.mutex.requirements/thread.mutex.requirements.mutex/thread.mutex.class/default.pass.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,8 @@
2121

2222
int main(int, char**)
2323
{
24-
static_assert(std::is_nothrow_default_constructible<std::mutex>::value, "");
25-
std::mutex m;
26-
24+
static_assert(std::is_nothrow_default_constructible<std::mutex>::value, "");
25+
std::mutex m;
26+
((void)m);
2727
return 0;
2828
}

0 commit comments

Comments
 (0)