aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86_64
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86_64')
-rw-r--r--arch/x86_64/kernel/io_apic.c126
1 files changed, 34 insertions, 92 deletions
diff --git a/arch/x86_64/kernel/io_apic.c b/arch/x86_64/kernel/io_apic.c
index a69c38b62629..e0648380fdde 100644
--- a/arch/x86_64/kernel/io_apic.c
+++ b/arch/x86_64/kernel/io_apic.c
@@ -615,22 +615,6 @@ static int pin_2_irq(int idx, int apic, int pin)
615 return irq; 615 return irq;
616} 616}
617 617
618static inline int IO_APIC_irq_trigger(int irq)
619{
620 int apic, idx, pin;
621
622 for (apic = 0; apic < nr_ioapics; apic++) {
623 for (pin = 0; pin < nr_ioapic_registers[apic]; pin++) {
624 idx = find_irq_entry(apic,pin,mp_INT);
625 if ((idx != -1) && (irq == pin_2_irq(idx,apic,pin)))
626 return irq_trigger(idx);
627 }
628 }
629 /*
630 * nonexistent IRQs are edge default
631 */
632 return 0;
633}
634 618
635/* irq_vectors is indexed by the sum of all RTEs in all I/O APICs. */ 619/* irq_vectors is indexed by the sum of all RTEs in all I/O APICs. */
636static u8 irq_vector[NR_IRQ_VECTORS] __read_mostly = { 620static u8 irq_vector[NR_IRQ_VECTORS] __read_mostly = {
@@ -795,26 +779,36 @@ void __setup_vector_irq(int cpu)
795 779
796static struct irq_chip ioapic_chip; 780static struct irq_chip ioapic_chip;
797 781
798#define IOAPIC_AUTO -1
799#define IOAPIC_EDGE 0
800#define IOAPIC_LEVEL 1
801
802static void ioapic_register_intr(int irq, unsigned long trigger) 782static void ioapic_register_intr(int irq, unsigned long trigger)
803{ 783{
804 if ((trigger == IOAPIC_AUTO && IO_APIC_irq_trigger(irq)) || 784 if (trigger)
805 trigger == IOAPIC_LEVEL)
806 set_irq_chip_and_handler_name(irq, &ioapic_chip, 785 set_irq_chip_and_handler_name(irq, &ioapic_chip,
807 handle_fasteoi_irq, "fasteoi"); 786 handle_fasteoi_irq, "fasteoi");
808 else 787 else
809 set_irq_chip_and_handler_name(irq, &ioapic_chip, 788 set_irq_chip_and_handler_name(irq, &ioapic_chip,
810 handle_edge_irq, "edge"); 789 handle_edge_irq, "edge");
811} 790}
812static void __init setup_IO_APIC_irq(int apic, int pin, int idx, int irq) 791
792static void setup_IO_APIC_irq(int apic, int pin, unsigned int irq,
793 int trigger, int polarity)
813{ 794{
814 struct IO_APIC_route_entry entry; 795 struct IO_APIC_route_entry entry;
796 cpumask_t mask;
815 int vector; 797 int vector;
816 unsigned long flags; 798 unsigned long flags;
817 799
800 if (!IO_APIC_IRQ(irq))
801 return;
802
803 vector = assign_irq_vector(irq, TARGET_CPUS, &mask);
804 if (vector < 0)
805 return;
806
807 apic_printk(APIC_VERBOSE,KERN_DEBUG
808 "IOAPIC[%d]: Set routing entry (%d-%d -> 0x%x -> "
809 "IRQ %d Mode:%i Active:%i)\n",
810 apic, mp_ioapics[apic].mpc_apicid, pin, vector,
811 irq, trigger, polarity);
818 812
819 /* 813 /*
820 * add it to the IO-APIC irq-routing table: 814 * add it to the IO-APIC irq-routing table:
@@ -823,41 +817,27 @@ static void __init setup_IO_APIC_irq(int apic, int pin, int idx, int irq)
823 817
824 entry.delivery_mode = INT_DELIVERY_MODE; 818 entry.delivery_mode = INT_DELIVERY_MODE;
825 entry.dest_mode = INT_DEST_MODE; 819 entry.dest_mode = INT_DEST_MODE;
820 entry.dest = cpu_mask_to_apicid(mask);
826 entry.mask = 0; /* enable IRQ */ 821 entry.mask = 0; /* enable IRQ */
827 entry.dest = cpu_mask_to_apicid(TARGET_CPUS); 822 entry.trigger = trigger;
828 823 entry.polarity = polarity;
829 entry.trigger = irq_trigger(idx); 824 entry.vector = vector;
830 entry.polarity = irq_polarity(idx);
831 825
832 if (irq_trigger(idx)) { 826 /* Mask level triggered irqs.
833 entry.trigger = 1; 827 * Use IRQ_DELAYED_DISABLE for edge triggered irqs.
828 */
829 if (trigger)
834 entry.mask = 1; 830 entry.mask = 1;
835 entry.dest = cpu_mask_to_apicid(TARGET_CPUS);
836 }
837
838 if (!apic && !IO_APIC_IRQ(irq))
839 return;
840 831
841 if (IO_APIC_IRQ(irq)) { 832 ioapic_register_intr(irq, trigger);
842 cpumask_t mask; 833 if (irq < 16)
843 vector = assign_irq_vector(irq, TARGET_CPUS, &mask); 834 disable_8259A_irq(irq);
844 if (vector < 0)
845 return;
846
847 entry.dest = cpu_mask_to_apicid(mask);
848 entry.vector = vector;
849
850 ioapic_register_intr(irq, IOAPIC_AUTO);
851 if (!apic && (irq < 16))
852 disable_8259A_irq(irq);
853 }
854 835
855 ioapic_write_entry(apic, pin, entry); 836 ioapic_write_entry(apic, pin, entry);
856 837
857 spin_lock_irqsave(&ioapic_lock, flags); 838 spin_lock_irqsave(&ioapic_lock, flags);
858 irq_desc[irq].affinity = TARGET_CPUS; 839 irq_desc[irq].affinity = TARGET_CPUS;
859 spin_unlock_irqrestore(&ioapic_lock, flags); 840 spin_unlock_irqrestore(&ioapic_lock, flags);
860
861} 841}
862 842
863static void __init setup_IO_APIC_irqs(void) 843static void __init setup_IO_APIC_irqs(void)
@@ -882,8 +862,8 @@ static void __init setup_IO_APIC_irqs(void)
882 irq = pin_2_irq(idx, apic, pin); 862 irq = pin_2_irq(idx, apic, pin);
883 add_pin_to_irq(irq, apic, pin); 863 add_pin_to_irq(irq, apic, pin);
884 864
885 setup_IO_APIC_irq(apic, pin, idx, irq); 865 setup_IO_APIC_irq(apic, pin, irq,
886 866 irq_trigger(idx), irq_polarity(idx));
887 } 867 }
888 } 868 }
889 869
@@ -2090,11 +2070,6 @@ int __init io_apic_get_redir_entries (int ioapic)
2090 2070
2091int io_apic_set_pci_routing (int ioapic, int pin, int irq, int triggering, int polarity) 2071int io_apic_set_pci_routing (int ioapic, int pin, int irq, int triggering, int polarity)
2092{ 2072{
2093 struct IO_APIC_route_entry entry;
2094 unsigned long flags;
2095 int vector;
2096 cpumask_t mask;
2097
2098 if (!IO_APIC_IRQ(irq)) { 2073 if (!IO_APIC_IRQ(irq)) {
2099 apic_printk(APIC_QUIET,KERN_ERR "IOAPIC[%d]: Invalid reference to IRQ 0\n", 2074 apic_printk(APIC_QUIET,KERN_ERR "IOAPIC[%d]: Invalid reference to IRQ 0\n",
2100 ioapic); 2075 ioapic);
@@ -2107,42 +2082,7 @@ int io_apic_set_pci_routing (int ioapic, int pin, int irq, int triggering, int p
2107 if (irq >= 16) 2082 if (irq >= 16)
2108 add_pin_to_irq(irq, ioapic, pin); 2083 add_pin_to_irq(irq, ioapic, pin);
2109 2084
2110 2085 setup_IO_APIC_irq(ioapic, pin, irq, triggering, polarity);
2111 vector = assign_irq_vector(irq, TARGET_CPUS, &mask);
2112 if (vector < 0)
2113 return vector;
2114
2115 /*
2116 * Generate a PCI IRQ routing entry and program the IOAPIC accordingly.
2117 * Note that we mask (disable) IRQs now -- these get enabled when the
2118 * corresponding device driver registers for this IRQ.
2119 */
2120
2121 memset(&entry,0,sizeof(entry));
2122
2123 entry.delivery_mode = INT_DELIVERY_MODE;
2124 entry.dest_mode = INT_DEST_MODE;
2125 entry.dest = cpu_mask_to_apicid(mask);
2126 entry.trigger = triggering;
2127 entry.polarity = polarity;
2128 entry.mask = 1; /* Disabled (masked) */
2129 entry.vector = vector & 0xff;
2130
2131 apic_printk(APIC_VERBOSE,KERN_DEBUG "IOAPIC[%d]: Set PCI routing entry (%d-%d -> 0x%x -> "
2132 "IRQ %d Mode:%i Active:%i)\n", ioapic,
2133 mp_ioapics[ioapic].mpc_apicid, pin, entry.vector, irq,
2134 triggering, polarity);
2135
2136 ioapic_register_intr(irq, triggering);
2137
2138 if (!ioapic && (irq < 16))
2139 disable_8259A_irq(irq);
2140
2141 ioapic_write_entry(ioapic, pin, entry);
2142
2143 spin_lock_irqsave(&ioapic_lock, flags);
2144 irq_desc[irq].affinity = TARGET_CPUS;
2145 spin_unlock_irqrestore(&ioapic_lock, flags);
2146 2086
2147 return 0; 2087 return 0;
2148} 2088}
@@ -2175,7 +2115,9 @@ void __init setup_ioapic_dest(void)
2175 * cpu is online. 2115 * cpu is online.
2176 */ 2116 */
2177 if(!irq_vector[irq]) 2117 if(!irq_vector[irq])
2178 setup_IO_APIC_irq(ioapic, pin, irq_entry, irq); 2118 setup_IO_APIC_irq(ioapic, pin, irq,
2119 irq_trigger(irq_entry),
2120 irq_polarity(irq_entry));
2179 else 2121 else
2180 set_ioapic_affinity_irq(irq, TARGET_CPUS); 2122 set_ioapic_affinity_irq(irq, TARGET_CPUS);
2181 } 2123 }