summaryrefslogtreecommitdiffstats
path: root/llvm/unittests/Support/ThreadPool.cpp
diff options
context:
space:
mode:
authorMehdi Amini <mehdi.amini@apple.com>2015-12-12 22:55:25 +0000
committerMehdi Amini <mehdi.amini@apple.com>2015-12-12 22:55:25 +0000
commit396abbb6f045232f92439dbce052db3177a474f9 (patch)
tree133708b122b2cee8e713bce070270b9a3b37862b /llvm/unittests/Support/ThreadPool.cpp
parentb627d9f8fcaab50cbcc39d41f98269b7389f896c (diff)
downloadbcm5719-llvm-396abbb6f045232f92439dbce052db3177a474f9.tar.gz
bcm5719-llvm-396abbb6f045232f92439dbce052db3177a474f9.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. Differential Revision: http://reviews.llvm.org/D15464 From: Mehdi Amini <mehdi.amini@apple.com> llvm-svn: 255444
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