diff --git a/libcxx/docs/Status/Cxx2bIssues.csv b/libcxx/docs/Status/Cxx2bIssues.csv --- a/libcxx/docs/Status/Cxx2bIssues.csv +++ b/libcxx/docs/Status/Cxx2bIssues.csv @@ -19,6 +19,7 @@ "`3036 `__","``polymorphic_allocator::destroy`` is extraneous","November 2020","","" "`3171 `__","LWG2989 breaks ``directory_entry`` stream insertion","November 2020","|Complete|","14.0" "`3306 `__","``ranges::advance`` violates its preconditions","November 2020","|Complete|","14.0","|ranges|" +"`3343 `__","Ordering of calls to ``unlock()`` and ``notify_all()`` in Effects element of ``notify_all_at_thread_exit()`` should be reversed","Not Yet Adopted","|Complete|","16.0" "`3403 `__","Domain of ``ranges::ssize(E)`` doesn't ``match ranges::size(E)``","November 2020","","","|ranges|" "`3404 `__","Finish removing subrange's conversions from pair-like","November 2020","","","|ranges|" "`3405 `__","``common_view``'s converting constructor is bad, too","November 2020","|Complete|","14.0","|ranges|" diff --git a/libcxx/src/thread.cpp b/libcxx/src/thread.cpp --- a/libcxx/src/thread.cpp +++ b/libcxx/src/thread.cpp @@ -164,8 +164,8 @@ for (_Notify::iterator i = notify_.begin(), e = notify_.end(); i != e; ++i) { - i->second->unlock(); i->first->notify_all(); + i->second->unlock(); } for (_AsyncStates::iterator i = async_states_.begin(), e = async_states_.end(); i != e; ++i) diff --git a/libcxx/test/std/thread/thread.condition/notify_all_at_thread_exit.pass.cpp b/libcxx/test/std/thread/thread.condition/notify_all_at_thread_exit.pass.cpp --- a/libcxx/test/std/thread/thread.condition/notify_all_at_thread_exit.pass.cpp +++ b/libcxx/test/std/thread/thread.condition/notify_all_at_thread_exit.pass.cpp @@ -8,14 +8,12 @@ // // UNSUPPORTED: no-threads -// notify_all_at_thread_exit(...) requires move semantics to transfer the -// unique_lock. +// notify_all_at_thread_exit(...) requires move semantics to transfer the unique_lock. // UNSUPPORTED: c++03 // - -// void -// notify_all_at_thread_exit(condition_variable& cond, unique_lock lk); +// +// void notify_all_at_thread_exit(condition_variable& cond, unique_lock lk); #include #include diff --git a/libcxx/test/std/thread/thread.condition/notify_all_at_thread_exit_lwg3343.pass.cpp b/libcxx/test/std/thread/thread.condition/notify_all_at_thread_exit_lwg3343.pass.cpp new file mode 100644 --- /dev/null +++ b/libcxx/test/std/thread/thread.condition/notify_all_at_thread_exit_lwg3343.pass.cpp @@ -0,0 +1,94 @@ +//===----------------------------------------------------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// UNSUPPORTED: libcpp-has-no-threads + +// notify_all_at_thread_exit(...) requires move semantics to transfer the unique_lock. +// UNSUPPORTED: c++03 + +// This is a regression test for LWG3343. +// +// +// +// void notify_all_at_thread_exit(condition_variable& cond, unique_lock lk); + +#include "make_test_thread.h" +#include "test_macros.h" + +#include +#include +#include +#include +#include +#include + +union X { + X() : cv_() {} + ~X() {} + std::condition_variable cv_; + unsigned char bytes_[sizeof(std::condition_variable)]; +}; + +void test() +{ + constexpr int N = 3; + + X x; + std::mutex m; + int threads_active = N; + + for (int i = 0; i < N; ++i) { + std::thread t = support::make_test_thread([&] { + // Signal thread completion + std::unique_lock lk(m); + --threads_active; + std::notify_all_at_thread_exit(x.cv_, std::move(lk)); + std::this_thread::sleep_for(std::chrono::milliseconds(1)); + }); + t.detach(); + } + + // Wait until all threads complete, i.e. until they've all + // decremented `threads_active` and then unlocked `m` at thread exit. + // It is possible that this `wait` may spuriously wake up, + // but it won't be able to continue until the last thread + // unlocks `m`. + { + std::unique_lock lk(m); + x.cv_.wait(lk, [&]() { return threads_active == 0; }); + } + + // Destroy the condition_variable and shred the bytes. + // Simulate reusing the memory for something else. + x.cv_.~condition_variable(); + for (unsigned char& c : x.bytes_) { + c = 0xcd; + } + + DoNotOptimize(x.bytes_); + + // Check that the bytes still have the same value we just wrote to them. + // If any thread wrongly unlocked `m` before calling cv.notify_all(), and + // cv.notify_all() writes to the memory of the cv, then we have a chance + // to detect the problem here. + int sum = 0; + for (unsigned char c : x.bytes_) { + sum += c; + } + DoNotOptimize(sum); + assert(sum == (0xcd * sizeof(std::condition_variable))); +} + +int main(int, char**) +{ + for (int i = 0; i < 10000; ++i) { + test(); + } + + return 0; +}