Commit 3e730dad authored by Dou Liyang's avatar Dou Liyang Committed by Thomas Gleixner

x86/apic: Unify interrupt mode setup for SMP-capable system

On a SMP-capable system, the kernel enables and sets up the APIC interrupt
delivery mode in native_smp_prepare_cpus(). The decision how to setup the
APIC is intermingled with the decision of setting up SMP or not.

Split the initialization of the APIC interrupt mode independent from other
decisions and have a separate apic_intr_mode_init() function for it.

The invocation time stays the same for now.
Signed-off-by: default avatarDou Liyang <douly.fnst@cn.fujitsu.com>
Signed-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
Cc: yinghai@kernel.org
Cc: bhe@redhat.com
Link: https://lkml.kernel.org/r/1505293975-26005-6-git-send-email-douly.fnst@cn.fujitsu.com
parent 4b1244b4
...@@ -1221,7 +1221,9 @@ void __init sync_Arb_IDs(void) ...@@ -1221,7 +1221,9 @@ void __init sync_Arb_IDs(void)
enum apic_intr_mode { enum apic_intr_mode {
APIC_PIC, APIC_PIC,
APIC_VIRTUAL_WIRE, APIC_VIRTUAL_WIRE,
APIC_VIRTUAL_WIRE_NO_CONFIG,
APIC_SYMMETRIC_IO, APIC_SYMMETRIC_IO,
APIC_SYMMETRIC_IO_NO_ROUTING,
}; };
static int __init apic_intr_mode_select(void) static int __init apic_intr_mode_select(void)
...@@ -1262,11 +1264,27 @@ static int __init apic_intr_mode_select(void) ...@@ -1262,11 +1264,27 @@ static int __init apic_intr_mode_select(void)
/* Check MP table or ACPI MADT configuration */ /* Check MP table or ACPI MADT configuration */
if (!smp_found_config) { if (!smp_found_config) {
disable_ioapic_support(); disable_ioapic_support();
if (!acpi_lapic) if (!acpi_lapic) {
pr_info("APIC: ACPI MADT or MP tables are not detected\n"); pr_info("APIC: ACPI MADT or MP tables are not detected\n");
return APIC_VIRTUAL_WIRE_NO_CONFIG;
}
return APIC_VIRTUAL_WIRE; return APIC_VIRTUAL_WIRE;
} }
#ifdef CONFIG_SMP
/* If SMP should be disabled, then really disable it! */
if (!setup_max_cpus) {
pr_info("APIC: SMP mode deactivated\n");
return APIC_SYMMETRIC_IO_NO_ROUTING;
}
if (read_apic_id() != boot_cpu_physical_apicid) {
panic("Boot APIC ID in local APIC unexpected (%d vs %d)",
read_apic_id(), boot_cpu_physical_apicid);
/* Or can we switch back to PIC here? */
}
#endif
return APIC_SYMMETRIC_IO; return APIC_SYMMETRIC_IO;
} }
...@@ -1322,17 +1340,31 @@ void __init init_bsp_APIC(void) ...@@ -1322,17 +1340,31 @@ void __init init_bsp_APIC(void)
/* Init the interrupt delivery mode for the BSP */ /* Init the interrupt delivery mode for the BSP */
void __init apic_intr_mode_init(void) void __init apic_intr_mode_init(void)
{ {
bool upmode = false;
switch (apic_intr_mode_select()) { switch (apic_intr_mode_select()) {
case APIC_PIC: case APIC_PIC:
pr_info("APIC: Keep in PIC mode(8259)\n"); pr_info("APIC: Keep in PIC mode(8259)\n");
return; return;
case APIC_VIRTUAL_WIRE: case APIC_VIRTUAL_WIRE:
pr_info("APIC: Switch to virtual wire mode setup\n"); pr_info("APIC: Switch to virtual wire mode setup\n");
return; default_setup_apic_routing();
break;
case APIC_VIRTUAL_WIRE_NO_CONFIG:
pr_info("APIC: Switch to virtual wire mode setup with no configuration\n");
upmode = true;
default_setup_apic_routing();
break;
case APIC_SYMMETRIC_IO: case APIC_SYMMETRIC_IO:
pr_info("APIC: Switch to symmectic I/O mode setup\n"); pr_info("APIC: Switch to symmectic I/O mode setup\n");
return; default_setup_apic_routing();
break;
case APIC_SYMMETRIC_IO_NO_ROUTING:
pr_info("APIC: Switch to symmectic I/O mode setup in no SMP routine\n");
break;
} }
apic_bsp_setup(upmode);
} }
static void lapic_setup_esr(void) static void lapic_setup_esr(void)
......
...@@ -1336,18 +1336,17 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus) ...@@ -1336,18 +1336,17 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus)
set_cpu_sibling_map(0); set_cpu_sibling_map(0);
apic_intr_mode_init();
switch (smp_sanity_check(max_cpus)) { switch (smp_sanity_check(max_cpus)) {
case SMP_NO_CONFIG: case SMP_NO_CONFIG:
disable_smp(); disable_smp();
if (APIC_init_uniprocessor())
pr_notice("Local APIC not detected. Using dummy APIC emulation.\n");
return; return;
case SMP_NO_APIC: case SMP_NO_APIC:
disable_smp(); disable_smp();
return; return;
case SMP_FORCE_UP: case SMP_FORCE_UP:
disable_smp(); disable_smp();
apic_bsp_setup(false);
/* Setup local timer */ /* Setup local timer */
x86_init.timers.setup_percpu_clockev(); x86_init.timers.setup_percpu_clockev();
return; return;
...@@ -1355,15 +1354,6 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus) ...@@ -1355,15 +1354,6 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus)
break; break;
} }
if (read_apic_id() != boot_cpu_physical_apicid) {
panic("Boot APIC ID in local APIC unexpected (%d vs %d)",
read_apic_id(), boot_cpu_physical_apicid);
/* Or can we switch back to PIC here? */
}
default_setup_apic_routing();
apic_bsp_setup(false);
/* Setup local timer */ /* Setup local timer */
x86_init.timers.setup_percpu_clockev(); x86_init.timers.setup_percpu_clockev();
......
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