diff options
Diffstat (limited to 'arch')
36 files changed, 2126 insertions, 492 deletions
diff --git a/arch/um/sys-x86_64/syscall_table.c b/arch/um/sys-x86_64/syscall_table.c index c128eb897008..32f5fbe2d0d2 100644 --- a/arch/um/sys-x86_64/syscall_table.c +++ b/arch/um/sys-x86_64/syscall_table.c | |||
| @@ -41,12 +41,12 @@ | |||
| 41 | #define stub_rt_sigreturn sys_rt_sigreturn | 41 | #define stub_rt_sigreturn sys_rt_sigreturn |
| 42 | 42 | ||
| 43 | #define __SYSCALL(nr, sym) extern asmlinkage void sym(void) ; | 43 | #define __SYSCALL(nr, sym) extern asmlinkage void sym(void) ; |
| 44 | #undef _ASM_X86_64_UNISTD_H_ | 44 | #undef ASM_X86__UNISTD_64_H |
| 45 | #include <asm-x86/unistd_64.h> | 45 | #include <asm-x86/unistd_64.h> |
| 46 | 46 | ||
| 47 | #undef __SYSCALL | 47 | #undef __SYSCALL |
| 48 | #define __SYSCALL(nr, sym) [ nr ] = sym, | 48 | #define __SYSCALL(nr, sym) [ nr ] = sym, |
| 49 | #undef _ASM_X86_64_UNISTD_H_ | 49 | #undef ASM_X86__UNISTD_64_H |
| 50 | 50 | ||
| 51 | typedef void (*sys_call_ptr_t)(void); | 51 | typedef void (*sys_call_ptr_t)(void); |
| 52 | 52 | ||
diff --git a/arch/x86/Kconfig b/arch/x86/Kconfig index 0d7cdbbfc1ee..44d4f2130d01 100644 --- a/arch/x86/Kconfig +++ b/arch/x86/Kconfig | |||
| @@ -1689,6 +1689,14 @@ config DMAR_FLOPPY_WA | |||
| 1689 | workaround will setup a 1:1 mapping for the first | 1689 | workaround will setup a 1:1 mapping for the first |
| 1690 | 16M to make floppy (an ISA device) work. | 1690 | 16M to make floppy (an ISA device) work. |
| 1691 | 1691 | ||
| 1692 | config INTR_REMAP | ||
| 1693 | bool "Support for Interrupt Remapping (EXPERIMENTAL)" | ||
| 1694 | depends on X86_64 && X86_IO_APIC && PCI_MSI && ACPI && EXPERIMENTAL | ||
| 1695 | help | ||
| 1696 | Supports Interrupt remapping for IO-APIC and MSI devices. | ||
| 1697 | To use x2apic mode in the CPU's which support x2APIC enhancements or | ||
| 1698 | to support platforms with CPU's having > 8 bit APIC ID, say Y. | ||
| 1699 | |||
| 1692 | source "drivers/pci/pcie/Kconfig" | 1700 | source "drivers/pci/pcie/Kconfig" |
| 1693 | 1701 | ||
| 1694 | source "drivers/pci/Kconfig" | 1702 | source "drivers/pci/Kconfig" |
diff --git a/arch/x86/mach-es7000/Makefile b/arch/x86/es7000/Makefile index 3ef8b43b62fc..3ef8b43b62fc 100644 --- a/arch/x86/mach-es7000/Makefile +++ b/arch/x86/es7000/Makefile | |||
diff --git a/arch/x86/mach-es7000/es7000.h b/arch/x86/es7000/es7000.h index c8d5aa132fa0..4e62f6fa95b8 100644 --- a/arch/x86/mach-es7000/es7000.h +++ b/arch/x86/es7000/es7000.h | |||
| @@ -1,7 +1,7 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Written by: Garry Forsgren, Unisys Corporation | 2 | * Written by: Garry Forsgren, Unisys Corporation |
| 3 | * Natalie Protasevich, Unisys Corporation | 3 | * Natalie Protasevich, Unisys Corporation |
| 4 | * This file contains the code to configure and interface | 4 | * This file contains the code to configure and interface |
| 5 | * with Unisys ES7000 series hardware system manager. | 5 | * with Unisys ES7000 series hardware system manager. |
| 6 | * | 6 | * |
| 7 | * Copyright (c) 2003 Unisys Corporation. All Rights Reserved. | 7 | * Copyright (c) 2003 Unisys Corporation. All Rights Reserved. |
| @@ -18,7 +18,7 @@ | |||
| 18 | * with this program; if not, write the Free Software Foundation, Inc., 59 | 18 | * with this program; if not, write the Free Software Foundation, Inc., 59 |
| 19 | * Temple Place - Suite 330, Boston MA 02111-1307, USA. | 19 | * Temple Place - Suite 330, Boston MA 02111-1307, USA. |
| 20 | * | 20 | * |
| 21 | * Contact information: Unisys Corporation, Township Line & Union Meeting | 21 | * Contact information: Unisys Corporation, Township Line & Union Meeting |
| 22 | * Roads-A, Unisys Way, Blue Bell, Pennsylvania, 19424, or: | 22 | * Roads-A, Unisys Way, Blue Bell, Pennsylvania, 19424, or: |
| 23 | * | 23 | * |
| 24 | * http://www.unisys.com | 24 | * http://www.unisys.com |
| @@ -41,7 +41,7 @@ | |||
| 41 | #define MIP_VALID 0x0100000000000000ULL | 41 | #define MIP_VALID 0x0100000000000000ULL |
| 42 | #define MIP_PORT(VALUE) ((VALUE >> 32) & 0xffff) | 42 | #define MIP_PORT(VALUE) ((VALUE >> 32) & 0xffff) |
| 43 | 43 | ||
| 44 | #define MIP_RD_LO(VALUE) (VALUE & 0xffffffff) | 44 | #define MIP_RD_LO(VALUE) (VALUE & 0xffffffff) |
| 45 | 45 | ||
| 46 | struct mip_reg_info { | 46 | struct mip_reg_info { |
| 47 | unsigned long long mip_info; | 47 | unsigned long long mip_info; |
| @@ -51,11 +51,11 @@ struct mip_reg_info { | |||
| 51 | }; | 51 | }; |
| 52 | 52 | ||
| 53 | struct part_info { | 53 | struct part_info { |
| 54 | unsigned char type; | 54 | unsigned char type; |
| 55 | unsigned char length; | 55 | unsigned char length; |
| 56 | unsigned char part_id; | 56 | unsigned char part_id; |
| 57 | unsigned char apic_mode; | 57 | unsigned char apic_mode; |
| 58 | unsigned long snum; | 58 | unsigned long snum; |
| 59 | char ptype[16]; | 59 | char ptype[16]; |
| 60 | char sname[64]; | 60 | char sname[64]; |
| 61 | char pname[64]; | 61 | char pname[64]; |
| @@ -68,11 +68,11 @@ struct psai { | |||
| 68 | }; | 68 | }; |
| 69 | 69 | ||
| 70 | struct es7000_mem_info { | 70 | struct es7000_mem_info { |
| 71 | unsigned char type; | 71 | unsigned char type; |
| 72 | unsigned char length; | 72 | unsigned char length; |
| 73 | unsigned char resv[6]; | 73 | unsigned char resv[6]; |
| 74 | unsigned long long start; | 74 | unsigned long long start; |
| 75 | unsigned long long size; | 75 | unsigned long long size; |
| 76 | }; | 76 | }; |
| 77 | 77 | ||
| 78 | struct es7000_oem_table { | 78 | struct es7000_oem_table { |
| @@ -106,7 +106,7 @@ struct mip_reg { | |||
| 106 | }; | 106 | }; |
| 107 | 107 | ||
| 108 | #define MIP_SW_APIC 0x1020b | 108 | #define MIP_SW_APIC 0x1020b |
| 109 | #define MIP_FUNC(VALUE) (VALUE & 0xff) | 109 | #define MIP_FUNC(VALUE) (VALUE & 0xff) |
| 110 | 110 | ||
| 111 | extern int parse_unisys_oem (char *oemptr); | 111 | extern int parse_unisys_oem (char *oemptr); |
| 112 | extern void setup_unisys(void); | 112 | extern void setup_unisys(void); |
diff --git a/arch/x86/mach-es7000/es7000plat.c b/arch/x86/es7000/es7000plat.c index 50189af14b85..7789fde13c3f 100644 --- a/arch/x86/mach-es7000/es7000plat.c +++ b/arch/x86/es7000/es7000plat.c | |||
| @@ -72,7 +72,7 @@ es7000_rename_gsi(int ioapic, int gsi) | |||
| 72 | base += nr_ioapic_registers[i]; | 72 | base += nr_ioapic_registers[i]; |
| 73 | } | 73 | } |
| 74 | 74 | ||
| 75 | if (!ioapic && (gsi < 16)) | 75 | if (!ioapic && (gsi < 16)) |
| 76 | gsi += base; | 76 | gsi += base; |
| 77 | return gsi; | 77 | return gsi; |
| 78 | } | 78 | } |
diff --git a/arch/x86/kernel/Makefile b/arch/x86/kernel/Makefile index 3db651fc8ec5..a07ec14f3312 100644 --- a/arch/x86/kernel/Makefile +++ b/arch/x86/kernel/Makefile | |||
| @@ -104,6 +104,8 @@ obj-$(CONFIG_OLPC) += olpc.o | |||
| 104 | ifeq ($(CONFIG_X86_64),y) | 104 | ifeq ($(CONFIG_X86_64),y) |
| 105 | obj-y += genapic_64.o genapic_flat_64.o genx2apic_uv_x.o tlb_uv.o | 105 | obj-y += genapic_64.o genapic_flat_64.o genx2apic_uv_x.o tlb_uv.o |
| 106 | obj-y += bios_uv.o | 106 | obj-y += bios_uv.o |
| 107 | obj-y += genx2apic_cluster.o | ||
| 108 | obj-y += genx2apic_phys.o | ||
| 107 | obj-$(CONFIG_X86_PM_TIMER) += pmtimer_64.o | 109 | obj-$(CONFIG_X86_PM_TIMER) += pmtimer_64.o |
| 108 | obj-$(CONFIG_AUDIT) += audit_64.o | 110 | obj-$(CONFIG_AUDIT) += audit_64.o |
| 109 | 111 | ||
diff --git a/arch/x86/kernel/acpi/boot.c b/arch/x86/kernel/acpi/boot.c index 7d40ef7b36e3..c2ac1b4515a0 100644 --- a/arch/x86/kernel/acpi/boot.c +++ b/arch/x86/kernel/acpi/boot.c | |||
| @@ -252,10 +252,8 @@ static void __cpuinit acpi_register_lapic(int id, u8 enabled) | |||
| 252 | return; | 252 | return; |
| 253 | } | 253 | } |
| 254 | 254 | ||
| 255 | #ifdef CONFIG_X86_32 | ||
| 256 | if (boot_cpu_physical_apicid != -1U) | 255 | if (boot_cpu_physical_apicid != -1U) |
| 257 | ver = apic_version[boot_cpu_physical_apicid]; | 256 | ver = apic_version[boot_cpu_physical_apicid]; |
| 258 | #endif | ||
| 259 | 257 | ||
| 260 | generic_processor_info(id, ver); | 258 | generic_processor_info(id, ver); |
| 261 | } | 259 | } |
| @@ -774,11 +772,9 @@ static void __init acpi_register_lapic_address(unsigned long address) | |||
| 774 | 772 | ||
| 775 | set_fixmap_nocache(FIX_APIC_BASE, address); | 773 | set_fixmap_nocache(FIX_APIC_BASE, address); |
| 776 | if (boot_cpu_physical_apicid == -1U) { | 774 | if (boot_cpu_physical_apicid == -1U) { |
| 777 | boot_cpu_physical_apicid = GET_APIC_ID(read_apic_id()); | 775 | boot_cpu_physical_apicid = read_apic_id(); |
| 778 | #ifdef CONFIG_X86_32 | ||
| 779 | apic_version[boot_cpu_physical_apicid] = | 776 | apic_version[boot_cpu_physical_apicid] = |
| 780 | GET_APIC_VERSION(apic_read(APIC_LVR)); | 777 | GET_APIC_VERSION(apic_read(APIC_LVR)); |
| 781 | #endif | ||
| 782 | } | 778 | } |
| 783 | } | 779 | } |
| 784 | 780 | ||
| @@ -1350,7 +1346,9 @@ static void __init acpi_process_madt(void) | |||
| 1350 | acpi_ioapic = 1; | 1346 | acpi_ioapic = 1; |
| 1351 | 1347 | ||
| 1352 | smp_found_config = 1; | 1348 | smp_found_config = 1; |
| 1349 | #ifdef CONFIG_X86_32 | ||
| 1353 | setup_apic_routing(); | 1350 | setup_apic_routing(); |
| 1351 | #endif | ||
| 1354 | } | 1352 | } |
| 1355 | } | 1353 | } |
| 1356 | if (error == -EINVAL) { | 1354 | if (error == -EINVAL) { |
diff --git a/arch/x86/kernel/apic_32.c b/arch/x86/kernel/apic_32.c index f88bd0d982b0..a91c57cb666a 100644 --- a/arch/x86/kernel/apic_32.c +++ b/arch/x86/kernel/apic_32.c | |||
| @@ -60,10 +60,8 @@ unsigned long mp_lapic_addr; | |||
| 60 | static int force_enable_local_apic; | 60 | static int force_enable_local_apic; |
| 61 | int disable_apic; | 61 | int disable_apic; |
| 62 | 62 | ||
| 63 | /* Local APIC timer verification ok */ | ||
| 64 | static int local_apic_timer_verify_ok; | ||
| 65 | /* Disable local APIC timer from the kernel commandline or via dmi quirk */ | 63 | /* Disable local APIC timer from the kernel commandline or via dmi quirk */ |
| 66 | static int local_apic_timer_disabled; | 64 | static int disable_apic_timer __cpuinitdata; |
| 67 | /* Local APIC timer works in C2 */ | 65 | /* Local APIC timer works in C2 */ |
| 68 | int local_apic_timer_c2_ok; | 66 | int local_apic_timer_c2_ok; |
| 69 | EXPORT_SYMBOL_GPL(local_apic_timer_c2_ok); | 67 | EXPORT_SYMBOL_GPL(local_apic_timer_c2_ok); |
| @@ -130,7 +128,11 @@ static inline int lapic_get_version(void) | |||
| 130 | */ | 128 | */ |
| 131 | static inline int lapic_is_integrated(void) | 129 | static inline int lapic_is_integrated(void) |
| 132 | { | 130 | { |
| 131 | #ifdef CONFIG_X86_64 | ||
| 132 | return 1; | ||
| 133 | #else | ||
| 133 | return APIC_INTEGRATED(lapic_get_version()); | 134 | return APIC_INTEGRATED(lapic_get_version()); |
| 135 | #endif | ||
| 134 | } | 136 | } |
| 135 | 137 | ||
| 136 | /* | 138 | /* |
| @@ -145,13 +147,18 @@ static int modern_apic(void) | |||
| 145 | return lapic_get_version() >= 0x14; | 147 | return lapic_get_version() >= 0x14; |
| 146 | } | 148 | } |
| 147 | 149 | ||
| 148 | void apic_wait_icr_idle(void) | 150 | /* |
| 151 | * Paravirt kernels also might be using these below ops. So we still | ||
| 152 | * use generic apic_read()/apic_write(), which might be pointing to different | ||
| 153 | * ops in PARAVIRT case. | ||
| 154 | */ | ||
| 155 | void xapic_wait_icr_idle(void) | ||
| 149 | { | 156 | { |
| 150 | while (apic_read(APIC_ICR) & APIC_ICR_BUSY) | 157 | while (apic_read(APIC_ICR) & APIC_ICR_BUSY) |
| 151 | cpu_relax(); | 158 | cpu_relax(); |
| 152 | } | 159 | } |
| 153 | 160 | ||
| 154 | u32 safe_apic_wait_icr_idle(void) | 161 | u32 safe_xapic_wait_icr_idle(void) |
| 155 | { | 162 | { |
| 156 | u32 send_status; | 163 | u32 send_status; |
| 157 | int timeout; | 164 | int timeout; |
| @@ -167,16 +174,48 @@ u32 safe_apic_wait_icr_idle(void) | |||
| 167 | return send_status; | 174 | return send_status; |
| 168 | } | 175 | } |
| 169 | 176 | ||
| 177 | void xapic_icr_write(u32 low, u32 id) | ||
| 178 | { | ||
| 179 | apic_write(APIC_ICR2, SET_APIC_DEST_FIELD(id)); | ||
| 180 | apic_write(APIC_ICR, low); | ||
| 181 | } | ||
| 182 | |||
| 183 | u64 xapic_icr_read(void) | ||
| 184 | { | ||
| 185 | u32 icr1, icr2; | ||
| 186 | |||
| 187 | icr2 = apic_read(APIC_ICR2); | ||
| 188 | icr1 = apic_read(APIC_ICR); | ||
| 189 | |||
| 190 | return icr1 | ((u64)icr2 << 32); | ||
| 191 | } | ||
| 192 | |||
| 193 | static struct apic_ops xapic_ops = { | ||
| 194 | .read = native_apic_mem_read, | ||
| 195 | .write = native_apic_mem_write, | ||
| 196 | .icr_read = xapic_icr_read, | ||
| 197 | .icr_write = xapic_icr_write, | ||
| 198 | .wait_icr_idle = xapic_wait_icr_idle, | ||
| 199 | .safe_wait_icr_idle = safe_xapic_wait_icr_idle, | ||
| 200 | }; | ||
| 201 | |||
| 202 | struct apic_ops __read_mostly *apic_ops = &xapic_ops; | ||
| 203 | EXPORT_SYMBOL_GPL(apic_ops); | ||
| 204 | |||
| 170 | /** | 205 | /** |
| 171 | * enable_NMI_through_LVT0 - enable NMI through local vector table 0 | 206 | * enable_NMI_through_LVT0 - enable NMI through local vector table 0 |
| 172 | */ | 207 | */ |
| 173 | void __cpuinit enable_NMI_through_LVT0(void) | 208 | void __cpuinit enable_NMI_through_LVT0(void) |
| 174 | { | 209 | { |
| 175 | unsigned int v = APIC_DM_NMI; | 210 | unsigned int v; |
| 176 | 211 | ||
| 177 | /* Level triggered for 82489DX */ | 212 | /* unmask and set to NMI */ |
| 213 | v = APIC_DM_NMI; | ||
| 214 | |||
| 215 | /* Level triggered for 82489DX (32bit mode) */ | ||
| 178 | if (!lapic_is_integrated()) | 216 | if (!lapic_is_integrated()) |
| 179 | v |= APIC_LVT_LEVEL_TRIGGER; | 217 | v |= APIC_LVT_LEVEL_TRIGGER; |
| 218 | |||
| 180 | apic_write(APIC_LVT0, v); | 219 | apic_write(APIC_LVT0, v); |
| 181 | } | 220 | } |
| 182 | 221 | ||
| @@ -193,9 +232,13 @@ int get_physical_broadcast(void) | |||
| 193 | */ | 232 | */ |
| 194 | int lapic_get_maxlvt(void) | 233 | int lapic_get_maxlvt(void) |
| 195 | { | 234 | { |
| 196 | unsigned int v = apic_read(APIC_LVR); | 235 | unsigned int v; |
| 197 | 236 | ||
| 198 | /* 82489DXs do not report # of LVT entries. */ | 237 | v = apic_read(APIC_LVR); |
| 238 | /* | ||
| 239 | * - we always have APIC integrated on 64bit mode | ||
| 240 | * - 82489DXs do not report # of LVT entries | ||
| 241 | */ | ||
| 199 | return APIC_INTEGRATED(GET_APIC_VERSION(v)) ? GET_APIC_MAXLVT(v) : 2; | 242 | return APIC_INTEGRATED(GET_APIC_VERSION(v)) ? GET_APIC_MAXLVT(v) : 2; |
| 200 | } | 243 | } |
| 201 | 244 | ||
| @@ -203,8 +246,12 @@ int lapic_get_maxlvt(void) | |||
| 203 | * Local APIC timer | 246 | * Local APIC timer |
| 204 | */ | 247 | */ |
| 205 | 248 | ||
| 206 | /* Clock divisor is set to 16 */ | 249 | /* Clock divisor */ |
| 250 | #ifdef CONFG_X86_64 | ||
| 251 | #define APIC_DIVISOR 1 | ||
| 252 | #else | ||
| 207 | #define APIC_DIVISOR 16 | 253 | #define APIC_DIVISOR 16 |
| 254 | #endif | ||
| 208 | 255 | ||
| 209 | /* | 256 | /* |
| 210 | * This function sets up the local APIC timer, with a timeout of | 257 | * This function sets up the local APIC timer, with a timeout of |
| @@ -212,6 +259,9 @@ int lapic_get_maxlvt(void) | |||
| 212 | * this function twice on the boot CPU, once with a bogus timeout | 259 | * this function twice on the boot CPU, once with a bogus timeout |
| 213 | * value, second time for real. The other (noncalibrating) CPUs | 260 | * value, second time for real. The other (noncalibrating) CPUs |
| 214 | * call this function only once, with the real, calibrated value. | 261 | * call this function only once, with the real, calibrated value. |
| 262 | * | ||
| 263 | * We do reads before writes even if unnecessary, to get around the | ||
| 264 | * P5 APIC double write bug. | ||
| 215 | */ | 265 | */ |
| 216 | static void __setup_APIC_LVTT(unsigned int clocks, int oneshot, int irqen) | 266 | static void __setup_APIC_LVTT(unsigned int clocks, int oneshot, int irqen) |
| 217 | { | 267 | { |
| @@ -233,14 +283,44 @@ static void __setup_APIC_LVTT(unsigned int clocks, int oneshot, int irqen) | |||
| 233 | */ | 283 | */ |
| 234 | tmp_value = apic_read(APIC_TDCR); | 284 | tmp_value = apic_read(APIC_TDCR); |
| 235 | apic_write(APIC_TDCR, | 285 | apic_write(APIC_TDCR, |
| 236 | (tmp_value & ~(APIC_TDR_DIV_1 | APIC_TDR_DIV_TMBASE)) | | 286 | (tmp_value & ~(APIC_TDR_DIV_1 | APIC_TDR_DIV_TMBASE)) | |
| 237 | APIC_TDR_DIV_16); | 287 | APIC_TDR_DIV_16); |
| 238 | 288 | ||
| 239 | if (!oneshot) | 289 | if (!oneshot) |
| 240 | apic_write(APIC_TMICT, clocks / APIC_DIVISOR); | 290 | apic_write(APIC_TMICT, clocks / APIC_DIVISOR); |
| 241 | } | 291 | } |
| 242 | 292 | ||
| 243 | /* | 293 | /* |
| 294 | * Setup extended LVT, AMD specific (K8, family 10h) | ||
| 295 | * | ||
| 296 | * Vector mappings are hard coded. On K8 only offset 0 (APIC500) and | ||
| 297 | * MCE interrupts are supported. Thus MCE offset must be set to 0. | ||
| 298 | */ | ||
| 299 | |||
| 300 | #define APIC_EILVT_LVTOFF_MCE 0 | ||
| 301 | #define APIC_EILVT_LVTOFF_IBS 1 | ||
| 302 | |||
| 303 | static void setup_APIC_eilvt(u8 lvt_off, u8 vector, u8 msg_type, u8 mask) | ||
| 304 | { | ||
| 305 | unsigned long reg = (lvt_off << 4) + APIC_EILVT0; | ||
| 306 | unsigned int v = (mask << 16) | (msg_type << 8) | vector; | ||
| 307 | |||
| 308 | apic_write(reg, v); | ||
| 309 | } | ||
| 310 | |||
| 311 | u8 setup_APIC_eilvt_mce(u8 vector, u8 msg_type, u8 mask) | ||
| 312 | { | ||
| 313 | setup_APIC_eilvt(APIC_EILVT_LVTOFF_MCE, vector, msg_type, mask); | ||
| 314 | return APIC_EILVT_LVTOFF_MCE; | ||
| 315 | } | ||
| 316 | |||
| 317 | u8 setup_APIC_eilvt_ibs(u8 vector, u8 msg_type, u8 mask) | ||
| 318 | { | ||
| 319 | setup_APIC_eilvt(APIC_EILVT_LVTOFF_IBS, vector, msg_type, mask); | ||
| 320 | return APIC_EILVT_LVTOFF_IBS; | ||
| 321 | } | ||
| 322 | |||
| 323 | /* | ||
| 244 | * Program the next event, relative to now | 324 | * Program the next event, relative to now |
| 245 | */ | 325 | */ |
| 246 | static int lapic_next_event(unsigned long delta, | 326 | static int lapic_next_event(unsigned long delta, |
| @@ -259,8 +339,8 @@ static void lapic_timer_setup(enum clock_event_mode mode, | |||
| 259 | unsigned long flags; | 339 | unsigned long flags; |
| 260 | unsigned int v; | 340 | unsigned int v; |
| 261 | 341 | ||
| 262 | /* Lapic used for broadcast ? */ | 342 | /* Lapic used as dummy for broadcast ? */ |
| 263 | if (!local_apic_timer_verify_ok) | 343 | if (evt->features & CLOCK_EVT_FEAT_DUMMY) |
| 264 | return; | 344 | return; |
| 265 | 345 | ||
| 266 | local_irq_save(flags); | 346 | local_irq_save(flags); |
| @@ -473,7 +553,7 @@ static int __init calibrate_APIC_clock(void) | |||
| 473 | return -1; | 553 | return -1; |
| 474 | } | 554 | } |
| 475 | 555 | ||
| 476 | local_apic_timer_verify_ok = 1; | 556 | levt->features &= ~CLOCK_EVT_FEAT_DUMMY; |
| 477 | 557 | ||
| 478 | /* We trust the pm timer based calibration */ | 558 | /* We trust the pm timer based calibration */ |
| 479 | if (!pm_referenced) { | 559 | if (!pm_referenced) { |
| @@ -507,11 +587,11 @@ static int __init calibrate_APIC_clock(void) | |||
| 507 | if (deltaj >= LAPIC_CAL_LOOPS-2 && deltaj <= LAPIC_CAL_LOOPS+2) | 587 | if (deltaj >= LAPIC_CAL_LOOPS-2 && deltaj <= LAPIC_CAL_LOOPS+2) |
| 508 | apic_printk(APIC_VERBOSE, "... jiffies result ok\n"); | 588 | apic_printk(APIC_VERBOSE, "... jiffies result ok\n"); |
| 509 | else | 589 | else |
| 510 | local_apic_timer_verify_ok = 0; | 590 | levt->features |= CLOCK_EVT_FEAT_DUMMY; |
| 511 | } else | 591 | } else |
| 512 | local_irq_enable(); | 592 | local_irq_enable(); |
| 513 | 593 | ||
| 514 | if (!local_apic_timer_verify_ok) { | 594 | if (levt->features & CLOCK_EVT_FEAT_DUMMY) { |
| 515 | printk(KERN_WARNING | 595 | printk(KERN_WARNING |
| 516 | "APIC timer disabled due to verification failure.\n"); | 596 | "APIC timer disabled due to verification failure.\n"); |
| 517 | return -1; | 597 | return -1; |
| @@ -533,7 +613,8 @@ void __init setup_boot_APIC_clock(void) | |||
| 533 | * timer as a dummy clock event source on SMP systems, so the | 613 | * timer as a dummy clock event source on SMP systems, so the |
| 534 | * broadcast mechanism is used. On UP systems simply ignore it. | 614 | * broadcast mechanism is used. On UP systems simply ignore it. |
| 535 | */ | 615 | */ |
| 536 | if (local_apic_timer_disabled) { | 616 | if (disable_apic_timer) { |
| 617 | printk(KERN_INFO "Disabling APIC timer\n"); | ||
| 537 | /* No broadcast on UP ! */ | 618 | /* No broadcast on UP ! */ |
| 538 | if (num_possible_cpus() > 1) { | 619 | if (num_possible_cpus() > 1) { |
| 539 | lapic_clockevent.mult = 1; | 620 | lapic_clockevent.mult = 1; |
| @@ -602,7 +683,11 @@ static void local_apic_timer_interrupt(void) | |||
| 602 | /* | 683 | /* |
| 603 | * the NMI deadlock-detector uses this. | 684 | * the NMI deadlock-detector uses this. |
| 604 | */ | 685 | */ |
| 686 | #ifdef CONFIG_X86_64 | ||
| 687 | add_pda(apic_timer_irqs, 1); | ||
| 688 | #else | ||
| 605 | per_cpu(irq_stat, cpu).apic_timer_irqs++; | 689 | per_cpu(irq_stat, cpu).apic_timer_irqs++; |
| 690 | #endif | ||
| 606 | 691 | ||
| 607 | evt->event_handler(evt); | 692 | evt->event_handler(evt); |
| 608 | } | 693 | } |
| @@ -642,35 +727,6 @@ int setup_profiling_timer(unsigned int multiplier) | |||
| 642 | } | 727 | } |
| 643 | 728 | ||
| 644 | /* | 729 | /* |
| 645 | * Setup extended LVT, AMD specific (K8, family 10h) | ||
| 646 | * | ||
| 647 | * Vector mappings are hard coded. On K8 only offset 0 (APIC500) and | ||
| 648 | * MCE interrupts are supported. Thus MCE offset must be set to 0. | ||
| 649 | */ | ||
| 650 | |||
| 651 | #define APIC_EILVT_LVTOFF_MCE 0 | ||
| 652 | #define APIC_EILVT_LVTOFF_IBS 1 | ||
| 653 | |||
| 654 | static void setup_APIC_eilvt(u8 lvt_off, u8 vector, u8 msg_type, u8 mask) | ||
| 655 | { | ||
| 656 | unsigned long reg = (lvt_off << 4) + APIC_EILVT0; | ||
| 657 | unsigned int v = (mask << 16) | (msg_type << 8) | vector; | ||
| 658 | apic_write(reg, v); | ||
| 659 | } | ||
| 660 | |||
| 661 | u8 setup_APIC_eilvt_mce(u8 vector, u8 msg_type, u8 mask) | ||
| 662 | { | ||
| 663 | setup_APIC_eilvt(APIC_EILVT_LVTOFF_MCE, vector, msg_type, mask); | ||
| 664 | return APIC_EILVT_LVTOFF_MCE; | ||
| 665 | } | ||
| 666 | |||
| 667 | u8 setup_APIC_eilvt_ibs(u8 vector, u8 msg_type, u8 mask) | ||
| 668 | { | ||
| 669 | setup_APIC_eilvt(APIC_EILVT_LVTOFF_IBS, vector, msg_type, mask); | ||
| 670 | return APIC_EILVT_LVTOFF_IBS; | ||
| 671 | } | ||
| 672 | |||
| 673 | /* | ||
| 674 | * Local APIC start and shutdown | 730 | * Local APIC start and shutdown |
| 675 | */ | 731 | */ |
| 676 | 732 | ||
| @@ -715,7 +771,7 @@ void clear_local_APIC(void) | |||
| 715 | } | 771 | } |
| 716 | 772 | ||
| 717 | /* lets not touch this if we didn't frob it */ | 773 | /* lets not touch this if we didn't frob it */ |
| 718 | #ifdef CONFIG_X86_MCE_P4THERMAL | 774 | #if defined(CONFIG_X86_MCE_P4THERMAL) || defined(X86_MCE_INTEL) |
| 719 | if (maxlvt >= 5) { | 775 | if (maxlvt >= 5) { |
| 720 | v = apic_read(APIC_LVTTHMR); | 776 | v = apic_read(APIC_LVTTHMR); |
| 721 | apic_write(APIC_LVTTHMR, v | APIC_LVT_MASKED); | 777 | apic_write(APIC_LVTTHMR, v | APIC_LVT_MASKED); |
| @@ -732,10 +788,6 @@ void clear_local_APIC(void) | |||
| 732 | if (maxlvt >= 4) | 788 | if (maxlvt >= 4) |
| 733 | apic_write(APIC_LVTPC, APIC_LVT_MASKED); | 789 | apic_write(APIC_LVTPC, APIC_LVT_MASKED); |
| 734 | 790 | ||
| 735 | #ifdef CONFIG_X86_MCE_P4THERMAL | ||
| 736 | if (maxlvt >= 5) | ||
| 737 | apic_write(APIC_LVTTHMR, APIC_LVT_MASKED); | ||
| 738 | #endif | ||
| 739 | /* Integrated APIC (!82489DX) ? */ | 791 | /* Integrated APIC (!82489DX) ? */ |
| 740 | if (lapic_is_integrated()) { | 792 | if (lapic_is_integrated()) { |
| 741 | if (maxlvt > 3) | 793 | if (maxlvt > 3) |
| @@ -750,7 +802,7 @@ void clear_local_APIC(void) | |||
| 750 | */ | 802 | */ |
| 751 | void disable_local_APIC(void) | 803 | void disable_local_APIC(void) |
| 752 | { | 804 | { |
| 753 | unsigned long value; | 805 | unsigned int value; |
| 754 | 806 | ||
| 755 | clear_local_APIC(); | 807 | clear_local_APIC(); |
| 756 | 808 | ||
| @@ -762,6 +814,7 @@ void disable_local_APIC(void) | |||
| 762 | value &= ~APIC_SPIV_APIC_ENABLED; | 814 | value &= ~APIC_SPIV_APIC_ENABLED; |
| 763 | apic_write(APIC_SPIV, value); | 815 | apic_write(APIC_SPIV, value); |
| 764 | 816 | ||
| 817 | #ifdef CONFIG_X86_32 | ||
| 765 | /* | 818 | /* |
| 766 | * When LAPIC was disabled by the BIOS and enabled by the kernel, | 819 | * When LAPIC was disabled by the BIOS and enabled by the kernel, |
| 767 | * restore the disabled state. | 820 | * restore the disabled state. |
| @@ -773,6 +826,7 @@ void disable_local_APIC(void) | |||
| 773 | l &= ~MSR_IA32_APICBASE_ENABLE; | 826 | l &= ~MSR_IA32_APICBASE_ENABLE; |
| 774 | wrmsr(MSR_IA32_APICBASE, l, h); | 827 | wrmsr(MSR_IA32_APICBASE, l, h); |
| 775 | } | 828 | } |
| 829 | #endif | ||
| 776 | } | 830 | } |
| 777 | 831 | ||
| 778 | /* | 832 | /* |
| @@ -789,11 +843,15 @@ void lapic_shutdown(void) | |||
| 789 | return; | 843 | return; |
| 790 | 844 | ||
| 791 | local_irq_save(flags); | 845 | local_irq_save(flags); |
| 792 | clear_local_APIC(); | ||
| 793 | 846 | ||
| 794 | if (enabled_via_apicbase) | 847 | #ifdef CONFIG_X86_32 |
| 848 | if (!enabled_via_apicbase) | ||
| 849 | clear_local_APIC(); | ||
| 850 | else | ||
| 851 | #endif | ||
| 795 | disable_local_APIC(); | 852 | disable_local_APIC(); |
| 796 | 853 | ||
| 854 | |||
| 797 | local_irq_restore(flags); | 855 | local_irq_restore(flags); |
| 798 | } | 856 | } |
| 799 | 857 | ||
| @@ -838,6 +896,12 @@ int __init verify_local_APIC(void) | |||
| 838 | */ | 896 | */ |
| 839 | reg0 = apic_read(APIC_ID); | 897 | reg0 = apic_read(APIC_ID); |
| 840 | apic_printk(APIC_DEBUG, "Getting ID: %x\n", reg0); | 898 | apic_printk(APIC_DEBUG, "Getting ID: %x\n", reg0); |
| 899 | apic_write(APIC_ID, reg0 ^ APIC_ID_MASK); | ||
| 900 | reg1 = apic_read(APIC_ID); | ||
| 901 | apic_printk(APIC_DEBUG, "Getting ID: %x\n", reg1); | ||
| 902 | apic_write(APIC_ID, reg0); | ||
| 903 | if (reg1 != (reg0 ^ APIC_ID_MASK)) | ||
| 904 | return 0; | ||
| 841 | 905 | ||
| 842 | /* | 906 | /* |
| 843 | * The next two are just to see if we have sane values. | 907 | * The next two are just to see if we have sane values. |
| @@ -863,14 +927,15 @@ void __init sync_Arb_IDs(void) | |||
| 863 | */ | 927 | */ |
| 864 | if (modern_apic() || boot_cpu_data.x86_vendor == X86_VENDOR_AMD) | 928 | if (modern_apic() || boot_cpu_data.x86_vendor == X86_VENDOR_AMD) |
| 865 | return; | 929 | return; |
| 930 | |||
| 866 | /* | 931 | /* |
| 867 | * Wait for idle. | 932 | * Wait for idle. |
| 868 | */ | 933 | */ |
| 869 | apic_wait_icr_idle(); | 934 | apic_wait_icr_idle(); |
| 870 | 935 | ||
| 871 | apic_printk(APIC_DEBUG, "Synchronizing Arb IDs.\n"); | 936 | apic_printk(APIC_DEBUG, "Synchronizing Arb IDs.\n"); |
| 872 | apic_write(APIC_ICR, | 937 | apic_write(APIC_ICR, APIC_DEST_ALLINC | |
| 873 | APIC_DEST_ALLINC | APIC_INT_LEVELTRIG | APIC_DM_INIT); | 938 | APIC_INT_LEVELTRIG | APIC_DM_INIT); |
| 874 | } | 939 | } |
| 875 | 940 | ||
| 876 | /* | 941 | /* |
| @@ -878,7 +943,7 @@ void __init sync_Arb_IDs(void) | |||
| 878 | */ | 943 | */ |
| 879 | void __init init_bsp_APIC(void) | 944 | void __init init_bsp_APIC(void) |
| 880 | { | 945 | { |
| 881 | unsigned long value; | 946 | unsigned int value; |
| 882 | 947 | ||
| 883 | /* | 948 | /* |
| 884 | * Don't do the setup now if we have a SMP BIOS as the | 949 | * Don't do the setup now if we have a SMP BIOS as the |
| @@ -899,11 +964,13 @@ void __init init_bsp_APIC(void) | |||
| 899 | value &= ~APIC_VECTOR_MASK; | 964 | value &= ~APIC_VECTOR_MASK; |
| 900 | value |= APIC_SPIV_APIC_ENABLED; | 965 | value |= APIC_SPIV_APIC_ENABLED; |
| 901 | 966 | ||
| 967 | #ifdef CONFIG_X86_32 | ||
| 902 | /* This bit is reserved on P4/Xeon and should be cleared */ | 968 | /* This bit is reserved on P4/Xeon and should be cleared */ |
| 903 | if ((boot_cpu_data.x86_vendor == X86_VENDOR_INTEL) && | 969 | if ((boot_cpu_data.x86_vendor == X86_VENDOR_INTEL) && |
| 904 | (boot_cpu_data.x86 == 15)) | 970 | (boot_cpu_data.x86 == 15)) |
| 905 | value &= ~APIC_SPIV_FOCUS_DISABLED; | 971 | value &= ~APIC_SPIV_FOCUS_DISABLED; |
| 906 | else | 972 | else |
| 973 | #endif | ||
| 907 | value |= APIC_SPIV_FOCUS_DISABLED; | 974 | value |= APIC_SPIV_FOCUS_DISABLED; |
| 908 | value |= SPURIOUS_APIC_VECTOR; | 975 | value |= SPURIOUS_APIC_VECTOR; |
| 909 | apic_write(APIC_SPIV, value); | 976 | apic_write(APIC_SPIV, value); |
| @@ -922,6 +989,16 @@ static void __cpuinit lapic_setup_esr(void) | |||
| 922 | { | 989 | { |
| 923 | unsigned long oldvalue, value, maxlvt; | 990 | unsigned long oldvalue, value, maxlvt; |
| 924 | if (lapic_is_integrated() && !esr_disable) { | 991 | if (lapic_is_integrated() && !esr_disable) { |
| 992 | if (esr_disable) { | ||
| 993 | /* | ||
| 994 | * Something untraceable is creating bad interrupts on | ||
| 995 | * secondary quads ... for the moment, just leave the | ||
| 996 | * ESR disabled - we can't do anything useful with the | ||
| 997 | * errors anyway - mbligh | ||
| 998 | */ | ||
| 999 | printk(KERN_INFO "Leaving ESR disabled.\n"); | ||
| 1000 | return; | ||
| 1001 | } | ||
| 925 | /* !82489DX */ | 1002 | /* !82489DX */ |
| 926 | maxlvt = lapic_get_maxlvt(); | 1003 | maxlvt = lapic_get_maxlvt(); |
| 927 | if (maxlvt > 3) /* Due to the Pentium erratum 3AP. */ | 1004 | if (maxlvt > 3) /* Due to the Pentium erratum 3AP. */ |
| @@ -942,16 +1019,7 @@ static void __cpuinit lapic_setup_esr(void) | |||
| 942 | "vector: 0x%08lx after: 0x%08lx\n", | 1019 | "vector: 0x%08lx after: 0x%08lx\n", |
| 943 | oldvalue, value); | 1020 | oldvalue, value); |
| 944 | } else { | 1021 | } else { |
| 945 | if (esr_disable) | 1022 | printk(KERN_INFO "No ESR for 82489DX.\n"); |
| 946 | /* | ||
| 947 | * Something untraceable is creating bad interrupts on | ||
| 948 | * secondary quads ... for the moment, just leave the | ||
| 949 | * ESR disabled - we can't do anything useful with the | ||
| 950 | * errors anyway - mbligh | ||
| 951 | */ | ||
| 952 | printk(KERN_INFO "Leaving ESR disabled.\n"); | ||
| 953 | else | ||
| 954 | printk(KERN_INFO "No ESR for 82489DX.\n"); | ||
| 955 | } | 1023 | } |
| 956 | } | 1024 | } |
| 957 | 1025 | ||
| @@ -1089,13 +1157,17 @@ void __cpuinit setup_local_APIC(void) | |||
| 1089 | 1157 | ||
| 1090 | void __cpuinit end_local_APIC_setup(void) | 1158 | void __cpuinit end_local_APIC_setup(void) |
| 1091 | { | 1159 | { |
| 1092 | unsigned long value; | ||
| 1093 | |||
| 1094 | lapic_setup_esr(); | 1160 | lapic_setup_esr(); |
| 1095 | /* Disable the local apic timer */ | 1161 | |
| 1096 | value = apic_read(APIC_LVTT); | 1162 | #ifdef CONFIG_X86_32 |
| 1097 | value |= (APIC_LVT_MASKED | LOCAL_TIMER_VECTOR); | 1163 | { |
| 1098 | apic_write(APIC_LVTT, value); | 1164 | unsigned int value; |
| 1165 | /* Disable the local apic timer */ | ||
| 1166 | value = apic_read(APIC_LVTT); | ||
| 1167 | value |= (APIC_LVT_MASKED | LOCAL_TIMER_VECTOR); | ||
| 1168 | apic_write(APIC_LVTT, value); | ||
| 1169 | } | ||
| 1170 | #endif | ||
| 1099 | 1171 | ||
| 1100 | setup_apic_nmi_watchdog(NULL); | 1172 | setup_apic_nmi_watchdog(NULL); |
| 1101 | apic_pm_activate(); | 1173 | apic_pm_activate(); |
| @@ -1205,7 +1277,7 @@ void __init init_apic_mappings(void) | |||
| 1205 | * default configuration (or the MP table is broken). | 1277 | * default configuration (or the MP table is broken). |
| 1206 | */ | 1278 | */ |
| 1207 | if (boot_cpu_physical_apicid == -1U) | 1279 | if (boot_cpu_physical_apicid == -1U) |
| 1208 | boot_cpu_physical_apicid = GET_APIC_ID(read_apic_id()); | 1280 | boot_cpu_physical_apicid = read_apic_id(); |
| 1209 | 1281 | ||
| 1210 | } | 1282 | } |
| 1211 | 1283 | ||
| @@ -1242,7 +1314,7 @@ int __init APIC_init_uniprocessor(void) | |||
| 1242 | * might be zero if read from MP tables. Get it from LAPIC. | 1314 | * might be zero if read from MP tables. Get it from LAPIC. |
| 1243 | */ | 1315 | */ |
| 1244 | #ifdef CONFIG_CRASH_DUMP | 1316 | #ifdef CONFIG_CRASH_DUMP |
| 1245 | boot_cpu_physical_apicid = GET_APIC_ID(read_apic_id()); | 1317 | boot_cpu_physical_apicid = read_apic_id(); |
| 1246 | #endif | 1318 | #endif |
| 1247 | physid_set_mask_of_physid(boot_cpu_physical_apicid, &phys_cpu_present_map); | 1319 | physid_set_mask_of_physid(boot_cpu_physical_apicid, &phys_cpu_present_map); |
| 1248 | 1320 | ||
| @@ -1321,59 +1393,12 @@ void smp_error_interrupt(struct pt_regs *regs) | |||
| 1321 | irq_exit(); | 1393 | irq_exit(); |
| 1322 | } | 1394 | } |
| 1323 | 1395 | ||
| 1324 | #ifdef CONFIG_SMP | ||
| 1325 | void __init smp_intr_init(void) | ||
| 1326 | { | ||
| 1327 | /* | ||
| 1328 | * IRQ0 must be given a fixed assignment and initialized, | ||
| 1329 | * because it's used before the IO-APIC is set up. | ||
| 1330 | */ | ||
| 1331 | set_intr_gate(FIRST_DEVICE_VECTOR, interrupt[0]); | ||
| 1332 | |||
| 1333 | /* | ||
| 1334 | * The reschedule interrupt is a CPU-to-CPU reschedule-helper | ||
| 1335 | * IPI, driven by wakeup. | ||
| 1336 | */ | ||
| 1337 | alloc_intr_gate(RESCHEDULE_VECTOR, reschedule_interrupt); | ||
| 1338 | |||
| 1339 | /* IPI for invalidation */ | ||
| 1340 | alloc_intr_gate(INVALIDATE_TLB_VECTOR, invalidate_interrupt); | ||
| 1341 | |||
| 1342 | /* IPI for generic function call */ | ||
| 1343 | alloc_intr_gate(CALL_FUNCTION_VECTOR, call_function_interrupt); | ||
| 1344 | |||
| 1345 | /* IPI for single call function */ | ||
| 1346 | set_intr_gate(CALL_FUNCTION_SINGLE_VECTOR, | ||
| 1347 | call_function_single_interrupt); | ||
| 1348 | } | ||
| 1349 | #endif | ||
| 1350 | |||
| 1351 | /* | ||
| 1352 | * Initialize APIC interrupts | ||
| 1353 | */ | ||
| 1354 | void __init apic_intr_init(void) | ||
| 1355 | { | ||
| 1356 | #ifdef CONFIG_SMP | ||
| 1357 | smp_intr_init(); | ||
| 1358 | #endif | ||
| 1359 | /* self generated IPI for local APIC timer */ | ||
| 1360 | alloc_intr_gate(LOCAL_TIMER_VECTOR, apic_timer_interrupt); | ||
| 1361 | |||
| 1362 | /* IPI vectors for APIC spurious and error interrupts */ | ||
| 1363 | alloc_intr_gate(SPURIOUS_APIC_VECTOR, spurious_interrupt); | ||
| 1364 | alloc_intr_gate(ERROR_APIC_VECTOR, error_interrupt); | ||
| 1365 | |||
| 1366 | /* thermal monitor LVT interrupt */ | ||
| 1367 | #ifdef CONFIG_X86_MCE_P4THERMAL | ||
| 1368 | alloc_intr_gate(THERMAL_APIC_VECTOR, thermal_interrupt); | ||
| 1369 | #endif | ||
| 1370 | } | ||
| 1371 | |||
| 1372 | /** | 1396 | /** |
| 1373 | * connect_bsp_APIC - attach the APIC to the interrupt system | 1397 | * connect_bsp_APIC - attach the APIC to the interrupt system |
| 1374 | */ | 1398 | */ |
| 1375 | void __init connect_bsp_APIC(void) | 1399 | void __init connect_bsp_APIC(void) |
| 1376 | { | 1400 | { |
| 1401 | #ifdef CONFIG_X86_32 | ||
| 1377 | if (pic_mode) { | 1402 | if (pic_mode) { |
| 1378 | /* | 1403 | /* |
| 1379 | * Do not trust the local APIC being empty at bootup. | 1404 | * Do not trust the local APIC being empty at bootup. |
| @@ -1388,6 +1413,7 @@ void __init connect_bsp_APIC(void) | |||
| 1388 | outb(0x70, 0x22); | 1413 | outb(0x70, 0x22); |
| 1389 | outb(0x01, 0x23); | 1414 | outb(0x01, 0x23); |
| 1390 | } | 1415 | } |
| 1416 | #endif | ||
| 1391 | enable_apic_mode(); | 1417 | enable_apic_mode(); |
| 1392 | } | 1418 | } |
| 1393 | 1419 | ||
| @@ -1400,6 +1426,9 @@ void __init connect_bsp_APIC(void) | |||
| 1400 | */ | 1426 | */ |
| 1401 | void disconnect_bsp_APIC(int virt_wire_setup) | 1427 | void disconnect_bsp_APIC(int virt_wire_setup) |
| 1402 | { | 1428 | { |
| 1429 | unsigned int value; | ||
| 1430 | |||
| 1431 | #ifdef CONFIG_X86_32 | ||
| 1403 | if (pic_mode) { | 1432 | if (pic_mode) { |
| 1404 | /* | 1433 | /* |
| 1405 | * Put the board back into PIC mode (has an effect only on | 1434 | * Put the board back into PIC mode (has an effect only on |
| @@ -1411,54 +1440,53 @@ void disconnect_bsp_APIC(int virt_wire_setup) | |||
| 1411 | "entering PIC mode.\n"); | 1440 | "entering PIC mode.\n"); |
| 1412 | outb(0x70, 0x22); | 1441 | outb(0x70, 0x22); |
| 1413 | outb(0x00, 0x23); | 1442 | outb(0x00, 0x23); |
| 1414 | } else { | 1443 | return; |
| 1415 | /* Go back to Virtual Wire compatibility mode */ | 1444 | } |
| 1416 | unsigned long value; | 1445 | #endif |
| 1417 | 1446 | ||
| 1418 | /* For the spurious interrupt use vector F, and enable it */ | 1447 | /* Go back to Virtual Wire compatibility mode */ |
| 1419 | value = apic_read(APIC_SPIV); | ||
| 1420 | value &= ~APIC_VECTOR_MASK; | ||
| 1421 | value |= APIC_SPIV_APIC_ENABLED; | ||
| 1422 | value |= 0xf; | ||
| 1423 | apic_write(APIC_SPIV, value); | ||
| 1424 | 1448 | ||
| 1425 | if (!virt_wire_setup) { | 1449 | /* For the spurious interrupt use vector F, and enable it */ |
| 1426 | /* | 1450 | value = apic_read(APIC_SPIV); |
| 1427 | * For LVT0 make it edge triggered, active high, | 1451 | value &= ~APIC_VECTOR_MASK; |
| 1428 | * external and enabled | 1452 | value |= APIC_SPIV_APIC_ENABLED; |
| 1429 | */ | 1453 | value |= 0xf; |
| 1430 | value = apic_read(APIC_LVT0); | 1454 | apic_write(APIC_SPIV, value); |
| 1431 | value &= ~(APIC_MODE_MASK | APIC_SEND_PENDING | | ||
| 1432 | APIC_INPUT_POLARITY | APIC_LVT_REMOTE_IRR | | ||
| 1433 | APIC_LVT_LEVEL_TRIGGER | APIC_LVT_MASKED); | ||
| 1434 | value |= APIC_LVT_REMOTE_IRR | APIC_SEND_PENDING; | ||
| 1435 | value = SET_APIC_DELIVERY_MODE(value, APIC_MODE_EXTINT); | ||
| 1436 | apic_write(APIC_LVT0, value); | ||
| 1437 | } else { | ||
| 1438 | /* Disable LVT0 */ | ||
| 1439 | apic_write(APIC_LVT0, APIC_LVT_MASKED); | ||
| 1440 | } | ||
| 1441 | 1455 | ||
| 1456 | if (!virt_wire_setup) { | ||
| 1442 | /* | 1457 | /* |
| 1443 | * For LVT1 make it edge triggered, active high, nmi and | 1458 | * For LVT0 make it edge triggered, active high, |
| 1444 | * enabled | 1459 | * external and enabled |
| 1445 | */ | 1460 | */ |
| 1446 | value = apic_read(APIC_LVT1); | 1461 | value = apic_read(APIC_LVT0); |
| 1447 | value &= ~( | 1462 | value &= ~(APIC_MODE_MASK | APIC_SEND_PENDING | |
| 1448 | APIC_MODE_MASK | APIC_SEND_PENDING | | ||
| 1449 | APIC_INPUT_POLARITY | APIC_LVT_REMOTE_IRR | | 1463 | APIC_INPUT_POLARITY | APIC_LVT_REMOTE_IRR | |
| 1450 | APIC_LVT_LEVEL_TRIGGER | APIC_LVT_MASKED); | 1464 | APIC_LVT_LEVEL_TRIGGER | APIC_LVT_MASKED); |
| 1451 | value |= APIC_LVT_REMOTE_IRR | APIC_SEND_PENDING; | 1465 | value |= APIC_LVT_REMOTE_IRR | APIC_SEND_PENDING; |
| 1452 | value = SET_APIC_DELIVERY_MODE(value, APIC_MODE_NMI); | 1466 | value = SET_APIC_DELIVERY_MODE(value, APIC_MODE_EXTINT); |
| 1453 | apic_write(APIC_LVT1, value); | 1467 | apic_write(APIC_LVT0, value); |
| 1468 | } else { | ||
| 1469 | /* Disable LVT0 */ | ||
| 1470 | apic_write(APIC_LVT0, APIC_LVT_MASKED); | ||
| 1454 | } | 1471 | } |
| 1472 | |||
| 1473 | /* | ||
| 1474 | * For LVT1 make it edge triggered, active high, | ||
| 1475 | * nmi and enabled | ||
| 1476 | */ | ||
| 1477 | value = apic_read(APIC_LVT1); | ||
| 1478 | value &= ~(APIC_MODE_MASK | APIC_SEND_PENDING | | ||
| 1479 | APIC_INPUT_POLARITY | APIC_LVT_REMOTE_IRR | | ||
| 1480 | APIC_LVT_LEVEL_TRIGGER | APIC_LVT_MASKED); | ||
| 1481 | value |= APIC_LVT_REMOTE_IRR | APIC_SEND_PENDING; | ||
| 1482 | value = SET_APIC_DELIVERY_MODE(value, APIC_MODE_NMI); | ||
| 1483 | apic_write(APIC_LVT1, value); | ||
| 1455 | } | 1484 | } |
| 1456 | 1485 | ||
| 1457 | void __cpuinit generic_processor_info(int apicid, int version) | 1486 | void __cpuinit generic_processor_info(int apicid, int version) |
| 1458 | { | 1487 | { |
| 1459 | int cpu; | 1488 | int cpu; |
| 1460 | cpumask_t tmp_map; | 1489 | cpumask_t tmp_map; |
| 1461 | physid_mask_t phys_cpu; | ||
| 1462 | 1490 | ||
| 1463 | /* | 1491 | /* |
| 1464 | * Validate version | 1492 | * Validate version |
| @@ -1471,9 +1499,6 @@ void __cpuinit generic_processor_info(int apicid, int version) | |||
| 1471 | } | 1499 | } |
| 1472 | apic_version[apicid] = version; | 1500 | apic_version[apicid] = version; |
| 1473 | 1501 | ||
| 1474 | phys_cpu = apicid_to_cpu_present(apicid); | ||
| 1475 | physids_or(phys_cpu_present_map, phys_cpu_present_map, phys_cpu); | ||
| 1476 | |||
| 1477 | if (num_processors >= NR_CPUS) { | 1502 | if (num_processors >= NR_CPUS) { |
| 1478 | printk(KERN_WARNING "WARNING: NR_CPUS limit of %i reached." | 1503 | printk(KERN_WARNING "WARNING: NR_CPUS limit of %i reached." |
| 1479 | " Processor ignored.\n", NR_CPUS); | 1504 | " Processor ignored.\n", NR_CPUS); |
| @@ -1484,17 +1509,19 @@ void __cpuinit generic_processor_info(int apicid, int version) | |||
| 1484 | cpus_complement(tmp_map, cpu_present_map); | 1509 | cpus_complement(tmp_map, cpu_present_map); |
| 1485 | cpu = first_cpu(tmp_map); | 1510 | cpu = first_cpu(tmp_map); |
| 1486 | 1511 | ||
| 1487 | if (apicid == boot_cpu_physical_apicid) | 1512 | physid_set(apicid, phys_cpu_present_map); |
| 1513 | if (apicid == boot_cpu_physical_apicid) { | ||
| 1488 | /* | 1514 | /* |
| 1489 | * x86_bios_cpu_apicid is required to have processors listed | 1515 | * x86_bios_cpu_apicid is required to have processors listed |
| 1490 | * in same order as logical cpu numbers. Hence the first | 1516 | * in same order as logical cpu numbers. Hence the first |
| 1491 | * entry is BSP, and so on. | 1517 | * entry is BSP, and so on. |
| 1492 | */ | 1518 | */ |
| 1493 | cpu = 0; | 1519 | cpu = 0; |
| 1494 | 1520 | } | |
| 1495 | if (apicid > max_physical_apicid) | 1521 | if (apicid > max_physical_apicid) |
| 1496 | max_physical_apicid = apicid; | 1522 | max_physical_apicid = apicid; |
| 1497 | 1523 | ||
| 1524 | #ifdef CONFIG_X86_32 | ||
| 1498 | /* | 1525 | /* |
| 1499 | * Would be preferable to switch to bigsmp when CONFIG_HOTPLUG_CPU=y | 1526 | * Would be preferable to switch to bigsmp when CONFIG_HOTPLUG_CPU=y |
| 1500 | * but we need to work other dependencies like SMP_SUSPEND etc | 1527 | * but we need to work other dependencies like SMP_SUSPEND etc |
| @@ -1514,7 +1541,9 @@ void __cpuinit generic_processor_info(int apicid, int version) | |||
| 1514 | def_to_bigsmp = 1; | 1541 | def_to_bigsmp = 1; |
| 1515 | } | 1542 | } |
| 1516 | } | 1543 | } |
| 1517 | #ifdef CONFIG_SMP | 1544 | #endif |
| 1545 | |||
| 1546 | #if defined(CONFIG_X86_SMP) || defined(CONFIG_X86_64) | ||
| 1518 | /* are we being called early in kernel startup? */ | 1547 | /* are we being called early in kernel startup? */ |
| 1519 | if (early_per_cpu_ptr(x86_cpu_to_apicid)) { | 1548 | if (early_per_cpu_ptr(x86_cpu_to_apicid)) { |
| 1520 | u16 *cpu_to_apicid = early_per_cpu_ptr(x86_cpu_to_apicid); | 1549 | u16 *cpu_to_apicid = early_per_cpu_ptr(x86_cpu_to_apicid); |
| @@ -1527,6 +1556,7 @@ void __cpuinit generic_processor_info(int apicid, int version) | |||
| 1527 | per_cpu(x86_bios_cpu_apicid, cpu) = apicid; | 1556 | per_cpu(x86_bios_cpu_apicid, cpu) = apicid; |
| 1528 | } | 1557 | } |
| 1529 | #endif | 1558 | #endif |
| 1559 | |||
| 1530 | cpu_set(cpu, cpu_possible_map); | 1560 | cpu_set(cpu, cpu_possible_map); |
| 1531 | cpu_set(cpu, cpu_present_map); | 1561 | cpu_set(cpu, cpu_present_map); |
| 1532 | } | 1562 | } |
| @@ -1537,6 +1567,11 @@ void __cpuinit generic_processor_info(int apicid, int version) | |||
| 1537 | #ifdef CONFIG_PM | 1567 | #ifdef CONFIG_PM |
| 1538 | 1568 | ||
| 1539 | static struct { | 1569 | static struct { |
| 1570 | /* | ||
| 1571 | * 'active' is true if the local APIC was enabled by us and | ||
| 1572 | * not the BIOS; this signifies that we are also responsible | ||
| 1573 | * for disabling it before entering apm/acpi suspend | ||
| 1574 | */ | ||
| 1540 | int active; | 1575 | int active; |
| 1541 | /* r/w apic fields */ | 1576 | /* r/w apic fields */ |
| 1542 | unsigned int apic_id; | 1577 | unsigned int apic_id; |
| @@ -1577,7 +1612,7 @@ static int lapic_suspend(struct sys_device *dev, pm_message_t state) | |||
| 1577 | apic_pm_state.apic_lvterr = apic_read(APIC_LVTERR); | 1612 | apic_pm_state.apic_lvterr = apic_read(APIC_LVTERR); |
| 1578 | apic_pm_state.apic_tmict = apic_read(APIC_TMICT); | 1613 | apic_pm_state.apic_tmict = apic_read(APIC_TMICT); |
| 1579 | apic_pm_state.apic_tdcr = apic_read(APIC_TDCR); | 1614 | apic_pm_state.apic_tdcr = apic_read(APIC_TDCR); |
| 1580 | #ifdef CONFIG_X86_MCE_P4THERMAL | 1615 | #if defined(CONFIG_X86_MCE_P4THERMAL) || defined(CONFIG_X86_MCE_INTEL) |
| 1581 | if (maxlvt >= 5) | 1616 | if (maxlvt >= 5) |
| 1582 | apic_pm_state.apic_thmr = apic_read(APIC_LVTTHMR); | 1617 | apic_pm_state.apic_thmr = apic_read(APIC_LVTTHMR); |
| 1583 | #endif | 1618 | #endif |
| @@ -1601,16 +1636,23 @@ static int lapic_resume(struct sys_device *dev) | |||
| 1601 | 1636 | ||
| 1602 | local_irq_save(flags); | 1637 | local_irq_save(flags); |
| 1603 | 1638 | ||
| 1604 | /* | 1639 | #ifdef CONFIG_X86_64 |
| 1605 | * Make sure the APICBASE points to the right address | 1640 | if (x2apic) |
| 1606 | * | 1641 | enable_x2apic(); |
| 1607 | * FIXME! This will be wrong if we ever support suspend on | 1642 | else |
| 1608 | * SMP! We'll need to do this as part of the CPU restore! | 1643 | #endif |
| 1609 | */ | 1644 | { |
| 1610 | rdmsr(MSR_IA32_APICBASE, l, h); | 1645 | /* |
| 1611 | l &= ~MSR_IA32_APICBASE_BASE; | 1646 | * Make sure the APICBASE points to the right address |
| 1612 | l |= MSR_IA32_APICBASE_ENABLE | mp_lapic_addr; | 1647 | * |
| 1613 | wrmsr(MSR_IA32_APICBASE, l, h); | 1648 | * FIXME! This will be wrong if we ever support suspend on |
| 1649 | * SMP! We'll need to do this as part of the CPU restore! | ||
| 1650 | */ | ||
| 1651 | rdmsr(MSR_IA32_APICBASE, l, h); | ||
| 1652 | l &= ~MSR_IA32_APICBASE_BASE; | ||
| 1653 | l |= MSR_IA32_APICBASE_ENABLE | mp_lapic_addr; | ||
| 1654 | wrmsr(MSR_IA32_APICBASE, l, h); | ||
| 1655 | } | ||
| 1614 | 1656 | ||
| 1615 | apic_write(APIC_LVTERR, ERROR_APIC_VECTOR | APIC_LVT_MASKED); | 1657 | apic_write(APIC_LVTERR, ERROR_APIC_VECTOR | APIC_LVT_MASKED); |
| 1616 | apic_write(APIC_ID, apic_pm_state.apic_id); | 1658 | apic_write(APIC_ID, apic_pm_state.apic_id); |
| @@ -1620,7 +1662,7 @@ static int lapic_resume(struct sys_device *dev) | |||
| 1620 | apic_write(APIC_SPIV, apic_pm_state.apic_spiv); | 1662 | apic_write(APIC_SPIV, apic_pm_state.apic_spiv); |
| 1621 | apic_write(APIC_LVT0, apic_pm_state.apic_lvt0); | 1663 | apic_write(APIC_LVT0, apic_pm_state.apic_lvt0); |
| 1622 | apic_write(APIC_LVT1, apic_pm_state.apic_lvt1); | 1664 | apic_write(APIC_LVT1, apic_pm_state.apic_lvt1); |
| 1623 | #ifdef CONFIG_X86_MCE_P4THERMAL | 1665 | #if defined(CONFIG_X86_MCE_P4THERMAL) || defined(CONFIG_X86_MCE_INTEL) |
| 1624 | if (maxlvt >= 5) | 1666 | if (maxlvt >= 5) |
| 1625 | apic_write(APIC_LVTTHMR, apic_pm_state.apic_thmr); | 1667 | apic_write(APIC_LVTTHMR, apic_pm_state.apic_thmr); |
| 1626 | #endif | 1668 | #endif |
| @@ -1634,7 +1676,9 @@ static int lapic_resume(struct sys_device *dev) | |||
| 1634 | apic_write(APIC_LVTERR, apic_pm_state.apic_lvterr); | 1676 | apic_write(APIC_LVTERR, apic_pm_state.apic_lvterr); |
| 1635 | apic_write(APIC_ESR, 0); | 1677 | apic_write(APIC_ESR, 0); |
| 1636 | apic_read(APIC_ESR); | 1678 | apic_read(APIC_ESR); |
| 1679 | |||
| 1637 | local_irq_restore(flags); | 1680 | local_irq_restore(flags); |
| 1681 | |||
| 1638 | return 0; | 1682 | return 0; |
| 1639 | } | 1683 | } |
| 1640 | 1684 | ||
| @@ -1690,20 +1734,20 @@ static int __init parse_lapic(char *arg) | |||
| 1690 | } | 1734 | } |
| 1691 | early_param("lapic", parse_lapic); | 1735 | early_param("lapic", parse_lapic); |
| 1692 | 1736 | ||
| 1693 | static int __init parse_nolapic(char *arg) | 1737 | static int __init setup_disableapic(char *arg) |
| 1694 | { | 1738 | { |
| 1695 | disable_apic = 1; | 1739 | disable_apic = 1; |
| 1696 | setup_clear_cpu_cap(X86_FEATURE_APIC); | 1740 | setup_clear_cpu_cap(X86_FEATURE_APIC); |
| 1697 | return 0; | 1741 | return 0; |
| 1698 | } | 1742 | } |
| 1699 | early_param("nolapic", parse_nolapic); | 1743 | early_param("disableapic", setup_disableapic); |
| 1700 | 1744 | ||
| 1701 | static int __init parse_disable_lapic_timer(char *arg) | 1745 | /* same as disableapic, for compatibility */ |
| 1746 | static int __init setup_nolapic(char *arg) | ||
| 1702 | { | 1747 | { |
| 1703 | local_apic_timer_disabled = 1; | 1748 | return setup_disableapic(arg); |
| 1704 | return 0; | ||
| 1705 | } | 1749 | } |
| 1706 | early_param("nolapic_timer", parse_disable_lapic_timer); | 1750 | early_param("nolapic", setup_nolapic); |
| 1707 | 1751 | ||
| 1708 | static int __init parse_lapic_timer_c2_ok(char *arg) | 1752 | static int __init parse_lapic_timer_c2_ok(char *arg) |
| 1709 | { | 1753 | { |
| @@ -1712,15 +1756,40 @@ static int __init parse_lapic_timer_c2_ok(char *arg) | |||
| 1712 | } | 1756 | } |
| 1713 | early_param("lapic_timer_c2_ok", parse_lapic_timer_c2_ok); | 1757 | early_param("lapic_timer_c2_ok", parse_lapic_timer_c2_ok); |
| 1714 | 1758 | ||
| 1759 | static int __init parse_disable_apic_timer(char *arg) | ||
| 1760 | { | ||
| 1761 | disable_apic_timer = 1; | ||
| 1762 | return 0; | ||
| 1763 | } | ||
| 1764 | early_param("noapictimer", parse_disable_apic_timer); | ||
| 1765 | |||
| 1766 | static int __init parse_nolapic_timer(char *arg) | ||
| 1767 | { | ||
| 1768 | disable_apic_timer = 1; | ||
| 1769 | return 0; | ||
| 1770 | } | ||
| 1771 | early_param("nolapic_timer", parse_nolapic_timer); | ||
| 1772 | |||
| 1715 | static int __init apic_set_verbosity(char *arg) | 1773 | static int __init apic_set_verbosity(char *arg) |
| 1716 | { | 1774 | { |
| 1717 | if (!arg) | 1775 | if (!arg) { |
| 1776 | #ifdef CONFIG_X86_64 | ||
| 1777 | skip_ioapic_setup = 0; | ||
| 1778 | ioapic_force = 1; | ||
| 1779 | return 0; | ||
| 1780 | #endif | ||
| 1718 | return -EINVAL; | 1781 | return -EINVAL; |
| 1782 | } | ||
| 1719 | 1783 | ||
| 1720 | if (strcmp(arg, "debug") == 0) | 1784 | if (strcmp("debug", arg) == 0) |
| 1721 | apic_verbosity = APIC_DEBUG; | 1785 | apic_verbosity = APIC_DEBUG; |
| 1722 | else if (strcmp(arg, "verbose") == 0) | 1786 | else if (strcmp("verbose", arg) == 0) |
| 1723 | apic_verbosity = APIC_VERBOSE; | 1787 | apic_verbosity = APIC_VERBOSE; |
| 1788 | else { | ||
| 1789 | printk(KERN_WARNING "APIC Verbosity level %s not recognised" | ||
| 1790 | " use apic=verbose or apic=debug\n", arg); | ||
| 1791 | return -EINVAL; | ||
| 1792 | } | ||
| 1724 | 1793 | ||
| 1725 | return 0; | 1794 | return 0; |
| 1726 | } | 1795 | } |
diff --git a/arch/x86/kernel/apic_64.c b/arch/x86/kernel/apic_64.c index 446c062e831c..53898b65a6ae 100644 --- a/arch/x86/kernel/apic_64.c +++ b/arch/x86/kernel/apic_64.c | |||
| @@ -27,6 +27,7 @@ | |||
| 27 | #include <linux/clockchips.h> | 27 | #include <linux/clockchips.h> |
| 28 | #include <linux/acpi_pmtmr.h> | 28 | #include <linux/acpi_pmtmr.h> |
| 29 | #include <linux/module.h> | 29 | #include <linux/module.h> |
| 30 | #include <linux/dmar.h> | ||
| 30 | 31 | ||
| 31 | #include <asm/atomic.h> | 32 | #include <asm/atomic.h> |
| 32 | #include <asm/smp.h> | 33 | #include <asm/smp.h> |
| @@ -39,13 +40,20 @@ | |||
| 39 | #include <asm/proto.h> | 40 | #include <asm/proto.h> |
| 40 | #include <asm/timex.h> | 41 | #include <asm/timex.h> |
| 41 | #include <asm/apic.h> | 42 | #include <asm/apic.h> |
| 43 | #include <asm/i8259.h> | ||
| 42 | 44 | ||
| 43 | #include <mach_ipi.h> | 45 | #include <mach_ipi.h> |
| 44 | #include <mach_apic.h> | 46 | #include <mach_apic.h> |
| 45 | 47 | ||
| 48 | /* Disable local APIC timer from the kernel commandline or via dmi quirk */ | ||
| 46 | static int disable_apic_timer __cpuinitdata; | 49 | static int disable_apic_timer __cpuinitdata; |
| 47 | static int apic_calibrate_pmtmr __initdata; | 50 | static int apic_calibrate_pmtmr __initdata; |
| 48 | int disable_apic; | 51 | int disable_apic; |
| 52 | int disable_x2apic; | ||
| 53 | int x2apic; | ||
| 54 | |||
| 55 | /* x2apic enabled before OS handover */ | ||
| 56 | int x2apic_preenabled; | ||
| 49 | 57 | ||
| 50 | /* Local APIC timer works in C2 */ | 58 | /* Local APIC timer works in C2 */ |
| 51 | int local_apic_timer_c2_ok; | 59 | int local_apic_timer_c2_ok; |
| @@ -73,6 +81,9 @@ static void lapic_timer_setup(enum clock_event_mode mode, | |||
| 73 | static void lapic_timer_broadcast(cpumask_t mask); | 81 | static void lapic_timer_broadcast(cpumask_t mask); |
| 74 | static void apic_pm_activate(void); | 82 | static void apic_pm_activate(void); |
| 75 | 83 | ||
| 84 | /* | ||
| 85 | * The local apic timer can be used for any function which is CPU local. | ||
| 86 | */ | ||
| 76 | static struct clock_event_device lapic_clockevent = { | 87 | static struct clock_event_device lapic_clockevent = { |
| 77 | .name = "lapic", | 88 | .name = "lapic", |
| 78 | .features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT | 89 | .features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT |
| @@ -99,11 +110,15 @@ static inline int lapic_get_version(void) | |||
| 99 | } | 110 | } |
| 100 | 111 | ||
| 101 | /* | 112 | /* |
| 102 | * Check, if the APIC is integrated or a seperate chip | 113 | * Check, if the APIC is integrated or a separate chip |
| 103 | */ | 114 | */ |
| 104 | static inline int lapic_is_integrated(void) | 115 | static inline int lapic_is_integrated(void) |
| 105 | { | 116 | { |
| 117 | #ifdef CONFIG_X86_64 | ||
| 106 | return 1; | 118 | return 1; |
| 119 | #else | ||
| 120 | return APIC_INTEGRATED(lapic_get_version()); | ||
| 121 | #endif | ||
| 107 | } | 122 | } |
| 108 | 123 | ||
| 109 | /* | 124 | /* |
| @@ -118,13 +133,18 @@ static int modern_apic(void) | |||
| 118 | return lapic_get_version() >= 0x14; | 133 | return lapic_get_version() >= 0x14; |
| 119 | } | 134 | } |
| 120 | 135 | ||
| 121 | void apic_wait_icr_idle(void) | 136 | /* |
| 137 | * Paravirt kernels also might be using these below ops. So we still | ||
| 138 | * use generic apic_read()/apic_write(), which might be pointing to different | ||
| 139 | * ops in PARAVIRT case. | ||
| 140 | */ | ||
| 141 | void xapic_wait_icr_idle(void) | ||
| 122 | { | 142 | { |
| 123 | while (apic_read(APIC_ICR) & APIC_ICR_BUSY) | 143 | while (apic_read(APIC_ICR) & APIC_ICR_BUSY) |
| 124 | cpu_relax(); | 144 | cpu_relax(); |
| 125 | } | 145 | } |
| 126 | 146 | ||
| 127 | u32 safe_apic_wait_icr_idle(void) | 147 | u32 safe_xapic_wait_icr_idle(void) |
| 128 | { | 148 | { |
| 129 | u32 send_status; | 149 | u32 send_status; |
| 130 | int timeout; | 150 | int timeout; |
| @@ -140,6 +160,68 @@ u32 safe_apic_wait_icr_idle(void) | |||
| 140 | return send_status; | 160 | return send_status; |
| 141 | } | 161 | } |
| 142 | 162 | ||
| 163 | void xapic_icr_write(u32 low, u32 id) | ||
| 164 | { | ||
| 165 | apic_write(APIC_ICR2, SET_APIC_DEST_FIELD(id)); | ||
| 166 | apic_write(APIC_ICR, low); | ||
| 167 | } | ||
| 168 | |||
| 169 | u64 xapic_icr_read(void) | ||
| 170 | { | ||
| 171 | u32 icr1, icr2; | ||
| 172 | |||
| 173 | icr2 = apic_read(APIC_ICR2); | ||
| 174 | icr1 = apic_read(APIC_ICR); | ||
| 175 | |||
| 176 | return icr1 | ((u64)icr2 << 32); | ||
| 177 | } | ||
| 178 | |||
| 179 | static struct apic_ops xapic_ops = { | ||
| 180 | .read = native_apic_mem_read, | ||
| 181 | .write = native_apic_mem_write, | ||
| 182 | .icr_read = xapic_icr_read, | ||
| 183 | .icr_write = xapic_icr_write, | ||
| 184 | .wait_icr_idle = xapic_wait_icr_idle, | ||
| 185 | .safe_wait_icr_idle = safe_xapic_wait_icr_idle, | ||
| 186 | }; | ||
| 187 | |||
| 188 | struct apic_ops __read_mostly *apic_ops = &xapic_ops; | ||
| 189 | EXPORT_SYMBOL_GPL(apic_ops); | ||
| 190 | |||
| 191 | static void x2apic_wait_icr_idle(void) | ||
| 192 | { | ||
| 193 | /* no need to wait for icr idle in x2apic */ | ||
| 194 | return; | ||
| 195 | } | ||
| 196 | |||
| 197 | static u32 safe_x2apic_wait_icr_idle(void) | ||
| 198 | { | ||
| 199 | /* no need to wait for icr idle in x2apic */ | ||
| 200 | return 0; | ||
| 201 | } | ||
| 202 | |||
| 203 | void x2apic_icr_write(u32 low, u32 id) | ||
| 204 | { | ||
| 205 | wrmsrl(APIC_BASE_MSR + (APIC_ICR >> 4), ((__u64) id) << 32 | low); | ||
| 206 | } | ||
| 207 | |||
| 208 | u64 x2apic_icr_read(void) | ||
| 209 | { | ||
| 210 | unsigned long val; | ||
| 211 | |||
| 212 | rdmsrl(APIC_BASE_MSR + (APIC_ICR >> 4), val); | ||
| 213 | return val; | ||
| 214 | } | ||
| 215 | |||
| 216 | static struct apic_ops x2apic_ops = { | ||
| 217 | .read = native_apic_msr_read, | ||
| 218 | .write = native_apic_msr_write, | ||
| 219 | .icr_read = x2apic_icr_read, | ||
| 220 | .icr_write = x2apic_icr_write, | ||
| 221 | .wait_icr_idle = x2apic_wait_icr_idle, | ||
| 222 | .safe_wait_icr_idle = safe_x2apic_wait_icr_idle, | ||
| 223 | }; | ||
| 224 | |||
| 143 | /** | 225 | /** |
| 144 | * enable_NMI_through_LVT0 - enable NMI through local vector table 0 | 226 | * enable_NMI_through_LVT0 - enable NMI through local vector table 0 |
| 145 | */ | 227 | */ |
| @@ -149,6 +231,11 @@ void __cpuinit enable_NMI_through_LVT0(void) | |||
| 149 | 231 | ||
| 150 | /* unmask and set to NMI */ | 232 | /* unmask and set to NMI */ |
| 151 | v = APIC_DM_NMI; | 233 | v = APIC_DM_NMI; |
| 234 | |||
| 235 | /* Level triggered for 82489DX (32bit mode) */ | ||
| 236 | if (!lapic_is_integrated()) | ||
| 237 | v |= APIC_LVT_LEVEL_TRIGGER; | ||
| 238 | |||
| 152 | apic_write(APIC_LVT0, v); | 239 | apic_write(APIC_LVT0, v); |
| 153 | } | 240 | } |
| 154 | 241 | ||
| @@ -157,14 +244,28 @@ void __cpuinit enable_NMI_through_LVT0(void) | |||
| 157 | */ | 244 | */ |
| 158 | int lapic_get_maxlvt(void) | 245 | int lapic_get_maxlvt(void) |
| 159 | { | 246 | { |
| 160 | unsigned int v, maxlvt; | 247 | unsigned int v; |
| 161 | 248 | ||
| 162 | v = apic_read(APIC_LVR); | 249 | v = apic_read(APIC_LVR); |
| 163 | maxlvt = GET_APIC_MAXLVT(v); | 250 | /* |
| 164 | return maxlvt; | 251 | * - we always have APIC integrated on 64bit mode |
| 252 | * - 82489DXs do not report # of LVT entries | ||
| 253 | */ | ||
| 254 | return APIC_INTEGRATED(GET_APIC_VERSION(v)) ? GET_APIC_MAXLVT(v) : 2; | ||
| 165 | } | 255 | } |
| 166 | 256 | ||
| 167 | /* | 257 | /* |
| 258 | * Local APIC timer | ||
| 259 | */ | ||
| 260 | |||
| 261 | /* Clock divisor */ | ||
| 262 | #ifdef CONFG_X86_64 | ||
| 263 | #define APIC_DIVISOR 1 | ||
| 264 | #else | ||
| 265 | #define APIC_DIVISOR 16 | ||
| 266 | #endif | ||
| 267 | |||
| 268 | /* | ||
| 168 | * This function sets up the local APIC timer, with a timeout of | 269 | * This function sets up the local APIC timer, with a timeout of |
| 169 | * 'clocks' APIC bus clock. During calibration we actually call | 270 | * 'clocks' APIC bus clock. During calibration we actually call |
| 170 | * this function twice on the boot CPU, once with a bogus timeout | 271 | * this function twice on the boot CPU, once with a bogus timeout |
| @@ -174,7 +275,6 @@ int lapic_get_maxlvt(void) | |||
| 174 | * We do reads before writes even if unnecessary, to get around the | 275 | * We do reads before writes even if unnecessary, to get around the |
| 175 | * P5 APIC double write bug. | 276 | * P5 APIC double write bug. |
| 176 | */ | 277 | */ |
| 177 | |||
| 178 | static void __setup_APIC_LVTT(unsigned int clocks, int oneshot, int irqen) | 278 | static void __setup_APIC_LVTT(unsigned int clocks, int oneshot, int irqen) |
| 179 | { | 279 | { |
| 180 | unsigned int lvtt_value, tmp_value; | 280 | unsigned int lvtt_value, tmp_value; |
| @@ -182,6 +282,9 @@ static void __setup_APIC_LVTT(unsigned int clocks, int oneshot, int irqen) | |||
| 182 | lvtt_value = LOCAL_TIMER_VECTOR; | 282 | lvtt_value = LOCAL_TIMER_VECTOR; |
| 183 | if (!oneshot) | 283 | if (!oneshot) |
| 184 | lvtt_value |= APIC_LVT_TIMER_PERIODIC; | 284 | lvtt_value |= APIC_LVT_TIMER_PERIODIC; |
| 285 | if (!lapic_is_integrated()) | ||
| 286 | lvtt_value |= SET_APIC_TIMER_BASE(APIC_TIMER_BASE_DIV); | ||
| 287 | |||
| 185 | if (!irqen) | 288 | if (!irqen) |
| 186 | lvtt_value |= APIC_LVT_MASKED; | 289 | lvtt_value |= APIC_LVT_MASKED; |
| 187 | 290 | ||
| @@ -191,12 +294,12 @@ static void __setup_APIC_LVTT(unsigned int clocks, int oneshot, int irqen) | |||
| 191 | * Divide PICLK by 16 | 294 | * Divide PICLK by 16 |
| 192 | */ | 295 | */ |
| 193 | tmp_value = apic_read(APIC_TDCR); | 296 | tmp_value = apic_read(APIC_TDCR); |
| 194 | apic_write(APIC_TDCR, (tmp_value | 297 | apic_write(APIC_TDCR, |
| 195 | & ~(APIC_TDR_DIV_1 | APIC_TDR_DIV_TMBASE)) | 298 | (tmp_value & ~(APIC_TDR_DIV_1 | APIC_TDR_DIV_TMBASE)) | |
| 196 | | APIC_TDR_DIV_16); | 299 | APIC_TDR_DIV_16); |
| 197 | 300 | ||
| 198 | if (!oneshot) | 301 | if (!oneshot) |
| 199 | apic_write(APIC_TMICT, clocks); | 302 | apic_write(APIC_TMICT, clocks / APIC_DIVISOR); |
| 200 | } | 303 | } |
| 201 | 304 | ||
| 202 | /* | 305 | /* |
| @@ -366,7 +469,7 @@ static int __init calibrate_APIC_clock(void) | |||
| 366 | lapic_clockevent.min_delta_ns = | 469 | lapic_clockevent.min_delta_ns = |
| 367 | clockevent_delta2ns(0xF, &lapic_clockevent); | 470 | clockevent_delta2ns(0xF, &lapic_clockevent); |
| 368 | 471 | ||
| 369 | calibration_result = result / HZ; | 472 | calibration_result = (result * APIC_DIVISOR) / HZ; |
| 370 | 473 | ||
| 371 | /* | 474 | /* |
| 372 | * Do a sanity check on the APIC calibration result | 475 | * Do a sanity check on the APIC calibration result |
| @@ -388,10 +491,10 @@ static int __init calibrate_APIC_clock(void) | |||
| 388 | void __init setup_boot_APIC_clock(void) | 491 | void __init setup_boot_APIC_clock(void) |
| 389 | { | 492 | { |
| 390 | /* | 493 | /* |
| 391 | * The local apic timer can be disabled via the kernel commandline. | 494 | * The local apic timer can be disabled via the kernel |
| 392 | * Register the lapic timer as a dummy clock event source on SMP | 495 | * commandline or from the CPU detection code. Register the lapic |
| 393 | * systems, so the broadcast mechanism is used. On UP systems simply | 496 | * timer as a dummy clock event source on SMP systems, so the |
| 394 | * ignore it. | 497 | * broadcast mechanism is used. On UP systems simply ignore it. |
| 395 | */ | 498 | */ |
| 396 | if (disable_apic_timer) { | 499 | if (disable_apic_timer) { |
| 397 | printk(KERN_INFO "Disabling APIC timer\n"); | 500 | printk(KERN_INFO "Disabling APIC timer\n"); |
| @@ -403,7 +506,9 @@ void __init setup_boot_APIC_clock(void) | |||
| 403 | return; | 506 | return; |
| 404 | } | 507 | } |
| 405 | 508 | ||
| 406 | printk(KERN_INFO "Using local APIC timer interrupts.\n"); | 509 | apic_printk(APIC_VERBOSE, "Using local APIC timer interrupts.\n" |
| 510 | "calibrating APIC timer ...\n"); | ||
| 511 | |||
| 407 | if (calibrate_APIC_clock()) { | 512 | if (calibrate_APIC_clock()) { |
| 408 | /* No broadcast on UP ! */ | 513 | /* No broadcast on UP ! */ |
| 409 | if (num_possible_cpus() > 1) | 514 | if (num_possible_cpus() > 1) |
| @@ -422,6 +527,7 @@ void __init setup_boot_APIC_clock(void) | |||
| 422 | printk(KERN_WARNING "APIC timer registered as dummy," | 527 | printk(KERN_WARNING "APIC timer registered as dummy," |
| 423 | " due to nmi_watchdog=%d!\n", nmi_watchdog); | 528 | " due to nmi_watchdog=%d!\n", nmi_watchdog); |
| 424 | 529 | ||
| 530 | /* Setup the lapic or request the broadcast */ | ||
| 425 | setup_APIC_timer(); | 531 | setup_APIC_timer(); |
| 426 | } | 532 | } |
| 427 | 533 | ||
| @@ -460,7 +566,11 @@ static void local_apic_timer_interrupt(void) | |||
| 460 | /* | 566 | /* |
| 461 | * the NMI deadlock-detector uses this. | 567 | * the NMI deadlock-detector uses this. |
| 462 | */ | 568 | */ |
| 569 | #ifdef CONFIG_X86_64 | ||
| 463 | add_pda(apic_timer_irqs, 1); | 570 | add_pda(apic_timer_irqs, 1); |
| 571 | #else | ||
| 572 | per_cpu(irq_stat, cpu).apic_timer_irqs++; | ||
| 573 | #endif | ||
| 464 | 574 | ||
| 465 | evt->event_handler(evt); | 575 | evt->event_handler(evt); |
| 466 | } | 576 | } |
| @@ -491,6 +601,7 @@ void smp_apic_timer_interrupt(struct pt_regs *regs) | |||
| 491 | irq_enter(); | 601 | irq_enter(); |
| 492 | local_apic_timer_interrupt(); | 602 | local_apic_timer_interrupt(); |
| 493 | irq_exit(); | 603 | irq_exit(); |
| 604 | |||
| 494 | set_irq_regs(old_regs); | 605 | set_irq_regs(old_regs); |
| 495 | } | 606 | } |
| 496 | 607 | ||
| @@ -544,6 +655,13 @@ void clear_local_APIC(void) | |||
| 544 | apic_write(APIC_LVTPC, v | APIC_LVT_MASKED); | 655 | apic_write(APIC_LVTPC, v | APIC_LVT_MASKED); |
| 545 | } | 656 | } |
| 546 | 657 | ||
| 658 | /* lets not touch this if we didn't frob it */ | ||
| 659 | #if defined(CONFIG_X86_MCE_P4THERMAL) || defined(X86_MCE_INTEL) | ||
| 660 | if (maxlvt >= 5) { | ||
| 661 | v = apic_read(APIC_LVTTHMR); | ||
| 662 | apic_write(APIC_LVTTHMR, v | APIC_LVT_MASKED); | ||
| 663 | } | ||
| 664 | #endif | ||
| 547 | /* | 665 | /* |
| 548 | * Clean APIC state for other OSs: | 666 | * Clean APIC state for other OSs: |
| 549 | */ | 667 | */ |
| @@ -554,8 +672,14 @@ void clear_local_APIC(void) | |||
| 554 | apic_write(APIC_LVTERR, APIC_LVT_MASKED); | 672 | apic_write(APIC_LVTERR, APIC_LVT_MASKED); |
| 555 | if (maxlvt >= 4) | 673 | if (maxlvt >= 4) |
| 556 | apic_write(APIC_LVTPC, APIC_LVT_MASKED); | 674 | apic_write(APIC_LVTPC, APIC_LVT_MASKED); |
| 557 | apic_write(APIC_ESR, 0); | 675 | |
| 558 | apic_read(APIC_ESR); | 676 | /* Integrated APIC (!82489DX) ? */ |
| 677 | if (lapic_is_integrated()) { | ||
| 678 | if (maxlvt > 3) | ||
| 679 | /* Clear ESR due to Pentium errata 3AP and 11AP */ | ||
| 680 | apic_write(APIC_ESR, 0); | ||
| 681 | apic_read(APIC_ESR); | ||
| 682 | } | ||
| 559 | } | 683 | } |
| 560 | 684 | ||
| 561 | /** | 685 | /** |
| @@ -574,8 +698,28 @@ void disable_local_APIC(void) | |||
| 574 | value = apic_read(APIC_SPIV); | 698 | value = apic_read(APIC_SPIV); |
| 575 | value &= ~APIC_SPIV_APIC_ENABLED; | 699 | value &= ~APIC_SPIV_APIC_ENABLED; |
| 576 | apic_write(APIC_SPIV, value); | 700 | apic_write(APIC_SPIV, value); |
| 701 | |||
| 702 | #ifdef CONFIG_X86_32 | ||
| 703 | /* | ||
| 704 | * When LAPIC was disabled by the BIOS and enabled by the kernel, | ||
| 705 | * restore the disabled state. | ||
| 706 | */ | ||
| 707 | if (enabled_via_apicbase) { | ||
| 708 | unsigned int l, h; | ||
| 709 | |||
| 710 | rdmsr(MSR_IA32_APICBASE, l, h); | ||
| 711 | l &= ~MSR_IA32_APICBASE_ENABLE; | ||
| 712 | wrmsr(MSR_IA32_APICBASE, l, h); | ||
| 713 | } | ||
| 714 | #endif | ||
| 577 | } | 715 | } |
| 578 | 716 | ||
| 717 | /* | ||
| 718 | * If Linux enabled the LAPIC against the BIOS default disable it down before | ||
| 719 | * re-entering the BIOS on shutdown. Otherwise the BIOS may get confused and | ||
| 720 | * not power-off. Additionally clear all LVT entries before disable_local_APIC | ||
| 721 | * for the case where Linux didn't enable the LAPIC. | ||
| 722 | */ | ||
| 579 | void lapic_shutdown(void) | 723 | void lapic_shutdown(void) |
| 580 | { | 724 | { |
| 581 | unsigned long flags; | 725 | unsigned long flags; |
| @@ -585,7 +729,13 @@ void lapic_shutdown(void) | |||
| 585 | 729 | ||
| 586 | local_irq_save(flags); | 730 | local_irq_save(flags); |
| 587 | 731 | ||
| 588 | disable_local_APIC(); | 732 | #ifdef CONFIG_X86_32 |
| 733 | if (!enabled_via_apicbase) | ||
| 734 | clear_local_APIC(); | ||
| 735 | else | ||
| 736 | #endif | ||
| 737 | disable_local_APIC(); | ||
| 738 | |||
| 589 | 739 | ||
| 590 | local_irq_restore(flags); | 740 | local_irq_restore(flags); |
| 591 | } | 741 | } |
| @@ -629,10 +779,10 @@ int __init verify_local_APIC(void) | |||
| 629 | /* | 779 | /* |
| 630 | * The ID register is read/write in a real APIC. | 780 | * The ID register is read/write in a real APIC. |
| 631 | */ | 781 | */ |
| 632 | reg0 = read_apic_id(); | 782 | reg0 = apic_read(APIC_ID); |
| 633 | apic_printk(APIC_DEBUG, "Getting ID: %x\n", reg0); | 783 | apic_printk(APIC_DEBUG, "Getting ID: %x\n", reg0); |
| 634 | apic_write(APIC_ID, reg0 ^ APIC_ID_MASK); | 784 | apic_write(APIC_ID, reg0 ^ APIC_ID_MASK); |
| 635 | reg1 = read_apic_id(); | 785 | reg1 = apic_read(APIC_ID); |
| 636 | apic_printk(APIC_DEBUG, "Getting ID: %x\n", reg1); | 786 | apic_printk(APIC_DEBUG, "Getting ID: %x\n", reg1); |
| 637 | apic_write(APIC_ID, reg0); | 787 | apic_write(APIC_ID, reg0); |
| 638 | if (reg1 != (reg0 ^ APIC_ID_MASK)) | 788 | if (reg1 != (reg0 ^ APIC_ID_MASK)) |
| @@ -656,8 +806,11 @@ int __init verify_local_APIC(void) | |||
| 656 | */ | 806 | */ |
| 657 | void __init sync_Arb_IDs(void) | 807 | void __init sync_Arb_IDs(void) |
| 658 | { | 808 | { |
| 659 | /* Unsupported on P4 - see Intel Dev. Manual Vol. 3, Ch. 8.6.1 */ | 809 | /* |
| 660 | if (modern_apic()) | 810 | * Unsupported on P4 - see Intel Dev. Manual Vol. 3, Ch. 8.6.1 And not |
| 811 | * needed on AMD. | ||
| 812 | */ | ||
| 813 | if (modern_apic() || boot_cpu_data.x86_vendor == X86_VENDOR_AMD) | ||
| 661 | return; | 814 | return; |
| 662 | 815 | ||
| 663 | /* | 816 | /* |
| @@ -666,8 +819,8 @@ void __init sync_Arb_IDs(void) | |||
| 666 | apic_wait_icr_idle(); | 819 | apic_wait_icr_idle(); |
| 667 | 820 | ||
| 668 | apic_printk(APIC_DEBUG, "Synchronizing Arb IDs.\n"); | 821 | apic_printk(APIC_DEBUG, "Synchronizing Arb IDs.\n"); |
| 669 | apic_write(APIC_ICR, APIC_DEST_ALLINC | APIC_INT_LEVELTRIG | 822 | apic_write(APIC_ICR, APIC_DEST_ALLINC | |
| 670 | | APIC_DM_INIT); | 823 | APIC_INT_LEVELTRIG | APIC_DM_INIT); |
| 671 | } | 824 | } |
| 672 | 825 | ||
| 673 | /* | 826 | /* |
| @@ -684,8 +837,6 @@ void __init init_bsp_APIC(void) | |||
| 684 | if (smp_found_config || !cpu_has_apic) | 837 | if (smp_found_config || !cpu_has_apic) |
| 685 | return; | 838 | return; |
| 686 | 839 | ||
| 687 | value = apic_read(APIC_LVR); | ||
| 688 | |||
| 689 | /* | 840 | /* |
| 690 | * Do not trust the local APIC being empty at bootup. | 841 | * Do not trust the local APIC being empty at bootup. |
| 691 | */ | 842 | */ |
| @@ -697,7 +848,15 @@ void __init init_bsp_APIC(void) | |||
| 697 | value = apic_read(APIC_SPIV); | 848 | value = apic_read(APIC_SPIV); |
| 698 | value &= ~APIC_VECTOR_MASK; | 849 | value &= ~APIC_VECTOR_MASK; |
| 699 | value |= APIC_SPIV_APIC_ENABLED; | 850 | value |= APIC_SPIV_APIC_ENABLED; |
| 700 | value |= APIC_SPIV_FOCUS_DISABLED; | 851 | |
| 852 | #ifdef CONFIG_X86_32 | ||
| 853 | /* This bit is reserved on P4/Xeon and should be cleared */ | ||
| 854 | if ((boot_cpu_data.x86_vendor == X86_VENDOR_INTEL) && | ||
| 855 | (boot_cpu_data.x86 == 15)) | ||
| 856 | value &= ~APIC_SPIV_FOCUS_DISABLED; | ||
| 857 | else | ||
| 858 | #endif | ||
| 859 | value |= APIC_SPIV_FOCUS_DISABLED; | ||
| 701 | value |= SPURIOUS_APIC_VECTOR; | 860 | value |= SPURIOUS_APIC_VECTOR; |
| 702 | apic_write(APIC_SPIV, value); | 861 | apic_write(APIC_SPIV, value); |
| 703 | 862 | ||
| @@ -706,9 +865,50 @@ void __init init_bsp_APIC(void) | |||
| 706 | */ | 865 | */ |
| 707 | apic_write(APIC_LVT0, APIC_DM_EXTINT); | 866 | apic_write(APIC_LVT0, APIC_DM_EXTINT); |
| 708 | value = APIC_DM_NMI; | 867 | value = APIC_DM_NMI; |
| 868 | if (!lapic_is_integrated()) /* 82489DX */ | ||
| 869 | value |= APIC_LVT_LEVEL_TRIGGER; | ||
| 709 | apic_write(APIC_LVT1, value); | 870 | apic_write(APIC_LVT1, value); |
| 710 | } | 871 | } |
| 711 | 872 | ||
| 873 | static void __cpuinit lapic_setup_esr(void) | ||
| 874 | { | ||
| 875 | unsigned long oldvalue, value, maxlvt; | ||
| 876 | if (lapic_is_integrated() && !esr_disable) { | ||
| 877 | if (esr_disable) { | ||
| 878 | /* | ||
| 879 | * Something untraceable is creating bad interrupts on | ||
| 880 | * secondary quads ... for the moment, just leave the | ||
| 881 | * ESR disabled - we can't do anything useful with the | ||
| 882 | * errors anyway - mbligh | ||
| 883 | */ | ||
| 884 | printk(KERN_INFO "Leaving ESR disabled.\n"); | ||
| 885 | return; | ||
| 886 | } | ||
| 887 | /* !82489DX */ | ||
| 888 | maxlvt = lapic_get_maxlvt(); | ||
| 889 | if (maxlvt > 3) /* Due to the Pentium erratum 3AP. */ | ||
| 890 | apic_write(APIC_ESR, 0); | ||
| 891 | oldvalue = apic_read(APIC_ESR); | ||
| 892 | |||
| 893 | /* enables sending errors */ | ||
| 894 | value = ERROR_APIC_VECTOR; | ||
| 895 | apic_write(APIC_LVTERR, value); | ||
| 896 | /* | ||
| 897 | * spec says clear errors after enabling vector. | ||
| 898 | */ | ||
| 899 | if (maxlvt > 3) | ||
| 900 | apic_write(APIC_ESR, 0); | ||
| 901 | value = apic_read(APIC_ESR); | ||
| 902 | if (value != oldvalue) | ||
| 903 | apic_printk(APIC_VERBOSE, "ESR value before enabling " | ||
| 904 | "vector: 0x%08lx after: 0x%08lx\n", | ||
| 905 | oldvalue, value); | ||
| 906 | } else { | ||
| 907 | printk(KERN_INFO "No ESR for 82489DX.\n"); | ||
| 908 | } | ||
| 909 | } | ||
| 910 | |||
| 911 | |||
| 712 | /** | 912 | /** |
| 713 | * setup_local_APIC - setup the local APIC | 913 | * setup_local_APIC - setup the local APIC |
| 714 | */ | 914 | */ |
| @@ -814,25 +1014,143 @@ void __cpuinit setup_local_APIC(void) | |||
| 814 | preempt_enable(); | 1014 | preempt_enable(); |
| 815 | } | 1015 | } |
| 816 | 1016 | ||
| 817 | static void __cpuinit lapic_setup_esr(void) | ||
| 818 | { | ||
| 819 | unsigned maxlvt = lapic_get_maxlvt(); | ||
| 820 | |||
| 821 | apic_write(APIC_LVTERR, ERROR_APIC_VECTOR); | ||
| 822 | /* | ||
| 823 | * spec says clear errors after enabling vector. | ||
| 824 | */ | ||
| 825 | if (maxlvt > 3) | ||
| 826 | apic_write(APIC_ESR, 0); | ||
| 827 | } | ||
| 828 | |||
| 829 | void __cpuinit end_local_APIC_setup(void) | 1017 | void __cpuinit end_local_APIC_setup(void) |
| 830 | { | 1018 | { |
| 831 | lapic_setup_esr(); | 1019 | lapic_setup_esr(); |
| 1020 | |||
| 1021 | #ifdef CONFIG_X86_32 | ||
| 1022 | { | ||
| 1023 | unsigned int value; | ||
| 1024 | /* Disable the local apic timer */ | ||
| 1025 | value = apic_read(APIC_LVTT); | ||
| 1026 | value |= (APIC_LVT_MASKED | LOCAL_TIMER_VECTOR); | ||
| 1027 | apic_write(APIC_LVTT, value); | ||
| 1028 | } | ||
| 1029 | #endif | ||
| 1030 | |||
| 832 | setup_apic_nmi_watchdog(NULL); | 1031 | setup_apic_nmi_watchdog(NULL); |
| 833 | apic_pm_activate(); | 1032 | apic_pm_activate(); |
| 834 | } | 1033 | } |
| 835 | 1034 | ||
| 1035 | void check_x2apic(void) | ||
| 1036 | { | ||
| 1037 | int msr, msr2; | ||
| 1038 | |||
| 1039 | rdmsr(MSR_IA32_APICBASE, msr, msr2); | ||
| 1040 | |||
| 1041 | if (msr & X2APIC_ENABLE) { | ||
| 1042 | printk("x2apic enabled by BIOS, switching to x2apic ops\n"); | ||
| 1043 | x2apic_preenabled = x2apic = 1; | ||
| 1044 | apic_ops = &x2apic_ops; | ||
| 1045 | } | ||
| 1046 | } | ||
| 1047 | |||
| 1048 | void enable_x2apic(void) | ||
| 1049 | { | ||
| 1050 | int msr, msr2; | ||
| 1051 | |||
| 1052 | rdmsr(MSR_IA32_APICBASE, msr, msr2); | ||
| 1053 | if (!(msr & X2APIC_ENABLE)) { | ||
| 1054 | printk("Enabling x2apic\n"); | ||
| 1055 | wrmsr(MSR_IA32_APICBASE, msr | X2APIC_ENABLE, 0); | ||
| 1056 | } | ||
| 1057 | } | ||
| 1058 | |||
| 1059 | void enable_IR_x2apic(void) | ||
| 1060 | { | ||
| 1061 | #ifdef CONFIG_INTR_REMAP | ||
| 1062 | int ret; | ||
| 1063 | unsigned long flags; | ||
| 1064 | |||
| 1065 | if (!cpu_has_x2apic) | ||
| 1066 | return; | ||
| 1067 | |||
| 1068 | if (!x2apic_preenabled && disable_x2apic) { | ||
| 1069 | printk(KERN_INFO | ||
| 1070 | "Skipped enabling x2apic and Interrupt-remapping " | ||
| 1071 | "because of nox2apic\n"); | ||
| 1072 | return; | ||
| 1073 | } | ||
| 1074 | |||
| 1075 | if (x2apic_preenabled && disable_x2apic) | ||
| 1076 | panic("Bios already enabled x2apic, can't enforce nox2apic"); | ||
| 1077 | |||
| 1078 | if (!x2apic_preenabled && skip_ioapic_setup) { | ||
| 1079 | printk(KERN_INFO | ||
| 1080 | "Skipped enabling x2apic and Interrupt-remapping " | ||
| 1081 | "because of skipping io-apic setup\n"); | ||
| 1082 | return; | ||
| 1083 | } | ||
| 1084 | |||
| 1085 | ret = dmar_table_init(); | ||
| 1086 | if (ret) { | ||
| 1087 | printk(KERN_INFO | ||
| 1088 | "dmar_table_init() failed with %d:\n", ret); | ||
| 1089 | |||
| 1090 | if (x2apic_preenabled) | ||
| 1091 | panic("x2apic enabled by bios. But IR enabling failed"); | ||
| 1092 | else | ||
| 1093 | printk(KERN_INFO | ||
| 1094 | "Not enabling x2apic,Intr-remapping\n"); | ||
| 1095 | return; | ||
| 1096 | } | ||
| 1097 | |||
| 1098 | local_irq_save(flags); | ||
| 1099 | mask_8259A(); | ||
| 1100 | save_mask_IO_APIC_setup(); | ||
| 1101 | |||
| 1102 | ret = enable_intr_remapping(1); | ||
| 1103 | |||
| 1104 | if (ret && x2apic_preenabled) { | ||
| 1105 | local_irq_restore(flags); | ||
| 1106 | panic("x2apic enabled by bios. But IR enabling failed"); | ||
| 1107 | } | ||
| 1108 | |||
| 1109 | if (ret) | ||
| 1110 | goto end; | ||
| 1111 | |||
| 1112 | if (!x2apic) { | ||
| 1113 | x2apic = 1; | ||
| 1114 | apic_ops = &x2apic_ops; | ||
| 1115 | enable_x2apic(); | ||
| 1116 | } | ||
| 1117 | end: | ||
| 1118 | if (ret) | ||
| 1119 | /* | ||
| 1120 | * IR enabling failed | ||
| 1121 | */ | ||
| 1122 | restore_IO_APIC_setup(); | ||
| 1123 | else | ||
| 1124 | reinit_intr_remapped_IO_APIC(x2apic_preenabled); | ||
| 1125 | |||
| 1126 | unmask_8259A(); | ||
| 1127 | local_irq_restore(flags); | ||
| 1128 | |||
| 1129 | if (!ret) { | ||
| 1130 | if (!x2apic_preenabled) | ||
| 1131 | printk(KERN_INFO | ||
| 1132 | "Enabled x2apic and interrupt-remapping\n"); | ||
| 1133 | else | ||
| 1134 | printk(KERN_INFO | ||
| 1135 | "Enabled Interrupt-remapping\n"); | ||
| 1136 | } else | ||
| 1137 | printk(KERN_ERR | ||
| 1138 | "Failed to enable Interrupt-remapping and x2apic\n"); | ||
| 1139 | #else | ||
| 1140 | if (!cpu_has_x2apic) | ||
| 1141 | return; | ||
| 1142 | |||
| 1143 | if (x2apic_preenabled) | ||
| 1144 | panic("x2apic enabled prior OS handover," | ||
| 1145 | " enable CONFIG_INTR_REMAP"); | ||
| 1146 | |||
| 1147 | printk(KERN_INFO "Enable CONFIG_INTR_REMAP for enabling intr-remapping " | ||
| 1148 | " and x2apic\n"); | ||
| 1149 | #endif | ||
| 1150 | |||
| 1151 | return; | ||
| 1152 | } | ||
| 1153 | |||
| 836 | /* | 1154 | /* |
| 837 | * Detect and enable local APICs on non-SMP boards. | 1155 | * Detect and enable local APICs on non-SMP boards. |
| 838 | * Original code written by Keir Fraser. | 1156 | * Original code written by Keir Fraser. |
| @@ -872,7 +1190,7 @@ void __init early_init_lapic_mapping(void) | |||
| 872 | * Fetch the APIC ID of the BSP in case we have a | 1190 | * Fetch the APIC ID of the BSP in case we have a |
| 873 | * default configuration (or the MP table is broken). | 1191 | * default configuration (or the MP table is broken). |
| 874 | */ | 1192 | */ |
| 875 | boot_cpu_physical_apicid = GET_APIC_ID(read_apic_id()); | 1193 | boot_cpu_physical_apicid = read_apic_id(); |
| 876 | } | 1194 | } |
| 877 | 1195 | ||
| 878 | /** | 1196 | /** |
| @@ -880,6 +1198,11 @@ void __init early_init_lapic_mapping(void) | |||
| 880 | */ | 1198 | */ |
| 881 | void __init init_apic_mappings(void) | 1199 | void __init init_apic_mappings(void) |
| 882 | { | 1200 | { |
| 1201 | if (x2apic) { | ||
| 1202 | boot_cpu_physical_apicid = read_apic_id(); | ||
| 1203 | return; | ||
| 1204 | } | ||
| 1205 | |||
| 883 | /* | 1206 | /* |
| 884 | * If no local APIC can be found then set up a fake all | 1207 | * If no local APIC can be found then set up a fake all |
| 885 | * zeroes page to simulate the local APIC and another | 1208 | * zeroes page to simulate the local APIC and another |
| @@ -899,13 +1222,15 @@ void __init init_apic_mappings(void) | |||
| 899 | * Fetch the APIC ID of the BSP in case we have a | 1222 | * Fetch the APIC ID of the BSP in case we have a |
| 900 | * default configuration (or the MP table is broken). | 1223 | * default configuration (or the MP table is broken). |
| 901 | */ | 1224 | */ |
| 902 | boot_cpu_physical_apicid = GET_APIC_ID(read_apic_id()); | 1225 | boot_cpu_physical_apicid = read_apic_id(); |
| 903 | } | 1226 | } |
| 904 | 1227 | ||
| 905 | /* | 1228 | /* |
| 906 | * This initializes the IO-APIC and APIC hardware if this is | 1229 | * This initializes the IO-APIC and APIC hardware if this is |
| 907 | * a UP kernel. | 1230 | * a UP kernel. |
| 908 | */ | 1231 | */ |
| 1232 | int apic_version[MAX_APICS]; | ||
| 1233 | |||
| 909 | int __init APIC_init_uniprocessor(void) | 1234 | int __init APIC_init_uniprocessor(void) |
| 910 | { | 1235 | { |
| 911 | if (disable_apic) { | 1236 | if (disable_apic) { |
| @@ -918,6 +1243,9 @@ int __init APIC_init_uniprocessor(void) | |||
| 918 | return -1; | 1243 | return -1; |
| 919 | } | 1244 | } |
| 920 | 1245 | ||
| 1246 | enable_IR_x2apic(); | ||
| 1247 | setup_apic_routing(); | ||
| 1248 | |||
| 921 | verify_local_APIC(); | 1249 | verify_local_APIC(); |
| 922 | 1250 | ||
| 923 | connect_bsp_APIC(); | 1251 | connect_bsp_APIC(); |
| @@ -1004,17 +1332,57 @@ asmlinkage void smp_error_interrupt(void) | |||
| 1004 | } | 1332 | } |
| 1005 | 1333 | ||
| 1006 | /** | 1334 | /** |
| 1007 | * * connect_bsp_APIC - attach the APIC to the interrupt system | 1335 | * connect_bsp_APIC - attach the APIC to the interrupt system |
| 1008 | * */ | 1336 | */ |
| 1009 | void __init connect_bsp_APIC(void) | 1337 | void __init connect_bsp_APIC(void) |
| 1010 | { | 1338 | { |
| 1339 | #ifdef CONFIG_X86_32 | ||
| 1340 | if (pic_mode) { | ||
| 1341 | /* | ||
| 1342 | * Do not trust the local APIC being empty at bootup. | ||
| 1343 | */ | ||
| 1344 | clear_local_APIC(); | ||
| 1345 | /* | ||
| 1346 | * PIC mode, enable APIC mode in the IMCR, i.e. connect BSP's | ||
| 1347 | * local APIC to INT and NMI lines. | ||
| 1348 | */ | ||
| 1349 | apic_printk(APIC_VERBOSE, "leaving PIC mode, " | ||
| 1350 | "enabling APIC mode.\n"); | ||
| 1351 | outb(0x70, 0x22); | ||
| 1352 | outb(0x01, 0x23); | ||
| 1353 | } | ||
| 1354 | #endif | ||
| 1011 | enable_apic_mode(); | 1355 | enable_apic_mode(); |
| 1012 | } | 1356 | } |
| 1013 | 1357 | ||
| 1358 | /** | ||
| 1359 | * disconnect_bsp_APIC - detach the APIC from the interrupt system | ||
| 1360 | * @virt_wire_setup: indicates, whether virtual wire mode is selected | ||
| 1361 | * | ||
| 1362 | * Virtual wire mode is necessary to deliver legacy interrupts even when the | ||
| 1363 | * APIC is disabled. | ||
| 1364 | */ | ||
| 1014 | void disconnect_bsp_APIC(int virt_wire_setup) | 1365 | void disconnect_bsp_APIC(int virt_wire_setup) |
| 1015 | { | 1366 | { |
| 1367 | unsigned int value; | ||
| 1368 | |||
| 1369 | #ifdef CONFIG_X86_32 | ||
| 1370 | if (pic_mode) { | ||
| 1371 | /* | ||
| 1372 | * Put the board back into PIC mode (has an effect only on | ||
| 1373 | * certain older boards). Note that APIC interrupts, including | ||
| 1374 | * IPIs, won't work beyond this point! The only exception are | ||
| 1375 | * INIT IPIs. | ||
| 1376 | */ | ||
| 1377 | apic_printk(APIC_VERBOSE, "disabling APIC mode, " | ||
| 1378 | "entering PIC mode.\n"); | ||
| 1379 | outb(0x70, 0x22); | ||
| 1380 | outb(0x00, 0x23); | ||
| 1381 | return; | ||
| 1382 | } | ||
| 1383 | #endif | ||
| 1384 | |||
| 1016 | /* Go back to Virtual Wire compatibility mode */ | 1385 | /* Go back to Virtual Wire compatibility mode */ |
| 1017 | unsigned long value; | ||
| 1018 | 1386 | ||
| 1019 | /* For the spurious interrupt use vector F, and enable it */ | 1387 | /* For the spurious interrupt use vector F, and enable it */ |
| 1020 | value = apic_read(APIC_SPIV); | 1388 | value = apic_read(APIC_SPIV); |
| @@ -1040,7 +1408,10 @@ void disconnect_bsp_APIC(int virt_wire_setup) | |||
| 1040 | apic_write(APIC_LVT0, APIC_LVT_MASKED); | 1408 | apic_write(APIC_LVT0, APIC_LVT_MASKED); |
| 1041 | } | 1409 | } |
| 1042 | 1410 | ||
| 1043 | /* For LVT1 make it edge triggered, active high, nmi and enabled */ | 1411 | /* |
| 1412 | * For LVT1 make it edge triggered, active high, | ||
| 1413 | * nmi and enabled | ||
| 1414 | */ | ||
| 1044 | value = apic_read(APIC_LVT1); | 1415 | value = apic_read(APIC_LVT1); |
| 1045 | value &= ~(APIC_MODE_MASK | APIC_SEND_PENDING | | 1416 | value &= ~(APIC_MODE_MASK | APIC_SEND_PENDING | |
| 1046 | APIC_INPUT_POLARITY | APIC_LVT_REMOTE_IRR | | 1417 | APIC_INPUT_POLARITY | APIC_LVT_REMOTE_IRR | |
| @@ -1055,9 +1426,20 @@ void __cpuinit generic_processor_info(int apicid, int version) | |||
| 1055 | int cpu; | 1426 | int cpu; |
| 1056 | cpumask_t tmp_map; | 1427 | cpumask_t tmp_map; |
| 1057 | 1428 | ||
| 1429 | /* | ||
| 1430 | * Validate version | ||
| 1431 | */ | ||
| 1432 | if (version == 0x0) { | ||
| 1433 | printk(KERN_WARNING "BIOS bug, APIC version is 0 for CPU#%d! " | ||
| 1434 | "fixing up to 0x10. (tell your hw vendor)\n", | ||
| 1435 | version); | ||
| 1436 | version = 0x10; | ||
| 1437 | } | ||
| 1438 | apic_version[apicid] = version; | ||
| 1439 | |||
| 1058 | if (num_processors >= NR_CPUS) { | 1440 | if (num_processors >= NR_CPUS) { |
| 1059 | printk(KERN_WARNING "WARNING: NR_CPUS limit of %i reached." | 1441 | printk(KERN_WARNING "WARNING: NR_CPUS limit of %i reached." |
| 1060 | " Processor ignored.\n", NR_CPUS); | 1442 | " Processor ignored.\n", NR_CPUS); |
| 1061 | return; | 1443 | return; |
| 1062 | } | 1444 | } |
| 1063 | 1445 | ||
| @@ -1077,6 +1459,29 @@ void __cpuinit generic_processor_info(int apicid, int version) | |||
| 1077 | if (apicid > max_physical_apicid) | 1459 | if (apicid > max_physical_apicid) |
| 1078 | max_physical_apicid = apicid; | 1460 | max_physical_apicid = apicid; |
| 1079 | 1461 | ||
| 1462 | #ifdef CONFIG_X86_32 | ||
| 1463 | /* | ||
| 1464 | * Would be preferable to switch to bigsmp when CONFIG_HOTPLUG_CPU=y | ||
| 1465 | * but we need to work other dependencies like SMP_SUSPEND etc | ||
| 1466 | * before this can be done without some confusion. | ||
| 1467 | * if (CPU_HOTPLUG_ENABLED || num_processors > 8) | ||
| 1468 | * - Ashok Raj <ashok.raj@intel.com> | ||
| 1469 | */ | ||
| 1470 | if (max_physical_apicid >= 8) { | ||
| 1471 | switch (boot_cpu_data.x86_vendor) { | ||
| 1472 | case X86_VENDOR_INTEL: | ||
| 1473 | if (!APIC_XAPIC(version)) { | ||
| 1474 | def_to_bigsmp = 0; | ||
| 1475 | break; | ||
| 1476 | } | ||
| 1477 | /* If P4 and above fall through */ | ||
| 1478 | case X86_VENDOR_AMD: | ||
| 1479 | def_to_bigsmp = 1; | ||
| 1480 | } | ||
| 1481 | } | ||
| 1482 | #endif | ||
| 1483 | |||
| 1484 | #if defined(CONFIG_X86_SMP) || defined(CONFIG_X86_64) | ||
| 1080 | /* are we being called early in kernel startup? */ | 1485 | /* are we being called early in kernel startup? */ |
| 1081 | if (early_per_cpu_ptr(x86_cpu_to_apicid)) { | 1486 | if (early_per_cpu_ptr(x86_cpu_to_apicid)) { |
| 1082 | u16 *cpu_to_apicid = early_per_cpu_ptr(x86_cpu_to_apicid); | 1487 | u16 *cpu_to_apicid = early_per_cpu_ptr(x86_cpu_to_apicid); |
| @@ -1088,20 +1493,28 @@ void __cpuinit generic_processor_info(int apicid, int version) | |||
| 1088 | per_cpu(x86_cpu_to_apicid, cpu) = apicid; | 1493 | per_cpu(x86_cpu_to_apicid, cpu) = apicid; |
| 1089 | per_cpu(x86_bios_cpu_apicid, cpu) = apicid; | 1494 | per_cpu(x86_bios_cpu_apicid, cpu) = apicid; |
| 1090 | } | 1495 | } |
| 1496 | #endif | ||
| 1091 | 1497 | ||
| 1092 | cpu_set(cpu, cpu_possible_map); | 1498 | cpu_set(cpu, cpu_possible_map); |
| 1093 | cpu_set(cpu, cpu_present_map); | 1499 | cpu_set(cpu, cpu_present_map); |
| 1094 | } | 1500 | } |
| 1095 | 1501 | ||
| 1502 | int hard_smp_processor_id(void) | ||
| 1503 | { | ||
| 1504 | return read_apic_id(); | ||
| 1505 | } | ||
| 1506 | |||
| 1096 | /* | 1507 | /* |
| 1097 | * Power management | 1508 | * Power management |
| 1098 | */ | 1509 | */ |
| 1099 | #ifdef CONFIG_PM | 1510 | #ifdef CONFIG_PM |
| 1100 | 1511 | ||
| 1101 | static struct { | 1512 | static struct { |
| 1102 | /* 'active' is true if the local APIC was enabled by us and | 1513 | /* |
| 1103 | not the BIOS; this signifies that we are also responsible | 1514 | * 'active' is true if the local APIC was enabled by us and |
| 1104 | for disabling it before entering apm/acpi suspend */ | 1515 | * not the BIOS; this signifies that we are also responsible |
| 1516 | * for disabling it before entering apm/acpi suspend | ||
| 1517 | */ | ||
| 1105 | int active; | 1518 | int active; |
| 1106 | /* r/w apic fields */ | 1519 | /* r/w apic fields */ |
| 1107 | unsigned int apic_id; | 1520 | unsigned int apic_id; |
| @@ -1129,7 +1542,7 @@ static int lapic_suspend(struct sys_device *dev, pm_message_t state) | |||
| 1129 | 1542 | ||
| 1130 | maxlvt = lapic_get_maxlvt(); | 1543 | maxlvt = lapic_get_maxlvt(); |
| 1131 | 1544 | ||
| 1132 | apic_pm_state.apic_id = read_apic_id(); | 1545 | apic_pm_state.apic_id = apic_read(APIC_ID); |
| 1133 | apic_pm_state.apic_taskpri = apic_read(APIC_TASKPRI); | 1546 | apic_pm_state.apic_taskpri = apic_read(APIC_TASKPRI); |
| 1134 | apic_pm_state.apic_ldr = apic_read(APIC_LDR); | 1547 | apic_pm_state.apic_ldr = apic_read(APIC_LDR); |
| 1135 | apic_pm_state.apic_dfr = apic_read(APIC_DFR); | 1548 | apic_pm_state.apic_dfr = apic_read(APIC_DFR); |
| @@ -1142,10 +1555,11 @@ static int lapic_suspend(struct sys_device *dev, pm_message_t state) | |||
| 1142 | apic_pm_state.apic_lvterr = apic_read(APIC_LVTERR); | 1555 | apic_pm_state.apic_lvterr = apic_read(APIC_LVTERR); |
| 1143 | apic_pm_state.apic_tmict = apic_read(APIC_TMICT); | 1556 | apic_pm_state.apic_tmict = apic_read(APIC_TMICT); |
| 1144 | apic_pm_state.apic_tdcr = apic_read(APIC_TDCR); | 1557 | apic_pm_state.apic_tdcr = apic_read(APIC_TDCR); |
| 1145 | #ifdef CONFIG_X86_MCE_INTEL | 1558 | #if defined(CONFIG_X86_MCE_P4THERMAL) || defined(CONFIG_X86_MCE_INTEL) |
| 1146 | if (maxlvt >= 5) | 1559 | if (maxlvt >= 5) |
| 1147 | apic_pm_state.apic_thmr = apic_read(APIC_LVTTHMR); | 1560 | apic_pm_state.apic_thmr = apic_read(APIC_LVTTHMR); |
| 1148 | #endif | 1561 | #endif |
| 1562 | |||
| 1149 | local_irq_save(flags); | 1563 | local_irq_save(flags); |
| 1150 | disable_local_APIC(); | 1564 | disable_local_APIC(); |
| 1151 | local_irq_restore(flags); | 1565 | local_irq_restore(flags); |
| @@ -1164,10 +1578,25 @@ static int lapic_resume(struct sys_device *dev) | |||
| 1164 | maxlvt = lapic_get_maxlvt(); | 1578 | maxlvt = lapic_get_maxlvt(); |
| 1165 | 1579 | ||
| 1166 | local_irq_save(flags); | 1580 | local_irq_save(flags); |
| 1167 | rdmsr(MSR_IA32_APICBASE, l, h); | 1581 | |
| 1168 | l &= ~MSR_IA32_APICBASE_BASE; | 1582 | #ifdef CONFIG_X86_64 |
| 1169 | l |= MSR_IA32_APICBASE_ENABLE | mp_lapic_addr; | 1583 | if (x2apic) |
| 1170 | wrmsr(MSR_IA32_APICBASE, l, h); | 1584 | enable_x2apic(); |
| 1585 | else | ||
| 1586 | #endif | ||
| 1587 | { | ||
| 1588 | /* | ||
| 1589 | * Make sure the APICBASE points to the right address | ||
| 1590 | * | ||
| 1591 | * FIXME! This will be wrong if we ever support suspend on | ||
| 1592 | * SMP! We'll need to do this as part of the CPU restore! | ||
| 1593 | */ | ||
| 1594 | rdmsr(MSR_IA32_APICBASE, l, h); | ||
| 1595 | l &= ~MSR_IA32_APICBASE_BASE; | ||
| 1596 | l |= MSR_IA32_APICBASE_ENABLE | mp_lapic_addr; | ||
| 1597 | wrmsr(MSR_IA32_APICBASE, l, h); | ||
| 1598 | } | ||
| 1599 | |||
| 1171 | apic_write(APIC_LVTERR, ERROR_APIC_VECTOR | APIC_LVT_MASKED); | 1600 | apic_write(APIC_LVTERR, ERROR_APIC_VECTOR | APIC_LVT_MASKED); |
| 1172 | apic_write(APIC_ID, apic_pm_state.apic_id); | 1601 | apic_write(APIC_ID, apic_pm_state.apic_id); |
| 1173 | apic_write(APIC_DFR, apic_pm_state.apic_dfr); | 1602 | apic_write(APIC_DFR, apic_pm_state.apic_dfr); |
| @@ -1176,7 +1605,7 @@ static int lapic_resume(struct sys_device *dev) | |||
| 1176 | apic_write(APIC_SPIV, apic_pm_state.apic_spiv); | 1605 | apic_write(APIC_SPIV, apic_pm_state.apic_spiv); |
| 1177 | apic_write(APIC_LVT0, apic_pm_state.apic_lvt0); | 1606 | apic_write(APIC_LVT0, apic_pm_state.apic_lvt0); |
| 1178 | apic_write(APIC_LVT1, apic_pm_state.apic_lvt1); | 1607 | apic_write(APIC_LVT1, apic_pm_state.apic_lvt1); |
| 1179 | #ifdef CONFIG_X86_MCE_INTEL | 1608 | #if defined(CONFIG_X86_MCE_P4THERMAL) || defined(CONFIG_X86_MCE_INTEL) |
| 1180 | if (maxlvt >= 5) | 1609 | if (maxlvt >= 5) |
| 1181 | apic_write(APIC_LVTTHMR, apic_pm_state.apic_thmr); | 1610 | apic_write(APIC_LVTTHMR, apic_pm_state.apic_thmr); |
| 1182 | #endif | 1611 | #endif |
| @@ -1190,10 +1619,17 @@ static int lapic_resume(struct sys_device *dev) | |||
| 1190 | apic_write(APIC_LVTERR, apic_pm_state.apic_lvterr); | 1619 | apic_write(APIC_LVTERR, apic_pm_state.apic_lvterr); |
| 1191 | apic_write(APIC_ESR, 0); | 1620 | apic_write(APIC_ESR, 0); |
| 1192 | apic_read(APIC_ESR); | 1621 | apic_read(APIC_ESR); |
| 1622 | |||
| 1193 | local_irq_restore(flags); | 1623 | local_irq_restore(flags); |
| 1624 | |||
| 1194 | return 0; | 1625 | return 0; |
| 1195 | } | 1626 | } |
| 1196 | 1627 | ||
| 1628 | /* | ||
| 1629 | * This device has no shutdown method - fully functioning local APICs | ||
| 1630 | * are needed on every CPU up until machine_halt/restart/poweroff. | ||
| 1631 | */ | ||
| 1632 | |||
| 1197 | static struct sysdev_class lapic_sysclass = { | 1633 | static struct sysdev_class lapic_sysclass = { |
| 1198 | .name = "lapic", | 1634 | .name = "lapic", |
| 1199 | .resume = lapic_resume, | 1635 | .resume = lapic_resume, |
| @@ -1307,31 +1743,19 @@ __cpuinit int apic_is_clustered_box(void) | |||
| 1307 | return (clusters > 2); | 1743 | return (clusters > 2); |
| 1308 | } | 1744 | } |
| 1309 | 1745 | ||
| 1310 | /* | 1746 | static __init int setup_nox2apic(char *str) |
| 1311 | * APIC command line parameters | ||
| 1312 | */ | ||
| 1313 | static int __init apic_set_verbosity(char *str) | ||
| 1314 | { | 1747 | { |
| 1315 | if (str == NULL) { | 1748 | disable_x2apic = 1; |
| 1316 | skip_ioapic_setup = 0; | 1749 | clear_cpu_cap(&boot_cpu_data, X86_FEATURE_X2APIC); |
| 1317 | ioapic_force = 1; | ||
| 1318 | return 0; | ||
| 1319 | } | ||
| 1320 | if (strcmp("debug", str) == 0) | ||
| 1321 | apic_verbosity = APIC_DEBUG; | ||
| 1322 | else if (strcmp("verbose", str) == 0) | ||
| 1323 | apic_verbosity = APIC_VERBOSE; | ||
| 1324 | else { | ||
| 1325 | printk(KERN_WARNING "APIC Verbosity level %s not recognised" | ||
| 1326 | " use apic=verbose or apic=debug\n", str); | ||
| 1327 | return -EINVAL; | ||
| 1328 | } | ||
| 1329 | |||
| 1330 | return 0; | 1750 | return 0; |
| 1331 | } | 1751 | } |
| 1332 | early_param("apic", apic_set_verbosity); | 1752 | early_param("nox2apic", setup_nox2apic); |
| 1753 | |||
| 1333 | 1754 | ||
| 1334 | static __init int setup_disableapic(char *str) | 1755 | /* |
| 1756 | * APIC command line parameters | ||
| 1757 | */ | ||
| 1758 | static int __init setup_disableapic(char *arg) | ||
| 1335 | { | 1759 | { |
| 1336 | disable_apic = 1; | 1760 | disable_apic = 1; |
| 1337 | setup_clear_cpu_cap(X86_FEATURE_APIC); | 1761 | setup_clear_cpu_cap(X86_FEATURE_APIC); |
| @@ -1340,9 +1764,9 @@ static __init int setup_disableapic(char *str) | |||
| 1340 | early_param("disableapic", setup_disableapic); | 1764 | early_param("disableapic", setup_disableapic); |
| 1341 | 1765 | ||
| 1342 | /* same as disableapic, for compatibility */ | 1766 | /* same as disableapic, for compatibility */ |
| 1343 | static __init int setup_nolapic(char *str) | 1767 | static int __init setup_nolapic(char *arg) |
| 1344 | { | 1768 | { |
| 1345 | return setup_disableapic(str); | 1769 | return setup_disableapic(arg); |
| 1346 | } | 1770 | } |
| 1347 | early_param("nolapic", setup_nolapic); | 1771 | early_param("nolapic", setup_nolapic); |
| 1348 | 1772 | ||
| @@ -1353,14 +1777,19 @@ static int __init parse_lapic_timer_c2_ok(char *arg) | |||
| 1353 | } | 1777 | } |
| 1354 | early_param("lapic_timer_c2_ok", parse_lapic_timer_c2_ok); | 1778 | early_param("lapic_timer_c2_ok", parse_lapic_timer_c2_ok); |
| 1355 | 1779 | ||
| 1356 | static __init int setup_noapictimer(char *str) | 1780 | static int __init parse_disable_apic_timer(char *arg) |
| 1357 | { | 1781 | { |
| 1358 | if (str[0] != ' ' && str[0] != 0) | ||
| 1359 | return 0; | ||
| 1360 | disable_apic_timer = 1; | 1782 | disable_apic_timer = 1; |
| 1361 | return 1; | 1783 | return 0; |
| 1362 | } | 1784 | } |
| 1363 | __setup("noapictimer", setup_noapictimer); | 1785 | early_param("noapictimer", parse_disable_apic_timer); |
| 1786 | |||
| 1787 | static int __init parse_nolapic_timer(char *arg) | ||
| 1788 | { | ||
| 1789 | disable_apic_timer = 1; | ||
| 1790 | return 0; | ||
| 1791 | } | ||
| 1792 | early_param("nolapic_timer", parse_nolapic_timer); | ||
| 1364 | 1793 | ||
| 1365 | static __init int setup_apicpmtimer(char *s) | 1794 | static __init int setup_apicpmtimer(char *s) |
| 1366 | { | 1795 | { |
| @@ -1370,6 +1799,31 @@ static __init int setup_apicpmtimer(char *s) | |||
| 1370 | } | 1799 | } |
| 1371 | __setup("apicpmtimer", setup_apicpmtimer); | 1800 | __setup("apicpmtimer", setup_apicpmtimer); |
| 1372 | 1801 | ||
| 1802 | static int __init apic_set_verbosity(char *arg) | ||
| 1803 | { | ||
| 1804 | if (!arg) { | ||
| 1805 | #ifdef CONFIG_X86_64 | ||
| 1806 | skip_ioapic_setup = 0; | ||
| 1807 | ioapic_force = 1; | ||
| 1808 | return 0; | ||
| 1809 | #endif | ||
| 1810 | return -EINVAL; | ||
| 1811 | } | ||
| 1812 | |||
| 1813 | if (strcmp("debug", arg) == 0) | ||
| 1814 | apic_verbosity = APIC_DEBUG; | ||
| 1815 | else if (strcmp("verbose", arg) == 0) | ||
| 1816 | apic_verbosity = APIC_VERBOSE; | ||
| 1817 | else { | ||
| 1818 | printk(KERN_WARNING "APIC Verbosity level %s not recognised" | ||
| 1819 | " use apic=verbose or apic=debug\n", arg); | ||
| 1820 | return -EINVAL; | ||
| 1821 | } | ||
| 1822 | |||
| 1823 | return 0; | ||
| 1824 | } | ||
| 1825 | early_param("apic", apic_set_verbosity); | ||
| 1826 | |||
| 1373 | static int __init lapic_insert_resource(void) | 1827 | static int __init lapic_insert_resource(void) |
| 1374 | { | 1828 | { |
| 1375 | if (!apic_phys) | 1829 | if (!apic_phys) |
diff --git a/arch/x86/kernel/cpu/common_64.c b/arch/x86/kernel/cpu/common_64.c index 305b465889b0..43f1aa51da5d 100644 --- a/arch/x86/kernel/cpu/common_64.c +++ b/arch/x86/kernel/cpu/common_64.c | |||
| @@ -687,6 +687,8 @@ void __cpuinit cpu_init(void) | |||
| 687 | barrier(); | 687 | barrier(); |
| 688 | 688 | ||
| 689 | check_efer(); | 689 | check_efer(); |
| 690 | if (cpu != 0 && x2apic) | ||
| 691 | enable_x2apic(); | ||
| 690 | 692 | ||
| 691 | /* | 693 | /* |
| 692 | * set up and load the per-CPU TSS | 694 | * set up and load the per-CPU TSS |
diff --git a/arch/x86/kernel/cpu/feature_names.c b/arch/x86/kernel/cpu/feature_names.c index c9017799497c..b96b69545fbf 100644 --- a/arch/x86/kernel/cpu/feature_names.c +++ b/arch/x86/kernel/cpu/feature_names.c | |||
| @@ -46,7 +46,7 @@ const char * const x86_cap_flags[NCAPINTS*32] = { | |||
| 46 | /* Intel-defined (#2) */ | 46 | /* Intel-defined (#2) */ |
| 47 | "pni", NULL, NULL, "monitor", "ds_cpl", "vmx", "smx", "est", | 47 | "pni", NULL, NULL, "monitor", "ds_cpl", "vmx", "smx", "est", |
| 48 | "tm2", "ssse3", "cid", NULL, NULL, "cx16", "xtpr", NULL, | 48 | "tm2", "ssse3", "cid", NULL, NULL, "cx16", "xtpr", NULL, |
| 49 | NULL, NULL, "dca", "sse4_1", "sse4_2", NULL, NULL, "popcnt", | 49 | NULL, NULL, "dca", "sse4_1", "sse4_2", "x2apic", NULL, "popcnt", |
| 50 | NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, | 50 | NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, |
| 51 | 51 | ||
| 52 | /* VIA/Cyrix/Centaur-defined */ | 52 | /* VIA/Cyrix/Centaur-defined */ |
diff --git a/arch/x86/kernel/genapic_64.c b/arch/x86/kernel/genapic_64.c index eaff0bbb1444..6c9bfc9e1e95 100644 --- a/arch/x86/kernel/genapic_64.c +++ b/arch/x86/kernel/genapic_64.c | |||
| @@ -16,87 +16,63 @@ | |||
| 16 | #include <linux/ctype.h> | 16 | #include <linux/ctype.h> |
| 17 | #include <linux/init.h> | 17 | #include <linux/init.h> |
| 18 | #include <linux/hardirq.h> | 18 | #include <linux/hardirq.h> |
| 19 | #include <linux/dmar.h> | ||
| 19 | 20 | ||
| 20 | #include <asm/smp.h> | 21 | #include <asm/smp.h> |
| 21 | #include <asm/ipi.h> | 22 | #include <asm/ipi.h> |
| 22 | #include <asm/genapic.h> | 23 | #include <asm/genapic.h> |
| 23 | 24 | ||
| 24 | #ifdef CONFIG_ACPI | 25 | extern struct genapic apic_flat; |
| 25 | #include <acpi/acpi_bus.h> | 26 | extern struct genapic apic_physflat; |
| 26 | #endif | 27 | extern struct genapic apic_x2xpic_uv_x; |
| 27 | 28 | extern struct genapic apic_x2apic_phys; | |
| 28 | DEFINE_PER_CPU(int, x2apic_extra_bits); | 29 | extern struct genapic apic_x2apic_cluster; |
| 29 | 30 | ||
| 30 | struct genapic __read_mostly *genapic = &apic_flat; | 31 | struct genapic __read_mostly *genapic = &apic_flat; |
| 31 | 32 | ||
| 32 | static enum uv_system_type uv_system_type; | 33 | static struct genapic *apic_probe[] __initdata = { |
| 34 | &apic_x2apic_uv_x, | ||
| 35 | &apic_x2apic_phys, | ||
| 36 | &apic_x2apic_cluster, | ||
| 37 | &apic_physflat, | ||
| 38 | NULL, | ||
| 39 | }; | ||
| 33 | 40 | ||
| 34 | /* | 41 | /* |
| 35 | * Check the APIC IDs in bios_cpu_apicid and choose the APIC mode. | 42 | * Check the APIC IDs in bios_cpu_apicid and choose the APIC mode. |
| 36 | */ | 43 | */ |
| 37 | void __init setup_apic_routing(void) | 44 | void __init setup_apic_routing(void) |
| 38 | { | 45 | { |
| 39 | if (uv_system_type == UV_NON_UNIQUE_APIC) | 46 | if (genapic == &apic_x2apic_phys || genapic == &apic_x2apic_cluster) { |
| 40 | genapic = &apic_x2apic_uv_x; | 47 | if (!intr_remapping_enabled) |
| 41 | else | 48 | genapic = &apic_flat; |
| 42 | #ifdef CONFIG_ACPI | 49 | } |
| 43 | /* | ||
| 44 | * Quirk: some x86_64 machines can only use physical APIC mode | ||
| 45 | * regardless of how many processors are present (x86_64 ES7000 | ||
| 46 | * is an example). | ||
| 47 | */ | ||
| 48 | if (acpi_gbl_FADT.header.revision > FADT2_REVISION_ID && | ||
| 49 | (acpi_gbl_FADT.flags & ACPI_FADT_APIC_PHYSICAL)) | ||
| 50 | genapic = &apic_physflat; | ||
| 51 | else | ||
| 52 | #endif | ||
| 53 | |||
| 54 | if (max_physical_apicid < 8) | ||
| 55 | genapic = &apic_flat; | ||
| 56 | else | ||
| 57 | genapic = &apic_physflat; | ||
| 58 | 50 | ||
| 59 | printk(KERN_INFO "Setting APIC routing to %s\n", genapic->name); | 51 | if (genapic == &apic_flat) { |
| 52 | if (max_physical_apicid >= 8) | ||
| 53 | genapic = &apic_physflat; | ||
| 54 | printk(KERN_INFO "Setting APIC routing to %s\n", genapic->name); | ||
| 55 | } | ||
| 60 | } | 56 | } |
| 61 | 57 | ||
| 62 | /* Same for both flat and physical. */ | 58 | /* Same for both flat and physical. */ |
| 63 | 59 | ||
| 64 | void send_IPI_self(int vector) | 60 | void apic_send_IPI_self(int vector) |
| 65 | { | 61 | { |
| 66 | __send_IPI_shortcut(APIC_DEST_SELF, vector, APIC_DEST_PHYSICAL); | 62 | __send_IPI_shortcut(APIC_DEST_SELF, vector, APIC_DEST_PHYSICAL); |
| 67 | } | 63 | } |
| 68 | 64 | ||
| 69 | int __init acpi_madt_oem_check(char *oem_id, char *oem_table_id) | 65 | int __init acpi_madt_oem_check(char *oem_id, char *oem_table_id) |
| 70 | { | 66 | { |
| 71 | if (!strcmp(oem_id, "SGI")) { | 67 | int i; |
| 72 | if (!strcmp(oem_table_id, "UVL")) | 68 | |
| 73 | uv_system_type = UV_LEGACY_APIC; | 69 | for (i = 0; apic_probe[i]; ++i) { |
| 74 | else if (!strcmp(oem_table_id, "UVX")) | 70 | if (apic_probe[i]->acpi_madt_oem_check(oem_id, oem_table_id)) { |
| 75 | uv_system_type = UV_X2APIC; | 71 | genapic = apic_probe[i]; |
| 76 | else if (!strcmp(oem_table_id, "UVH")) | 72 | printk(KERN_INFO "Setting APIC routing to %s.\n", |
| 77 | uv_system_type = UV_NON_UNIQUE_APIC; | 73 | genapic->name); |
| 74 | return 1; | ||
| 75 | } | ||
| 78 | } | 76 | } |
| 79 | return 0; | 77 | return 0; |
| 80 | } | 78 | } |
| 81 | |||
| 82 | unsigned int read_apic_id(void) | ||
| 83 | { | ||
| 84 | unsigned int id; | ||
| 85 | |||
| 86 | WARN_ON(preemptible() && num_online_cpus() > 1); | ||
| 87 | id = apic_read(APIC_ID); | ||
| 88 | if (uv_system_type >= UV_X2APIC) | ||
| 89 | id |= __get_cpu_var(x2apic_extra_bits); | ||
| 90 | return id; | ||
| 91 | } | ||
| 92 | |||
| 93 | enum uv_system_type get_uv_system_type(void) | ||
| 94 | { | ||
| 95 | return uv_system_type; | ||
| 96 | } | ||
| 97 | |||
| 98 | int is_uv_system(void) | ||
| 99 | { | ||
| 100 | return uv_system_type != UV_NONE; | ||
| 101 | } | ||
| 102 | EXPORT_SYMBOL_GPL(is_uv_system); | ||
diff --git a/arch/x86/kernel/genapic_flat_64.c b/arch/x86/kernel/genapic_flat_64.c index 786548a62d38..9eca5ba7a6b1 100644 --- a/arch/x86/kernel/genapic_flat_64.c +++ b/arch/x86/kernel/genapic_flat_64.c | |||
| @@ -15,9 +15,20 @@ | |||
| 15 | #include <linux/kernel.h> | 15 | #include <linux/kernel.h> |
| 16 | #include <linux/ctype.h> | 16 | #include <linux/ctype.h> |
| 17 | #include <linux/init.h> | 17 | #include <linux/init.h> |
| 18 | #include <linux/hardirq.h> | ||
| 18 | #include <asm/smp.h> | 19 | #include <asm/smp.h> |
| 19 | #include <asm/ipi.h> | 20 | #include <asm/ipi.h> |
| 20 | #include <asm/genapic.h> | 21 | #include <asm/genapic.h> |
| 22 | #include <mach_apicdef.h> | ||
| 23 | |||
| 24 | #ifdef CONFIG_ACPI | ||
| 25 | #include <acpi/acpi_bus.h> | ||
| 26 | #endif | ||
| 27 | |||
| 28 | static int __init flat_acpi_madt_oem_check(char *oem_id, char *oem_table_id) | ||
| 29 | { | ||
| 30 | return 1; | ||
| 31 | } | ||
| 21 | 32 | ||
| 22 | static cpumask_t flat_target_cpus(void) | 33 | static cpumask_t flat_target_cpus(void) |
| 23 | { | 34 | { |
| @@ -95,9 +106,33 @@ static void flat_send_IPI_all(int vector) | |||
| 95 | __send_IPI_shortcut(APIC_DEST_ALLINC, vector, APIC_DEST_LOGICAL); | 106 | __send_IPI_shortcut(APIC_DEST_ALLINC, vector, APIC_DEST_LOGICAL); |
| 96 | } | 107 | } |
| 97 | 108 | ||
| 109 | static unsigned int get_apic_id(unsigned long x) | ||
| 110 | { | ||
| 111 | unsigned int id; | ||
| 112 | |||
| 113 | id = (((x)>>24) & 0xFFu); | ||
| 114 | return id; | ||
| 115 | } | ||
| 116 | |||
| 117 | static unsigned long set_apic_id(unsigned int id) | ||
| 118 | { | ||
| 119 | unsigned long x; | ||
| 120 | |||
| 121 | x = ((id & 0xFFu)<<24); | ||
| 122 | return x; | ||
| 123 | } | ||
| 124 | |||
| 125 | static unsigned int read_xapic_id(void) | ||
| 126 | { | ||
| 127 | unsigned int id; | ||
| 128 | |||
| 129 | id = get_apic_id(apic_read(APIC_ID)); | ||
| 130 | return id; | ||
| 131 | } | ||
| 132 | |||
| 98 | static int flat_apic_id_registered(void) | 133 | static int flat_apic_id_registered(void) |
| 99 | { | 134 | { |
| 100 | return physid_isset(GET_APIC_ID(read_apic_id()), phys_cpu_present_map); | 135 | return physid_isset(read_xapic_id(), phys_cpu_present_map); |
| 101 | } | 136 | } |
| 102 | 137 | ||
| 103 | static unsigned int flat_cpu_mask_to_apicid(cpumask_t cpumask) | 138 | static unsigned int flat_cpu_mask_to_apicid(cpumask_t cpumask) |
| @@ -112,6 +147,7 @@ static unsigned int phys_pkg_id(int index_msb) | |||
| 112 | 147 | ||
| 113 | struct genapic apic_flat = { | 148 | struct genapic apic_flat = { |
| 114 | .name = "flat", | 149 | .name = "flat", |
| 150 | .acpi_madt_oem_check = flat_acpi_madt_oem_check, | ||
| 115 | .int_delivery_mode = dest_LowestPrio, | 151 | .int_delivery_mode = dest_LowestPrio, |
| 116 | .int_dest_mode = (APIC_DEST_LOGICAL != 0), | 152 | .int_dest_mode = (APIC_DEST_LOGICAL != 0), |
| 117 | .target_cpus = flat_target_cpus, | 153 | .target_cpus = flat_target_cpus, |
| @@ -121,8 +157,12 @@ struct genapic apic_flat = { | |||
| 121 | .send_IPI_all = flat_send_IPI_all, | 157 | .send_IPI_all = flat_send_IPI_all, |
| 122 | .send_IPI_allbutself = flat_send_IPI_allbutself, | 158 | .send_IPI_allbutself = flat_send_IPI_allbutself, |
| 123 | .send_IPI_mask = flat_send_IPI_mask, | 159 | .send_IPI_mask = flat_send_IPI_mask, |
| 160 | .send_IPI_self = apic_send_IPI_self, | ||
| 124 | .cpu_mask_to_apicid = flat_cpu_mask_to_apicid, | 161 | .cpu_mask_to_apicid = flat_cpu_mask_to_apicid, |
| 125 | .phys_pkg_id = phys_pkg_id, | 162 | .phys_pkg_id = phys_pkg_id, |
| 163 | .get_apic_id = get_apic_id, | ||
| 164 | .set_apic_id = set_apic_id, | ||
| 165 | .apic_id_mask = (0xFFu<<24), | ||
| 126 | }; | 166 | }; |
| 127 | 167 | ||
| 128 | /* | 168 | /* |
| @@ -130,6 +170,21 @@ struct genapic apic_flat = { | |||
| 130 | * We cannot use logical delivery in this case because the mask | 170 | * We cannot use logical delivery in this case because the mask |
| 131 | * overflows, so use physical mode. | 171 | * overflows, so use physical mode. |
| 132 | */ | 172 | */ |
| 173 | static int __init physflat_acpi_madt_oem_check(char *oem_id, char *oem_table_id) | ||
| 174 | { | ||
| 175 | #ifdef CONFIG_ACPI | ||
| 176 | /* | ||
| 177 | * Quirk: some x86_64 machines can only use physical APIC mode | ||
| 178 | * regardless of how many processors are present (x86_64 ES7000 | ||
| 179 | * is an example). | ||
| 180 | */ | ||
| 181 | if (acpi_gbl_FADT.header.revision > FADT2_REVISION_ID && | ||
| 182 | (acpi_gbl_FADT.flags & ACPI_FADT_APIC_PHYSICAL)) | ||
| 183 | return 1; | ||
| 184 | #endif | ||
| 185 | |||
| 186 | return 0; | ||
| 187 | } | ||
| 133 | 188 | ||
| 134 | static cpumask_t physflat_target_cpus(void) | 189 | static cpumask_t physflat_target_cpus(void) |
| 135 | { | 190 | { |
| @@ -176,6 +231,7 @@ static unsigned int physflat_cpu_mask_to_apicid(cpumask_t cpumask) | |||
| 176 | 231 | ||
| 177 | struct genapic apic_physflat = { | 232 | struct genapic apic_physflat = { |
| 178 | .name = "physical flat", | 233 | .name = "physical flat", |
| 234 | .acpi_madt_oem_check = physflat_acpi_madt_oem_check, | ||
| 179 | .int_delivery_mode = dest_Fixed, | 235 | .int_delivery_mode = dest_Fixed, |
| 180 | .int_dest_mode = (APIC_DEST_PHYSICAL != 0), | 236 | .int_dest_mode = (APIC_DEST_PHYSICAL != 0), |
| 181 | .target_cpus = physflat_target_cpus, | 237 | .target_cpus = physflat_target_cpus, |
| @@ -185,6 +241,10 @@ struct genapic apic_physflat = { | |||
| 185 | .send_IPI_all = physflat_send_IPI_all, | 241 | .send_IPI_all = physflat_send_IPI_all, |
| 186 | .send_IPI_allbutself = physflat_send_IPI_allbutself, | 242 | .send_IPI_allbutself = physflat_send_IPI_allbutself, |
| 187 | .send_IPI_mask = physflat_send_IPI_mask, | 243 | .send_IPI_mask = physflat_send_IPI_mask, |
| 244 | .send_IPI_self = apic_send_IPI_self, | ||
| 188 | .cpu_mask_to_apicid = physflat_cpu_mask_to_apicid, | 245 | .cpu_mask_to_apicid = physflat_cpu_mask_to_apicid, |
| 189 | .phys_pkg_id = phys_pkg_id, | 246 | .phys_pkg_id = phys_pkg_id, |
| 247 | .get_apic_id = get_apic_id, | ||
| 248 | .set_apic_id = set_apic_id, | ||
| 249 | .apic_id_mask = (0xFFu<<24), | ||
| 190 | }; | 250 | }; |
diff --git a/arch/x86/kernel/genx2apic_cluster.c b/arch/x86/kernel/genx2apic_cluster.c new file mode 100644 index 000000000000..fed9f68efd66 --- /dev/null +++ b/arch/x86/kernel/genx2apic_cluster.c | |||
| @@ -0,0 +1,164 @@ | |||
| 1 | #include <linux/threads.h> | ||
| 2 | #include <linux/cpumask.h> | ||
| 3 | #include <linux/string.h> | ||
| 4 | #include <linux/kernel.h> | ||
| 5 | #include <linux/ctype.h> | ||
| 6 | #include <linux/init.h> | ||
| 7 | #include <linux/dmar.h> | ||
| 8 | |||
| 9 | #include <asm/smp.h> | ||
| 10 | #include <asm/ipi.h> | ||
| 11 | #include <asm/genapic.h> | ||
| 12 | |||
| 13 | DEFINE_PER_CPU(u32, x86_cpu_to_logical_apicid); | ||
| 14 | |||
| 15 | static int __init x2apic_acpi_madt_oem_check(char *oem_id, char *oem_table_id) | ||
| 16 | { | ||
| 17 | if (cpu_has_x2apic) | ||
| 18 | return 1; | ||
| 19 | |||
| 20 | return 0; | ||
| 21 | } | ||
| 22 | |||
| 23 | /* Start with all IRQs pointing to boot CPU. IRQ balancing will shift them. */ | ||
| 24 | |||
| 25 | static cpumask_t x2apic_target_cpus(void) | ||
| 26 | { | ||
| 27 | return cpumask_of_cpu(0); | ||
| 28 | } | ||
| 29 | |||
| 30 | /* | ||
| 31 | * for now each logical cpu is in its own vector allocation domain. | ||
| 32 | */ | ||
| 33 | static cpumask_t x2apic_vector_allocation_domain(int cpu) | ||
| 34 | { | ||
| 35 | cpumask_t domain = CPU_MASK_NONE; | ||
| 36 | cpu_set(cpu, domain); | ||
| 37 | return domain; | ||
| 38 | } | ||
| 39 | |||
| 40 | static void __x2apic_send_IPI_dest(unsigned int apicid, int vector, | ||
| 41 | unsigned int dest) | ||
| 42 | { | ||
| 43 | unsigned long cfg; | ||
| 44 | |||
| 45 | cfg = __prepare_ICR(0, vector, dest); | ||
| 46 | |||
| 47 | /* | ||
| 48 | * send the IPI. | ||
| 49 | */ | ||
| 50 | x2apic_icr_write(cfg, apicid); | ||
| 51 | } | ||
| 52 | |||
| 53 | /* | ||
| 54 | * for now, we send the IPI's one by one in the cpumask. | ||
| 55 | * TBD: Based on the cpu mask, we can send the IPI's to the cluster group | ||
| 56 | * at once. We have 16 cpu's in a cluster. This will minimize IPI register | ||
| 57 | * writes. | ||
| 58 | */ | ||
| 59 | static void x2apic_send_IPI_mask(cpumask_t mask, int vector) | ||
| 60 | { | ||
| 61 | unsigned long flags; | ||
| 62 | unsigned long query_cpu; | ||
| 63 | |||
| 64 | local_irq_save(flags); | ||
| 65 | for_each_cpu_mask(query_cpu, mask) { | ||
| 66 | __x2apic_send_IPI_dest(per_cpu(x86_cpu_to_logical_apicid, query_cpu), | ||
| 67 | vector, APIC_DEST_LOGICAL); | ||
| 68 | } | ||
| 69 | local_irq_restore(flags); | ||
| 70 | } | ||
| 71 | |||
| 72 | static void x2apic_send_IPI_allbutself(int vector) | ||
| 73 | { | ||
| 74 | cpumask_t mask = cpu_online_map; | ||
| 75 | |||
| 76 | cpu_clear(smp_processor_id(), mask); | ||
| 77 | |||
| 78 | if (!cpus_empty(mask)) | ||
| 79 | x2apic_send_IPI_mask(mask, vector); | ||
| 80 | } | ||
| 81 | |||
| 82 | static void x2apic_send_IPI_all(int vector) | ||
| 83 | { | ||
| 84 | x2apic_send_IPI_mask(cpu_online_map, vector); | ||
| 85 | } | ||
| 86 | |||
| 87 | static int x2apic_apic_id_registered(void) | ||
| 88 | { | ||
| 89 | return 1; | ||
| 90 | } | ||
| 91 | |||
| 92 | static unsigned int x2apic_cpu_mask_to_apicid(cpumask_t cpumask) | ||
| 93 | { | ||
| 94 | int cpu; | ||
| 95 | |||
| 96 | /* | ||
| 97 | * We're using fixed IRQ delivery, can only return one phys APIC ID. | ||
| 98 | * May as well be the first. | ||
| 99 | */ | ||
| 100 | cpu = first_cpu(cpumask); | ||
| 101 | if ((unsigned)cpu < NR_CPUS) | ||
| 102 | return per_cpu(x86_cpu_to_logical_apicid, cpu); | ||
| 103 | else | ||
| 104 | return BAD_APICID; | ||
| 105 | } | ||
| 106 | |||
| 107 | static unsigned int get_apic_id(unsigned long x) | ||
| 108 | { | ||
| 109 | unsigned int id; | ||
| 110 | |||
| 111 | id = x; | ||
| 112 | return id; | ||
| 113 | } | ||
| 114 | |||
| 115 | static unsigned long set_apic_id(unsigned int id) | ||
| 116 | { | ||
| 117 | unsigned long x; | ||
| 118 | |||
| 119 | x = id; | ||
| 120 | return x; | ||
| 121 | } | ||
| 122 | |||
| 123 | static unsigned int x2apic_read_id(void) | ||
| 124 | { | ||
| 125 | return apic_read(APIC_ID); | ||
| 126 | } | ||
| 127 | |||
| 128 | static unsigned int phys_pkg_id(int index_msb) | ||
| 129 | { | ||
| 130 | return x2apic_read_id() >> index_msb; | ||
| 131 | } | ||
| 132 | |||
| 133 | static void x2apic_send_IPI_self(int vector) | ||
| 134 | { | ||
| 135 | apic_write(APIC_SELF_IPI, vector); | ||
| 136 | } | ||
| 137 | |||
| 138 | static void init_x2apic_ldr(void) | ||
| 139 | { | ||
| 140 | int cpu = smp_processor_id(); | ||
| 141 | |||
| 142 | per_cpu(x86_cpu_to_logical_apicid, cpu) = apic_read(APIC_LDR); | ||
| 143 | return; | ||
| 144 | } | ||
| 145 | |||
| 146 | struct genapic apic_x2apic_cluster = { | ||
| 147 | .name = "cluster x2apic", | ||
| 148 | .acpi_madt_oem_check = x2apic_acpi_madt_oem_check, | ||
| 149 | .int_delivery_mode = dest_LowestPrio, | ||
| 150 | .int_dest_mode = (APIC_DEST_LOGICAL != 0), | ||
| 151 | .target_cpus = x2apic_target_cpus, | ||
| 152 | .vector_allocation_domain = x2apic_vector_allocation_domain, | ||
| 153 | .apic_id_registered = x2apic_apic_id_registered, | ||
| 154 | .init_apic_ldr = init_x2apic_ldr, | ||
| 155 | .send_IPI_all = x2apic_send_IPI_all, | ||
| 156 | .send_IPI_allbutself = x2apic_send_IPI_allbutself, | ||
| 157 | .send_IPI_mask = x2apic_send_IPI_mask, | ||
| 158 | .send_IPI_self = x2apic_send_IPI_self, | ||
| 159 | .cpu_mask_to_apicid = x2apic_cpu_mask_to_apicid, | ||
| 160 | .phys_pkg_id = phys_pkg_id, | ||
| 161 | .get_apic_id = get_apic_id, | ||
| 162 | .set_apic_id = set_apic_id, | ||
| 163 | .apic_id_mask = (0xFFFFFFFFu), | ||
| 164 | }; | ||
diff --git a/arch/x86/kernel/genx2apic_phys.c b/arch/x86/kernel/genx2apic_phys.c new file mode 100644 index 000000000000..958d537b4cc9 --- /dev/null +++ b/arch/x86/kernel/genx2apic_phys.c | |||
| @@ -0,0 +1,159 @@ | |||
| 1 | #include <linux/threads.h> | ||
| 2 | #include <linux/cpumask.h> | ||
| 3 | #include <linux/string.h> | ||
| 4 | #include <linux/kernel.h> | ||
| 5 | #include <linux/ctype.h> | ||
| 6 | #include <linux/init.h> | ||
| 7 | #include <linux/dmar.h> | ||
| 8 | |||
| 9 | #include <asm/smp.h> | ||
| 10 | #include <asm/ipi.h> | ||
| 11 | #include <asm/genapic.h> | ||
| 12 | |||
| 13 | static int x2apic_phys; | ||
| 14 | |||
| 15 | static int set_x2apic_phys_mode(char *arg) | ||
| 16 | { | ||
| 17 | x2apic_phys = 1; | ||
| 18 | return 0; | ||
| 19 | } | ||
| 20 | early_param("x2apic_phys", set_x2apic_phys_mode); | ||
| 21 | |||
| 22 | static int __init x2apic_acpi_madt_oem_check(char *oem_id, char *oem_table_id) | ||
| 23 | { | ||
| 24 | if (cpu_has_x2apic && x2apic_phys) | ||
| 25 | return 1; | ||
| 26 | |||
| 27 | return 0; | ||
| 28 | } | ||
| 29 | |||
| 30 | /* Start with all IRQs pointing to boot CPU. IRQ balancing will shift them. */ | ||
| 31 | |||
| 32 | static cpumask_t x2apic_target_cpus(void) | ||
| 33 | { | ||
| 34 | return cpumask_of_cpu(0); | ||
| 35 | } | ||
| 36 | |||
| 37 | static cpumask_t x2apic_vector_allocation_domain(int cpu) | ||
| 38 | { | ||
| 39 | cpumask_t domain = CPU_MASK_NONE; | ||
| 40 | cpu_set(cpu, domain); | ||
| 41 | return domain; | ||
| 42 | } | ||
| 43 | |||
| 44 | static void __x2apic_send_IPI_dest(unsigned int apicid, int vector, | ||
| 45 | unsigned int dest) | ||
| 46 | { | ||
| 47 | unsigned long cfg; | ||
| 48 | |||
| 49 | cfg = __prepare_ICR(0, vector, dest); | ||
| 50 | |||
| 51 | /* | ||
| 52 | * send the IPI. | ||
| 53 | */ | ||
| 54 | x2apic_icr_write(cfg, apicid); | ||
| 55 | } | ||
| 56 | |||
| 57 | static void x2apic_send_IPI_mask(cpumask_t mask, int vector) | ||
| 58 | { | ||
| 59 | unsigned long flags; | ||
| 60 | unsigned long query_cpu; | ||
| 61 | |||
| 62 | local_irq_save(flags); | ||
| 63 | for_each_cpu_mask(query_cpu, mask) { | ||
| 64 | __x2apic_send_IPI_dest(per_cpu(x86_cpu_to_apicid, query_cpu), | ||
| 65 | vector, APIC_DEST_PHYSICAL); | ||
| 66 | } | ||
| 67 | local_irq_restore(flags); | ||
| 68 | } | ||
| 69 | |||
| 70 | static void x2apic_send_IPI_allbutself(int vector) | ||
| 71 | { | ||
| 72 | cpumask_t mask = cpu_online_map; | ||
| 73 | |||
| 74 | cpu_clear(smp_processor_id(), mask); | ||
| 75 | |||
| 76 | if (!cpus_empty(mask)) | ||
| 77 | x2apic_send_IPI_mask(mask, vector); | ||
| 78 | } | ||
| 79 | |||
| 80 | static void x2apic_send_IPI_all(int vector) | ||
| 81 | { | ||
| 82 | x2apic_send_IPI_mask(cpu_online_map, vector); | ||
| 83 | } | ||
| 84 | |||
| 85 | static int x2apic_apic_id_registered(void) | ||
| 86 | { | ||
| 87 | return 1; | ||
| 88 | } | ||
| 89 | |||
| 90 | static unsigned int x2apic_cpu_mask_to_apicid(cpumask_t cpumask) | ||
| 91 | { | ||
| 92 | int cpu; | ||
| 93 | |||
| 94 | /* | ||
| 95 | * We're using fixed IRQ delivery, can only return one phys APIC ID. | ||
| 96 | * May as well be the first. | ||
| 97 | */ | ||
| 98 | cpu = first_cpu(cpumask); | ||
| 99 | if ((unsigned)cpu < NR_CPUS) | ||
| 100 | return per_cpu(x86_cpu_to_apicid, cpu); | ||
| 101 | else | ||
| 102 | return BAD_APICID; | ||
| 103 | } | ||
| 104 | |||
| 105 | static unsigned int get_apic_id(unsigned long x) | ||
| 106 | { | ||
| 107 | unsigned int id; | ||
| 108 | |||
| 109 | id = x; | ||
| 110 | return id; | ||
| 111 | } | ||
| 112 | |||
| 113 | static unsigned long set_apic_id(unsigned int id) | ||
| 114 | { | ||
| 115 | unsigned long x; | ||
| 116 | |||
| 117 | x = id; | ||
| 118 | return x; | ||
| 119 | } | ||
| 120 | |||
| 121 | static unsigned int x2apic_read_id(void) | ||
| 122 | { | ||
| 123 | return apic_read(APIC_ID); | ||
| 124 | } | ||
| 125 | |||
| 126 | static unsigned int phys_pkg_id(int index_msb) | ||
| 127 | { | ||
| 128 | return x2apic_read_id() >> index_msb; | ||
| 129 | } | ||
| 130 | |||
| 131 | void x2apic_send_IPI_self(int vector) | ||
| 132 | { | ||
| 133 | apic_write(APIC_SELF_IPI, vector); | ||
| 134 | } | ||
| 135 | |||
| 136 | void init_x2apic_ldr(void) | ||
| 137 | { | ||
| 138 | return; | ||
| 139 | } | ||
| 140 | |||
| 141 | struct genapic apic_x2apic_phys = { | ||
| 142 | .name = "physical x2apic", | ||
| 143 | .acpi_madt_oem_check = x2apic_acpi_madt_oem_check, | ||
| 144 | .int_delivery_mode = dest_Fixed, | ||
| 145 | .int_dest_mode = (APIC_DEST_PHYSICAL != 0), | ||
| 146 | .target_cpus = x2apic_target_cpus, | ||
| 147 | .vector_allocation_domain = x2apic_vector_allocation_domain, | ||
| 148 | .apic_id_registered = x2apic_apic_id_registered, | ||
| 149 | .init_apic_ldr = init_x2apic_ldr, | ||
| 150 | .send_IPI_all = x2apic_send_IPI_all, | ||
| 151 | .send_IPI_allbutself = x2apic_send_IPI_allbutself, | ||
| 152 | .send_IPI_mask = x2apic_send_IPI_mask, | ||
| 153 | .send_IPI_self = x2apic_send_IPI_self, | ||
| 154 | .cpu_mask_to_apicid = x2apic_cpu_mask_to_apicid, | ||
| 155 | .phys_pkg_id = phys_pkg_id, | ||
| 156 | .get_apic_id = get_apic_id, | ||
| 157 | .set_apic_id = set_apic_id, | ||
| 158 | .apic_id_mask = (0xFFFFFFFFu), | ||
| 159 | }; | ||
diff --git a/arch/x86/kernel/genx2apic_uv_x.c b/arch/x86/kernel/genx2apic_uv_x.c index bfa837cb16be..ae2ffc8a400c 100644 --- a/arch/x86/kernel/genx2apic_uv_x.c +++ b/arch/x86/kernel/genx2apic_uv_x.c | |||
| @@ -12,12 +12,12 @@ | |||
| 12 | #include <linux/threads.h> | 12 | #include <linux/threads.h> |
| 13 | #include <linux/cpumask.h> | 13 | #include <linux/cpumask.h> |
| 14 | #include <linux/string.h> | 14 | #include <linux/string.h> |
| 15 | #include <linux/kernel.h> | ||
| 16 | #include <linux/ctype.h> | 15 | #include <linux/ctype.h> |
| 17 | #include <linux/init.h> | 16 | #include <linux/init.h> |
| 18 | #include <linux/sched.h> | 17 | #include <linux/sched.h> |
| 19 | #include <linux/bootmem.h> | 18 | #include <linux/bootmem.h> |
| 20 | #include <linux/module.h> | 19 | #include <linux/module.h> |
| 20 | #include <linux/hardirq.h> | ||
| 21 | #include <asm/smp.h> | 21 | #include <asm/smp.h> |
| 22 | #include <asm/ipi.h> | 22 | #include <asm/ipi.h> |
| 23 | #include <asm/genapic.h> | 23 | #include <asm/genapic.h> |
| @@ -26,6 +26,36 @@ | |||
| 26 | #include <asm/uv/uv_hub.h> | 26 | #include <asm/uv/uv_hub.h> |
| 27 | #include <asm/uv/bios.h> | 27 | #include <asm/uv/bios.h> |
| 28 | 28 | ||
| 29 | DEFINE_PER_CPU(int, x2apic_extra_bits); | ||
| 30 | |||
| 31 | static enum uv_system_type uv_system_type; | ||
| 32 | |||
| 33 | static int __init uv_acpi_madt_oem_check(char *oem_id, char *oem_table_id) | ||
| 34 | { | ||
| 35 | if (!strcmp(oem_id, "SGI")) { | ||
| 36 | if (!strcmp(oem_table_id, "UVL")) | ||
| 37 | uv_system_type = UV_LEGACY_APIC; | ||
| 38 | else if (!strcmp(oem_table_id, "UVX")) | ||
| 39 | uv_system_type = UV_X2APIC; | ||
| 40 | else if (!strcmp(oem_table_id, "UVH")) { | ||
| 41 | uv_system_type = UV_NON_UNIQUE_APIC; | ||
| 42 | return 1; | ||
| 43 | } | ||
| 44 | } | ||
| 45 | return 0; | ||
| 46 | } | ||
| 47 | |||
| 48 | enum uv_system_type get_uv_system_type(void) | ||
| 49 | { | ||
| 50 | return uv_system_type; | ||
| 51 | } | ||
| 52 | |||
| 53 | int is_uv_system(void) | ||
| 54 | { | ||
| 55 | return uv_system_type != UV_NONE; | ||
| 56 | } | ||
| 57 | EXPORT_SYMBOL_GPL(is_uv_system); | ||
| 58 | |||
| 29 | DEFINE_PER_CPU(struct uv_hub_info_s, __uv_hub_info); | 59 | DEFINE_PER_CPU(struct uv_hub_info_s, __uv_hub_info); |
| 30 | EXPORT_PER_CPU_SYMBOL_GPL(__uv_hub_info); | 60 | EXPORT_PER_CPU_SYMBOL_GPL(__uv_hub_info); |
| 31 | 61 | ||
| @@ -123,6 +153,10 @@ static int uv_apic_id_registered(void) | |||
| 123 | return 1; | 153 | return 1; |
| 124 | } | 154 | } |
| 125 | 155 | ||
| 156 | static void uv_init_apic_ldr(void) | ||
| 157 | { | ||
| 158 | } | ||
| 159 | |||
| 126 | static unsigned int uv_cpu_mask_to_apicid(cpumask_t cpumask) | 160 | static unsigned int uv_cpu_mask_to_apicid(cpumask_t cpumask) |
| 127 | { | 161 | { |
| 128 | int cpu; | 162 | int cpu; |
| @@ -138,9 +172,34 @@ static unsigned int uv_cpu_mask_to_apicid(cpumask_t cpumask) | |||
| 138 | return BAD_APICID; | 172 | return BAD_APICID; |
| 139 | } | 173 | } |
| 140 | 174 | ||
| 175 | static unsigned int get_apic_id(unsigned long x) | ||
| 176 | { | ||
| 177 | unsigned int id; | ||
| 178 | |||
| 179 | WARN_ON(preemptible() && num_online_cpus() > 1); | ||
| 180 | id = x | __get_cpu_var(x2apic_extra_bits); | ||
| 181 | |||
| 182 | return id; | ||
| 183 | } | ||
| 184 | |||
| 185 | static unsigned long set_apic_id(unsigned int id) | ||
| 186 | { | ||
| 187 | unsigned long x; | ||
| 188 | |||
| 189 | /* maskout x2apic_extra_bits ? */ | ||
| 190 | x = id; | ||
| 191 | return x; | ||
| 192 | } | ||
| 193 | |||
| 194 | static unsigned int uv_read_apic_id(void) | ||
| 195 | { | ||
| 196 | |||
| 197 | return get_apic_id(apic_read(APIC_ID)); | ||
| 198 | } | ||
| 199 | |||
| 141 | static unsigned int phys_pkg_id(int index_msb) | 200 | static unsigned int phys_pkg_id(int index_msb) |
| 142 | { | 201 | { |
| 143 | return GET_APIC_ID(read_apic_id()) >> index_msb; | 202 | return uv_read_apic_id() >> index_msb; |
| 144 | } | 203 | } |
| 145 | 204 | ||
| 146 | #ifdef ZZZ /* Needs x2apic patch */ | 205 | #ifdef ZZZ /* Needs x2apic patch */ |
| @@ -152,17 +211,22 @@ static void uv_send_IPI_self(int vector) | |||
| 152 | 211 | ||
| 153 | struct genapic apic_x2apic_uv_x = { | 212 | struct genapic apic_x2apic_uv_x = { |
| 154 | .name = "UV large system", | 213 | .name = "UV large system", |
| 214 | .acpi_madt_oem_check = uv_acpi_madt_oem_check, | ||
| 155 | .int_delivery_mode = dest_Fixed, | 215 | .int_delivery_mode = dest_Fixed, |
| 156 | .int_dest_mode = (APIC_DEST_PHYSICAL != 0), | 216 | .int_dest_mode = (APIC_DEST_PHYSICAL != 0), |
| 157 | .target_cpus = uv_target_cpus, | 217 | .target_cpus = uv_target_cpus, |
| 158 | .vector_allocation_domain = uv_vector_allocation_domain,/* Fixme ZZZ */ | 218 | .vector_allocation_domain = uv_vector_allocation_domain,/* Fixme ZZZ */ |
| 159 | .apic_id_registered = uv_apic_id_registered, | 219 | .apic_id_registered = uv_apic_id_registered, |
| 220 | .init_apic_ldr = uv_init_apic_ldr, | ||
| 160 | .send_IPI_all = uv_send_IPI_all, | 221 | .send_IPI_all = uv_send_IPI_all, |
| 161 | .send_IPI_allbutself = uv_send_IPI_allbutself, | 222 | .send_IPI_allbutself = uv_send_IPI_allbutself, |
| 162 | .send_IPI_mask = uv_send_IPI_mask, | 223 | .send_IPI_mask = uv_send_IPI_mask, |
| 163 | /* ZZZ.send_IPI_self = uv_send_IPI_self, */ | 224 | /* ZZZ.send_IPI_self = uv_send_IPI_self, */ |
| 164 | .cpu_mask_to_apicid = uv_cpu_mask_to_apicid, | 225 | .cpu_mask_to_apicid = uv_cpu_mask_to_apicid, |
| 165 | .phys_pkg_id = phys_pkg_id, /* Fixme ZZZ */ | 226 | .phys_pkg_id = phys_pkg_id, /* Fixme ZZZ */ |
| 227 | .get_apic_id = get_apic_id, | ||
| 228 | .set_apic_id = set_apic_id, | ||
| 229 | .apic_id_mask = (0xFFFFFFFFu), | ||
| 166 | }; | 230 | }; |
| 167 | 231 | ||
| 168 | static __cpuinit void set_x2apic_extra_bits(int pnode) | 232 | static __cpuinit void set_x2apic_extra_bits(int pnode) |
| @@ -401,3 +465,5 @@ void __cpuinit uv_cpu_init(void) | |||
| 401 | if (get_uv_system_type() == UV_NON_UNIQUE_APIC) | 465 | if (get_uv_system_type() == UV_NON_UNIQUE_APIC) |
| 402 | set_x2apic_extra_bits(uv_hub_info->pnode); | 466 | set_x2apic_extra_bits(uv_hub_info->pnode); |
| 403 | } | 467 | } |
| 468 | |||
| 469 | |||
diff --git a/arch/x86/kernel/i8259.c b/arch/x86/kernel/i8259.c index dc92b49d9204..4b8a53d841f7 100644 --- a/arch/x86/kernel/i8259.c +++ b/arch/x86/kernel/i8259.c | |||
| @@ -282,6 +282,30 @@ static int __init i8259A_init_sysfs(void) | |||
| 282 | 282 | ||
| 283 | device_initcall(i8259A_init_sysfs); | 283 | device_initcall(i8259A_init_sysfs); |
| 284 | 284 | ||
| 285 | void mask_8259A(void) | ||
| 286 | { | ||
| 287 | unsigned long flags; | ||
| 288 | |||
| 289 | spin_lock_irqsave(&i8259A_lock, flags); | ||
| 290 | |||
| 291 | outb(0xff, PIC_MASTER_IMR); /* mask all of 8259A-1 */ | ||
| 292 | outb(0xff, PIC_SLAVE_IMR); /* mask all of 8259A-2 */ | ||
| 293 | |||
| 294 | spin_unlock_irqrestore(&i8259A_lock, flags); | ||
| 295 | } | ||
| 296 | |||
| 297 | void unmask_8259A(void) | ||
| 298 | { | ||
| 299 | unsigned long flags; | ||
| 300 | |||
| 301 | spin_lock_irqsave(&i8259A_lock, flags); | ||
| 302 | |||
| 303 | outb(cached_master_mask, PIC_MASTER_IMR); /* restore master IRQ mask */ | ||
| 304 | outb(cached_slave_mask, PIC_SLAVE_IMR); /* restore slave IRQ mask */ | ||
| 305 | |||
| 306 | spin_unlock_irqrestore(&i8259A_lock, flags); | ||
| 307 | } | ||
| 308 | |||
| 285 | void init_8259A(int auto_eoi) | 309 | void init_8259A(int auto_eoi) |
| 286 | { | 310 | { |
| 287 | unsigned long flags; | 311 | unsigned long flags; |
diff --git a/arch/x86/kernel/io_apic_32.c b/arch/x86/kernel/io_apic_32.c index 09cddb57bec4..e710289f673e 100644 --- a/arch/x86/kernel/io_apic_32.c +++ b/arch/x86/kernel/io_apic_32.c | |||
| @@ -46,10 +46,13 @@ | |||
| 46 | #include <asm/nmi.h> | 46 | #include <asm/nmi.h> |
| 47 | #include <asm/msidef.h> | 47 | #include <asm/msidef.h> |
| 48 | #include <asm/hypertransport.h> | 48 | #include <asm/hypertransport.h> |
| 49 | #include <asm/setup.h> | ||
| 49 | 50 | ||
| 50 | #include <mach_apic.h> | 51 | #include <mach_apic.h> |
| 51 | #include <mach_apicdef.h> | 52 | #include <mach_apicdef.h> |
| 52 | 53 | ||
| 54 | #define __apicdebuginit(type) static type __init | ||
| 55 | |||
| 53 | int (*ioapic_renumber_irq)(int ioapic, int irq); | 56 | int (*ioapic_renumber_irq)(int ioapic, int irq); |
| 54 | atomic_t irq_mis_count; | 57 | atomic_t irq_mis_count; |
| 55 | 58 | ||
| @@ -1341,7 +1344,8 @@ static void __init setup_timer_IRQ0_pin(unsigned int apic, unsigned int pin, | |||
| 1341 | ioapic_write_entry(apic, pin, entry); | 1344 | ioapic_write_entry(apic, pin, entry); |
| 1342 | } | 1345 | } |
| 1343 | 1346 | ||
| 1344 | void __init print_IO_APIC(void) | 1347 | |
| 1348 | __apicdebuginit(void) print_IO_APIC(void) | ||
| 1345 | { | 1349 | { |
| 1346 | int apic, i; | 1350 | int apic, i; |
| 1347 | union IO_APIC_reg_00 reg_00; | 1351 | union IO_APIC_reg_00 reg_00; |
| @@ -1456,9 +1460,7 @@ void __init print_IO_APIC(void) | |||
| 1456 | return; | 1460 | return; |
| 1457 | } | 1461 | } |
| 1458 | 1462 | ||
| 1459 | #if 0 | 1463 | __apicdebuginit(void) print_APIC_bitfield(int base) |
| 1460 | |||
| 1461 | static void print_APIC_bitfield(int base) | ||
| 1462 | { | 1464 | { |
| 1463 | unsigned int v; | 1465 | unsigned int v; |
| 1464 | int i, j; | 1466 | int i, j; |
| @@ -1479,9 +1481,10 @@ static void print_APIC_bitfield(int base) | |||
| 1479 | } | 1481 | } |
| 1480 | } | 1482 | } |
| 1481 | 1483 | ||
| 1482 | void /*__init*/ print_local_APIC(void *dummy) | 1484 | __apicdebuginit(void) print_local_APIC(void *dummy) |
| 1483 | { | 1485 | { |
| 1484 | unsigned int v, ver, maxlvt; | 1486 | unsigned int v, ver, maxlvt; |
| 1487 | u64 icr; | ||
| 1485 | 1488 | ||
| 1486 | if (apic_verbosity == APIC_QUIET) | 1489 | if (apic_verbosity == APIC_QUIET) |
| 1487 | return; | 1490 | return; |
| @@ -1490,7 +1493,7 @@ void /*__init*/ print_local_APIC(void *dummy) | |||
| 1490 | smp_processor_id(), hard_smp_processor_id()); | 1493 | smp_processor_id(), hard_smp_processor_id()); |
| 1491 | v = apic_read(APIC_ID); | 1494 | v = apic_read(APIC_ID); |
| 1492 | printk(KERN_INFO "... APIC ID: %08x (%01x)\n", v, | 1495 | printk(KERN_INFO "... APIC ID: %08x (%01x)\n", v, |
| 1493 | GET_APIC_ID(read_apic_id())); | 1496 | GET_APIC_ID(v)); |
| 1494 | v = apic_read(APIC_LVR); | 1497 | v = apic_read(APIC_LVR); |
| 1495 | printk(KERN_INFO "... APIC VERSION: %08x\n", v); | 1498 | printk(KERN_INFO "... APIC VERSION: %08x\n", v); |
| 1496 | ver = GET_APIC_VERSION(v); | 1499 | ver = GET_APIC_VERSION(v); |
| @@ -1532,10 +1535,9 @@ void /*__init*/ print_local_APIC(void *dummy) | |||
| 1532 | printk(KERN_DEBUG "... APIC ESR: %08x\n", v); | 1535 | printk(KERN_DEBUG "... APIC ESR: %08x\n", v); |
| 1533 | } | 1536 | } |
| 1534 | 1537 | ||
| 1535 | v = apic_read(APIC_ICR); | 1538 | icr = apic_icr_read(); |
| 1536 | printk(KERN_DEBUG "... APIC ICR: %08x\n", v); | 1539 | printk(KERN_DEBUG "... APIC ICR: %08x\n", icr); |
| 1537 | v = apic_read(APIC_ICR2); | 1540 | printk(KERN_DEBUG "... APIC ICR2: %08x\n", icr >> 32); |
| 1538 | printk(KERN_DEBUG "... APIC ICR2: %08x\n", v); | ||
| 1539 | 1541 | ||
| 1540 | v = apic_read(APIC_LVTT); | 1542 | v = apic_read(APIC_LVTT); |
| 1541 | printk(KERN_DEBUG "... APIC LVTT: %08x\n", v); | 1543 | printk(KERN_DEBUG "... APIC LVTT: %08x\n", v); |
| @@ -1563,12 +1565,12 @@ void /*__init*/ print_local_APIC(void *dummy) | |||
| 1563 | printk("\n"); | 1565 | printk("\n"); |
| 1564 | } | 1566 | } |
| 1565 | 1567 | ||
| 1566 | void print_all_local_APICs(void) | 1568 | __apicdebuginit(void) print_all_local_APICs(void) |
| 1567 | { | 1569 | { |
| 1568 | on_each_cpu(print_local_APIC, NULL, 1); | 1570 | on_each_cpu(print_local_APIC, NULL, 1); |
| 1569 | } | 1571 | } |
| 1570 | 1572 | ||
| 1571 | void /*__init*/ print_PIC(void) | 1573 | __apicdebuginit(void) print_PIC(void) |
| 1572 | { | 1574 | { |
| 1573 | unsigned int v; | 1575 | unsigned int v; |
| 1574 | unsigned long flags; | 1576 | unsigned long flags; |
| @@ -1600,7 +1602,17 @@ void /*__init*/ print_PIC(void) | |||
| 1600 | printk(KERN_DEBUG "... PIC ELCR: %04x\n", v); | 1602 | printk(KERN_DEBUG "... PIC ELCR: %04x\n", v); |
| 1601 | } | 1603 | } |
| 1602 | 1604 | ||
| 1603 | #endif /* 0 */ | 1605 | __apicdebuginit(int) print_all_ICs(void) |
| 1606 | { | ||
| 1607 | print_PIC(); | ||
| 1608 | print_all_local_APICs(); | ||
| 1609 | print_IO_APIC(); | ||
| 1610 | |||
| 1611 | return 0; | ||
| 1612 | } | ||
| 1613 | |||
| 1614 | fs_initcall(print_all_ICs); | ||
| 1615 | |||
| 1604 | 1616 | ||
| 1605 | static void __init enable_IO_APIC(void) | 1617 | static void __init enable_IO_APIC(void) |
| 1606 | { | 1618 | { |
| @@ -1698,8 +1710,7 @@ void disable_IO_APIC(void) | |||
| 1698 | entry.dest_mode = 0; /* Physical */ | 1710 | entry.dest_mode = 0; /* Physical */ |
| 1699 | entry.delivery_mode = dest_ExtINT; /* ExtInt */ | 1711 | entry.delivery_mode = dest_ExtINT; /* ExtInt */ |
| 1700 | entry.vector = 0; | 1712 | entry.vector = 0; |
| 1701 | entry.dest.physical.physical_dest = | 1713 | entry.dest.physical.physical_dest = read_apic_id(); |
| 1702 | GET_APIC_ID(read_apic_id()); | ||
| 1703 | 1714 | ||
| 1704 | /* | 1715 | /* |
| 1705 | * Add it to the IO-APIC irq-routing table: | 1716 | * Add it to the IO-APIC irq-routing table: |
| @@ -1725,10 +1736,8 @@ static void __init setup_ioapic_ids_from_mpc(void) | |||
| 1725 | unsigned char old_id; | 1736 | unsigned char old_id; |
| 1726 | unsigned long flags; | 1737 | unsigned long flags; |
| 1727 | 1738 | ||
| 1728 | #ifdef CONFIG_X86_NUMAQ | 1739 | if (x86_quirks->setup_ioapic_ids && x86_quirks->setup_ioapic_ids()) |
| 1729 | if (found_numaq) | ||
| 1730 | return; | 1740 | return; |
| 1731 | #endif | ||
| 1732 | 1741 | ||
| 1733 | /* | 1742 | /* |
| 1734 | * Don't check I/O APIC IDs for xAPIC systems. They have | 1743 | * Don't check I/O APIC IDs for xAPIC systems. They have |
| @@ -2329,8 +2338,6 @@ void __init setup_IO_APIC(void) | |||
| 2329 | setup_IO_APIC_irqs(); | 2338 | setup_IO_APIC_irqs(); |
| 2330 | init_IO_APIC_traps(); | 2339 | init_IO_APIC_traps(); |
| 2331 | check_timer(); | 2340 | check_timer(); |
| 2332 | if (!acpi_ioapic) | ||
| 2333 | print_IO_APIC(); | ||
| 2334 | } | 2341 | } |
| 2335 | 2342 | ||
| 2336 | /* | 2343 | /* |
diff --git a/arch/x86/kernel/io_apic_64.c b/arch/x86/kernel/io_apic_64.c index 61a83b70c18f..a1bec2969c6a 100644 --- a/arch/x86/kernel/io_apic_64.c +++ b/arch/x86/kernel/io_apic_64.c | |||
| @@ -37,6 +37,7 @@ | |||
| 37 | #include <acpi/acpi_bus.h> | 37 | #include <acpi/acpi_bus.h> |
| 38 | #endif | 38 | #endif |
| 39 | #include <linux/bootmem.h> | 39 | #include <linux/bootmem.h> |
| 40 | #include <linux/dmar.h> | ||
| 40 | 41 | ||
| 41 | #include <asm/idle.h> | 42 | #include <asm/idle.h> |
| 42 | #include <asm/io.h> | 43 | #include <asm/io.h> |
| @@ -49,10 +50,13 @@ | |||
| 49 | #include <asm/nmi.h> | 50 | #include <asm/nmi.h> |
| 50 | #include <asm/msidef.h> | 51 | #include <asm/msidef.h> |
| 51 | #include <asm/hypertransport.h> | 52 | #include <asm/hypertransport.h> |
| 53 | #include <asm/irq_remapping.h> | ||
| 52 | 54 | ||
| 53 | #include <mach_ipi.h> | 55 | #include <mach_ipi.h> |
| 54 | #include <mach_apic.h> | 56 | #include <mach_apic.h> |
| 55 | 57 | ||
| 58 | #define __apicdebuginit(type) static type __init | ||
| 59 | |||
| 56 | struct irq_cfg { | 60 | struct irq_cfg { |
| 57 | cpumask_t domain; | 61 | cpumask_t domain; |
| 58 | cpumask_t old_domain; | 62 | cpumask_t old_domain; |
| @@ -87,8 +91,6 @@ int first_system_vector = 0xfe; | |||
| 87 | 91 | ||
| 88 | char system_vectors[NR_VECTORS] = { [0 ... NR_VECTORS-1] = SYS_VECTOR_FREE}; | 92 | char system_vectors[NR_VECTORS] = { [0 ... NR_VECTORS-1] = SYS_VECTOR_FREE}; |
| 89 | 93 | ||
| 90 | #define __apicdebuginit __init | ||
| 91 | |||
| 92 | int sis_apic_bug; /* not actually supported, dummy for compile */ | 94 | int sis_apic_bug; /* not actually supported, dummy for compile */ |
| 93 | 95 | ||
| 94 | static int no_timer_check; | 96 | static int no_timer_check; |
| @@ -108,6 +110,9 @@ static DEFINE_SPINLOCK(vector_lock); | |||
| 108 | */ | 110 | */ |
| 109 | int nr_ioapic_registers[MAX_IO_APICS]; | 111 | int nr_ioapic_registers[MAX_IO_APICS]; |
| 110 | 112 | ||
| 113 | /* I/O APIC RTE contents at the OS boot up */ | ||
| 114 | struct IO_APIC_route_entry *early_ioapic_entries[MAX_IO_APICS]; | ||
| 115 | |||
| 111 | /* I/O APIC entries */ | 116 | /* I/O APIC entries */ |
| 112 | struct mp_config_ioapic mp_ioapics[MAX_IO_APICS]; | 117 | struct mp_config_ioapic mp_ioapics[MAX_IO_APICS]; |
| 113 | int nr_ioapics; | 118 | int nr_ioapics; |
| @@ -303,7 +308,12 @@ static void __target_IO_APIC_irq(unsigned int irq, unsigned int dest, u8 vector) | |||
| 303 | pin = entry->pin; | 308 | pin = entry->pin; |
| 304 | if (pin == -1) | 309 | if (pin == -1) |
| 305 | break; | 310 | break; |
| 306 | io_apic_write(apic, 0x11 + pin*2, dest); | 311 | /* |
| 312 | * With interrupt-remapping, destination information comes | ||
| 313 | * from interrupt-remapping table entry. | ||
| 314 | */ | ||
| 315 | if (!irq_remapped(irq)) | ||
| 316 | io_apic_write(apic, 0x11 + pin*2, dest); | ||
| 307 | reg = io_apic_read(apic, 0x10 + pin*2); | 317 | reg = io_apic_read(apic, 0x10 + pin*2); |
| 308 | reg &= ~IO_APIC_REDIR_VECTOR_MASK; | 318 | reg &= ~IO_APIC_REDIR_VECTOR_MASK; |
| 309 | reg |= vector; | 319 | reg |= vector; |
| @@ -440,6 +450,69 @@ static void clear_IO_APIC (void) | |||
| 440 | clear_IO_APIC_pin(apic, pin); | 450 | clear_IO_APIC_pin(apic, pin); |
| 441 | } | 451 | } |
| 442 | 452 | ||
| 453 | /* | ||
| 454 | * Saves and masks all the unmasked IO-APIC RTE's | ||
| 455 | */ | ||
| 456 | int save_mask_IO_APIC_setup(void) | ||
| 457 | { | ||
| 458 | union IO_APIC_reg_01 reg_01; | ||
| 459 | unsigned long flags; | ||
| 460 | int apic, pin; | ||
| 461 | |||
| 462 | /* | ||
| 463 | * The number of IO-APIC IRQ registers (== #pins): | ||
| 464 | */ | ||
| 465 | for (apic = 0; apic < nr_ioapics; apic++) { | ||
| 466 | spin_lock_irqsave(&ioapic_lock, flags); | ||
| 467 | reg_01.raw = io_apic_read(apic, 1); | ||
| 468 | spin_unlock_irqrestore(&ioapic_lock, flags); | ||
| 469 | nr_ioapic_registers[apic] = reg_01.bits.entries+1; | ||
| 470 | } | ||
| 471 | |||
| 472 | for (apic = 0; apic < nr_ioapics; apic++) { | ||
| 473 | early_ioapic_entries[apic] = | ||
| 474 | kzalloc(sizeof(struct IO_APIC_route_entry) * | ||
| 475 | nr_ioapic_registers[apic], GFP_KERNEL); | ||
| 476 | if (!early_ioapic_entries[apic]) | ||
| 477 | return -ENOMEM; | ||
| 478 | } | ||
| 479 | |||
| 480 | for (apic = 0; apic < nr_ioapics; apic++) | ||
| 481 | for (pin = 0; pin < nr_ioapic_registers[apic]; pin++) { | ||
| 482 | struct IO_APIC_route_entry entry; | ||
| 483 | |||
| 484 | entry = early_ioapic_entries[apic][pin] = | ||
| 485 | ioapic_read_entry(apic, pin); | ||
| 486 | if (!entry.mask) { | ||
| 487 | entry.mask = 1; | ||
| 488 | ioapic_write_entry(apic, pin, entry); | ||
| 489 | } | ||
| 490 | } | ||
| 491 | return 0; | ||
| 492 | } | ||
| 493 | |||
| 494 | void restore_IO_APIC_setup(void) | ||
| 495 | { | ||
| 496 | int apic, pin; | ||
| 497 | |||
| 498 | for (apic = 0; apic < nr_ioapics; apic++) | ||
| 499 | for (pin = 0; pin < nr_ioapic_registers[apic]; pin++) | ||
| 500 | ioapic_write_entry(apic, pin, | ||
| 501 | early_ioapic_entries[apic][pin]); | ||
| 502 | } | ||
| 503 | |||
| 504 | void reinit_intr_remapped_IO_APIC(int intr_remapping) | ||
| 505 | { | ||
| 506 | /* | ||
| 507 | * for now plain restore of previous settings. | ||
| 508 | * TBD: In the case of OS enabling interrupt-remapping, | ||
| 509 | * IO-APIC RTE's need to be setup to point to interrupt-remapping | ||
| 510 | * table entries. for now, do a plain restore, and wait for | ||
| 511 | * the setup_IO_APIC_irqs() to do proper initialization. | ||
| 512 | */ | ||
| 513 | restore_IO_APIC_setup(); | ||
| 514 | } | ||
| 515 | |||
| 443 | int skip_ioapic_setup; | 516 | int skip_ioapic_setup; |
| 444 | int ioapic_force; | 517 | int ioapic_force; |
| 445 | 518 | ||
| @@ -839,18 +912,98 @@ void __setup_vector_irq(int cpu) | |||
| 839 | } | 912 | } |
| 840 | 913 | ||
| 841 | static struct irq_chip ioapic_chip; | 914 | static struct irq_chip ioapic_chip; |
| 915 | #ifdef CONFIG_INTR_REMAP | ||
| 916 | static struct irq_chip ir_ioapic_chip; | ||
| 917 | #endif | ||
| 842 | 918 | ||
| 843 | static void ioapic_register_intr(int irq, unsigned long trigger) | 919 | static void ioapic_register_intr(int irq, unsigned long trigger) |
| 844 | { | 920 | { |
| 845 | if (trigger) { | 921 | if (trigger) |
| 846 | irq_desc[irq].status |= IRQ_LEVEL; | 922 | irq_desc[irq].status |= IRQ_LEVEL; |
| 847 | set_irq_chip_and_handler_name(irq, &ioapic_chip, | 923 | else |
| 848 | handle_fasteoi_irq, "fasteoi"); | ||
| 849 | } else { | ||
| 850 | irq_desc[irq].status &= ~IRQ_LEVEL; | 924 | irq_desc[irq].status &= ~IRQ_LEVEL; |
| 925 | |||
| 926 | #ifdef CONFIG_INTR_REMAP | ||
| 927 | if (irq_remapped(irq)) { | ||
| 928 | irq_desc[irq].status |= IRQ_MOVE_PCNTXT; | ||
| 929 | if (trigger) | ||
| 930 | set_irq_chip_and_handler_name(irq, &ir_ioapic_chip, | ||
| 931 | handle_fasteoi_irq, | ||
| 932 | "fasteoi"); | ||
| 933 | else | ||
| 934 | set_irq_chip_and_handler_name(irq, &ir_ioapic_chip, | ||
| 935 | handle_edge_irq, "edge"); | ||
| 936 | return; | ||
| 937 | } | ||
| 938 | #endif | ||
| 939 | if (trigger) | ||
| 940 | set_irq_chip_and_handler_name(irq, &ioapic_chip, | ||
| 941 | handle_fasteoi_irq, | ||
| 942 | "fasteoi"); | ||
| 943 | else | ||
| 851 | set_irq_chip_and_handler_name(irq, &ioapic_chip, | 944 | set_irq_chip_and_handler_name(irq, &ioapic_chip, |
| 852 | handle_edge_irq, "edge"); | 945 | handle_edge_irq, "edge"); |
| 946 | } | ||
| 947 | |||
| 948 | static int setup_ioapic_entry(int apic, int irq, | ||
| 949 | struct IO_APIC_route_entry *entry, | ||
| 950 | unsigned int destination, int trigger, | ||
| 951 | int polarity, int vector) | ||
| 952 | { | ||
| 953 | /* | ||
| 954 | * add it to the IO-APIC irq-routing table: | ||
| 955 | */ | ||
| 956 | memset(entry,0,sizeof(*entry)); | ||
| 957 | |||
| 958 | #ifdef CONFIG_INTR_REMAP | ||
| 959 | if (intr_remapping_enabled) { | ||
| 960 | struct intel_iommu *iommu = map_ioapic_to_ir(apic); | ||
| 961 | struct irte irte; | ||
| 962 | struct IR_IO_APIC_route_entry *ir_entry = | ||
| 963 | (struct IR_IO_APIC_route_entry *) entry; | ||
| 964 | int index; | ||
| 965 | |||
| 966 | if (!iommu) | ||
| 967 | panic("No mapping iommu for ioapic %d\n", apic); | ||
| 968 | |||
| 969 | index = alloc_irte(iommu, irq, 1); | ||
| 970 | if (index < 0) | ||
| 971 | panic("Failed to allocate IRTE for ioapic %d\n", apic); | ||
| 972 | |||
| 973 | memset(&irte, 0, sizeof(irte)); | ||
| 974 | |||
| 975 | irte.present = 1; | ||
| 976 | irte.dst_mode = INT_DEST_MODE; | ||
| 977 | irte.trigger_mode = trigger; | ||
| 978 | irte.dlvry_mode = INT_DELIVERY_MODE; | ||
| 979 | irte.vector = vector; | ||
| 980 | irte.dest_id = IRTE_DEST(destination); | ||
| 981 | |||
| 982 | modify_irte(irq, &irte); | ||
| 983 | |||
| 984 | ir_entry->index2 = (index >> 15) & 0x1; | ||
| 985 | ir_entry->zero = 0; | ||
| 986 | ir_entry->format = 1; | ||
| 987 | ir_entry->index = (index & 0x7fff); | ||
| 988 | } else | ||
| 989 | #endif | ||
| 990 | { | ||
| 991 | entry->delivery_mode = INT_DELIVERY_MODE; | ||
| 992 | entry->dest_mode = INT_DEST_MODE; | ||
| 993 | entry->dest = destination; | ||
| 853 | } | 994 | } |
| 995 | |||
| 996 | entry->mask = 0; /* enable IRQ */ | ||
| 997 | entry->trigger = trigger; | ||
| 998 | entry->polarity = polarity; | ||
| 999 | entry->vector = vector; | ||
| 1000 | |||
| 1001 | /* Mask level triggered irqs. | ||
| 1002 | * Use IRQ_DELAYED_DISABLE for edge triggered irqs. | ||
| 1003 | */ | ||
| 1004 | if (trigger) | ||
| 1005 | entry->mask = 1; | ||
| 1006 | return 0; | ||
| 854 | } | 1007 | } |
| 855 | 1008 | ||
| 856 | static void setup_IO_APIC_irq(int apic, int pin, unsigned int irq, | 1009 | static void setup_IO_APIC_irq(int apic, int pin, unsigned int irq, |
| @@ -875,24 +1028,15 @@ static void setup_IO_APIC_irq(int apic, int pin, unsigned int irq, | |||
| 875 | apic, mp_ioapics[apic].mp_apicid, pin, cfg->vector, | 1028 | apic, mp_ioapics[apic].mp_apicid, pin, cfg->vector, |
| 876 | irq, trigger, polarity); | 1029 | irq, trigger, polarity); |
| 877 | 1030 | ||
| 878 | /* | ||
| 879 | * add it to the IO-APIC irq-routing table: | ||
| 880 | */ | ||
| 881 | memset(&entry,0,sizeof(entry)); | ||
| 882 | |||
| 883 | entry.delivery_mode = INT_DELIVERY_MODE; | ||
| 884 | entry.dest_mode = INT_DEST_MODE; | ||
| 885 | entry.dest = cpu_mask_to_apicid(mask); | ||
| 886 | entry.mask = 0; /* enable IRQ */ | ||
| 887 | entry.trigger = trigger; | ||
| 888 | entry.polarity = polarity; | ||
| 889 | entry.vector = cfg->vector; | ||
| 890 | 1031 | ||
| 891 | /* Mask level triggered irqs. | 1032 | if (setup_ioapic_entry(mp_ioapics[apic].mp_apicid, irq, &entry, |
| 892 | * Use IRQ_DELAYED_DISABLE for edge triggered irqs. | 1033 | cpu_mask_to_apicid(mask), trigger, polarity, |
| 893 | */ | 1034 | cfg->vector)) { |
| 894 | if (trigger) | 1035 | printk("Failed to setup ioapic entry for ioapic %d, pin %d\n", |
| 895 | entry.mask = 1; | 1036 | mp_ioapics[apic].mp_apicid, pin); |
| 1037 | __clear_irq_vector(irq); | ||
| 1038 | return; | ||
| 1039 | } | ||
| 896 | 1040 | ||
| 897 | ioapic_register_intr(irq, trigger); | 1041 | ioapic_register_intr(irq, trigger); |
| 898 | if (irq < 16) | 1042 | if (irq < 16) |
| @@ -944,6 +1088,9 @@ static void __init setup_timer_IRQ0_pin(unsigned int apic, unsigned int pin, | |||
| 944 | { | 1088 | { |
| 945 | struct IO_APIC_route_entry entry; | 1089 | struct IO_APIC_route_entry entry; |
| 946 | 1090 | ||
| 1091 | if (intr_remapping_enabled) | ||
| 1092 | return; | ||
| 1093 | |||
| 947 | memset(&entry, 0, sizeof(entry)); | 1094 | memset(&entry, 0, sizeof(entry)); |
| 948 | 1095 | ||
| 949 | /* | 1096 | /* |
| @@ -970,7 +1117,8 @@ static void __init setup_timer_IRQ0_pin(unsigned int apic, unsigned int pin, | |||
| 970 | ioapic_write_entry(apic, pin, entry); | 1117 | ioapic_write_entry(apic, pin, entry); |
| 971 | } | 1118 | } |
| 972 | 1119 | ||
| 973 | void __apicdebuginit print_IO_APIC(void) | 1120 | |
| 1121 | __apicdebuginit(void) print_IO_APIC(void) | ||
| 974 | { | 1122 | { |
| 975 | int apic, i; | 1123 | int apic, i; |
| 976 | union IO_APIC_reg_00 reg_00; | 1124 | union IO_APIC_reg_00 reg_00; |
| @@ -1064,9 +1212,7 @@ void __apicdebuginit print_IO_APIC(void) | |||
| 1064 | return; | 1212 | return; |
| 1065 | } | 1213 | } |
| 1066 | 1214 | ||
| 1067 | #if 0 | 1215 | __apicdebuginit(void) print_APIC_bitfield(int base) |
| 1068 | |||
| 1069 | static __apicdebuginit void print_APIC_bitfield (int base) | ||
| 1070 | { | 1216 | { |
| 1071 | unsigned int v; | 1217 | unsigned int v; |
| 1072 | int i, j; | 1218 | int i, j; |
| @@ -1087,9 +1233,10 @@ static __apicdebuginit void print_APIC_bitfield (int base) | |||
| 1087 | } | 1233 | } |
| 1088 | } | 1234 | } |
| 1089 | 1235 | ||
| 1090 | void __apicdebuginit print_local_APIC(void * dummy) | 1236 | __apicdebuginit(void) print_local_APIC(void *dummy) |
| 1091 | { | 1237 | { |
| 1092 | unsigned int v, ver, maxlvt; | 1238 | unsigned int v, ver, maxlvt; |
| 1239 | unsigned long icr; | ||
| 1093 | 1240 | ||
| 1094 | if (apic_verbosity == APIC_QUIET) | 1241 | if (apic_verbosity == APIC_QUIET) |
| 1095 | return; | 1242 | return; |
| @@ -1097,7 +1244,7 @@ void __apicdebuginit print_local_APIC(void * dummy) | |||
| 1097 | printk("\n" KERN_DEBUG "printing local APIC contents on CPU#%d/%d:\n", | 1244 | printk("\n" KERN_DEBUG "printing local APIC contents on CPU#%d/%d:\n", |
| 1098 | smp_processor_id(), hard_smp_processor_id()); | 1245 | smp_processor_id(), hard_smp_processor_id()); |
| 1099 | v = apic_read(APIC_ID); | 1246 | v = apic_read(APIC_ID); |
| 1100 | printk(KERN_INFO "... APIC ID: %08x (%01x)\n", v, GET_APIC_ID(read_apic_id())); | 1247 | printk(KERN_INFO "... APIC ID: %08x (%01x)\n", v, read_apic_id()); |
| 1101 | v = apic_read(APIC_LVR); | 1248 | v = apic_read(APIC_LVR); |
| 1102 | printk(KERN_INFO "... APIC VERSION: %08x\n", v); | 1249 | printk(KERN_INFO "... APIC VERSION: %08x\n", v); |
| 1103 | ver = GET_APIC_VERSION(v); | 1250 | ver = GET_APIC_VERSION(v); |
| @@ -1133,10 +1280,9 @@ void __apicdebuginit print_local_APIC(void * dummy) | |||
| 1133 | v = apic_read(APIC_ESR); | 1280 | v = apic_read(APIC_ESR); |
| 1134 | printk(KERN_DEBUG "... APIC ESR: %08x\n", v); | 1281 | printk(KERN_DEBUG "... APIC ESR: %08x\n", v); |
| 1135 | 1282 | ||
| 1136 | v = apic_read(APIC_ICR); | 1283 | icr = apic_icr_read(); |
| 1137 | printk(KERN_DEBUG "... APIC ICR: %08x\n", v); | 1284 | printk(KERN_DEBUG "... APIC ICR: %08x\n", icr); |
| 1138 | v = apic_read(APIC_ICR2); | 1285 | printk(KERN_DEBUG "... APIC ICR2: %08x\n", icr >> 32); |
| 1139 | printk(KERN_DEBUG "... APIC ICR2: %08x\n", v); | ||
| 1140 | 1286 | ||
| 1141 | v = apic_read(APIC_LVTT); | 1287 | v = apic_read(APIC_LVTT); |
| 1142 | printk(KERN_DEBUG "... APIC LVTT: %08x\n", v); | 1288 | printk(KERN_DEBUG "... APIC LVTT: %08x\n", v); |
| @@ -1164,12 +1310,12 @@ void __apicdebuginit print_local_APIC(void * dummy) | |||
| 1164 | printk("\n"); | 1310 | printk("\n"); |
| 1165 | } | 1311 | } |
| 1166 | 1312 | ||
| 1167 | void print_all_local_APICs (void) | 1313 | __apicdebuginit(void) print_all_local_APICs(void) |
| 1168 | { | 1314 | { |
| 1169 | on_each_cpu(print_local_APIC, NULL, 1); | 1315 | on_each_cpu(print_local_APIC, NULL, 1); |
| 1170 | } | 1316 | } |
| 1171 | 1317 | ||
| 1172 | void __apicdebuginit print_PIC(void) | 1318 | __apicdebuginit(void) print_PIC(void) |
| 1173 | { | 1319 | { |
| 1174 | unsigned int v; | 1320 | unsigned int v; |
| 1175 | unsigned long flags; | 1321 | unsigned long flags; |
| @@ -1201,7 +1347,17 @@ void __apicdebuginit print_PIC(void) | |||
| 1201 | printk(KERN_DEBUG "... PIC ELCR: %04x\n", v); | 1347 | printk(KERN_DEBUG "... PIC ELCR: %04x\n", v); |
| 1202 | } | 1348 | } |
| 1203 | 1349 | ||
| 1204 | #endif /* 0 */ | 1350 | __apicdebuginit(int) print_all_ICs(void) |
| 1351 | { | ||
| 1352 | print_PIC(); | ||
| 1353 | print_all_local_APICs(); | ||
| 1354 | print_IO_APIC(); | ||
| 1355 | |||
| 1356 | return 0; | ||
| 1357 | } | ||
| 1358 | |||
| 1359 | fs_initcall(print_all_ICs); | ||
| 1360 | |||
| 1205 | 1361 | ||
| 1206 | void __init enable_IO_APIC(void) | 1362 | void __init enable_IO_APIC(void) |
| 1207 | { | 1363 | { |
| @@ -1291,7 +1447,7 @@ void disable_IO_APIC(void) | |||
| 1291 | entry.dest_mode = 0; /* Physical */ | 1447 | entry.dest_mode = 0; /* Physical */ |
| 1292 | entry.delivery_mode = dest_ExtINT; /* ExtInt */ | 1448 | entry.delivery_mode = dest_ExtINT; /* ExtInt */ |
| 1293 | entry.vector = 0; | 1449 | entry.vector = 0; |
| 1294 | entry.dest = GET_APIC_ID(read_apic_id()); | 1450 | entry.dest = read_apic_id(); |
| 1295 | 1451 | ||
| 1296 | /* | 1452 | /* |
| 1297 | * Add it to the IO-APIC irq-routing table: | 1453 | * Add it to the IO-APIC irq-routing table: |
| @@ -1397,6 +1553,147 @@ static int ioapic_retrigger_irq(unsigned int irq) | |||
| 1397 | */ | 1553 | */ |
| 1398 | 1554 | ||
| 1399 | #ifdef CONFIG_SMP | 1555 | #ifdef CONFIG_SMP |
| 1556 | |||
| 1557 | #ifdef CONFIG_INTR_REMAP | ||
| 1558 | static void ir_irq_migration(struct work_struct *work); | ||
| 1559 | |||
| 1560 | static DECLARE_DELAYED_WORK(ir_migration_work, ir_irq_migration); | ||
| 1561 | |||
| 1562 | /* | ||
| 1563 | * Migrate the IO-APIC irq in the presence of intr-remapping. | ||
| 1564 | * | ||
| 1565 | * For edge triggered, irq migration is a simple atomic update(of vector | ||
| 1566 | * and cpu destination) of IRTE and flush the hardware cache. | ||
| 1567 | * | ||
| 1568 | * For level triggered, we need to modify the io-apic RTE aswell with the update | ||
| 1569 | * vector information, along with modifying IRTE with vector and destination. | ||
| 1570 | * So irq migration for level triggered is little bit more complex compared to | ||
| 1571 | * edge triggered migration. But the good news is, we use the same algorithm | ||
| 1572 | * for level triggered migration as we have today, only difference being, | ||
| 1573 | * we now initiate the irq migration from process context instead of the | ||
| 1574 | * interrupt context. | ||
| 1575 | * | ||
| 1576 | * In future, when we do a directed EOI (combined with cpu EOI broadcast | ||
| 1577 | * suppression) to the IO-APIC, level triggered irq migration will also be | ||
| 1578 | * as simple as edge triggered migration and we can do the irq migration | ||
| 1579 | * with a simple atomic update to IO-APIC RTE. | ||
| 1580 | */ | ||
| 1581 | static void migrate_ioapic_irq(int irq, cpumask_t mask) | ||
| 1582 | { | ||
| 1583 | struct irq_cfg *cfg = irq_cfg + irq; | ||
| 1584 | struct irq_desc *desc = irq_desc + irq; | ||
| 1585 | cpumask_t tmp, cleanup_mask; | ||
| 1586 | struct irte irte; | ||
| 1587 | int modify_ioapic_rte = desc->status & IRQ_LEVEL; | ||
| 1588 | unsigned int dest; | ||
| 1589 | unsigned long flags; | ||
| 1590 | |||
| 1591 | cpus_and(tmp, mask, cpu_online_map); | ||
| 1592 | if (cpus_empty(tmp)) | ||
| 1593 | return; | ||
| 1594 | |||
| 1595 | if (get_irte(irq, &irte)) | ||
| 1596 | return; | ||
| 1597 | |||
| 1598 | if (assign_irq_vector(irq, mask)) | ||
| 1599 | return; | ||
| 1600 | |||
| 1601 | cpus_and(tmp, cfg->domain, mask); | ||
| 1602 | dest = cpu_mask_to_apicid(tmp); | ||
| 1603 | |||
| 1604 | if (modify_ioapic_rte) { | ||
| 1605 | spin_lock_irqsave(&ioapic_lock, flags); | ||
| 1606 | __target_IO_APIC_irq(irq, dest, cfg->vector); | ||
| 1607 | spin_unlock_irqrestore(&ioapic_lock, flags); | ||
| 1608 | } | ||
| 1609 | |||
| 1610 | irte.vector = cfg->vector; | ||
| 1611 | irte.dest_id = IRTE_DEST(dest); | ||
| 1612 | |||
| 1613 | /* | ||
| 1614 | * Modified the IRTE and flushes the Interrupt entry cache. | ||
| 1615 | */ | ||
| 1616 | modify_irte(irq, &irte); | ||
| 1617 | |||
| 1618 | if (cfg->move_in_progress) { | ||
| 1619 | cpus_and(cleanup_mask, cfg->old_domain, cpu_online_map); | ||
| 1620 | cfg->move_cleanup_count = cpus_weight(cleanup_mask); | ||
| 1621 | send_IPI_mask(cleanup_mask, IRQ_MOVE_CLEANUP_VECTOR); | ||
| 1622 | cfg->move_in_progress = 0; | ||
| 1623 | } | ||
| 1624 | |||
| 1625 | irq_desc[irq].affinity = mask; | ||
| 1626 | } | ||
| 1627 | |||
| 1628 | static int migrate_irq_remapped_level(int irq) | ||
| 1629 | { | ||
| 1630 | int ret = -1; | ||
| 1631 | |||
| 1632 | mask_IO_APIC_irq(irq); | ||
| 1633 | |||
| 1634 | if (io_apic_level_ack_pending(irq)) { | ||
| 1635 | /* | ||
| 1636 | * Interrupt in progress. Migrating irq now will change the | ||
| 1637 | * vector information in the IO-APIC RTE and that will confuse | ||
| 1638 | * the EOI broadcast performed by cpu. | ||
| 1639 | * So, delay the irq migration to the next instance. | ||
| 1640 | */ | ||
| 1641 | schedule_delayed_work(&ir_migration_work, 1); | ||
| 1642 | goto unmask; | ||
| 1643 | } | ||
| 1644 | |||
| 1645 | /* everthing is clear. we have right of way */ | ||
| 1646 | migrate_ioapic_irq(irq, irq_desc[irq].pending_mask); | ||
| 1647 | |||
| 1648 | ret = 0; | ||
| 1649 | irq_desc[irq].status &= ~IRQ_MOVE_PENDING; | ||
| 1650 | cpus_clear(irq_desc[irq].pending_mask); | ||
| 1651 | |||
| 1652 | unmask: | ||
| 1653 | unmask_IO_APIC_irq(irq); | ||
| 1654 | return ret; | ||
| 1655 | } | ||
| 1656 | |||
| 1657 | static void ir_irq_migration(struct work_struct *work) | ||
| 1658 | { | ||
| 1659 | int irq; | ||
| 1660 | |||
| 1661 | for (irq = 0; irq < NR_IRQS; irq++) { | ||
| 1662 | struct irq_desc *desc = irq_desc + irq; | ||
| 1663 | if (desc->status & IRQ_MOVE_PENDING) { | ||
| 1664 | unsigned long flags; | ||
| 1665 | |||
| 1666 | spin_lock_irqsave(&desc->lock, flags); | ||
| 1667 | if (!desc->chip->set_affinity || | ||
| 1668 | !(desc->status & IRQ_MOVE_PENDING)) { | ||
| 1669 | desc->status &= ~IRQ_MOVE_PENDING; | ||
| 1670 | spin_unlock_irqrestore(&desc->lock, flags); | ||
| 1671 | continue; | ||
| 1672 | } | ||
| 1673 | |||
| 1674 | desc->chip->set_affinity(irq, | ||
| 1675 | irq_desc[irq].pending_mask); | ||
| 1676 | spin_unlock_irqrestore(&desc->lock, flags); | ||
| 1677 | } | ||
| 1678 | } | ||
| 1679 | } | ||
| 1680 | |||
| 1681 | /* | ||
| 1682 | * Migrates the IRQ destination in the process context. | ||
| 1683 | */ | ||
| 1684 | static void set_ir_ioapic_affinity_irq(unsigned int irq, cpumask_t mask) | ||
| 1685 | { | ||
| 1686 | if (irq_desc[irq].status & IRQ_LEVEL) { | ||
| 1687 | irq_desc[irq].status |= IRQ_MOVE_PENDING; | ||
| 1688 | irq_desc[irq].pending_mask = mask; | ||
| 1689 | migrate_irq_remapped_level(irq); | ||
| 1690 | return; | ||
| 1691 | } | ||
| 1692 | |||
| 1693 | migrate_ioapic_irq(irq, mask); | ||
| 1694 | } | ||
| 1695 | #endif | ||
| 1696 | |||
| 1400 | asmlinkage void smp_irq_move_cleanup_interrupt(void) | 1697 | asmlinkage void smp_irq_move_cleanup_interrupt(void) |
| 1401 | { | 1698 | { |
| 1402 | unsigned vector, me; | 1699 | unsigned vector, me; |
| @@ -1453,6 +1750,17 @@ static void irq_complete_move(unsigned int irq) | |||
| 1453 | #else | 1750 | #else |
| 1454 | static inline void irq_complete_move(unsigned int irq) {} | 1751 | static inline void irq_complete_move(unsigned int irq) {} |
| 1455 | #endif | 1752 | #endif |
| 1753 | #ifdef CONFIG_INTR_REMAP | ||
| 1754 | static void ack_x2apic_level(unsigned int irq) | ||
| 1755 | { | ||
| 1756 | ack_x2APIC_irq(); | ||
| 1757 | } | ||
| 1758 | |||
| 1759 | static void ack_x2apic_edge(unsigned int irq) | ||
| 1760 | { | ||
| 1761 | ack_x2APIC_irq(); | ||
| 1762 | } | ||
| 1763 | #endif | ||
| 1456 | 1764 | ||
| 1457 | static void ack_apic_edge(unsigned int irq) | 1765 | static void ack_apic_edge(unsigned int irq) |
| 1458 | { | 1766 | { |
| @@ -1527,6 +1835,21 @@ static struct irq_chip ioapic_chip __read_mostly = { | |||
| 1527 | .retrigger = ioapic_retrigger_irq, | 1835 | .retrigger = ioapic_retrigger_irq, |
| 1528 | }; | 1836 | }; |
| 1529 | 1837 | ||
| 1838 | #ifdef CONFIG_INTR_REMAP | ||
| 1839 | static struct irq_chip ir_ioapic_chip __read_mostly = { | ||
| 1840 | .name = "IR-IO-APIC", | ||
| 1841 | .startup = startup_ioapic_irq, | ||
| 1842 | .mask = mask_IO_APIC_irq, | ||
| 1843 | .unmask = unmask_IO_APIC_irq, | ||
| 1844 | .ack = ack_x2apic_edge, | ||
| 1845 | .eoi = ack_x2apic_level, | ||
| 1846 | #ifdef CONFIG_SMP | ||
| 1847 | .set_affinity = set_ir_ioapic_affinity_irq, | ||
| 1848 | #endif | ||
| 1849 | .retrigger = ioapic_retrigger_irq, | ||
| 1850 | }; | ||
| 1851 | #endif | ||
| 1852 | |||
| 1530 | static inline void init_IO_APIC_traps(void) | 1853 | static inline void init_IO_APIC_traps(void) |
| 1531 | { | 1854 | { |
| 1532 | int irq; | 1855 | int irq; |
| @@ -1712,6 +2035,8 @@ static inline void __init check_timer(void) | |||
| 1712 | * 8259A. | 2035 | * 8259A. |
| 1713 | */ | 2036 | */ |
| 1714 | if (pin1 == -1) { | 2037 | if (pin1 == -1) { |
| 2038 | if (intr_remapping_enabled) | ||
| 2039 | panic("BIOS bug: timer not connected to IO-APIC"); | ||
| 1715 | pin1 = pin2; | 2040 | pin1 = pin2; |
| 1716 | apic1 = apic2; | 2041 | apic1 = apic2; |
| 1717 | no_pin1 = 1; | 2042 | no_pin1 = 1; |
| @@ -1738,6 +2063,8 @@ static inline void __init check_timer(void) | |||
| 1738 | clear_IO_APIC_pin(0, pin1); | 2063 | clear_IO_APIC_pin(0, pin1); |
| 1739 | goto out; | 2064 | goto out; |
| 1740 | } | 2065 | } |
| 2066 | if (intr_remapping_enabled) | ||
| 2067 | panic("timer doesn't work through Interrupt-remapped IO-APIC"); | ||
| 1741 | clear_IO_APIC_pin(apic1, pin1); | 2068 | clear_IO_APIC_pin(apic1, pin1); |
| 1742 | if (!no_pin1) | 2069 | if (!no_pin1) |
| 1743 | apic_printk(APIC_QUIET, KERN_ERR "..MP-BIOS bug: " | 2070 | apic_printk(APIC_QUIET, KERN_ERR "..MP-BIOS bug: " |
| @@ -1854,8 +2181,6 @@ void __init setup_IO_APIC(void) | |||
| 1854 | setup_IO_APIC_irqs(); | 2181 | setup_IO_APIC_irqs(); |
| 1855 | init_IO_APIC_traps(); | 2182 | init_IO_APIC_traps(); |
| 1856 | check_timer(); | 2183 | check_timer(); |
| 1857 | if (!acpi_ioapic) | ||
| 1858 | print_IO_APIC(); | ||
| 1859 | } | 2184 | } |
| 1860 | 2185 | ||
| 1861 | struct sysfs_ioapic_data { | 2186 | struct sysfs_ioapic_data { |
| @@ -1977,6 +2302,9 @@ void destroy_irq(unsigned int irq) | |||
| 1977 | 2302 | ||
| 1978 | dynamic_irq_cleanup(irq); | 2303 | dynamic_irq_cleanup(irq); |
| 1979 | 2304 | ||
| 2305 | #ifdef CONFIG_INTR_REMAP | ||
| 2306 | free_irte(irq); | ||
| 2307 | #endif | ||
| 1980 | spin_lock_irqsave(&vector_lock, flags); | 2308 | spin_lock_irqsave(&vector_lock, flags); |
| 1981 | __clear_irq_vector(irq); | 2309 | __clear_irq_vector(irq); |
| 1982 | spin_unlock_irqrestore(&vector_lock, flags); | 2310 | spin_unlock_irqrestore(&vector_lock, flags); |
| @@ -1995,11 +2323,42 @@ static int msi_compose_msg(struct pci_dev *pdev, unsigned int irq, struct msi_ms | |||
| 1995 | 2323 | ||
| 1996 | tmp = TARGET_CPUS; | 2324 | tmp = TARGET_CPUS; |
| 1997 | err = assign_irq_vector(irq, tmp); | 2325 | err = assign_irq_vector(irq, tmp); |
| 1998 | if (!err) { | 2326 | if (err) |
| 1999 | cpus_and(tmp, cfg->domain, tmp); | 2327 | return err; |
| 2000 | dest = cpu_mask_to_apicid(tmp); | 2328 | |
| 2329 | cpus_and(tmp, cfg->domain, tmp); | ||
| 2330 | dest = cpu_mask_to_apicid(tmp); | ||
| 2331 | |||
| 2332 | #ifdef CONFIG_INTR_REMAP | ||
| 2333 | if (irq_remapped(irq)) { | ||
| 2334 | struct irte irte; | ||
| 2335 | int ir_index; | ||
| 2336 | u16 sub_handle; | ||
| 2337 | |||
| 2338 | ir_index = map_irq_to_irte_handle(irq, &sub_handle); | ||
| 2339 | BUG_ON(ir_index == -1); | ||
| 2340 | |||
| 2341 | memset (&irte, 0, sizeof(irte)); | ||
| 2342 | |||
| 2343 | irte.present = 1; | ||
| 2344 | irte.dst_mode = INT_DEST_MODE; | ||
| 2345 | irte.trigger_mode = 0; /* edge */ | ||
| 2346 | irte.dlvry_mode = INT_DELIVERY_MODE; | ||
| 2347 | irte.vector = cfg->vector; | ||
| 2348 | irte.dest_id = IRTE_DEST(dest); | ||
| 2349 | |||
| 2350 | modify_irte(irq, &irte); | ||
| 2001 | 2351 | ||
| 2002 | msg->address_hi = MSI_ADDR_BASE_HI; | 2352 | msg->address_hi = MSI_ADDR_BASE_HI; |
| 2353 | msg->data = sub_handle; | ||
| 2354 | msg->address_lo = MSI_ADDR_BASE_LO | MSI_ADDR_IR_EXT_INT | | ||
| 2355 | MSI_ADDR_IR_SHV | | ||
| 2356 | MSI_ADDR_IR_INDEX1(ir_index) | | ||
| 2357 | MSI_ADDR_IR_INDEX2(ir_index); | ||
| 2358 | } else | ||
| 2359 | #endif | ||
| 2360 | { | ||
| 2361 | msg->address_hi = MSI_ADDR_BASE_HI; | ||
| 2003 | msg->address_lo = | 2362 | msg->address_lo = |
| 2004 | MSI_ADDR_BASE_LO | | 2363 | MSI_ADDR_BASE_LO | |
| 2005 | ((INT_DEST_MODE == 0) ? | 2364 | ((INT_DEST_MODE == 0) ? |
| @@ -2049,6 +2408,55 @@ static void set_msi_irq_affinity(unsigned int irq, cpumask_t mask) | |||
| 2049 | write_msi_msg(irq, &msg); | 2408 | write_msi_msg(irq, &msg); |
| 2050 | irq_desc[irq].affinity = mask; | 2409 | irq_desc[irq].affinity = mask; |
| 2051 | } | 2410 | } |
| 2411 | |||
| 2412 | #ifdef CONFIG_INTR_REMAP | ||
| 2413 | /* | ||
| 2414 | * Migrate the MSI irq to another cpumask. This migration is | ||
| 2415 | * done in the process context using interrupt-remapping hardware. | ||
| 2416 | */ | ||
| 2417 | static void ir_set_msi_irq_affinity(unsigned int irq, cpumask_t mask) | ||
| 2418 | { | ||
| 2419 | struct irq_cfg *cfg = irq_cfg + irq; | ||
| 2420 | unsigned int dest; | ||
| 2421 | cpumask_t tmp, cleanup_mask; | ||
| 2422 | struct irte irte; | ||
| 2423 | |||
| 2424 | cpus_and(tmp, mask, cpu_online_map); | ||
| 2425 | if (cpus_empty(tmp)) | ||
| 2426 | return; | ||
| 2427 | |||
| 2428 | if (get_irte(irq, &irte)) | ||
| 2429 | return; | ||
| 2430 | |||
| 2431 | if (assign_irq_vector(irq, mask)) | ||
| 2432 | return; | ||
| 2433 | |||
| 2434 | cpus_and(tmp, cfg->domain, mask); | ||
| 2435 | dest = cpu_mask_to_apicid(tmp); | ||
| 2436 | |||
| 2437 | irte.vector = cfg->vector; | ||
| 2438 | irte.dest_id = IRTE_DEST(dest); | ||
| 2439 | |||
| 2440 | /* | ||
| 2441 | * atomically update the IRTE with the new destination and vector. | ||
| 2442 | */ | ||
| 2443 | modify_irte(irq, &irte); | ||
| 2444 | |||
| 2445 | /* | ||
| 2446 | * After this point, all the interrupts will start arriving | ||
| 2447 | * at the new destination. So, time to cleanup the previous | ||
| 2448 | * vector allocation. | ||
| 2449 | */ | ||
| 2450 | if (cfg->move_in_progress) { | ||
| 2451 | cpus_and(cleanup_mask, cfg->old_domain, cpu_online_map); | ||
| 2452 | cfg->move_cleanup_count = cpus_weight(cleanup_mask); | ||
| 2453 | send_IPI_mask(cleanup_mask, IRQ_MOVE_CLEANUP_VECTOR); | ||
| 2454 | cfg->move_in_progress = 0; | ||
| 2455 | } | ||
| 2456 | |||
| 2457 | irq_desc[irq].affinity = mask; | ||
| 2458 | } | ||
| 2459 | #endif | ||
| 2052 | #endif /* CONFIG_SMP */ | 2460 | #endif /* CONFIG_SMP */ |
| 2053 | 2461 | ||
| 2054 | /* | 2462 | /* |
| @@ -2066,26 +2474,157 @@ static struct irq_chip msi_chip = { | |||
| 2066 | .retrigger = ioapic_retrigger_irq, | 2474 | .retrigger = ioapic_retrigger_irq, |
| 2067 | }; | 2475 | }; |
| 2068 | 2476 | ||
| 2069 | int arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc) | 2477 | #ifdef CONFIG_INTR_REMAP |
| 2478 | static struct irq_chip msi_ir_chip = { | ||
| 2479 | .name = "IR-PCI-MSI", | ||
| 2480 | .unmask = unmask_msi_irq, | ||
| 2481 | .mask = mask_msi_irq, | ||
| 2482 | .ack = ack_x2apic_edge, | ||
| 2483 | #ifdef CONFIG_SMP | ||
| 2484 | .set_affinity = ir_set_msi_irq_affinity, | ||
| 2485 | #endif | ||
| 2486 | .retrigger = ioapic_retrigger_irq, | ||
| 2487 | }; | ||
| 2488 | |||
| 2489 | /* | ||
| 2490 | * Map the PCI dev to the corresponding remapping hardware unit | ||
| 2491 | * and allocate 'nvec' consecutive interrupt-remapping table entries | ||
| 2492 | * in it. | ||
| 2493 | */ | ||
| 2494 | static int msi_alloc_irte(struct pci_dev *dev, int irq, int nvec) | ||
| 2070 | { | 2495 | { |
| 2496 | struct intel_iommu *iommu; | ||
| 2497 | int index; | ||
| 2498 | |||
| 2499 | iommu = map_dev_to_ir(dev); | ||
| 2500 | if (!iommu) { | ||
| 2501 | printk(KERN_ERR | ||
| 2502 | "Unable to map PCI %s to iommu\n", pci_name(dev)); | ||
| 2503 | return -ENOENT; | ||
| 2504 | } | ||
| 2505 | |||
| 2506 | index = alloc_irte(iommu, irq, nvec); | ||
| 2507 | if (index < 0) { | ||
| 2508 | printk(KERN_ERR | ||
| 2509 | "Unable to allocate %d IRTE for PCI %s\n", nvec, | ||
| 2510 | pci_name(dev)); | ||
| 2511 | return -ENOSPC; | ||
| 2512 | } | ||
| 2513 | return index; | ||
| 2514 | } | ||
| 2515 | #endif | ||
| 2516 | |||
| 2517 | static int setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc, int irq) | ||
| 2518 | { | ||
| 2519 | int ret; | ||
| 2071 | struct msi_msg msg; | 2520 | struct msi_msg msg; |
| 2521 | |||
| 2522 | ret = msi_compose_msg(dev, irq, &msg); | ||
| 2523 | if (ret < 0) | ||
| 2524 | return ret; | ||
| 2525 | |||
| 2526 | set_irq_msi(irq, desc); | ||
| 2527 | write_msi_msg(irq, &msg); | ||
| 2528 | |||
| 2529 | #ifdef CONFIG_INTR_REMAP | ||
| 2530 | if (irq_remapped(irq)) { | ||
| 2531 | struct irq_desc *desc = irq_desc + irq; | ||
| 2532 | /* | ||
| 2533 | * irq migration in process context | ||
| 2534 | */ | ||
| 2535 | desc->status |= IRQ_MOVE_PCNTXT; | ||
| 2536 | set_irq_chip_and_handler_name(irq, &msi_ir_chip, handle_edge_irq, "edge"); | ||
| 2537 | } else | ||
| 2538 | #endif | ||
| 2539 | set_irq_chip_and_handler_name(irq, &msi_chip, handle_edge_irq, "edge"); | ||
| 2540 | |||
| 2541 | return 0; | ||
| 2542 | } | ||
| 2543 | |||
| 2544 | int arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc) | ||
| 2545 | { | ||
| 2072 | int irq, ret; | 2546 | int irq, ret; |
| 2547 | |||
| 2073 | irq = create_irq(); | 2548 | irq = create_irq(); |
| 2074 | if (irq < 0) | 2549 | if (irq < 0) |
| 2075 | return irq; | 2550 | return irq; |
| 2076 | 2551 | ||
| 2077 | ret = msi_compose_msg(dev, irq, &msg); | 2552 | #ifdef CONFIG_INTR_REMAP |
| 2553 | if (!intr_remapping_enabled) | ||
| 2554 | goto no_ir; | ||
| 2555 | |||
| 2556 | ret = msi_alloc_irte(dev, irq, 1); | ||
| 2557 | if (ret < 0) | ||
| 2558 | goto error; | ||
| 2559 | no_ir: | ||
| 2560 | #endif | ||
| 2561 | ret = setup_msi_irq(dev, desc, irq); | ||
| 2078 | if (ret < 0) { | 2562 | if (ret < 0) { |
| 2079 | destroy_irq(irq); | 2563 | destroy_irq(irq); |
| 2080 | return ret; | 2564 | return ret; |
| 2081 | } | 2565 | } |
| 2566 | return 0; | ||
| 2082 | 2567 | ||
| 2083 | set_irq_msi(irq, desc); | 2568 | #ifdef CONFIG_INTR_REMAP |
| 2084 | write_msi_msg(irq, &msg); | 2569 | error: |
| 2570 | destroy_irq(irq); | ||
| 2571 | return ret; | ||
| 2572 | #endif | ||
| 2573 | } | ||
| 2085 | 2574 | ||
| 2086 | set_irq_chip_and_handler_name(irq, &msi_chip, handle_edge_irq, "edge"); | 2575 | int arch_setup_msi_irqs(struct pci_dev *dev, int nvec, int type) |
| 2576 | { | ||
| 2577 | int irq, ret, sub_handle; | ||
| 2578 | struct msi_desc *desc; | ||
| 2579 | #ifdef CONFIG_INTR_REMAP | ||
| 2580 | struct intel_iommu *iommu = 0; | ||
| 2581 | int index = 0; | ||
| 2582 | #endif | ||
| 2583 | |||
| 2584 | sub_handle = 0; | ||
| 2585 | list_for_each_entry(desc, &dev->msi_list, list) { | ||
| 2586 | irq = create_irq(); | ||
| 2587 | if (irq < 0) | ||
| 2588 | return irq; | ||
| 2589 | #ifdef CONFIG_INTR_REMAP | ||
| 2590 | if (!intr_remapping_enabled) | ||
| 2591 | goto no_ir; | ||
| 2087 | 2592 | ||
| 2593 | if (!sub_handle) { | ||
| 2594 | /* | ||
| 2595 | * allocate the consecutive block of IRTE's | ||
| 2596 | * for 'nvec' | ||
| 2597 | */ | ||
| 2598 | index = msi_alloc_irte(dev, irq, nvec); | ||
| 2599 | if (index < 0) { | ||
| 2600 | ret = index; | ||
| 2601 | goto error; | ||
| 2602 | } | ||
| 2603 | } else { | ||
| 2604 | iommu = map_dev_to_ir(dev); | ||
| 2605 | if (!iommu) { | ||
| 2606 | ret = -ENOENT; | ||
| 2607 | goto error; | ||
| 2608 | } | ||
| 2609 | /* | ||
| 2610 | * setup the mapping between the irq and the IRTE | ||
| 2611 | * base index, the sub_handle pointing to the | ||
| 2612 | * appropriate interrupt remap table entry. | ||
| 2613 | */ | ||
| 2614 | set_irte_irq(irq, iommu, index, sub_handle); | ||
| 2615 | } | ||
| 2616 | no_ir: | ||
| 2617 | #endif | ||
| 2618 | ret = setup_msi_irq(dev, desc, irq); | ||
| 2619 | if (ret < 0) | ||
| 2620 | goto error; | ||
| 2621 | sub_handle++; | ||
| 2622 | } | ||
| 2088 | return 0; | 2623 | return 0; |
| 2624 | |||
| 2625 | error: | ||
| 2626 | destroy_irq(irq); | ||
| 2627 | return ret; | ||
| 2089 | } | 2628 | } |
| 2090 | 2629 | ||
| 2091 | void arch_teardown_msi_irq(unsigned int irq) | 2630 | void arch_teardown_msi_irq(unsigned int irq) |
| @@ -2333,6 +2872,10 @@ void __init setup_ioapic_dest(void) | |||
| 2333 | setup_IO_APIC_irq(ioapic, pin, irq, | 2872 | setup_IO_APIC_irq(ioapic, pin, irq, |
| 2334 | irq_trigger(irq_entry), | 2873 | irq_trigger(irq_entry), |
| 2335 | irq_polarity(irq_entry)); | 2874 | irq_polarity(irq_entry)); |
| 2875 | #ifdef CONFIG_INTR_REMAP | ||
| 2876 | else if (intr_remapping_enabled) | ||
| 2877 | set_ir_ioapic_affinity_irq(irq, TARGET_CPUS); | ||
| 2878 | #endif | ||
| 2336 | else | 2879 | else |
| 2337 | set_ioapic_affinity_irq(irq, TARGET_CPUS); | 2880 | set_ioapic_affinity_irq(irq, TARGET_CPUS); |
| 2338 | } | 2881 | } |
diff --git a/arch/x86/kernel/irqinit_32.c b/arch/x86/kernel/irqinit_32.c index d66914287ee1..9200a1e2752d 100644 --- a/arch/x86/kernel/irqinit_32.c +++ b/arch/x86/kernel/irqinit_32.c | |||
| @@ -74,6 +74,15 @@ void __init init_ISA_irqs (void) | |||
| 74 | } | 74 | } |
| 75 | } | 75 | } |
| 76 | 76 | ||
| 77 | /* | ||
| 78 | * IRQ2 is cascade interrupt to second interrupt controller | ||
| 79 | */ | ||
| 80 | static struct irqaction irq2 = { | ||
| 81 | .handler = no_action, | ||
| 82 | .mask = CPU_MASK_NONE, | ||
| 83 | .name = "cascade", | ||
| 84 | }; | ||
| 85 | |||
| 77 | /* Overridden in paravirt.c */ | 86 | /* Overridden in paravirt.c */ |
| 78 | void init_IRQ(void) __attribute__((weak, alias("native_init_IRQ"))); | 87 | void init_IRQ(void) __attribute__((weak, alias("native_init_IRQ"))); |
| 79 | 88 | ||
| @@ -98,6 +107,46 @@ void __init native_init_IRQ(void) | |||
| 98 | set_intr_gate(vector, interrupt[i]); | 107 | set_intr_gate(vector, interrupt[i]); |
| 99 | } | 108 | } |
| 100 | 109 | ||
| 110 | #if defined(CONFIG_X86_LOCAL_APIC) && defined(CONFIG_SMP) | ||
| 111 | /* | ||
| 112 | * IRQ0 must be given a fixed assignment and initialized, | ||
| 113 | * because it's used before the IO-APIC is set up. | ||
| 114 | */ | ||
| 115 | set_intr_gate(FIRST_DEVICE_VECTOR, interrupt[0]); | ||
| 116 | |||
| 117 | /* | ||
| 118 | * The reschedule interrupt is a CPU-to-CPU reschedule-helper | ||
| 119 | * IPI, driven by wakeup. | ||
| 120 | */ | ||
| 121 | alloc_intr_gate(RESCHEDULE_VECTOR, reschedule_interrupt); | ||
| 122 | |||
| 123 | /* IPI for invalidation */ | ||
| 124 | alloc_intr_gate(INVALIDATE_TLB_VECTOR, invalidate_interrupt); | ||
| 125 | |||
| 126 | /* IPI for generic function call */ | ||
| 127 | alloc_intr_gate(CALL_FUNCTION_VECTOR, call_function_interrupt); | ||
| 128 | |||
| 129 | /* IPI for single call function */ | ||
| 130 | set_intr_gate(CALL_FUNCTION_SINGLE_VECTOR, call_function_single_interrupt); | ||
| 131 | #endif | ||
| 132 | |||
| 133 | #ifdef CONFIG_X86_LOCAL_APIC | ||
| 134 | /* self generated IPI for local APIC timer */ | ||
| 135 | alloc_intr_gate(LOCAL_TIMER_VECTOR, apic_timer_interrupt); | ||
| 136 | |||
| 137 | /* IPI vectors for APIC spurious and error interrupts */ | ||
| 138 | alloc_intr_gate(SPURIOUS_APIC_VECTOR, spurious_interrupt); | ||
| 139 | alloc_intr_gate(ERROR_APIC_VECTOR, error_interrupt); | ||
| 140 | #endif | ||
| 141 | |||
| 142 | #if defined(CONFIG_X86_LOCAL_APIC) && defined(CONFIG_X86_MCE_P4THERMAL) | ||
| 143 | /* thermal monitor LVT interrupt */ | ||
| 144 | alloc_intr_gate(THERMAL_APIC_VECTOR, thermal_interrupt); | ||
| 145 | #endif | ||
| 146 | |||
| 147 | if (!acpi_ioapic) | ||
| 148 | setup_irq(2, &irq2); | ||
| 149 | |||
| 101 | /* setup after call gates are initialised (usually add in | 150 | /* setup after call gates are initialised (usually add in |
| 102 | * the architecture specific gates) | 151 | * the architecture specific gates) |
| 103 | */ | 152 | */ |
diff --git a/arch/x86/kernel/mpparse.c b/arch/x86/kernel/mpparse.c index b3fb430725cb..f98f4e1dba09 100644 --- a/arch/x86/kernel/mpparse.c +++ b/arch/x86/kernel/mpparse.c | |||
| @@ -397,7 +397,9 @@ static int __init smp_read_mpc(struct mp_config_table *mpc, unsigned early) | |||
| 397 | generic_bigsmp_probe(); | 397 | generic_bigsmp_probe(); |
| 398 | #endif | 398 | #endif |
| 399 | 399 | ||
| 400 | #ifdef CONFIG_X86_32 | ||
| 400 | setup_apic_routing(); | 401 | setup_apic_routing(); |
| 402 | #endif | ||
| 401 | if (!num_processors) | 403 | if (!num_processors) |
| 402 | printk(KERN_ERR "MPTABLE: no processors registered!\n"); | 404 | printk(KERN_ERR "MPTABLE: no processors registered!\n"); |
| 403 | return num_processors; | 405 | return num_processors; |
diff --git a/arch/x86/kernel/numaq_32.c b/arch/x86/kernel/numaq_32.c index eecc8c18f010..4caff39078e0 100644 --- a/arch/x86/kernel/numaq_32.c +++ b/arch/x86/kernel/numaq_32.c | |||
| @@ -229,6 +229,12 @@ static void __init smp_read_mpc_oem(struct mp_config_oemtable *oemtable, | |||
| 229 | } | 229 | } |
| 230 | } | 230 | } |
| 231 | 231 | ||
| 232 | static int __init numaq_setup_ioapic_ids(void) | ||
| 233 | { | ||
| 234 | /* so can skip it */ | ||
| 235 | return 1; | ||
| 236 | } | ||
| 237 | |||
| 232 | static struct x86_quirks numaq_x86_quirks __initdata = { | 238 | static struct x86_quirks numaq_x86_quirks __initdata = { |
| 233 | .arch_pre_time_init = numaq_pre_time_init, | 239 | .arch_pre_time_init = numaq_pre_time_init, |
| 234 | .arch_time_init = NULL, | 240 | .arch_time_init = NULL, |
| @@ -243,6 +249,7 @@ static struct x86_quirks numaq_x86_quirks __initdata = { | |||
| 243 | .mpc_oem_bus_info = mpc_oem_bus_info, | 249 | .mpc_oem_bus_info = mpc_oem_bus_info, |
| 244 | .mpc_oem_pci_bus = mpc_oem_pci_bus, | 250 | .mpc_oem_pci_bus = mpc_oem_pci_bus, |
| 245 | .smp_read_mpc_oem = smp_read_mpc_oem, | 251 | .smp_read_mpc_oem = smp_read_mpc_oem, |
| 252 | .setup_ioapic_ids = numaq_setup_ioapic_ids, | ||
| 246 | }; | 253 | }; |
| 247 | 254 | ||
| 248 | void numaq_mps_oem_check(struct mp_config_table *mpc, char *oem, | 255 | void numaq_mps_oem_check(struct mp_config_table *mpc, char *oem, |
diff --git a/arch/x86/kernel/paravirt.c b/arch/x86/kernel/paravirt.c index e2f43768723a..6b0bb73998dd 100644 --- a/arch/x86/kernel/paravirt.c +++ b/arch/x86/kernel/paravirt.c | |||
| @@ -374,8 +374,6 @@ struct pv_cpu_ops pv_cpu_ops = { | |||
| 374 | 374 | ||
| 375 | struct pv_apic_ops pv_apic_ops = { | 375 | struct pv_apic_ops pv_apic_ops = { |
| 376 | #ifdef CONFIG_X86_LOCAL_APIC | 376 | #ifdef CONFIG_X86_LOCAL_APIC |
| 377 | .apic_write = native_apic_write, | ||
| 378 | .apic_read = native_apic_read, | ||
| 379 | .setup_boot_clock = setup_boot_APIC_clock, | 377 | .setup_boot_clock = setup_boot_APIC_clock, |
| 380 | .setup_secondary_clock = setup_secondary_APIC_clock, | 378 | .setup_secondary_clock = setup_secondary_APIC_clock, |
| 381 | .startup_ipi_hook = paravirt_nop, | 379 | .startup_ipi_hook = paravirt_nop, |
diff --git a/arch/x86/kernel/setup.c b/arch/x86/kernel/setup.c index 141efab52400..46c98efbbf8d 100644 --- a/arch/x86/kernel/setup.c +++ b/arch/x86/kernel/setup.c | |||
| @@ -758,6 +758,8 @@ void __init setup_arch(char **cmdline_p) | |||
| 758 | #else | 758 | #else |
| 759 | num_physpages = max_pfn; | 759 | num_physpages = max_pfn; |
| 760 | 760 | ||
| 761 | if (cpu_has_x2apic) | ||
| 762 | check_x2apic(); | ||
| 761 | 763 | ||
| 762 | /* How many end-of-memory variables you have, grandma! */ | 764 | /* How many end-of-memory variables you have, grandma! */ |
| 763 | /* need this before calling reserve_initrd */ | 765 | /* need this before calling reserve_initrd */ |
diff --git a/arch/x86/kernel/smpboot.c b/arch/x86/kernel/smpboot.c index 45531e3ba194..2ff0bbcd5bd1 100644 --- a/arch/x86/kernel/smpboot.c +++ b/arch/x86/kernel/smpboot.c | |||
| @@ -123,7 +123,6 @@ EXPORT_PER_CPU_SYMBOL(cpu_info); | |||
| 123 | 123 | ||
| 124 | static atomic_t init_deasserted; | 124 | static atomic_t init_deasserted; |
| 125 | 125 | ||
| 126 | static int boot_cpu_logical_apicid; | ||
| 127 | 126 | ||
| 128 | /* representing cpus for which sibling maps can be computed */ | 127 | /* representing cpus for which sibling maps can be computed */ |
| 129 | static cpumask_t cpu_sibling_setup_map; | 128 | static cpumask_t cpu_sibling_setup_map; |
| @@ -165,6 +164,8 @@ static void unmap_cpu_to_node(int cpu) | |||
| 165 | #endif | 164 | #endif |
| 166 | 165 | ||
| 167 | #ifdef CONFIG_X86_32 | 166 | #ifdef CONFIG_X86_32 |
| 167 | static int boot_cpu_logical_apicid; | ||
| 168 | |||
| 168 | u8 cpu_2_logical_apicid[NR_CPUS] __read_mostly = | 169 | u8 cpu_2_logical_apicid[NR_CPUS] __read_mostly = |
| 169 | { [0 ... NR_CPUS-1] = BAD_APICID }; | 170 | { [0 ... NR_CPUS-1] = BAD_APICID }; |
| 170 | 171 | ||
| @@ -210,7 +211,7 @@ static void __cpuinit smp_callin(void) | |||
| 210 | /* | 211 | /* |
| 211 | * (This works even if the APIC is not enabled.) | 212 | * (This works even if the APIC is not enabled.) |
| 212 | */ | 213 | */ |
| 213 | phys_id = GET_APIC_ID(read_apic_id()); | 214 | phys_id = read_apic_id(); |
| 214 | cpuid = smp_processor_id(); | 215 | cpuid = smp_processor_id(); |
| 215 | if (cpu_isset(cpuid, cpu_callin_map)) { | 216 | if (cpu_isset(cpuid, cpu_callin_map)) { |
| 216 | panic("%s: phys CPU#%d, CPU#%d already present??\n", __func__, | 217 | panic("%s: phys CPU#%d, CPU#%d already present??\n", __func__, |
| @@ -550,8 +551,7 @@ static inline void __inquire_remote_apic(int apicid) | |||
| 550 | printk(KERN_CONT | 551 | printk(KERN_CONT |
| 551 | "a previous APIC delivery may have failed\n"); | 552 | "a previous APIC delivery may have failed\n"); |
| 552 | 553 | ||
| 553 | apic_write(APIC_ICR2, SET_APIC_DEST_FIELD(apicid)); | 554 | apic_icr_write(APIC_DM_REMRD | regs[i], apicid); |
| 554 | apic_write(APIC_ICR, APIC_DM_REMRD | regs[i]); | ||
| 555 | 555 | ||
| 556 | timeout = 0; | 556 | timeout = 0; |
| 557 | do { | 557 | do { |
| @@ -583,11 +583,9 @@ wakeup_secondary_cpu(int logical_apicid, unsigned long start_eip) | |||
| 583 | int maxlvt; | 583 | int maxlvt; |
| 584 | 584 | ||
| 585 | /* Target chip */ | 585 | /* Target chip */ |
| 586 | apic_write(APIC_ICR2, SET_APIC_DEST_FIELD(logical_apicid)); | ||
| 587 | |||
| 588 | /* Boot on the stack */ | 586 | /* Boot on the stack */ |
| 589 | /* Kick the second */ | 587 | /* Kick the second */ |
| 590 | apic_write(APIC_ICR, APIC_DM_NMI | APIC_DEST_LOGICAL); | 588 | apic_icr_write(APIC_DM_NMI | APIC_DEST_LOGICAL, logical_apicid); |
| 591 | 589 | ||
| 592 | pr_debug("Waiting for send to finish...\n"); | 590 | pr_debug("Waiting for send to finish...\n"); |
| 593 | send_status = safe_apic_wait_icr_idle(); | 591 | send_status = safe_apic_wait_icr_idle(); |
| @@ -640,13 +638,11 @@ wakeup_secondary_cpu(int phys_apicid, unsigned long start_eip) | |||
| 640 | /* | 638 | /* |
| 641 | * Turn INIT on target chip | 639 | * Turn INIT on target chip |
| 642 | */ | 640 | */ |
| 643 | apic_write(APIC_ICR2, SET_APIC_DEST_FIELD(phys_apicid)); | ||
| 644 | |||
| 645 | /* | 641 | /* |
| 646 | * Send IPI | 642 | * Send IPI |
| 647 | */ | 643 | */ |
| 648 | apic_write(APIC_ICR, | 644 | apic_icr_write(APIC_INT_LEVELTRIG | APIC_INT_ASSERT | APIC_DM_INIT, |
| 649 | APIC_INT_LEVELTRIG | APIC_INT_ASSERT | APIC_DM_INIT); | 645 | phys_apicid); |
| 650 | 646 | ||
| 651 | pr_debug("Waiting for send to finish...\n"); | 647 | pr_debug("Waiting for send to finish...\n"); |
| 652 | send_status = safe_apic_wait_icr_idle(); | 648 | send_status = safe_apic_wait_icr_idle(); |
| @@ -656,10 +652,8 @@ wakeup_secondary_cpu(int phys_apicid, unsigned long start_eip) | |||
| 656 | pr_debug("Deasserting INIT.\n"); | 652 | pr_debug("Deasserting INIT.\n"); |
| 657 | 653 | ||
| 658 | /* Target chip */ | 654 | /* Target chip */ |
| 659 | apic_write(APIC_ICR2, SET_APIC_DEST_FIELD(phys_apicid)); | ||
| 660 | |||
| 661 | /* Send IPI */ | 655 | /* Send IPI */ |
| 662 | apic_write(APIC_ICR, APIC_INT_LEVELTRIG | APIC_DM_INIT); | 656 | apic_icr_write(APIC_INT_LEVELTRIG | APIC_DM_INIT, phys_apicid); |
| 663 | 657 | ||
| 664 | pr_debug("Waiting for send to finish...\n"); | 658 | pr_debug("Waiting for send to finish...\n"); |
| 665 | send_status = safe_apic_wait_icr_idle(); | 659 | send_status = safe_apic_wait_icr_idle(); |
| @@ -702,11 +696,10 @@ wakeup_secondary_cpu(int phys_apicid, unsigned long start_eip) | |||
| 702 | */ | 696 | */ |
| 703 | 697 | ||
| 704 | /* Target chip */ | 698 | /* Target chip */ |
| 705 | apic_write(APIC_ICR2, SET_APIC_DEST_FIELD(phys_apicid)); | ||
| 706 | |||
| 707 | /* Boot on the stack */ | 699 | /* Boot on the stack */ |
| 708 | /* Kick the second */ | 700 | /* Kick the second */ |
| 709 | apic_write(APIC_ICR, APIC_DM_STARTUP | (start_eip >> 12)); | 701 | apic_icr_write(APIC_DM_STARTUP | (start_eip >> 12), |
| 702 | phys_apicid); | ||
| 710 | 703 | ||
| 711 | /* | 704 | /* |
| 712 | * Give the other CPU some time to accept the IPI. | 705 | * Give the other CPU some time to accept the IPI. |
| @@ -1175,10 +1168,17 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus) | |||
| 1175 | * Setup boot CPU information | 1168 | * Setup boot CPU information |
| 1176 | */ | 1169 | */ |
| 1177 | smp_store_cpu_info(0); /* Final full version of the data */ | 1170 | smp_store_cpu_info(0); /* Final full version of the data */ |
| 1171 | #ifdef CONFIG_X86_32 | ||
| 1178 | boot_cpu_logical_apicid = logical_smp_processor_id(); | 1172 | boot_cpu_logical_apicid = logical_smp_processor_id(); |
| 1173 | #endif | ||
| 1179 | current_thread_info()->cpu = 0; /* needed? */ | 1174 | current_thread_info()->cpu = 0; /* needed? */ |
| 1180 | set_cpu_sibling_map(0); | 1175 | set_cpu_sibling_map(0); |
| 1181 | 1176 | ||
| 1177 | #ifdef CONFIG_X86_64 | ||
| 1178 | enable_IR_x2apic(); | ||
| 1179 | setup_apic_routing(); | ||
| 1180 | #endif | ||
| 1181 | |||
| 1182 | if (smp_sanity_check(max_cpus) < 0) { | 1182 | if (smp_sanity_check(max_cpus) < 0) { |
| 1183 | printk(KERN_INFO "SMP disabled\n"); | 1183 | printk(KERN_INFO "SMP disabled\n"); |
| 1184 | disable_smp(); | 1184 | disable_smp(); |
| @@ -1186,9 +1186,9 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus) | |||
| 1186 | } | 1186 | } |
| 1187 | 1187 | ||
| 1188 | preempt_disable(); | 1188 | preempt_disable(); |
| 1189 | if (GET_APIC_ID(read_apic_id()) != boot_cpu_physical_apicid) { | 1189 | if (read_apic_id() != boot_cpu_physical_apicid) { |
| 1190 | panic("Boot APIC ID in local APIC unexpected (%d vs %d)", | 1190 | panic("Boot APIC ID in local APIC unexpected (%d vs %d)", |
| 1191 | GET_APIC_ID(read_apic_id()), boot_cpu_physical_apicid); | 1191 | read_apic_id(), boot_cpu_physical_apicid); |
| 1192 | /* Or can we switch back to PIC here? */ | 1192 | /* Or can we switch back to PIC here? */ |
| 1193 | } | 1193 | } |
| 1194 | preempt_enable(); | 1194 | preempt_enable(); |
diff --git a/arch/x86/kernel/summit_32.c b/arch/x86/kernel/summit_32.c index d67ce5f044ba..7b987852e876 100644 --- a/arch/x86/kernel/summit_32.c +++ b/arch/x86/kernel/summit_32.c | |||
| @@ -30,7 +30,7 @@ | |||
| 30 | #include <linux/init.h> | 30 | #include <linux/init.h> |
| 31 | #include <asm/io.h> | 31 | #include <asm/io.h> |
| 32 | #include <asm/bios_ebda.h> | 32 | #include <asm/bios_ebda.h> |
| 33 | #include <asm/mach-summit/mach_mpparse.h> | 33 | #include <asm/summit/mpparse.h> |
| 34 | 34 | ||
| 35 | static struct rio_table_hdr *rio_table_hdr __initdata; | 35 | static struct rio_table_hdr *rio_table_hdr __initdata; |
| 36 | static struct scal_detail *scal_devs[MAX_NUMNODES] __initdata; | 36 | static struct scal_detail *scal_devs[MAX_NUMNODES] __initdata; |
diff --git a/arch/x86/kernel/vmi_32.c b/arch/x86/kernel/vmi_32.c index 8c9ad02af5a2..8b6c393ab9fd 100644 --- a/arch/x86/kernel/vmi_32.c +++ b/arch/x86/kernel/vmi_32.c | |||
| @@ -905,8 +905,8 @@ static inline int __init activate_vmi(void) | |||
| 905 | #endif | 905 | #endif |
| 906 | 906 | ||
| 907 | #ifdef CONFIG_X86_LOCAL_APIC | 907 | #ifdef CONFIG_X86_LOCAL_APIC |
| 908 | para_fill(pv_apic_ops.apic_read, APICRead); | 908 | para_fill(apic_ops->read, APICRead); |
| 909 | para_fill(pv_apic_ops.apic_write, APICWrite); | 909 | para_fill(apic_ops->write, APICWrite); |
| 910 | #endif | 910 | #endif |
| 911 | 911 | ||
| 912 | /* | 912 | /* |
diff --git a/arch/x86/lguest/boot.c b/arch/x86/lguest/boot.c index d9249a882aa5..65f0b8a47bed 100644 --- a/arch/x86/lguest/boot.c +++ b/arch/x86/lguest/boot.c | |||
| @@ -55,6 +55,7 @@ | |||
| 55 | #include <linux/lguest_launcher.h> | 55 | #include <linux/lguest_launcher.h> |
| 56 | #include <linux/virtio_console.h> | 56 | #include <linux/virtio_console.h> |
| 57 | #include <linux/pm.h> | 57 | #include <linux/pm.h> |
| 58 | #include <asm/apic.h> | ||
| 58 | #include <asm/lguest.h> | 59 | #include <asm/lguest.h> |
| 59 | #include <asm/paravirt.h> | 60 | #include <asm/paravirt.h> |
| 60 | #include <asm/param.h> | 61 | #include <asm/param.h> |
| @@ -783,14 +784,44 @@ static void lguest_wbinvd(void) | |||
| 783 | * code qualifies for Advanced. It will also never interrupt anything. It | 784 | * code qualifies for Advanced. It will also never interrupt anything. It |
| 784 | * does, however, allow us to get through the Linux boot code. */ | 785 | * does, however, allow us to get through the Linux boot code. */ |
| 785 | #ifdef CONFIG_X86_LOCAL_APIC | 786 | #ifdef CONFIG_X86_LOCAL_APIC |
| 786 | static void lguest_apic_write(unsigned long reg, u32 v) | 787 | static void lguest_apic_write(u32 reg, u32 v) |
| 787 | { | 788 | { |
| 788 | } | 789 | } |
| 789 | 790 | ||
| 790 | static u32 lguest_apic_read(unsigned long reg) | 791 | static u32 lguest_apic_read(u32 reg) |
| 791 | { | 792 | { |
| 792 | return 0; | 793 | return 0; |
| 793 | } | 794 | } |
| 795 | |||
| 796 | static u64 lguest_apic_icr_read(void) | ||
| 797 | { | ||
| 798 | return 0; | ||
| 799 | } | ||
| 800 | |||
| 801 | static void lguest_apic_icr_write(u32 low, u32 id) | ||
| 802 | { | ||
| 803 | /* Warn to see if there's any stray references */ | ||
| 804 | WARN_ON(1); | ||
| 805 | } | ||
| 806 | |||
| 807 | static void lguest_apic_wait_icr_idle(void) | ||
| 808 | { | ||
| 809 | return; | ||
| 810 | } | ||
| 811 | |||
| 812 | static u32 lguest_apic_safe_wait_icr_idle(void) | ||
| 813 | { | ||
| 814 | return 0; | ||
| 815 | } | ||
| 816 | |||
| 817 | static struct apic_ops lguest_basic_apic_ops = { | ||
| 818 | .read = lguest_apic_read, | ||
| 819 | .write = lguest_apic_write, | ||
| 820 | .icr_read = lguest_apic_icr_read, | ||
| 821 | .icr_write = lguest_apic_icr_write, | ||
| 822 | .wait_icr_idle = lguest_apic_wait_icr_idle, | ||
| 823 | .safe_wait_icr_idle = lguest_apic_safe_wait_icr_idle, | ||
| 824 | }; | ||
| 794 | #endif | 825 | #endif |
| 795 | 826 | ||
| 796 | /* STOP! Until an interrupt comes in. */ | 827 | /* STOP! Until an interrupt comes in. */ |
| @@ -990,8 +1021,7 @@ __init void lguest_init(void) | |||
| 990 | 1021 | ||
| 991 | #ifdef CONFIG_X86_LOCAL_APIC | 1022 | #ifdef CONFIG_X86_LOCAL_APIC |
| 992 | /* apic read/write intercepts */ | 1023 | /* apic read/write intercepts */ |
| 993 | pv_apic_ops.apic_write = lguest_apic_write; | 1024 | apic_ops = &lguest_basic_apic_ops; |
| 994 | pv_apic_ops.apic_read = lguest_apic_read; | ||
| 995 | #endif | 1025 | #endif |
| 996 | 1026 | ||
| 997 | /* time operations */ | 1027 | /* time operations */ |
diff --git a/arch/x86/mach-default/setup.c b/arch/x86/mach-default/setup.c index 3f2cf11f201a..37b9ae4d44c5 100644 --- a/arch/x86/mach-default/setup.c +++ b/arch/x86/mach-default/setup.c | |||
| @@ -38,15 +38,6 @@ void __init pre_intr_init_hook(void) | |||
| 38 | init_ISA_irqs(); | 38 | init_ISA_irqs(); |
| 39 | } | 39 | } |
| 40 | 40 | ||
| 41 | /* | ||
| 42 | * IRQ2 is cascade interrupt to second interrupt controller | ||
| 43 | */ | ||
| 44 | static struct irqaction irq2 = { | ||
| 45 | .handler = no_action, | ||
| 46 | .mask = CPU_MASK_NONE, | ||
| 47 | .name = "cascade", | ||
| 48 | }; | ||
| 49 | |||
| 50 | /** | 41 | /** |
| 51 | * intr_init_hook - post gate setup interrupt initialisation | 42 | * intr_init_hook - post gate setup interrupt initialisation |
| 52 | * | 43 | * |
| @@ -62,12 +53,6 @@ void __init intr_init_hook(void) | |||
| 62 | if (x86_quirks->arch_intr_init()) | 53 | if (x86_quirks->arch_intr_init()) |
| 63 | return; | 54 | return; |
| 64 | } | 55 | } |
| 65 | #ifdef CONFIG_X86_LOCAL_APIC | ||
| 66 | apic_intr_init(); | ||
| 67 | #endif | ||
| 68 | |||
| 69 | if (!acpi_ioapic) | ||
| 70 | setup_irq(2, &irq2); | ||
| 71 | } | 56 | } |
| 72 | 57 | ||
| 73 | /** | 58 | /** |
diff --git a/arch/x86/mach-generic/Makefile b/arch/x86/mach-generic/Makefile index 0dbd7803a1d5..4706de7676b1 100644 --- a/arch/x86/mach-generic/Makefile +++ b/arch/x86/mach-generic/Makefile | |||
| @@ -9,4 +9,4 @@ obj-$(CONFIG_X86_NUMAQ) += numaq.o | |||
| 9 | obj-$(CONFIG_X86_SUMMIT) += summit.o | 9 | obj-$(CONFIG_X86_SUMMIT) += summit.o |
| 10 | obj-$(CONFIG_X86_BIGSMP) += bigsmp.o | 10 | obj-$(CONFIG_X86_BIGSMP) += bigsmp.o |
| 11 | obj-$(CONFIG_X86_ES7000) += es7000.o | 11 | obj-$(CONFIG_X86_ES7000) += es7000.o |
| 12 | obj-$(CONFIG_X86_ES7000) += ../../x86/mach-es7000/ | 12 | obj-$(CONFIG_X86_ES7000) += ../../x86/es7000/ |
diff --git a/arch/x86/mach-generic/bigsmp.c b/arch/x86/mach-generic/bigsmp.c index 59d771714559..df37fc9d6a26 100644 --- a/arch/x86/mach-generic/bigsmp.c +++ b/arch/x86/mach-generic/bigsmp.c | |||
| @@ -5,18 +5,17 @@ | |||
| 5 | #define APIC_DEFINITION 1 | 5 | #define APIC_DEFINITION 1 |
| 6 | #include <linux/threads.h> | 6 | #include <linux/threads.h> |
| 7 | #include <linux/cpumask.h> | 7 | #include <linux/cpumask.h> |
| 8 | #include <asm/smp.h> | ||
| 9 | #include <asm/mpspec.h> | 8 | #include <asm/mpspec.h> |
| 10 | #include <asm/genapic.h> | 9 | #include <asm/genapic.h> |
| 11 | #include <asm/fixmap.h> | 10 | #include <asm/fixmap.h> |
| 12 | #include <asm/apicdef.h> | 11 | #include <asm/apicdef.h> |
| 13 | #include <linux/kernel.h> | 12 | #include <linux/kernel.h> |
| 14 | #include <linux/smp.h> | ||
| 15 | #include <linux/init.h> | 13 | #include <linux/init.h> |
| 16 | #include <linux/dmi.h> | 14 | #include <linux/dmi.h> |
| 17 | #include <asm/mach-bigsmp/mach_apic.h> | 15 | #include <asm/bigsmp/apicdef.h> |
| 18 | #include <asm/mach-bigsmp/mach_apicdef.h> | 16 | #include <linux/smp.h> |
| 19 | #include <asm/mach-bigsmp/mach_ipi.h> | 17 | #include <asm/bigsmp/apic.h> |
| 18 | #include <asm/bigsmp/ipi.h> | ||
| 20 | #include <asm/mach-default/mach_mpparse.h> | 19 | #include <asm/mach-default/mach_mpparse.h> |
| 21 | 20 | ||
| 22 | static int dmi_bigsmp; /* can be set by dmi scanners */ | 21 | static int dmi_bigsmp; /* can be set by dmi scanners */ |
diff --git a/arch/x86/mach-generic/es7000.c b/arch/x86/mach-generic/es7000.c index 4742626f08c4..520cca0ee04e 100644 --- a/arch/x86/mach-generic/es7000.c +++ b/arch/x86/mach-generic/es7000.c | |||
| @@ -4,20 +4,19 @@ | |||
| 4 | #define APIC_DEFINITION 1 | 4 | #define APIC_DEFINITION 1 |
| 5 | #include <linux/threads.h> | 5 | #include <linux/threads.h> |
| 6 | #include <linux/cpumask.h> | 6 | #include <linux/cpumask.h> |
| 7 | #include <asm/smp.h> | ||
| 8 | #include <asm/mpspec.h> | 7 | #include <asm/mpspec.h> |
| 9 | #include <asm/genapic.h> | 8 | #include <asm/genapic.h> |
| 10 | #include <asm/fixmap.h> | 9 | #include <asm/fixmap.h> |
| 11 | #include <asm/apicdef.h> | 10 | #include <asm/apicdef.h> |
| 12 | #include <linux/kernel.h> | 11 | #include <linux/kernel.h> |
| 13 | #include <linux/string.h> | 12 | #include <linux/string.h> |
| 14 | #include <linux/smp.h> | ||
| 15 | #include <linux/init.h> | 13 | #include <linux/init.h> |
| 16 | #include <asm/mach-es7000/mach_apicdef.h> | 14 | #include <asm/es7000/apicdef.h> |
| 17 | #include <asm/mach-es7000/mach_apic.h> | 15 | #include <linux/smp.h> |
| 18 | #include <asm/mach-es7000/mach_ipi.h> | 16 | #include <asm/es7000/apic.h> |
| 19 | #include <asm/mach-es7000/mach_mpparse.h> | 17 | #include <asm/es7000/ipi.h> |
| 20 | #include <asm/mach-es7000/mach_wakecpu.h> | 18 | #include <asm/es7000/mpparse.h> |
| 19 | #include <asm/es7000/wakecpu.h> | ||
| 21 | 20 | ||
| 22 | static int probe_es7000(void) | 21 | static int probe_es7000(void) |
| 23 | { | 22 | { |
diff --git a/arch/x86/mach-generic/numaq.c b/arch/x86/mach-generic/numaq.c index 8091e68764c4..8cf58394975e 100644 --- a/arch/x86/mach-generic/numaq.c +++ b/arch/x86/mach-generic/numaq.c | |||
| @@ -4,7 +4,6 @@ | |||
| 4 | #define APIC_DEFINITION 1 | 4 | #define APIC_DEFINITION 1 |
| 5 | #include <linux/threads.h> | 5 | #include <linux/threads.h> |
| 6 | #include <linux/cpumask.h> | 6 | #include <linux/cpumask.h> |
| 7 | #include <linux/smp.h> | ||
| 8 | #include <asm/mpspec.h> | 7 | #include <asm/mpspec.h> |
| 9 | #include <asm/genapic.h> | 8 | #include <asm/genapic.h> |
| 10 | #include <asm/fixmap.h> | 9 | #include <asm/fixmap.h> |
| @@ -12,11 +11,12 @@ | |||
| 12 | #include <linux/kernel.h> | 11 | #include <linux/kernel.h> |
| 13 | #include <linux/string.h> | 12 | #include <linux/string.h> |
| 14 | #include <linux/init.h> | 13 | #include <linux/init.h> |
| 15 | #include <asm/mach-numaq/mach_apic.h> | 14 | #include <asm/numaq/apicdef.h> |
| 16 | #include <asm/mach-numaq/mach_apicdef.h> | 15 | #include <linux/smp.h> |
| 17 | #include <asm/mach-numaq/mach_ipi.h> | 16 | #include <asm/numaq/apic.h> |
| 18 | #include <asm/mach-numaq/mach_mpparse.h> | 17 | #include <asm/numaq/ipi.h> |
| 19 | #include <asm/mach-numaq/mach_wakecpu.h> | 18 | #include <asm/numaq/mpparse.h> |
| 19 | #include <asm/numaq/wakecpu.h> | ||
| 20 | #include <asm/numaq.h> | 20 | #include <asm/numaq.h> |
| 21 | 21 | ||
| 22 | static int mps_oem_check(struct mp_config_table *mpc, char *oem, | 22 | static int mps_oem_check(struct mp_config_table *mpc, char *oem, |
diff --git a/arch/x86/mach-generic/summit.c b/arch/x86/mach-generic/summit.c index a97ea0f35b1e..6ad6b67a723d 100644 --- a/arch/x86/mach-generic/summit.c +++ b/arch/x86/mach-generic/summit.c | |||
| @@ -4,19 +4,18 @@ | |||
| 4 | #define APIC_DEFINITION 1 | 4 | #define APIC_DEFINITION 1 |
| 5 | #include <linux/threads.h> | 5 | #include <linux/threads.h> |
| 6 | #include <linux/cpumask.h> | 6 | #include <linux/cpumask.h> |
| 7 | #include <asm/smp.h> | ||
| 8 | #include <asm/mpspec.h> | 7 | #include <asm/mpspec.h> |
| 9 | #include <asm/genapic.h> | 8 | #include <asm/genapic.h> |
| 10 | #include <asm/fixmap.h> | 9 | #include <asm/fixmap.h> |
| 11 | #include <asm/apicdef.h> | 10 | #include <asm/apicdef.h> |
| 12 | #include <linux/kernel.h> | 11 | #include <linux/kernel.h> |
| 13 | #include <linux/string.h> | 12 | #include <linux/string.h> |
| 14 | #include <linux/smp.h> | ||
| 15 | #include <linux/init.h> | 13 | #include <linux/init.h> |
| 16 | #include <asm/mach-summit/mach_apic.h> | 14 | #include <asm/summit/apicdef.h> |
| 17 | #include <asm/mach-summit/mach_apicdef.h> | 15 | #include <linux/smp.h> |
| 18 | #include <asm/mach-summit/mach_ipi.h> | 16 | #include <asm/summit/apic.h> |
| 19 | #include <asm/mach-summit/mach_mpparse.h> | 17 | #include <asm/summit/ipi.h> |
| 18 | #include <asm/summit/mpparse.h> | ||
| 20 | 19 | ||
| 21 | static int probe_summit(void) | 20 | static int probe_summit(void) |
| 22 | { | 21 | { |
diff --git a/arch/x86/pci/acpi.c b/arch/x86/pci/acpi.c index 19af06927fbc..1d88d2b39771 100644 --- a/arch/x86/pci/acpi.c +++ b/arch/x86/pci/acpi.c | |||
| @@ -250,10 +250,5 @@ int __init pci_acpi_init(void) | |||
| 250 | acpi_pci_irq_enable(dev); | 250 | acpi_pci_irq_enable(dev); |
| 251 | } | 251 | } |
| 252 | 252 | ||
| 253 | #ifdef CONFIG_X86_IO_APIC | ||
| 254 | if (acpi_ioapic) | ||
| 255 | print_IO_APIC(); | ||
| 256 | #endif | ||
| 257 | |||
| 258 | return 0; | 253 | return 0; |
| 259 | } | 254 | } |
diff --git a/arch/x86/xen/enlighten.c b/arch/x86/xen/enlighten.c index 7dcd321a0508..a27d562a9744 100644 --- a/arch/x86/xen/enlighten.c +++ b/arch/x86/xen/enlighten.c | |||
| @@ -36,6 +36,7 @@ | |||
| 36 | #include <xen/hvc-console.h> | 36 | #include <xen/hvc-console.h> |
| 37 | 37 | ||
| 38 | #include <asm/paravirt.h> | 38 | #include <asm/paravirt.h> |
| 39 | #include <asm/apic.h> | ||
| 39 | #include <asm/page.h> | 40 | #include <asm/page.h> |
| 40 | #include <asm/xen/hypercall.h> | 41 | #include <asm/xen/hypercall.h> |
| 41 | #include <asm/xen/hypervisor.h> | 42 | #include <asm/xen/hypervisor.h> |
| @@ -580,16 +581,47 @@ static void xen_io_delay(void) | |||
| 580 | } | 581 | } |
| 581 | 582 | ||
| 582 | #ifdef CONFIG_X86_LOCAL_APIC | 583 | #ifdef CONFIG_X86_LOCAL_APIC |
| 583 | static u32 xen_apic_read(unsigned long reg) | 584 | static u32 xen_apic_read(u32 reg) |
| 584 | { | 585 | { |
| 585 | return 0; | 586 | return 0; |
| 586 | } | 587 | } |
| 587 | 588 | ||
| 588 | static void xen_apic_write(unsigned long reg, u32 val) | 589 | static void xen_apic_write(u32 reg, u32 val) |
| 589 | { | 590 | { |
| 590 | /* Warn to see if there's any stray references */ | 591 | /* Warn to see if there's any stray references */ |
| 591 | WARN_ON(1); | 592 | WARN_ON(1); |
| 592 | } | 593 | } |
| 594 | |||
| 595 | static u64 xen_apic_icr_read(void) | ||
| 596 | { | ||
| 597 | return 0; | ||
| 598 | } | ||
| 599 | |||
| 600 | static void xen_apic_icr_write(u32 low, u32 id) | ||
| 601 | { | ||
| 602 | /* Warn to see if there's any stray references */ | ||
| 603 | WARN_ON(1); | ||
| 604 | } | ||
| 605 | |||
| 606 | static void xen_apic_wait_icr_idle(void) | ||
| 607 | { | ||
| 608 | return; | ||
| 609 | } | ||
| 610 | |||
| 611 | static u32 xen_safe_apic_wait_icr_idle(void) | ||
| 612 | { | ||
| 613 | return 0; | ||
| 614 | } | ||
| 615 | |||
| 616 | static struct apic_ops xen_basic_apic_ops = { | ||
| 617 | .read = xen_apic_read, | ||
| 618 | .write = xen_apic_write, | ||
| 619 | .icr_read = xen_apic_icr_read, | ||
| 620 | .icr_write = xen_apic_icr_write, | ||
| 621 | .wait_icr_idle = xen_apic_wait_icr_idle, | ||
| 622 | .safe_wait_icr_idle = xen_safe_apic_wait_icr_idle, | ||
| 623 | }; | ||
| 624 | |||
| 593 | #endif | 625 | #endif |
| 594 | 626 | ||
| 595 | static void xen_flush_tlb(void) | 627 | static void xen_flush_tlb(void) |
| @@ -1273,8 +1305,6 @@ static const struct pv_irq_ops xen_irq_ops __initdata = { | |||
| 1273 | 1305 | ||
| 1274 | static const struct pv_apic_ops xen_apic_ops __initdata = { | 1306 | static const struct pv_apic_ops xen_apic_ops __initdata = { |
| 1275 | #ifdef CONFIG_X86_LOCAL_APIC | 1307 | #ifdef CONFIG_X86_LOCAL_APIC |
| 1276 | .apic_write = xen_apic_write, | ||
| 1277 | .apic_read = xen_apic_read, | ||
| 1278 | .setup_boot_clock = paravirt_nop, | 1308 | .setup_boot_clock = paravirt_nop, |
| 1279 | .setup_secondary_clock = paravirt_nop, | 1309 | .setup_secondary_clock = paravirt_nop, |
| 1280 | .startup_ipi_hook = paravirt_nop, | 1310 | .startup_ipi_hook = paravirt_nop, |
| @@ -1677,6 +1707,13 @@ asmlinkage void __init xen_start_kernel(void) | |||
| 1677 | pv_apic_ops = xen_apic_ops; | 1707 | pv_apic_ops = xen_apic_ops; |
| 1678 | pv_mmu_ops = xen_mmu_ops; | 1708 | pv_mmu_ops = xen_mmu_ops; |
| 1679 | 1709 | ||
| 1710 | #ifdef CONFIG_X86_LOCAL_APIC | ||
| 1711 | /* | ||
| 1712 | * set up the basic apic ops. | ||
| 1713 | */ | ||
| 1714 | apic_ops = &xen_basic_apic_ops; | ||
| 1715 | #endif | ||
| 1716 | |||
| 1680 | if (xen_feature(XENFEAT_mmu_pt_update_preserve_ad)) { | 1717 | if (xen_feature(XENFEAT_mmu_pt_update_preserve_ad)) { |
| 1681 | pv_mmu_ops.ptep_modify_prot_start = xen_ptep_modify_prot_start; | 1718 | pv_mmu_ops.ptep_modify_prot_start = xen_ptep_modify_prot_start; |
| 1682 | pv_mmu_ops.ptep_modify_prot_commit = xen_ptep_modify_prot_commit; | 1719 | pv_mmu_ops.ptep_modify_prot_commit = xen_ptep_modify_prot_commit; |
