summaryrefslogtreecommitdiffstats
path: root/llvm/unittests/Support/ThreadPool.cpp
diff options
context:
space:
mode:
authorMehdi Amini <mehdi.amini@apple.com>2015-12-15 00:59:19 +0000
committerMehdi Amini <mehdi.amini@apple.com>2015-12-15 00:59:19 +0000
commit33a7ea4b9a1a81fd5b2b3036940bcfc38d438830 (patch)
tree46aa058b2e65870e31dbce56fcd1df4863fb3a2e /llvm/unittests/Support/ThreadPool.cpp
parentcec6b45aec8ac058b2123c1a54b4d96f23b6c1a9 (diff)
downloadbcm5719-llvm-33a7ea4b9a1a81fd5b2b3036940bcfc38d438830.tar.gz
bcm5719-llvm-33a7ea4b9a1a81fd5b2b3036940bcfc38d438830.zip
Add a C++11 ThreadPool implementation in LLVM
This is a very simple implementation of a thread pool using C++11 thread. It accepts any std::function<void()> for asynchronous execution. Individual task can be synchronize using the returned future, or the client can block on the full queue completion. In case LLVM is configured with Threading disabled, it falls back to sequential execution using std::async with launch:deferred. This is intended to support parallelism for ThinLTO processing in linker plugin, but is generic enough for any other uses. This is a recommit of r255444 ; trying to workaround a bug in the MSVC 2013 standard library. I think I was hit by: http://connect.microsoft.com/VisualStudio/feedbackdetail/view/791185/std-packaged-task-t-where-t-is-void-or-a-reference-class-are-not-movable Recommit of r255589, trying to please g++ as well. Differential Revision: http://reviews.llvm.org/D15464 From: mehdi_amini <mehdi_amini@91177308-0d34-0410-b5e6-96231b3b80d8> llvm-svn: 255593
Diffstat (limited to 'llvm/unittests/Support/ThreadPool.cpp')
-rw-r--r--llvm/unittests/Support/ThreadPool.cpp91
1 files changed, 91 insertions, 0 deletions
diff --git a/llvm/unittests/Support/ThreadPool.cpp b/llvm/unittests/Support/ThreadPool.cpp
new file mode 100644
index 00000000000..d36341e425d
--- /dev/null
+++ b/llvm/unittests/Support/ThreadPool.cpp
@@ -0,0 +1,91 @@
+//========- unittests/Support/ThreadPools.cpp - ThreadPools.h tests --========//
+//
+// The LLVM Compiler Infrastructure
+//
+// This file is distributed under the University of Illinois Open Source
+// License. See LICENSE.TXT for details.
+//
+//===----------------------------------------------------------------------===//
+
+#include "llvm/Support/ThreadPool.h"
+
+#include "llvm/ADT/STLExtras.h"
+
+#include "gtest/gtest.h"
+
+using namespace llvm;
+using namespace std::chrono;
+
+/// Try best to make this thread not progress faster than the main thread
+static void yield() {
+#ifdef LLVM_ENABLE_THREADS
+ std::this_thread::yield();
+#endif
+ std::this_thread::sleep_for(milliseconds(200));
+#ifdef LLVM_ENABLE_THREADS
+ std::this_thread::yield();
+#endif
+}
+
+TEST(ThreadPoolTest, AsyncBarrier) {
+ // test that async & barrier work together properly.
+
+ std::atomic_int checked_in{0};
+
+ ThreadPool Pool;
+ for (size_t i = 0; i < 5; ++i) {
+ Pool.async([&checked_in, i] {
+ yield();
+ ++checked_in;
+ });
+ }
+ ASSERT_EQ(0, checked_in);
+ Pool.wait();
+ ASSERT_EQ(5, checked_in);
+}
+
+TEST(ThreadPoolTest, Async) {
+ ThreadPool Pool;
+ std::atomic_int i{0};
+ // sleep here just to ensure that the not-equal is correct.
+ Pool.async([&i] {
+ yield();
+ ++i;
+ });
+ Pool.async([&i] { ++i; });
+ ASSERT_NE(2, i.load());
+ Pool.wait();
+ ASSERT_EQ(2, i.load());
+}
+
+TEST(ThreadPoolTest, GetFuture) {
+ ThreadPool Pool;
+ std::atomic_int i{0};
+ // sleep here just to ensure that the not-equal is correct.
+ Pool.async([&i] {
+ yield();
+ ++i;
+ });
+ // Force the future using get()
+ Pool.async([&i] { ++i; }).get();
+ ASSERT_NE(2, i.load());
+ Pool.wait();
+ ASSERT_EQ(2, i.load());
+}
+
+TEST(ThreadPoolTest, PoolDestruction) {
+ // Test that we are waiting on destruction
+ std::atomic_int checked_in{0};
+
+ {
+ ThreadPool Pool;
+ for (size_t i = 0; i < 5; ++i) {
+ Pool.async([&checked_in, i] {
+ yield();
+ ++checked_in;
+ });
+ }
+ ASSERT_EQ(0, checked_in);
+ }
+ ASSERT_EQ(5, checked_in);
+}
OpenPOWER on IntegriCloud