[PATCH] tools/sched_ext: scx_central: start timer from central dispatch

Zhao Mengmeng posted 1 patch 1 week ago
There is a newer version of this series
tools/sched_ext/scx_central.bpf.c | 60 ++++++++++++++++++++-----------
tools/sched_ext/scx_central.c     |  1 -
2 files changed, 40 insertions(+), 21 deletions(-)
[PATCH] tools/sched_ext: scx_central: start timer from central dispatch
Posted by Zhao Mengmeng 1 week ago
From: Zhao Mengmeng <zhaomengmeng@kylinos.cn>

scx_central currently assumes that ops.init() runs on the selected
central CPU and aborts otherwise. This is no longer true, as ops.init()
is invoked from the scx_enable_helper thread, which can run on any
CPU. As a result, sched_setaffinity() from userspace doesn't work, loading
scx_central can fail with:

[ 1985.319942] sched_ext: central: scx_central.bpf.c:314: init from non-central CPU
[ 1985.320317]    scx_exit+0xa3/0xd0
[ 1985.320535]    scx_bpf_error_bstr+0xbd/0x220
[ 1985.320840]    bpf_prog_3a445a8163fa8149_central_init+0x103/0x1ba
[ 1985.321073]    bpf__sched_ext_ops_init+0x40/0xa8
[ 1985.321286]    scx_root_enable_workfn+0x507/0x1650
[ 1985.321461]    kthread_worker_fn+0x260/0x940
[ 1985.321745]    kthread+0x303/0x3e0
[ 1985.321901]    ret_from_fork+0x589/0x7d0
[ 1985.322065]    ret_from_fork_asm+0x1a/0x30

DEBUG DUMP
===================================================================

central: root
scx_enable_help[134] triggered exit kind 1025:
  scx_bpf_error (scx_central.bpf.c:314: init from non-central CPU)

To fix it, initialize the BPF timer from ops.init(), but defer
bpf_timer_start() to the first dispatch on the central CPU. This preserves
the requirement that the timer be started from the central CPU when
BPF_F_TIMER_CPU_PIN is used, without depending on the CPU affinity of
the enable path.

Signed-off-by: Zhao Mengmeng <zhaomengmeng@kylinos.cn>
---
 tools/sched_ext/scx_central.bpf.c | 60 ++++++++++++++++++++-----------
 tools/sched_ext/scx_central.c     |  1 -
 2 files changed, 40 insertions(+), 21 deletions(-)

diff --git a/tools/sched_ext/scx_central.bpf.c b/tools/sched_ext/scx_central.bpf.c
index 399e8d3f8bec..b2d9b1164603 100644
--- a/tools/sched_ext/scx_central.bpf.c
+++ b/tools/sched_ext/scx_central.bpf.c
@@ -60,6 +60,7 @@ const volatile u32 nr_cpu_ids = 1;	/* !0 for veristat, set during init */
 const volatile u64 slice_ns;
 
 bool timer_pinned = true;
+bool timer_started;
 u64 nr_total, nr_locals, nr_queued, nr_lost_pids;
 u64 nr_timers, nr_dispatches, nr_mismatches, nr_retries;
 u64 nr_overflows;
@@ -179,9 +180,47 @@ static bool dispatch_to_cpu(s32 cpu)
 	return false;
 }
 
+static void start_central_timer(void)
+{
+	struct bpf_timer *timer;
+	u32 key = 0;
+	int ret;
+
+	if (likely(timer_started))
+		return;
+
+	timer = bpf_map_lookup_elem(&central_timer, &key);
+	if (!timer) {
+		scx_bpf_error("failed to lookup central timer");
+		return;
+	}
+
+	ret = bpf_timer_start(timer, TIMER_INTERVAL_NS, BPF_F_TIMER_CPU_PIN);
+	/*
+	 * BPF_F_TIMER_CPU_PIN is pretty new (>=6.7). If we're running in a
+	 * kernel which doesn't have it, bpf_timer_start() will return -EINVAL.
+	 * Retry without the PIN. This would be the perfect use case for
+	 * bpf_core_enum_value_exists() but the enum type doesn't have a name
+	 * and can't be used with bpf_core_enum_value_exists(). Oh well...
+	 */
+	if (ret == -EINVAL) {
+		timer_pinned = false;
+		ret = bpf_timer_start(timer, TIMER_INTERVAL_NS, 0);
+	}
+
+	if (ret) {
+		scx_bpf_error("bpf_timer_start failed (%d)", ret);
+		return;
+	}
+
+	timer_started = true;
+}
+
 void BPF_STRUCT_OPS(central_dispatch, s32 cpu, struct task_struct *prev)
 {
 	if (cpu == central_cpu) {
+		start_central_timer();
+
 		/* dispatch for all other CPUs first */
 		__sync_fetch_and_add(&nr_dispatches, 1);
 
@@ -310,29 +349,10 @@ int BPF_STRUCT_OPS_SLEEPABLE(central_init)
 	if (!timer)
 		return -ESRCH;
 
-	if (bpf_get_smp_processor_id() != central_cpu) {
-		scx_bpf_error("init from non-central CPU");
-		return -EINVAL;
-	}
-
 	bpf_timer_init(timer, &central_timer, CLOCK_MONOTONIC);
 	bpf_timer_set_callback(timer, central_timerfn);
 
-	ret = bpf_timer_start(timer, TIMER_INTERVAL_NS, BPF_F_TIMER_CPU_PIN);
-	/*
-	 * BPF_F_TIMER_CPU_PIN is pretty new (>=6.7). If we're running in a
-	 * kernel which doesn't have it, bpf_timer_start() will return -EINVAL.
-	 * Retry without the PIN. This would be the perfect use case for
-	 * bpf_core_enum_value_exists() but the enum type doesn't have a name
-	 * and can't be used with bpf_core_enum_value_exists(). Oh well...
-	 */
-	if (ret == -EINVAL) {
-		timer_pinned = false;
-		ret = bpf_timer_start(timer, TIMER_INTERVAL_NS, 0);
-	}
-	if (ret)
-		scx_bpf_error("bpf_timer_start failed (%d)", ret);
-	return ret;
+	return 0;
 }
 
 void BPF_STRUCT_OPS(central_exit, struct scx_exit_info *ei)
diff --git a/tools/sched_ext/scx_central.c b/tools/sched_ext/scx_central.c
index fd4c0eaa4326..f4bed0e69357 100644
--- a/tools/sched_ext/scx_central.c
+++ b/tools/sched_ext/scx_central.c
@@ -98,7 +98,6 @@ int main(int argc, char **argv)
 
 	/*
 	 * Affinitize the loading thread to the central CPU, as:
-	 * - That's where the BPF timer is first invoked in the BPF program.
 	 * - We probably don't want this user space component to take up a core
 	 *   from a task that would benefit from avoiding preemption on one of
 	 *   the tickless cores.
-- 
2.43.0
Re: [PATCH] tools/sched_ext: scx_central: start timer from central dispatch
Posted by Tejun Heo 6 days, 19 hours ago
Hello, Zhao.

On Thu, Mar 26, 2026 at 05:15:23PM +0800, Zhao Mengmeng wrote:
> From: Zhao Mengmeng <zhaomengmeng@kylinos.cn>
> 
> scx_central currently assumes that ops.init() runs on the selected
> central CPU and aborts otherwise. This is no longer true, as ops.init()
> is invoked from the scx_enable_helper thread, which can run on any
> CPU. As a result, sched_setaffinity() from userspace doesn't work, loading
> scx_central can fail with:
> 
> [ 1985.319942] sched_ext: central: scx_central.bpf.c:314: init from non-central CPU
> [ 1985.320317]    scx_exit+0xa3/0xd0
> [ 1985.320535]    scx_bpf_error_bstr+0xbd/0x220
> [ 1985.320840]    bpf_prog_3a445a8163fa8149_central_init+0x103/0x1ba
> [ 1985.321073]    bpf__sched_ext_ops_init+0x40/0xa8
> [ 1985.321286]    scx_root_enable_workfn+0x507/0x1650
> [ 1985.321461]    kthread_worker_fn+0x260/0x940
> [ 1985.321745]    kthread+0x303/0x3e0
> [ 1985.321901]    ret_from_fork+0x589/0x7d0
> [ 1985.322065]    ret_from_fork_asm+0x1a/0x30
> 
> DEBUG DUMP
> ===================================================================
> 
> central: root
> scx_enable_help[134] triggered exit kind 1025:
>   scx_bpf_error (scx_central.bpf.c:314: init from non-central CPU)
> 
> To fix it, initialize the BPF timer from ops.init(), but defer
> bpf_timer_start() to the first dispatch on the central CPU. This preserves
> the requirement that the timer be started from the central CPU when
> BPF_F_TIMER_CPU_PIN is used, without depending on the CPU affinity of
> the enable path.

Can you do the followings please?

- Remove now unnecessary sched_setaffinity() call.

- Do scx_bpf_kick_cpu(central_cpu, 0) so that we're guaranteed to enter the
  dispatch path on the central CPU immediately.

Thanks.

-- 
tejun
Re: [PATCH] tools/sched_ext: scx_central: start timer from central dispatch
Posted by Zhao Mengmeng 6 days, 19 hours ago
On 3/27/26 08:58, Tejun Heo wrote:
> Hello, Zhao.
> 
> On Thu, Mar 26, 2026 at 05:15:23PM +0800, Zhao Mengmeng wrote:
>> From: Zhao Mengmeng <zhaomengmeng@kylinos.cn>
>>
>> scx_central currently assumes that ops.init() runs on the selected
>> central CPU and aborts otherwise. This is no longer true, as ops.init()
>> is invoked from the scx_enable_helper thread, which can run on any
>> CPU. As a result, sched_setaffinity() from userspace doesn't work, loading
>> scx_central can fail with:
>>
>> [ 1985.319942] sched_ext: central: scx_central.bpf.c:314: init from non-central CPU
>> [ 1985.320317]    scx_exit+0xa3/0xd0
>> [ 1985.320535]    scx_bpf_error_bstr+0xbd/0x220
>> [ 1985.320840]    bpf_prog_3a445a8163fa8149_central_init+0x103/0x1ba
>> [ 1985.321073]    bpf__sched_ext_ops_init+0x40/0xa8
>> [ 1985.321286]    scx_root_enable_workfn+0x507/0x1650
>> [ 1985.321461]    kthread_worker_fn+0x260/0x940
>> [ 1985.321745]    kthread+0x303/0x3e0
>> [ 1985.321901]    ret_from_fork+0x589/0x7d0
>> [ 1985.322065]    ret_from_fork_asm+0x1a/0x30
>>
>> DEBUG DUMP
>> ===================================================================
>>
>> central: root
>> scx_enable_help[134] triggered exit kind 1025:
>>   scx_bpf_error (scx_central.bpf.c:314: init from non-central CPU)
>>
>> To fix it, initialize the BPF timer from ops.init(), but defer
>> bpf_timer_start() to the first dispatch on the central CPU. This preserves
>> the requirement that the timer be started from the central CPU when
>> BPF_F_TIMER_CPU_PIN is used, without depending on the CPU affinity of
>> the enable path.
> 
> Can you do the followings please?
> 
> - Remove now unnecessary sched_setaffinity() call.
> 
> - Do scx_bpf_kick_cpu(central_cpu, 0) so that we're guaranteed to enter the
>   dispatch path on the central CPU immediately.
> 
> Thanks.
> 
No Problem. Will try it.