From a80bc914fee11e0152efab6837517efa580415a8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Martin=20Kr=C3=B6ning?= Date: Wed, 13 Sep 2023 22:29:52 +0200 Subject: [PATCH] fix(task): don't access scheduler from BlockedTaskQueue::wakeup_task MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Martin Kröning --- src/scheduler/mod.rs | 17 ++++++++--- src/scheduler/task.rs | 65 ++++++++++++++++++++++++------------------- 2 files changed, 49 insertions(+), 33 deletions(-) diff --git a/src/scheduler/mod.rs b/src/scheduler/mod.rs index af687a65d4..03eae19910 100644 --- a/src/scheduler/mod.rs +++ b/src/scheduler/mod.rs @@ -293,19 +293,27 @@ impl PerCoreScheduler { pub fn handle_waiting_tasks(&mut self) { without_interrupts(|| { crate::executor::run(); - self.blocked_tasks.handle_waiting_tasks(); + for task in self.blocked_tasks.handle_waiting_tasks() { + self.ready_queue.push(task); + } }); } #[cfg(not(feature = "smp"))] pub fn custom_wakeup(&mut self, task: TaskHandle) { - without_interrupts(|| self.blocked_tasks.custom_wakeup(task)); + without_interrupts(|| { + let task = self.blocked_tasks.custom_wakeup(task); + self.ready_queue.push(task); + }); } #[cfg(feature = "smp")] pub fn custom_wakeup(&mut self, task: TaskHandle) { if task.get_core_id() == self.core_id { - without_interrupts(|| self.blocked_tasks.custom_wakeup(task)); + without_interrupts(|| { + let task = self.blocked_tasks.custom_wakeup(task); + self.ready_queue.push(task); + }); } else { get_scheduler_input(task.get_core_id()) .lock() @@ -447,7 +455,8 @@ impl PerCoreScheduler { let mut input_locked = CoreLocal::get().scheduler_input.lock(); while let Some(task) = input_locked.wakeup_tasks.pop_front() { - self.blocked_tasks.custom_wakeup(task); + let task = self.blocked_tasks.custom_wakeup(task); + self.ready_queue.push(task); } while let Some(new_task) = input_locked.new_tasks.pop_front() { diff --git a/src/scheduler/task.rs b/src/scheduler/task.rs index 755fbf71da..7078e9a3f0 100644 --- a/src/scheduler/task.rs +++ b/src/scheduler/task.rs @@ -1,6 +1,7 @@ use alloc::boxed::Box; use alloc::collections::{LinkedList, VecDeque}; use alloc::rc::Rc; +use alloc::vec::Vec; use core::cell::RefCell; use core::cmp::Ordering; use core::fmt; @@ -463,31 +464,26 @@ impl BlockedTaskQueue { } fn wakeup_task(task: Rc>) { - { - let mut borrowed = task.borrow_mut(); - debug!( - "Waking up task {} on core {}", - borrowed.id, borrowed.core_id - ); - - assert!( - borrowed.core_id == core_id(), - "Try to wake up task {} on the wrong core {} != {}", - borrowed.id, - borrowed.core_id, - core_id() - ); + let mut borrowed = task.borrow_mut(); + debug!( + "Waking up task {} on core {}", + borrowed.id, borrowed.core_id + ); - assert!( - borrowed.status == TaskStatus::Blocked, - "Trying to wake up task {} which is not blocked", - borrowed.id - ); - borrowed.status = TaskStatus::Ready; - } + assert!( + borrowed.core_id == core_id(), + "Try to wake up task {} on the wrong core {} != {}", + borrowed.id, + borrowed.core_id, + core_id() + ); - // Add the task to the ready queue. - core_scheduler().ready_queue.push(task); + assert!( + borrowed.status == TaskStatus::Blocked, + "Trying to wake up task {} which is not blocked", + borrowed.id + ); + borrowed.status = TaskStatus::Ready; } #[cfg(any(feature = "tcp", feature = "udp"))] @@ -560,7 +556,7 @@ impl BlockedTaskQueue { } /// Manually wake up a blocked task. - pub fn custom_wakeup(&mut self, task: TaskHandle) { + pub fn custom_wakeup(&mut self, task: TaskHandle) -> Rc> { let mut first_task = true; let mut cursor = self.list.cursor_front_mut(); @@ -574,8 +570,8 @@ impl BlockedTaskQueue { // Loop through all blocked tasks to find it. while let Some(node) = cursor.current() { if node.task.borrow().id == task.get_id() { - // Remove it from the list of blocked tasks and wake it up. - Self::wakeup_task(node.task.clone()); + // Remove it from the list of blocked tasks. + let task_ref = node.task.clone(); cursor.remove_current(); // If this is the first task, adjust the One-Shot Timer to fire at the @@ -609,7 +605,10 @@ impl BlockedTaskQueue { ); } - return; + // Wake it up. + Self::wakeup_task(task_ref.clone()); + + return task_ref; } first_task = false; @@ -623,7 +622,7 @@ impl BlockedTaskQueue { /// /// Should be called by the One-Shot Timer interrupt handler when the wakeup time for /// at least one task has elapsed. - pub fn handle_waiting_tasks(&mut self) { + pub fn handle_waiting_tasks(&mut self) -> Vec>> { // Get the current time. let time = arch::processor::get_timer_ticks(); @@ -636,6 +635,8 @@ impl BlockedTaskQueue { } } + let mut tasks = vec![]; + // Loop through all blocked tasks. let mut cursor = self.list.cursor_front_mut(); while let Some(node) = cursor.current() { @@ -647,7 +648,7 @@ impl BlockedTaskQueue { } // Otherwise, this task has elapsed, so remove it from the list and wake it up. - Self::wakeup_task(node.task.clone()); + tasks.push(node.task.clone()); cursor.remove_current(); } @@ -675,5 +676,11 @@ impl BlockedTaskQueue { .current() .map_or_else(|| None, |node| node.wakeup_time), ); + + for task in tasks.iter().cloned() { + Self::wakeup_task(task); + } + + tasks } }