Commit 5475abbd authored by Thomas Gleixner's avatar Thomas Gleixner Committed by Peter Zijlstra

x86/smpboot: Remove the CPU0 hotplug kludge

This was introduced with commit e1c467e6 ("x86, hotplug: Wake up CPU0
via NMI instead of INIT, SIPI, SIPI") to eventually support physical
hotplug of CPU0:

 "We'll change this code in the future to wake up hard offlined CPU0 if
  real platform and request are available."

11 years later this has not happened and physical hotplug is not officially
supported. Remove the cruft.
Signed-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
Signed-off-by: default avatarPeter Zijlstra (Intel) <peterz@infradead.org>
Tested-by: default avatarMichael Kelley <mikelley@microsoft.com>
Tested-by: default avatarOleksandr Natalenko <oleksandr@natalenko.name>
Tested-by: Helge Deller <deller@gmx.de> # parisc
Tested-by: Guilherme G. Piccoli <gpiccoli@igalia.com> # Steam Deck
Link: https://lore.kernel.org/r/20230512205255.768845190@linutronix.de
parent e59e74dc
...@@ -377,7 +377,6 @@ extern struct apic *__apicdrivers[], *__apicdrivers_end[]; ...@@ -377,7 +377,6 @@ extern struct apic *__apicdrivers[], *__apicdrivers_end[];
* APIC functionality to boot other CPUs - only used on SMP: * APIC functionality to boot other CPUs - only used on SMP:
*/ */
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
extern int wakeup_secondary_cpu_via_nmi(int apicid, unsigned long start_eip);
extern int lapic_can_unplug_cpu(void); extern int lapic_can_unplug_cpu(void);
#endif #endif
......
...@@ -130,7 +130,6 @@ void native_play_dead(void); ...@@ -130,7 +130,6 @@ void native_play_dead(void);
void play_dead_common(void); void play_dead_common(void);
void wbinvd_on_cpu(int cpu); void wbinvd_on_cpu(int cpu);
int wbinvd_on_all_cpus(void); int wbinvd_on_all_cpus(void);
void cond_wakeup_cpu0(void);
void native_smp_send_reschedule(int cpu); void native_smp_send_reschedule(int cpu);
void native_send_call_func_ipi(const struct cpumask *mask); void native_send_call_func_ipi(const struct cpumask *mask);
......
...@@ -216,9 +216,6 @@ static void ap_calibrate_delay(void) ...@@ -216,9 +216,6 @@ static void ap_calibrate_delay(void)
cpu_data(smp_processor_id()).loops_per_jiffy = loops_per_jiffy; cpu_data(smp_processor_id()).loops_per_jiffy = loops_per_jiffy;
} }
static int cpu0_logical_apicid;
static int enable_start_cpu0;
/* /*
* Activate a secondary processor. * Activate a secondary processor.
*/ */
...@@ -241,8 +238,6 @@ static void notrace start_secondary(void *unused) ...@@ -241,8 +238,6 @@ static void notrace start_secondary(void *unused)
x86_cpuinit.early_percpu_clock_init(); x86_cpuinit.early_percpu_clock_init();
smp_callin(); smp_callin();
enable_start_cpu0 = 0;
/* otherwise gcc will move up smp_processor_id before the cpu_init */ /* otherwise gcc will move up smp_processor_id before the cpu_init */
barrier(); barrier();
/* Check TSC synchronization with the control CPU: */ /* Check TSC synchronization with the control CPU: */
...@@ -410,7 +405,7 @@ void smp_store_cpu_info(int id) ...@@ -410,7 +405,7 @@ void smp_store_cpu_info(int id)
c->cpu_index = id; c->cpu_index = id;
/* /*
* During boot time, CPU0 has this setup already. Save the info when * During boot time, CPU0 has this setup already. Save the info when
* bringing up AP or offlined CPU0. * bringing up an AP.
*/ */
identify_secondary_cpu(c); identify_secondary_cpu(c);
c->initialized = true; c->initialized = true;
...@@ -807,51 +802,14 @@ static void __init smp_quirk_init_udelay(void) ...@@ -807,51 +802,14 @@ static void __init smp_quirk_init_udelay(void)
} }
/* /*
* Poke the other CPU in the eye via NMI to wake it up. Remember that the normal * Wake up AP by INIT, INIT, STARTUP sequence.
* INIT, INIT, STARTUP sequence will reset the chip hard for us, and this
* won't ... remember to clear down the APIC, etc later.
*/ */
int static int wakeup_secondary_cpu_via_init(int phys_apicid, unsigned long start_eip)
wakeup_secondary_cpu_via_nmi(int apicid, unsigned long start_eip)
{
u32 dm = apic->dest_mode_logical ? APIC_DEST_LOGICAL : APIC_DEST_PHYSICAL;
unsigned long send_status, accept_status = 0;
int maxlvt;
/* Target chip */
/* Boot on the stack */
/* Kick the second */
apic_icr_write(APIC_DM_NMI | dm, apicid);
pr_debug("Waiting for send to finish...\n");
send_status = safe_apic_wait_icr_idle();
/*
* Give the other CPU some time to accept the IPI.
*/
udelay(200);
if (APIC_INTEGRATED(boot_cpu_apic_version)) {
maxlvt = lapic_get_maxlvt();
if (maxlvt > 3) /* Due to the Pentium erratum 3AP. */
apic_write(APIC_ESR, 0);
accept_status = (apic_read(APIC_ESR) & 0xEF);
}
pr_debug("NMI sent\n");
if (send_status)
pr_err("APIC never delivered???\n");
if (accept_status)
pr_err("APIC delivery error (%lx)\n", accept_status);
return (send_status | accept_status);
}
static int
wakeup_secondary_cpu_via_init(int phys_apicid, unsigned long start_eip)
{ {
unsigned long send_status = 0, accept_status = 0; unsigned long send_status = 0, accept_status = 0;
int maxlvt, num_starts, j; int maxlvt, num_starts, j;
preempt_disable();
maxlvt = lapic_get_maxlvt(); maxlvt = lapic_get_maxlvt();
/* /*
...@@ -957,6 +915,7 @@ wakeup_secondary_cpu_via_init(int phys_apicid, unsigned long start_eip) ...@@ -957,6 +915,7 @@ wakeup_secondary_cpu_via_init(int phys_apicid, unsigned long start_eip)
if (accept_status) if (accept_status)
pr_err("APIC delivery error (%lx)\n", accept_status); pr_err("APIC delivery error (%lx)\n", accept_status);
preempt_enable();
return (send_status | accept_status); return (send_status | accept_status);
} }
...@@ -997,67 +956,6 @@ static void announce_cpu(int cpu, int apicid) ...@@ -997,67 +956,6 @@ static void announce_cpu(int cpu, int apicid)
node, cpu, apicid); node, cpu, apicid);
} }
static int wakeup_cpu0_nmi(unsigned int cmd, struct pt_regs *regs)
{
int cpu;
cpu = smp_processor_id();
if (cpu == 0 && !cpu_online(cpu) && enable_start_cpu0)
return NMI_HANDLED;
return NMI_DONE;
}
/*
* Wake up AP by INIT, INIT, STARTUP sequence.
*
* Instead of waiting for STARTUP after INITs, BSP will execute the BIOS
* boot-strap code which is not a desired behavior for waking up BSP. To
* void the boot-strap code, wake up CPU0 by NMI instead.
*
* This works to wake up soft offlined CPU0 only. If CPU0 is hard offlined
* (i.e. physically hot removed and then hot added), NMI won't wake it up.
* We'll change this code in the future to wake up hard offlined CPU0 if
* real platform and request are available.
*/
static int
wakeup_cpu_via_init_nmi(int cpu, unsigned long start_ip, int apicid,
int *cpu0_nmi_registered)
{
int id;
int boot_error;
preempt_disable();
/*
* Wake up AP by INIT, INIT, STARTUP sequence.
*/
if (cpu) {
boot_error = wakeup_secondary_cpu_via_init(apicid, start_ip);
goto out;
}
/*
* Wake up BSP by nmi.
*
* Register a NMI handler to help wake up CPU0.
*/
boot_error = register_nmi_handler(NMI_LOCAL,
wakeup_cpu0_nmi, 0, "wake_cpu0");
if (!boot_error) {
enable_start_cpu0 = 1;
*cpu0_nmi_registered = 1;
id = apic->dest_mode_logical ? cpu0_logical_apicid : apicid;
boot_error = wakeup_secondary_cpu_via_nmi(id, start_ip);
}
out:
preempt_enable();
return boot_error;
}
int common_cpu_up(unsigned int cpu, struct task_struct *idle) int common_cpu_up(unsigned int cpu, struct task_struct *idle)
{ {
int ret; int ret;
...@@ -1086,8 +984,7 @@ int common_cpu_up(unsigned int cpu, struct task_struct *idle) ...@@ -1086,8 +984,7 @@ int common_cpu_up(unsigned int cpu, struct task_struct *idle)
* Returns zero if CPU booted OK, else error code from * Returns zero if CPU booted OK, else error code from
* ->wakeup_secondary_cpu. * ->wakeup_secondary_cpu.
*/ */
static int do_boot_cpu(int apicid, int cpu, struct task_struct *idle, static int do_boot_cpu(int apicid, int cpu, struct task_struct *idle)
int *cpu0_nmi_registered)
{ {
/* start_ip had better be page-aligned! */ /* start_ip had better be page-aligned! */
unsigned long start_ip = real_mode_header->trampoline_start; unsigned long start_ip = real_mode_header->trampoline_start;
...@@ -1120,7 +1017,6 @@ static int do_boot_cpu(int apicid, int cpu, struct task_struct *idle, ...@@ -1120,7 +1017,6 @@ static int do_boot_cpu(int apicid, int cpu, struct task_struct *idle,
* This grunge runs the startup process for * This grunge runs the startup process for
* the targeted processor. * the targeted processor.
*/ */
if (x86_platform.legacy.warm_reset) { if (x86_platform.legacy.warm_reset) {
pr_debug("Setting warm reset code and vector.\n"); pr_debug("Setting warm reset code and vector.\n");
...@@ -1149,15 +1045,14 @@ static int do_boot_cpu(int apicid, int cpu, struct task_struct *idle, ...@@ -1149,15 +1045,14 @@ static int do_boot_cpu(int apicid, int cpu, struct task_struct *idle,
* - Use a method from the APIC driver if one defined, with wakeup * - Use a method from the APIC driver if one defined, with wakeup
* straight to 64-bit mode preferred over wakeup to RM. * straight to 64-bit mode preferred over wakeup to RM.
* Otherwise, * Otherwise,
* - Use an INIT boot APIC message for APs or NMI for BSP. * - Use an INIT boot APIC message
*/ */
if (apic->wakeup_secondary_cpu_64) if (apic->wakeup_secondary_cpu_64)
boot_error = apic->wakeup_secondary_cpu_64(apicid, start_ip); boot_error = apic->wakeup_secondary_cpu_64(apicid, start_ip);
else if (apic->wakeup_secondary_cpu) else if (apic->wakeup_secondary_cpu)
boot_error = apic->wakeup_secondary_cpu(apicid, start_ip); boot_error = apic->wakeup_secondary_cpu(apicid, start_ip);
else else
boot_error = wakeup_cpu_via_init_nmi(cpu, start_ip, apicid, boot_error = wakeup_secondary_cpu_via_init(apicid, start_ip);
cpu0_nmi_registered);
if (!boot_error) { if (!boot_error) {
/* /*
...@@ -1206,9 +1101,8 @@ static int do_boot_cpu(int apicid, int cpu, struct task_struct *idle, ...@@ -1206,9 +1101,8 @@ static int do_boot_cpu(int apicid, int cpu, struct task_struct *idle,
int native_cpu_up(unsigned int cpu, struct task_struct *tidle) int native_cpu_up(unsigned int cpu, struct task_struct *tidle)
{ {
int apicid = apic->cpu_present_to_apicid(cpu); int apicid = apic->cpu_present_to_apicid(cpu);
int cpu0_nmi_registered = 0;
unsigned long flags; unsigned long flags;
int err, ret = 0; int err;
lockdep_assert_irqs_enabled(); lockdep_assert_irqs_enabled();
...@@ -1247,11 +1141,10 @@ int native_cpu_up(unsigned int cpu, struct task_struct *tidle) ...@@ -1247,11 +1141,10 @@ int native_cpu_up(unsigned int cpu, struct task_struct *tidle)
if (err) if (err)
return err; return err;
err = do_boot_cpu(apicid, cpu, tidle, &cpu0_nmi_registered); err = do_boot_cpu(apicid, cpu, tidle);
if (err) { if (err) {
pr_err("do_boot_cpu failed(%d) to wakeup CPU#%u\n", err, cpu); pr_err("do_boot_cpu failed(%d) to wakeup CPU#%u\n", err, cpu);
ret = -EIO; return err;
goto unreg_nmi;
} }
/* /*
...@@ -1267,15 +1160,7 @@ int native_cpu_up(unsigned int cpu, struct task_struct *tidle) ...@@ -1267,15 +1160,7 @@ int native_cpu_up(unsigned int cpu, struct task_struct *tidle)
touch_nmi_watchdog(); touch_nmi_watchdog();
} }
unreg_nmi: return 0;
/*
* Clean up the nmi handler. Do this after the callin and callout sync
* to avoid impact of possible long unregister time.
*/
if (cpu0_nmi_registered)
unregister_nmi_handler(NMI_LOCAL, "wake_cpu0");
return ret;
} }
/** /**
...@@ -1373,14 +1258,6 @@ static void __init smp_cpu_index_default(void) ...@@ -1373,14 +1258,6 @@ static void __init smp_cpu_index_default(void)
} }
} }
static void __init smp_get_logical_apicid(void)
{
if (x2apic_mode)
cpu0_logical_apicid = apic_read(APIC_LDR);
else
cpu0_logical_apicid = GET_APIC_LOGICAL_ID(apic_read(APIC_LDR));
}
void __init smp_prepare_cpus_common(void) void __init smp_prepare_cpus_common(void)
{ {
unsigned int i; unsigned int i;
...@@ -1443,8 +1320,6 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus) ...@@ -1443,8 +1320,6 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus)
/* Setup local timer */ /* Setup local timer */
x86_init.timers.setup_percpu_clockev(); x86_init.timers.setup_percpu_clockev();
smp_get_logical_apicid();
pr_info("CPU0: "); pr_info("CPU0: ");
print_cpu_info(&cpu_data(0)); print_cpu_info(&cpu_data(0));
...@@ -1752,18 +1627,6 @@ void play_dead_common(void) ...@@ -1752,18 +1627,6 @@ void play_dead_common(void)
local_irq_disable(); local_irq_disable();
} }
/**
* cond_wakeup_cpu0 - Wake up CPU0 if needed.
*
* If NMI wants to wake up CPU0, start CPU0.
*/
void cond_wakeup_cpu0(void)
{
if (smp_processor_id() == 0 && enable_start_cpu0)
start_cpu0();
}
EXPORT_SYMBOL_GPL(cond_wakeup_cpu0);
/* /*
* We need to flush the caches before going to sleep, lest we have * We need to flush the caches before going to sleep, lest we have
* dirty data in our caches when we come back up. * dirty data in our caches when we come back up.
...@@ -1831,8 +1694,6 @@ static inline void mwait_play_dead(void) ...@@ -1831,8 +1694,6 @@ static inline void mwait_play_dead(void)
__monitor(mwait_ptr, 0, 0); __monitor(mwait_ptr, 0, 0);
mb(); mb();
__mwait(eax, 0); __mwait(eax, 0);
cond_wakeup_cpu0();
} }
} }
...@@ -1841,11 +1702,8 @@ void __noreturn hlt_play_dead(void) ...@@ -1841,11 +1702,8 @@ void __noreturn hlt_play_dead(void)
if (__this_cpu_read(cpu_info.x86) >= 4) if (__this_cpu_read(cpu_info.x86) >= 4)
wbinvd(); wbinvd();
while (1) { while (1)
native_halt(); native_halt();
cond_wakeup_cpu0();
}
} }
void native_play_dead(void) void native_play_dead(void)
......
...@@ -597,10 +597,6 @@ static int acpi_idle_play_dead(struct cpuidle_device *dev, int index) ...@@ -597,10 +597,6 @@ static int acpi_idle_play_dead(struct cpuidle_device *dev, int index)
io_idle(cx->address); io_idle(cx->address);
} else } else
return -ENODEV; return -ENODEV;
#if defined(CONFIG_X86) && defined(CONFIG_HOTPLUG_CPU)
cond_wakeup_cpu0();
#endif
} }
/* Never reached */ /* Never reached */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment