From 873cb8ae2fc291eaffbd71e3c83d17b2f0ed7abf Mon Sep 17 00:00:00 2001 From: Weijia Jiang Date: Thu, 30 May 2024 16:45:45 +0800 Subject: [PATCH] runtime: move task out of the `lifo_slot` in `block_in_place` (#6596) --- tokio/src/runtime/scheduler/multi_thread/worker.rs | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/tokio/src/runtime/scheduler/multi_thread/worker.rs b/tokio/src/runtime/scheduler/multi_thread/worker.rs index 65851b21e82..8ef487b09fd 100644 --- a/tokio/src/runtime/scheduler/multi_thread/worker.rs +++ b/tokio/src/runtime/scheduler/multi_thread/worker.rs @@ -395,11 +395,19 @@ where let cx = maybe_cx.expect("no .is_some() == false cases above should lead here"); // Get the worker core. If none is set, then blocking is fine! - let core = match cx.core.borrow_mut().take() { + let mut core = match cx.core.borrow_mut().take() { Some(core) => core, None => return Ok(()), }; + // If we heavily call `spawn_blocking`, there might be no available thread to + // run this core. Except for the task in the lifo_slot, all tasks can be + // stolen, so we move the task out of the lifo_slot to the run_queue. + if let Some(task) = core.lifo_slot.take() { + core.run_queue + .push_back_or_overflow(task, &*cx.worker.handle, &mut core.stats); + } + // We are taking the core from the context and sending it to another // thread. take_core = true;