Commit ef1f87aa authored by Suresh Siddha's avatar Suresh Siddha Committed by Ingo Molnar

x86: select x2apic ops in early apic probe only if x2apic mode is enabled

If BIOS hands over the control to OS in legacy xapic mode, select
legacy xapic related ops in the early apic probe and shift to x2apic
ops later in the boot sequence, only after enabling x2apic mode.

If BIOS hands over the control in x2apic mode, select x2apic related
ops in the early apic probe.

This fixes the early boot panic, where we were selecting x2apic ops,
while the cpu is still in legacy xapic mode.
Signed-off-by: default avatarSuresh Siddha <suresh.b.siddha@intel.com>
Cc: Yinghai Lu <yinghai@kernel.org>
Cc: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarIngo Molnar <mingo@elte.hu>
parent 9be1b56a
...@@ -146,7 +146,7 @@ static inline u64 native_x2apic_icr_read(void) ...@@ -146,7 +146,7 @@ static inline u64 native_x2apic_icr_read(void)
return val; return val;
} }
extern int x2apic; extern int x2apic, x2apic_phys;
extern void check_x2apic(void); extern void check_x2apic(void);
extern void enable_x2apic(void); extern void enable_x2apic(void);
extern void enable_IR_x2apic(void); extern void enable_IR_x2apic(void);
......
...@@ -1265,14 +1265,7 @@ void __cpuinit end_local_APIC_setup(void) ...@@ -1265,14 +1265,7 @@ void __cpuinit end_local_APIC_setup(void)
#ifdef CONFIG_X86_X2APIC #ifdef CONFIG_X86_X2APIC
void check_x2apic(void) void check_x2apic(void)
{ {
int msr, msr2; if (x2apic_enabled()) {
if (!cpu_has_x2apic)
return;
rdmsr(MSR_IA32_APICBASE, msr, msr2);
if (msr & X2APIC_ENABLE) {
pr_info("x2apic enabled by BIOS, switching to x2apic ops\n"); pr_info("x2apic enabled by BIOS, switching to x2apic ops\n");
x2apic_preenabled = x2apic = 1; x2apic_preenabled = x2apic = 1;
} }
......
...@@ -50,9 +50,16 @@ static struct apic *apic_probe[] __initdata = { ...@@ -50,9 +50,16 @@ static struct apic *apic_probe[] __initdata = {
void __init default_setup_apic_routing(void) void __init default_setup_apic_routing(void)
{ {
#ifdef CONFIG_X86_X2APIC #ifdef CONFIG_X86_X2APIC
if (apic == &apic_x2apic_phys || apic == &apic_x2apic_cluster) { if (x2apic && (apic != &apic_x2apic_phys &&
if (!intr_remapping_enabled) #ifdef CONFIG_X86_UV
apic = &apic_flat; apic != &apic_x2apic_uv_x &&
#endif
apic != &apic_x2apic_cluster)) {
if (x2apic_phys)
apic = &apic_x2apic_phys;
else
apic = &apic_x2apic_cluster;
printk(KERN_INFO "Setting APIC routing to %s\n", apic->name);
} }
#endif #endif
......
...@@ -14,10 +14,7 @@ DEFINE_PER_CPU(u32, x86_cpu_to_logical_apicid); ...@@ -14,10 +14,7 @@ DEFINE_PER_CPU(u32, x86_cpu_to_logical_apicid);
static int x2apic_acpi_madt_oem_check(char *oem_id, char *oem_table_id) static int x2apic_acpi_madt_oem_check(char *oem_id, char *oem_table_id)
{ {
if (cpu_has_x2apic) return x2apic_enabled();
return 1;
return 0;
} }
/* Start with all IRQs pointing to boot CPU. IRQ balancing will shift them. */ /* Start with all IRQs pointing to boot CPU. IRQ balancing will shift them. */
......
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
#include <asm/apic.h> #include <asm/apic.h>
#include <asm/ipi.h> #include <asm/ipi.h>
static int x2apic_phys; int x2apic_phys;
static int set_x2apic_phys_mode(char *arg) static int set_x2apic_phys_mode(char *arg)
{ {
...@@ -21,10 +21,10 @@ early_param("x2apic_phys", set_x2apic_phys_mode); ...@@ -21,10 +21,10 @@ early_param("x2apic_phys", set_x2apic_phys_mode);
static int x2apic_acpi_madt_oem_check(char *oem_id, char *oem_table_id) static int x2apic_acpi_madt_oem_check(char *oem_id, char *oem_table_id)
{ {
if (cpu_has_x2apic && x2apic_phys) if (x2apic_phys)
return 1; return x2apic_enabled();
else
return 0; return 0;
} }
/* Start with all IRQs pointing to boot CPU. IRQ balancing will shift them. */ /* Start with all IRQs pointing to boot CPU. IRQ balancing will shift them. */
......
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