diff options
| author | Suresh Siddha <suresh.b.siddha@intel.com> | 2009-12-01 18:31:16 -0500 |
|---|---|---|
| committer | Ingo Molnar <mingo@elte.hu> | 2009-12-02 04:11:01 -0500 |
| commit | c29d9db338db606c3335a03f337e1d4b7f6bb727 (patch) | |
| tree | d64763cd2a06af0d95b481a61c0ff6fcfa0de208 | |
| parent | ca64c47cecd0321b2e0dcbd7aaff44b68ce20654 (diff) | |
x86, ioapic: Fix the EOI register detection mechanism
Maciej W. Rozycki reported:
> 82093AA I/O APIC has its version set to 0x11 and it
> does not support the EOI register. Similarly I/O APICs
> integrated into the 82379AB south bridge and the 82374EB/SB
> EISA component.
IO-APIC versions below 0x20 don't support EOI register.
Some of the Intel ICH Specs (ICH2 to ICH5) documents the io-apic
version as 0x2. This is an error with documentation and these
ICH chips use io-apic's of version 0x20 and indeed has a working
EOI register for the io-apic.
Fix the EOI register detection mechanism to check for version
0x20 and beyond.
And also, a platform can potentially have io-apic's with
different versions. Make the EOI register check per io-apic.
Reported-by: Maciej W. Rozycki <macro@linux-mips.org>
Signed-off-by: Suresh Siddha <suresh.b.siddha@intel.com>
Cc: ebiederm@xmission.com
Cc: garyhade@us.ibm.com
LKML-Reference: <20091201233335.065361533@sbs-t61.sc.intel.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
| -rw-r--r-- | arch/x86/kernel/apic/io_apic.c | 115 |
1 files changed, 61 insertions, 54 deletions
diff --git a/arch/x86/kernel/apic/io_apic.c b/arch/x86/kernel/apic/io_apic.c index b377b973899e..78960a3b0ed0 100644 --- a/arch/x86/kernel/apic/io_apic.c +++ b/arch/x86/kernel/apic/io_apic.c | |||
| @@ -539,23 +539,41 @@ static void __init replace_pin_at_irq_node(struct irq_cfg *cfg, int node, | |||
| 539 | add_pin_to_irq_node(cfg, node, newapic, newpin); | 539 | add_pin_to_irq_node(cfg, node, newapic, newpin); |
| 540 | } | 540 | } |
| 541 | 541 | ||
| 542 | static void __io_apic_modify_irq(struct irq_pin_list *entry, | ||
| 543 | int mask_and, int mask_or, | ||
| 544 | void (*final)(struct irq_pin_list *entry)) | ||
| 545 | { | ||
| 546 | unsigned int reg, pin; | ||
| 547 | |||
| 548 | pin = entry->pin; | ||
| 549 | reg = io_apic_read(entry->apic, 0x10 + pin * 2); | ||
| 550 | reg &= mask_and; | ||
| 551 | reg |= mask_or; | ||
| 552 | io_apic_modify(entry->apic, 0x10 + pin * 2, reg); | ||
| 553 | if (final) | ||
| 554 | final(entry); | ||
| 555 | } | ||
| 556 | |||
| 542 | static void io_apic_modify_irq(struct irq_cfg *cfg, | 557 | static void io_apic_modify_irq(struct irq_cfg *cfg, |
| 543 | int mask_and, int mask_or, | 558 | int mask_and, int mask_or, |
| 544 | void (*final)(struct irq_pin_list *entry)) | 559 | void (*final)(struct irq_pin_list *entry)) |
| 545 | { | 560 | { |
| 546 | int pin; | ||
| 547 | struct irq_pin_list *entry; | 561 | struct irq_pin_list *entry; |
| 548 | 562 | ||
| 549 | for_each_irq_pin(entry, cfg->irq_2_pin) { | 563 | for_each_irq_pin(entry, cfg->irq_2_pin) |
| 550 | unsigned int reg; | 564 | __io_apic_modify_irq(entry, mask_and, mask_or, final); |
| 551 | pin = entry->pin; | 565 | } |
| 552 | reg = io_apic_read(entry->apic, 0x10 + pin * 2); | 566 | |
| 553 | reg &= mask_and; | 567 | static void __mask_and_edge_IO_APIC_irq(struct irq_pin_list *entry) |
| 554 | reg |= mask_or; | 568 | { |
| 555 | io_apic_modify(entry->apic, 0x10 + pin * 2, reg); | 569 | __io_apic_modify_irq(entry, ~IO_APIC_REDIR_LEVEL_TRIGGER, |
| 556 | if (final) | 570 | IO_APIC_REDIR_MASKED, NULL); |
| 557 | final(entry); | 571 | } |
| 558 | } | 572 | |
| 573 | static void __unmask_and_level_IO_APIC_irq(struct irq_pin_list *entry) | ||
| 574 | { | ||
| 575 | __io_apic_modify_irq(entry, ~IO_APIC_REDIR_MASKED, | ||
| 576 | IO_APIC_REDIR_LEVEL_TRIGGER, NULL); | ||
| 559 | } | 577 | } |
| 560 | 578 | ||
| 561 | static void __unmask_IO_APIC_irq(struct irq_cfg *cfg) | 579 | static void __unmask_IO_APIC_irq(struct irq_cfg *cfg) |
| @@ -579,18 +597,6 @@ static void __mask_IO_APIC_irq(struct irq_cfg *cfg) | |||
| 579 | io_apic_modify_irq(cfg, ~0, IO_APIC_REDIR_MASKED, &io_apic_sync); | 597 | io_apic_modify_irq(cfg, ~0, IO_APIC_REDIR_MASKED, &io_apic_sync); |
| 580 | } | 598 | } |
| 581 | 599 | ||
| 582 | static void __mask_and_edge_IO_APIC_irq(struct irq_cfg *cfg) | ||
| 583 | { | ||
| 584 | io_apic_modify_irq(cfg, ~IO_APIC_REDIR_LEVEL_TRIGGER, | ||
| 585 | IO_APIC_REDIR_MASKED, NULL); | ||
| 586 | } | ||
| 587 | |||
| 588 | static void __unmask_and_level_IO_APIC_irq(struct irq_cfg *cfg) | ||
| 589 | { | ||
| 590 | io_apic_modify_irq(cfg, ~IO_APIC_REDIR_MASKED, | ||
| 591 | IO_APIC_REDIR_LEVEL_TRIGGER, NULL); | ||
| 592 | } | ||
| 593 | |||
| 594 | static void mask_IO_APIC_irq_desc(struct irq_desc *desc) | 600 | static void mask_IO_APIC_irq_desc(struct irq_desc *desc) |
| 595 | { | 601 | { |
| 596 | struct irq_cfg *cfg = desc->chip_data; | 602 | struct irq_cfg *cfg = desc->chip_data; |
| @@ -2492,17 +2498,42 @@ static void ack_apic_edge(unsigned int irq) | |||
| 2492 | 2498 | ||
| 2493 | atomic_t irq_mis_count; | 2499 | atomic_t irq_mis_count; |
| 2494 | 2500 | ||
| 2495 | static int use_eoi_reg __read_mostly; | 2501 | /* |
| 2496 | 2502 | * IO-APIC versions below 0x20 don't support EOI register. | |
| 2503 | * For the record, here is the information about various versions: | ||
| 2504 | * 0Xh 82489DX | ||
| 2505 | * 1Xh I/OAPIC or I/O(x)APIC which are not PCI 2.2 Compliant | ||
| 2506 | * 2Xh I/O(x)APIC which is PCI 2.2 Compliant | ||
| 2507 | * 30h-FFh Reserved | ||
| 2508 | * | ||
| 2509 | * Some of the Intel ICH Specs (ICH2 to ICH5) documents the io-apic | ||
| 2510 | * version as 0x2. This is an error with documentation and these ICH chips | ||
| 2511 | * use io-apic's of version 0x20. | ||
| 2512 | * | ||
| 2513 | * For IO-APIC's with EOI register, we use that to do an explicit EOI. | ||
| 2514 | * Otherwise, we simulate the EOI message manually by changing the trigger | ||
| 2515 | * mode to edge and then back to level, with RTE being masked during this. | ||
| 2516 | */ | ||
| 2497 | static void __eoi_ioapic_irq(unsigned int irq, struct irq_cfg *cfg) | 2517 | static void __eoi_ioapic_irq(unsigned int irq, struct irq_cfg *cfg) |
| 2498 | { | 2518 | { |
| 2499 | struct irq_pin_list *entry; | 2519 | struct irq_pin_list *entry; |
| 2500 | 2520 | ||
| 2501 | for_each_irq_pin(entry, cfg->irq_2_pin) { | 2521 | for_each_irq_pin(entry, cfg->irq_2_pin) { |
| 2502 | if (irq_remapped(irq)) | 2522 | if (mp_ioapics[entry->apic].apicver >= 0x20) { |
| 2503 | io_apic_eoi(entry->apic, entry->pin); | 2523 | /* |
| 2504 | else | 2524 | * Intr-remapping uses pin number as the virtual vector |
| 2505 | io_apic_eoi(entry->apic, cfg->vector); | 2525 | * in the RTE. Actual vector is programmed in |
| 2526 | * intr-remapping table entry. Hence for the io-apic | ||
| 2527 | * EOI we use the pin number. | ||
| 2528 | */ | ||
| 2529 | if (irq_remapped(irq)) | ||
| 2530 | io_apic_eoi(entry->apic, entry->pin); | ||
| 2531 | else | ||
| 2532 | io_apic_eoi(entry->apic, cfg->vector); | ||
| 2533 | } else { | ||
| 2534 | __mask_and_edge_IO_APIC_irq(entry); | ||
| 2535 | __unmask_and_level_IO_APIC_irq(entry); | ||
| 2536 | } | ||
| 2506 | } | 2537 | } |
| 2507 | } | 2538 | } |
| 2508 | 2539 | ||
| @@ -2520,23 +2551,6 @@ static void eoi_ioapic_irq(struct irq_desc *desc) | |||
| 2520 | spin_unlock_irqrestore(&ioapic_lock, flags); | 2551 | spin_unlock_irqrestore(&ioapic_lock, flags); |
| 2521 | } | 2552 | } |
| 2522 | 2553 | ||
| 2523 | static int ioapic_supports_eoi(void) | ||
| 2524 | { | ||
| 2525 | struct pci_dev *root; | ||
| 2526 | |||
| 2527 | root = pci_get_bus_and_slot(0, PCI_DEVFN(0, 0)); | ||
| 2528 | if (root && root->vendor == PCI_VENDOR_ID_INTEL && | ||
| 2529 | mp_ioapics[0].apicver >= 0x2) { | ||
| 2530 | use_eoi_reg = 1; | ||
| 2531 | printk(KERN_INFO "IO-APIC supports EOI register\n"); | ||
| 2532 | } else | ||
| 2533 | printk(KERN_INFO "IO-APIC doesn't support EOI\n"); | ||
| 2534 | |||
| 2535 | return 0; | ||
| 2536 | } | ||
| 2537 | |||
| 2538 | fs_initcall(ioapic_supports_eoi); | ||
| 2539 | |||
| 2540 | static void ack_apic_level(unsigned int irq) | 2554 | static void ack_apic_level(unsigned int irq) |
| 2541 | { | 2555 | { |
| 2542 | struct irq_desc *desc = irq_to_desc(irq); | 2556 | struct irq_desc *desc = irq_to_desc(irq); |
| @@ -2587,14 +2601,7 @@ static void ack_apic_level(unsigned int irq) | |||
| 2587 | if (!(v & (1 << (i & 0x1f)))) { | 2601 | if (!(v & (1 << (i & 0x1f)))) { |
| 2588 | atomic_inc(&irq_mis_count); | 2602 | atomic_inc(&irq_mis_count); |
| 2589 | 2603 | ||
| 2590 | if (use_eoi_reg) | 2604 | eoi_ioapic_irq(desc); |
| 2591 | eoi_ioapic_irq(desc); | ||
| 2592 | else { | ||
| 2593 | spin_lock(&ioapic_lock); | ||
| 2594 | __mask_and_edge_IO_APIC_irq(cfg); | ||
| 2595 | __unmask_and_level_IO_APIC_irq(cfg); | ||
| 2596 | spin_unlock(&ioapic_lock); | ||
| 2597 | } | ||
| 2598 | } | 2605 | } |
| 2599 | 2606 | ||
| 2600 | /* Now we can move and renable the irq */ | 2607 | /* Now we can move and renable the irq */ |
