From f0db5929ffd118d8ee8ab520da16448b491af071 Mon Sep 17 00:00:00 2001 From: Ali Ijaz Sheikh Date: Wed, 26 Sep 2018 15:58:45 -0700 Subject: [PATCH] src: ready background workers before bootstrap Make sure background workers are ready before proceeding with the bootstrap or post-bootstrap execution of any code that may trigger `process.exit()`. Fixes: https://github.com/nodejs/node/issues/23065 --- src/node_platform.cc | 31 +++++++++++++++++++++++++++++-- 1 file changed, 29 insertions(+), 2 deletions(-) diff --git a/src/node_platform.cc b/src/node_platform.cc index 1c237159f2d2e9..4d5552280318f6 100644 --- a/src/node_platform.cc +++ b/src/node_platform.cc @@ -18,10 +18,23 @@ using v8::TracingController; namespace { +struct PlatformWorkerData { + TaskQueue* task_queue; + uv_barrier_t* barrier; + int id; +}; + static void PlatformWorkerThread(void* data) { + PlatformWorkerData* worker_data = static_cast(data); + TaskQueue* pending_worker_tasks = worker_data->task_queue; TRACE_EVENT_METADATA1("__metadata", "thread_name", "name", "PlatformWorkerThread"); - TaskQueue* pending_worker_tasks = static_cast*>(data); + + if (uv_barrier_wait(worker_data->barrier) > 0) { + uv_barrier_destroy(worker_data->barrier); + delete worker_data->barrier; + worker_data->barrier = nullptr; + } while (std::unique_ptr task = pending_worker_tasks->BlockingPop()) { task->Run(); pending_worker_tasks->NotifyOfCompletion(); @@ -148,17 +161,31 @@ class WorkerThreadsTaskRunner::DelayedTaskScheduler { }; WorkerThreadsTaskRunner::WorkerThreadsTaskRunner(int thread_pool_size) { + uv_barrier_t* barrier = new uv_barrier_t; + uv_barrier_init(barrier, thread_pool_size + 1); + delayed_task_scheduler_.reset( new DelayedTaskScheduler(&pending_worker_tasks_)); threads_.push_back(delayed_task_scheduler_->Start()); + for (int i = 0; i < thread_pool_size; i++) { + // FIXME(ofrobots): need to delete upon shutdown. + PlatformWorkerData* worker_data = new PlatformWorkerData{ + &pending_worker_tasks_, barrier, i + }; std::unique_ptr t { new uv_thread_t() }; if (uv_thread_create(t.get(), PlatformWorkerThread, - &pending_worker_tasks_) != 0) { + worker_data) != 0) { break; } threads_.push_back(std::move(t)); } + + // Wait for all the worker threads to be initialized. + if (uv_barrier_wait(barrier) > 0) { + uv_barrier_destroy(barrier); + delete barrier; + } } void WorkerThreadsTaskRunner::PostTask(std::unique_ptr task) {