Commit 9d7e8ea3 authored by Andreas Haas's avatar Andreas Haas Committed by V8 LUCI CQ

[d8] Clean up DefaultForegroundTaskRunner::Terminate

Follow-up to post-submit comments in
https://chromium-review.googlesource.com/c/v8/v8/+/3782796

Bug: chromium:1346250, v8:12926
Change-Id: I09a8601c600b24fbc92489224ad69602e557bf7f
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3784604
Commit-Queue: Andreas Haas <ahaas@chromium.org>
Reviewed-by: 's avatarClemens Backes <clemensb@chromium.org>
Reviewed-by: 's avatarMichael Lippautz <mlippautz@chromium.org>
Cr-Commit-Position: refs/heads/main@{#82154}
parent 9afba3ff
...@@ -27,39 +27,24 @@ DefaultForegroundTaskRunner::DefaultForegroundTaskRunner( ...@@ -27,39 +27,24 @@ DefaultForegroundTaskRunner::DefaultForegroundTaskRunner(
: idle_task_support_(idle_task_support), time_function_(time_function) {} : idle_task_support_(idle_task_support), time_function_(time_function) {}
void DefaultForegroundTaskRunner::Terminate() { void DefaultForegroundTaskRunner::Terminate() {
{
base::MutexGuard guard(&lock_);
terminated_ = true;
}
// Drain the task queues. // Drain the task queues.
// We have to do this lock dance here to avoid a lock order inversion warning // We make sure to delete tasks outside the TaskRunner lock, to avoid
// of TSAN: during initialization, the WasmEngine lock gets taken before the // potential deadlocks.
// TaskRunner lock. The TaskRunner lock is taken during WasmEngine std::deque<TaskQueueEntry> obsolete_tasks;
// initialization when the ForegroundTaskRunner gets acquire. During shutdown, std::priority_queue<DelayedEntry, std::vector<DelayedEntry>,
// the TaskRunner lock gets taken before the WasmEngine lock. The WasmEngine DelayedEntryCompare>
// lock gets taken in the destructor of the LogCode task. obsolete_delayed_tasks;
// In the code below we pop one task at a time from the task queue while std::queue<std::unique_ptr<IdleTask>> obsolete_idle_tasks;
// holding the TaskRunner lock, but delete the task only after releasing the
// TaskRunner lock. Thereby we avoid holding the TaskRunner lock when the
// destructor of the task may take other locks.
while (true) {
std::unique_ptr<Task> task;
{ {
base::MutexGuard guard(&lock_); base::MutexGuard guard(&lock_);
if (task_queue_.empty()) break; terminated_ = true;
task = std::move(task_queue_.front().second); task_queue_.swap(obsolete_tasks);
task_queue_.pop_front(); delayed_task_queue_.swap(obsolete_delayed_tasks);
} idle_task_queue_.swap(obsolete_idle_tasks);
// Reset the unique_ptr just for readability, to show that the task gets
// deallocated here without holding the lock.
task.reset();
} }
while (!obsolete_tasks.empty()) obsolete_tasks.pop_front();
// TODO(v8): If it ever becomes necessary, do the same lock dance for delayed while (!obsolete_delayed_tasks.empty()) obsolete_delayed_tasks.pop();
// tasks and idle tasks as above. while (!obsolete_idle_tasks.empty()) obsolete_idle_tasks.pop();
base::MutexGuard guard(&lock_);
while (!delayed_task_queue_.empty()) delayed_task_queue_.pop();
while (!idle_task_queue_.empty()) idle_task_queue_.pop();
} }
void DefaultForegroundTaskRunner::PostTaskLocked(std::unique_ptr<Task> task, void DefaultForegroundTaskRunner::PostTaskLocked(std::unique_ptr<Task> task,
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment