]> www.infradead.org Git - users/hch/configfs.git/commitdiff
tick: Start centralizing tick related CPU hotplug operations
authorFrederic Weisbecker <frederic@kernel.org>
Sun, 25 Feb 2024 22:54:59 +0000 (23:54 +0100)
committerThomas Gleixner <tglx@linutronix.de>
Mon, 26 Feb 2024 10:37:31 +0000 (11:37 +0100)
During the CPU offlining process, the various timer tick features are
shut down from scattered places, sometimes from teardown callbacks on
stop machine, sometimes through explicit calls, sometimes from the
control CPU after the CPU died. The reason why these shutdown operations
are spread around is not always clear and it makes the tick lifecycle
hard to follow.

The tick should be shut down in order from highest to lowest level:

On stop machine from the dying CPU (high-level):

 1) Hand-over the timekeeping duty (tick_handover_do_timer())
 2) Cancel the tick implementation called by the clockevent callback
    (tick_cancel_sched_timer())
 3) Shutdown broadcasting (tick_offline_cpu() / tick_broadcast_offline())

On stop machine from the dying CPU (low-level):

 4) Shutdown clockevents drivers (CPUHP_AP_*_TIMER_STARTING states)

From the control CPU after the CPU died (low-level):

 5) Shutdown/unregister/cleanup clockevents for the dead CPU
    (tick_cleanup_dead_cpu())

Instead the current order is 2, 4 (both from CPU hotplug states), then
1 and 3 through direct calls. This layout and order don't make much
sense. The operations 1, 2, 3 should be gathered together and in order.

Sort this situation with creating a new TICK shut-down CPU hotplug state
and start with introducing the timekeeping duty hand-over there. The
state must precede hrtimers migration because the tick hrtimer will be
stopped from it in a further patch.

Signed-off-by: Frederic Weisbecker <frederic@kernel.org>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Reviewed-by: Thomas Gleixner <tglx@linutronix.de>
Link: https://lore.kernel.org/r/20240225225508.11587-8-frederic@kernel.org
include/linux/cpuhotplug.h
include/linux/tick.h
kernel/cpu.c
kernel/time/tick-common.c

index 7651904c6db56113bbb0a36d76b29cf2f1710f1e..35e78ddb2b37c70357955bb6087a1d3899adb2cd 100644 (file)
@@ -184,6 +184,7 @@ enum cpuhp_state {
        CPUHP_AP_ARM64_ISNDEP_STARTING,
        CPUHP_AP_SMPCFD_DYING,
        CPUHP_AP_HRTIMERS_DYING,
+       CPUHP_AP_TICK_DYING,
        CPUHP_AP_X86_TBOOT_DYING,
        CPUHP_AP_ARM_CACHE_B15_RAC_DYING,
        CPUHP_AP_ONLINE,
index 716d17f31c4539d18bcdd94c993ef47bf1c42368..afff4c207bd8dad8e03546a8a013fa8af46619c4 100644 (file)
@@ -19,16 +19,20 @@ extern void __init tick_init(void);
 extern void tick_suspend_local(void);
 /* Should be core only, but XEN resume magic and ARM BL switcher require it */
 extern void tick_resume_local(void);
-extern void tick_handover_do_timer(void);
 extern void tick_cleanup_dead_cpu(int cpu);
 #else /* CONFIG_GENERIC_CLOCKEVENTS */
 static inline void tick_init(void) { }
 static inline void tick_suspend_local(void) { }
 static inline void tick_resume_local(void) { }
-static inline void tick_handover_do_timer(void) { }
 static inline void tick_cleanup_dead_cpu(int cpu) { }
 #endif /* !CONFIG_GENERIC_CLOCKEVENTS */
 
+#if defined(CONFIG_GENERIC_CLOCKEVENTS) && defined(CONFIG_HOTPLUG_CPU)
+extern int tick_cpu_dying(unsigned int cpu);
+#else
+#define tick_cpu_dying NULL
+#endif
+
 #if defined(CONFIG_GENERIC_CLOCKEVENTS) && defined(CONFIG_SUSPEND)
 extern void tick_freeze(void);
 extern void tick_unfreeze(void);
index e6ec3ba4950b48c0afa6665aba2824456dc409f5..263508073da896d17f1ce90906511c4e9f3c077a 100644 (file)
@@ -1324,8 +1324,6 @@ static int take_cpu_down(void *_param)
         */
        cpuhp_invoke_callback_range_nofail(false, cpu, st, target);
 
-       /* Give up timekeeping duties */
-       tick_handover_do_timer();
        /* Remove CPU from timer broadcasting */
        tick_offline_cpu(cpu);
        /* Park the stopper thread */
@@ -2205,7 +2203,11 @@ static struct cpuhp_step cpuhp_hp_states[] = {
                .startup.single         = NULL,
                .teardown.single        = hrtimers_cpu_dying,
        },
-
+       [CPUHP_AP_TICK_DYING] = {
+               .name                   = "tick:dying",
+               .startup.single         = NULL,
+               .teardown.single        = tick_cpu_dying,
+       },
        /* Entry state on starting. Interrupts enabled from here on. Transient
         * state for synchronsization */
        [CPUHP_AP_ONLINE] = {
index 0084e1ae25833af10426a3736301533d1a84f9ab..a89ef450fda7b5e01f37c56ee7daed3689ea6b7d 100644 (file)
@@ -397,15 +397,20 @@ EXPORT_SYMBOL_GPL(tick_broadcast_oneshot_control);
 
 #ifdef CONFIG_HOTPLUG_CPU
 /*
- * Transfer the do_timer job away from a dying cpu.
- *
- * Called with interrupts disabled. No locking required. If
- * tick_do_timer_cpu is owned by this cpu, nothing can change it.
+ * Stop the tick and transfer the timekeeping job away from a dying cpu.
  */
-void tick_handover_do_timer(void)
+int tick_cpu_dying(unsigned int dying_cpu)
 {
-       if (tick_do_timer_cpu == smp_processor_id())
+       /*
+        * If the current CPU is the timekeeper, it's the only one that
+        * can safely hand over its duty. Also all online CPUs are in
+        * stop machine, guaranteed not to be idle, therefore it's safe
+        * to pick any online successor.
+        */
+       if (tick_do_timer_cpu == dying_cpu)
                tick_do_timer_cpu = cpumask_first(cpu_online_mask);
+
+       return 0;
 }
 
 /*