diff options
author | Eric Fiselier <eric@efcs.ca> | 2015-06-12 00:41:34 +0000 |
---|---|---|
committer | Eric Fiselier <eric@efcs.ca> | 2015-06-12 00:41:34 +0000 |
commit | b3ec43d78aa2599e1592392844a7ec638fab7902 (patch) | |
tree | 3cd800dd82226b066717608877c112c0749afd78 /libcxx/test/std/thread/futures/futures.async/async_race.pass.cpp | |
parent | 03a9056f58e2f2d58fa8de12f836098f1488b29c (diff) | |
download | bcm5719-llvm-b3ec43d78aa2599e1592392844a7ec638fab7902.tar.gz bcm5719-llvm-b3ec43d78aa2599e1592392844a7ec638fab7902.zip |
Fix PR23293 - Do not unlock shared state before notifying consumers.
Within the shared state methods do not unlock the lock guards manually. This
could cause a race condition where the shared state is destroyed before the
method is complete.
llvm-svn: 239577
Diffstat (limited to 'libcxx/test/std/thread/futures/futures.async/async_race.pass.cpp')
-rw-r--r-- | libcxx/test/std/thread/futures/futures.async/async_race.pass.cpp | 67 |
1 files changed, 67 insertions, 0 deletions
diff --git a/libcxx/test/std/thread/futures/futures.async/async_race.pass.cpp b/libcxx/test/std/thread/futures/futures.async/async_race.pass.cpp new file mode 100644 index 00000000000..325a027132d --- /dev/null +++ b/libcxx/test/std/thread/futures/futures.async/async_race.pass.cpp @@ -0,0 +1,67 @@ +//===----------------------------------------------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is dual licensed under the MIT and the University of Illinois Open +// Source Licenses. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// UNSUPPORTED: libcpp-has-no-threads + +// <future> + +// template <class F, class... Args> +// future<typename result_of<F(Args...)>::type> +// async(F&& f, Args&&... args); + +// template <class F, class... Args> +// future<typename result_of<F(Args...)>::type> +// async(launch policy, F&& f, Args&&... args); + +// This test is designed to cause and allow TSAN to detect the race condition +// reported in PR23293. (http://llvm.org/PR23293). + +#include <future> +#include <chrono> +#include <thread> +#include <memory> +#include <cassert> + +int f_async() { + typedef std::chrono::milliseconds ms; + std::this_thread::sleep_for(ms(200)); + return 42; +} + +bool ran = false; + +int f_deferred() { + ran = true; + return 42; +} + +void test_each() { + { + std::future<int> f = std::async(f_async); + int const result = f.get(); + assert(result == 42); + } + { + std::future<int> f = std::async(std::launch::async, f_async); + int const result = f.get(); + assert(result == 42); + } + { + ran = false; + std::future<int> f = std::async(std::launch::deferred, f_deferred); + assert(ran == false); + int const result = f.get(); + assert(ran == true); + assert(result == 42); + } +} + +int main() { + for (int i=0; i < 25; ++i) test_each(); +} |