diff options
Diffstat (limited to 'arch/x86/kernel/apic/apic.c')
| -rw-r--r-- | arch/x86/kernel/apic/apic.c | 91 |
1 files changed, 69 insertions, 22 deletions
diff --git a/arch/x86/kernel/apic/apic.c b/arch/x86/kernel/apic/apic.c index e3b534cda49a..850657d1b0ed 100644 --- a/arch/x86/kernel/apic/apic.c +++ b/arch/x86/kernel/apic/apic.c | |||
| @@ -52,6 +52,7 @@ | |||
| 52 | #include <asm/mce.h> | 52 | #include <asm/mce.h> |
| 53 | #include <asm/kvm_para.h> | 53 | #include <asm/kvm_para.h> |
| 54 | #include <asm/tsc.h> | 54 | #include <asm/tsc.h> |
| 55 | #include <asm/atomic.h> | ||
| 55 | 56 | ||
| 56 | unsigned int num_processors; | 57 | unsigned int num_processors; |
| 57 | 58 | ||
| @@ -370,38 +371,87 @@ static void __setup_APIC_LVTT(unsigned int clocks, int oneshot, int irqen) | |||
| 370 | } | 371 | } |
| 371 | 372 | ||
| 372 | /* | 373 | /* |
| 373 | * Setup extended LVT, AMD specific (K8, family 10h) | 374 | * Setup extended LVT, AMD specific |
| 374 | * | 375 | * |
| 375 | * Vector mappings are hard coded. On K8 only offset 0 (APIC500) and | 376 | * Software should use the LVT offsets the BIOS provides. The offsets |
| 376 | * MCE interrupts are supported. Thus MCE offset must be set to 0. | 377 | * are determined by the subsystems using it like those for MCE |
| 378 | * threshold or IBS. On K8 only offset 0 (APIC500) and MCE interrupts | ||
| 379 | * are supported. Beginning with family 10h at least 4 offsets are | ||
| 380 | * available. | ||
| 377 | * | 381 | * |
| 378 | * If mask=1, the LVT entry does not generate interrupts while mask=0 | 382 | * Since the offsets must be consistent for all cores, we keep track |
| 379 | * enables the vector. See also the BKDGs. | 383 | * of the LVT offsets in software and reserve the offset for the same |
| 384 | * vector also to be used on other cores. An offset is freed by | ||
| 385 | * setting the entry to APIC_EILVT_MASKED. | ||
| 386 | * | ||
| 387 | * If the BIOS is right, there should be no conflicts. Otherwise a | ||
| 388 | * "[Firmware Bug]: ..." error message is generated. However, if | ||
| 389 | * software does not properly determines the offsets, it is not | ||
| 390 | * necessarily a BIOS bug. | ||
| 380 | */ | 391 | */ |
| 381 | 392 | ||
| 382 | #define APIC_EILVT_LVTOFF_MCE 0 | 393 | static atomic_t eilvt_offsets[APIC_EILVT_NR_MAX]; |
| 383 | #define APIC_EILVT_LVTOFF_IBS 1 | ||
| 384 | 394 | ||
| 385 | static void setup_APIC_eilvt(u8 lvt_off, u8 vector, u8 msg_type, u8 mask) | 395 | static inline int eilvt_entry_is_changeable(unsigned int old, unsigned int new) |
| 386 | { | 396 | { |
| 387 | unsigned long reg = (lvt_off << 4) + APIC_EILVTn(0); | 397 | return (old & APIC_EILVT_MASKED) |
| 388 | unsigned int v = (mask << 16) | (msg_type << 8) | vector; | 398 | || (new == APIC_EILVT_MASKED) |
| 389 | 399 | || ((new & ~APIC_EILVT_MASKED) == old); | |
| 390 | apic_write(reg, v); | ||
| 391 | } | 400 | } |
| 392 | 401 | ||
| 393 | u8 setup_APIC_eilvt_mce(u8 vector, u8 msg_type, u8 mask) | 402 | static unsigned int reserve_eilvt_offset(int offset, unsigned int new) |
| 394 | { | 403 | { |
| 395 | setup_APIC_eilvt(APIC_EILVT_LVTOFF_MCE, vector, msg_type, mask); | 404 | unsigned int rsvd; /* 0: uninitialized */ |
| 396 | return APIC_EILVT_LVTOFF_MCE; | 405 | |
| 406 | if (offset >= APIC_EILVT_NR_MAX) | ||
| 407 | return ~0; | ||
| 408 | |||
| 409 | rsvd = atomic_read(&eilvt_offsets[offset]) & ~APIC_EILVT_MASKED; | ||
| 410 | do { | ||
| 411 | if (rsvd && | ||
| 412 | !eilvt_entry_is_changeable(rsvd, new)) | ||
| 413 | /* may not change if vectors are different */ | ||
| 414 | return rsvd; | ||
| 415 | rsvd = atomic_cmpxchg(&eilvt_offsets[offset], rsvd, new); | ||
| 416 | } while (rsvd != new); | ||
| 417 | |||
| 418 | return new; | ||
| 397 | } | 419 | } |
| 398 | 420 | ||
| 399 | u8 setup_APIC_eilvt_ibs(u8 vector, u8 msg_type, u8 mask) | 421 | /* |
| 422 | * If mask=1, the LVT entry does not generate interrupts while mask=0 | ||
| 423 | * enables the vector. See also the BKDGs. | ||
| 424 | */ | ||
| 425 | |||
| 426 | int setup_APIC_eilvt(u8 offset, u8 vector, u8 msg_type, u8 mask) | ||
| 400 | { | 427 | { |
| 401 | setup_APIC_eilvt(APIC_EILVT_LVTOFF_IBS, vector, msg_type, mask); | 428 | unsigned long reg = APIC_EILVTn(offset); |
| 402 | return APIC_EILVT_LVTOFF_IBS; | 429 | unsigned int new, old, reserved; |
| 430 | |||
| 431 | new = (mask << 16) | (msg_type << 8) | vector; | ||
| 432 | old = apic_read(reg); | ||
| 433 | reserved = reserve_eilvt_offset(offset, new); | ||
| 434 | |||
| 435 | if (reserved != new) { | ||
| 436 | pr_err(FW_BUG "cpu %d, try to setup vector 0x%x, but " | ||
| 437 | "vector 0x%x was already reserved by another core, " | ||
| 438 | "APIC%lX=0x%x\n", | ||
| 439 | smp_processor_id(), new, reserved, reg, old); | ||
| 440 | return -EINVAL; | ||
| 441 | } | ||
| 442 | |||
| 443 | if (!eilvt_entry_is_changeable(old, new)) { | ||
| 444 | pr_err(FW_BUG "cpu %d, try to setup vector 0x%x but " | ||
| 445 | "register already in use, APIC%lX=0x%x\n", | ||
| 446 | smp_processor_id(), new, reg, old); | ||
| 447 | return -EBUSY; | ||
| 448 | } | ||
| 449 | |||
| 450 | apic_write(reg, new); | ||
| 451 | |||
| 452 | return 0; | ||
| 403 | } | 453 | } |
| 404 | EXPORT_SYMBOL_GPL(setup_APIC_eilvt_ibs); | 454 | EXPORT_SYMBOL_GPL(setup_APIC_eilvt); |
| 405 | 455 | ||
| 406 | /* | 456 | /* |
| 407 | * Program the next event, relative to now | 457 | * Program the next event, relative to now |
| @@ -1665,10 +1715,7 @@ int __init APIC_init_uniprocessor(void) | |||
| 1665 | } | 1715 | } |
| 1666 | #endif | 1716 | #endif |
| 1667 | 1717 | ||
| 1668 | #ifndef CONFIG_SMP | ||
| 1669 | enable_IR_x2apic(); | ||
| 1670 | default_setup_apic_routing(); | 1718 | default_setup_apic_routing(); |
| 1671 | #endif | ||
| 1672 | 1719 | ||
| 1673 | verify_local_APIC(); | 1720 | verify_local_APIC(); |
| 1674 | connect_bsp_APIC(); | 1721 | connect_bsp_APIC(); |
