]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
sched_ext: Make qmap dump operation non-destructive
authorTejun Heo <tj@kernel.org>
Tue, 23 Sep 2025 19:03:26 +0000 (09:03 -1000)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Sun, 2 Nov 2025 13:15:21 +0000 (22:15 +0900)
[ Upstream commit d452972858e5cfa4262320ab74fe8f016460b96f ]

The qmap dump operation was destructively consuming queue entries while
displaying them. As dump can be triggered anytime, this can easily lead to
stalls. Add a temporary dump_store queue and modify the dump logic to pop
entries, display them, and then restore them back to the original queue.
This allows dump operations to be performed without affecting the
scheduler's queue state.

Note that if racing against new enqueues during dump, ordering can get
mixed up, but this is acceptable for debugging purposes.

Acked-by: Andrea Righi <arighi@nvidia.com>
Signed-off-by: Tejun Heo <tj@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
tools/sched_ext/scx_qmap.bpf.c

index 5d1f880d1149e76ee47d0b044dc075df62442875..e952f525599bd163501b3e5edd8013df402dd509 100644 (file)
@@ -56,7 +56,8 @@ struct qmap {
   queue1 SEC(".maps"),
   queue2 SEC(".maps"),
   queue3 SEC(".maps"),
-  queue4 SEC(".maps");
+  queue4 SEC(".maps"),
+  dump_store SEC(".maps");
 
 struct {
        __uint(type, BPF_MAP_TYPE_ARRAY_OF_MAPS);
@@ -578,11 +579,26 @@ void BPF_STRUCT_OPS(qmap_dump, struct scx_dump_ctx *dctx)
                        return;
 
                scx_bpf_dump("QMAP FIFO[%d]:", i);
+
+               /*
+                * Dump can be invoked anytime and there is no way to iterate in
+                * a non-destructive way. Pop and store in dump_store and then
+                * restore afterwards. If racing against new enqueues, ordering
+                * can get mixed up.
+                */
                bpf_repeat(4096) {
                        if (bpf_map_pop_elem(fifo, &pid))
                                break;
+                       bpf_map_push_elem(&dump_store, &pid, 0);
                        scx_bpf_dump(" %d", pid);
                }
+
+               bpf_repeat(4096) {
+                       if (bpf_map_pop_elem(&dump_store, &pid))
+                               break;
+                       bpf_map_push_elem(fifo, &pid, 0);
+               }
+
                scx_bpf_dump("\n");
        }
 }