summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorAlon Zakai <alonzakai@gmail.com>2018-01-24 15:50:13 -0800
committerGitHub <noreply@github.com>2018-01-24 15:50:13 -0800
commit0ddfd3a397eefde12a2999111cbdda0e77ab5639 (patch)
tree258ebeb10d0e2165acae4351228c82a23a7053e1 /src
parent544cce0a37a124415b00a6b3a1dd2791d714a807 (diff)
downloadbinaryen-0ddfd3a397eefde12a2999111cbdda0e77ab5639.tar.gz
binaryen-0ddfd3a397eefde12a2999111cbdda0e77ab5639.tar.bz2
binaryen-0ddfd3a397eefde12a2999111cbdda0e77ab5639.zip
Threading fixes (#1377)
* threading fixes, be careful when creating the pool (more than one thread may try to) and don't create it just to check if its running in the thread constructor assertions * child threads will call ::get() - don't do initialize() under the lock
Diffstat (limited to 'src')
-rw-r--r--src/support/threads.cpp24
-rw-r--r--src/support/threads.h3
2 files changed, 21 insertions, 6 deletions
diff --git a/src/support/threads.cpp b/src/support/threads.cpp
index 31c900ceb..c4f714f8e 100644
--- a/src/support/threads.cpp
+++ b/src/support/threads.cpp
@@ -41,18 +41,19 @@ namespace wasm {
// Global thread information
+static std::mutex poolMutex;
static std::unique_ptr<ThreadPool> pool;
// Thread
Thread::Thread() {
- assert(!ThreadPool::get()->isRunning());
+ assert(!ThreadPool::isRunning());
thread = make_unique<std::thread>(mainLoop, this);
}
Thread::~Thread() {
- assert(!ThreadPool::get()->isRunning());
+ assert(!ThreadPool::isRunning());
{
std::lock_guard<std::mutex> lock(mutex);
// notify the thread that it can exit
@@ -138,9 +139,23 @@ size_t ThreadPool::getNumCores() {
}
ThreadPool* ThreadPool::get() {
- if (!pool) {
- pool = make_unique<ThreadPool>();
+ DEBUG_POOL("::get()\n");
+ bool created = false;
+ {
+ // lock on the creation
+ std::lock_guard<std::mutex> lock(poolMutex);
+ if (!pool) {
+ DEBUG_POOL("::get() creating\n");
+ created = true;
+ pool = make_unique<ThreadPool>();
+ }
+ }
+ if (created) {
+ // if we created it here, do the initialization too. this
+ // is outside of the mutex, as we create child threads who
+ // will call ::get() themselves
pool->initialize(getNumCores());
+ DEBUG_POOL("::get() created\n");
}
return pool.get();
}
@@ -178,6 +193,7 @@ size_t ThreadPool::size() {
}
bool ThreadPool::isRunning() {
+ DEBUG_POOL("check if running\n");
return pool && pool->running;
}
diff --git a/src/support/threads.h b/src/support/threads.h
index ec726370e..0ec109e4d 100644
--- a/src/support/threads.h
+++ b/src/support/threads.h
@@ -83,8 +83,7 @@ public:
// Get the number of cores we can use.
static size_t getNumCores();
- // Get the singleton threadpool. This can return null
- // if there is just one thread available.
+ // Get the singleton threadpool.
static ThreadPool* get();
// Execute a bunch of tasks by the pool. This calls