summaryrefslogtreecommitdiff
path: root/kernel
diff options
context:
space:
mode:
authorTejun Heo <tj@kernel.org>2024-08-07 10:17:38 -1000
committerTejun Heo <tj@kernel.org>2024-08-08 13:38:09 -1000
commit72763ea3d45c7f9fd69b825468afbf4d11c5ffc2 (patch)
tree61aa7f11ecbdca19d25c62a2ae10bcc2290a5cbb /kernel
parent2c390dda9e03d7936c492224453342d458e9bf98 (diff)
downloadlinux-72763ea3d45c7f9fd69b825468afbf4d11c5ffc2.tar.gz
linux-72763ea3d45c7f9fd69b825468afbf4d11c5ffc2.tar.bz2
linux-72763ea3d45c7f9fd69b825468afbf4d11c5ffc2.zip
sched_ext: Fix unsafe list iteration in process_ddsp_deferred_locals()
process_ddsp_deferred_locals() executes deferred direct dispatches to the local DSQs of remote CPUs. It iterates the tasks on rq->scx.ddsp_deferred_locals list, removing and calling dispatch_to_local_dsq() on each. However, the list is protected by the rq lock that can be dropped by dispatch_to_local_dsq() temporarily, so the list can be modified during the iteration, which can lead to oopses and other failures. Fix it by popping from the head of the list instead of iterating the list. Signed-off-by: Tejun Heo <tj@kernel.org> Fixes: 5b26f7b920f7 ("sched_ext: Allow SCX_DSQ_LOCAL_ON for direct dispatches") Acked-by: David Vernet <void@manifault.com>
Diffstat (limited to 'kernel')
-rw-r--r--kernel/sched/ext.c10
1 files changed, 6 insertions, 4 deletions
diff --git a/kernel/sched/ext.c b/kernel/sched/ext.c
index 60a7eb7d8a9e..9efb54172495 100644
--- a/kernel/sched/ext.c
+++ b/kernel/sched/ext.c
@@ -2726,17 +2726,19 @@ static void set_next_task_scx(struct rq *rq, struct task_struct *p, bool first)
static void process_ddsp_deferred_locals(struct rq *rq)
{
- struct task_struct *p, *tmp;
+ struct task_struct *p;
lockdep_assert_rq_held(rq);
/*
* Now that @rq can be unlocked, execute the deferred enqueueing of
* tasks directly dispatched to the local DSQs of other CPUs. See
- * direct_dispatch().
+ * direct_dispatch(). Keep popping from the head instead of using
+ * list_for_each_entry_safe() as dispatch_local_dsq() may unlock @rq
+ * temporarily.
*/
- list_for_each_entry_safe(p, tmp, &rq->scx.ddsp_deferred_locals,
- scx.dsq_list.node) {
+ while ((p = list_first_entry_or_null(&rq->scx.ddsp_deferred_locals,
+ struct task_struct, scx.dsq_list.node))) {
s32 ret;
list_del_init(&p->scx.dsq_list.node);