diff options
| -rw-r--r-- | arch/x86/include/asm/hpet.h | 5 | ||||
| -rw-r--r-- | arch/x86/include/asm/hw_irq.h | 13 | ||||
| -rw-r--r-- | arch/x86/include/asm/io_apic.h | 28 | ||||
| -rw-r--r-- | arch/x86/include/asm/irq_remapping.h | 40 | ||||
| -rw-r--r-- | arch/x86/include/asm/pci.h | 3 | ||||
| -rw-r--r-- | arch/x86/include/asm/x86_init.h | 27 | ||||
| -rw-r--r-- | arch/x86/kernel/apic/apic.c | 28 | ||||
| -rw-r--r-- | arch/x86/kernel/apic/io_apic.c | 446 | ||||
| -rw-r--r-- | arch/x86/kernel/hpet.c | 2 | ||||
| -rw-r--r-- | arch/x86/kernel/x86_init.c | 24 | ||||
| -rw-r--r-- | drivers/iommu/amd_iommu.c | 8 | ||||
| -rw-r--r-- | drivers/iommu/dmar.c | 2 | ||||
| -rw-r--r-- | drivers/iommu/intel-iommu.c | 2 | ||||
| -rw-r--r-- | drivers/iommu/intel_irq_remapping.c | 12 | ||||
| -rw-r--r-- | drivers/iommu/irq_remapping.c | 231 | ||||
| -rw-r--r-- | drivers/iommu/irq_remapping.h | 1 | ||||
| -rw-r--r-- | include/linux/irq.h | 3 |
17 files changed, 504 insertions, 371 deletions
diff --git a/arch/x86/include/asm/hpet.h b/arch/x86/include/asm/hpet.h index 434e2106cc87..b18df579c0e9 100644 --- a/arch/x86/include/asm/hpet.h +++ b/arch/x86/include/asm/hpet.h | |||
| @@ -80,9 +80,9 @@ extern void hpet_msi_write(struct hpet_dev *hdev, struct msi_msg *msg); | |||
| 80 | extern void hpet_msi_read(struct hpet_dev *hdev, struct msi_msg *msg); | 80 | extern void hpet_msi_read(struct hpet_dev *hdev, struct msi_msg *msg); |
| 81 | 81 | ||
| 82 | #ifdef CONFIG_PCI_MSI | 82 | #ifdef CONFIG_PCI_MSI |
| 83 | extern int arch_setup_hpet_msi(unsigned int irq, unsigned int id); | 83 | extern int default_setup_hpet_msi(unsigned int irq, unsigned int id); |
| 84 | #else | 84 | #else |
| 85 | static inline int arch_setup_hpet_msi(unsigned int irq, unsigned int id) | 85 | static inline int default_setup_hpet_msi(unsigned int irq, unsigned int id) |
| 86 | { | 86 | { |
| 87 | return -EINVAL; | 87 | return -EINVAL; |
| 88 | } | 88 | } |
| @@ -111,6 +111,7 @@ extern void hpet_unregister_irq_handler(rtc_irq_handler handler); | |||
| 111 | static inline int hpet_enable(void) { return 0; } | 111 | static inline int hpet_enable(void) { return 0; } |
| 112 | static inline int is_hpet_enabled(void) { return 0; } | 112 | static inline int is_hpet_enabled(void) { return 0; } |
| 113 | #define hpet_readl(a) 0 | 113 | #define hpet_readl(a) 0 |
| 114 | #define default_setup_hpet_msi NULL | ||
| 114 | 115 | ||
| 115 | #endif | 116 | #endif |
| 116 | #endif /* _ASM_X86_HPET_H */ | 117 | #endif /* _ASM_X86_HPET_H */ |
diff --git a/arch/x86/include/asm/hw_irq.h b/arch/x86/include/asm/hw_irq.h index eb92a6ed2be7..10a78c3d3d5a 100644 --- a/arch/x86/include/asm/hw_irq.h +++ b/arch/x86/include/asm/hw_irq.h | |||
| @@ -101,6 +101,7 @@ static inline void set_io_apic_irq_attr(struct io_apic_irq_attr *irq_attr, | |||
| 101 | irq_attr->polarity = polarity; | 101 | irq_attr->polarity = polarity; |
| 102 | } | 102 | } |
| 103 | 103 | ||
| 104 | /* Intel specific interrupt remapping information */ | ||
| 104 | struct irq_2_iommu { | 105 | struct irq_2_iommu { |
| 105 | struct intel_iommu *iommu; | 106 | struct intel_iommu *iommu; |
| 106 | u16 irte_index; | 107 | u16 irte_index; |
| @@ -108,6 +109,12 @@ struct irq_2_iommu { | |||
| 108 | u8 irte_mask; | 109 | u8 irte_mask; |
| 109 | }; | 110 | }; |
| 110 | 111 | ||
| 112 | /* AMD specific interrupt remapping information */ | ||
| 113 | struct irq_2_irte { | ||
| 114 | u16 devid; /* Device ID for IRTE table */ | ||
| 115 | u16 index; /* Index into IRTE table*/ | ||
| 116 | }; | ||
| 117 | |||
| 111 | /* | 118 | /* |
| 112 | * This is performance-critical, we want to do it O(1) | 119 | * This is performance-critical, we want to do it O(1) |
| 113 | * | 120 | * |
| @@ -120,7 +127,11 @@ struct irq_cfg { | |||
| 120 | u8 vector; | 127 | u8 vector; |
| 121 | u8 move_in_progress : 1; | 128 | u8 move_in_progress : 1; |
| 122 | #ifdef CONFIG_IRQ_REMAP | 129 | #ifdef CONFIG_IRQ_REMAP |
| 123 | struct irq_2_iommu irq_2_iommu; | 130 | u8 remapped : 1; |
| 131 | union { | ||
| 132 | struct irq_2_iommu irq_2_iommu; | ||
| 133 | struct irq_2_irte irq_2_irte; | ||
| 134 | }; | ||
| 124 | #endif | 135 | #endif |
| 125 | }; | 136 | }; |
| 126 | 137 | ||
diff --git a/arch/x86/include/asm/io_apic.h b/arch/x86/include/asm/io_apic.h index 73d8c5398ea9..459e50a424d1 100644 --- a/arch/x86/include/asm/io_apic.h +++ b/arch/x86/include/asm/io_apic.h | |||
| @@ -144,11 +144,24 @@ extern int timer_through_8259; | |||
| 144 | (mp_irq_entries && !skip_ioapic_setup && io_apic_irqs) | 144 | (mp_irq_entries && !skip_ioapic_setup && io_apic_irqs) |
| 145 | 145 | ||
| 146 | struct io_apic_irq_attr; | 146 | struct io_apic_irq_attr; |
| 147 | struct irq_cfg; | ||
| 147 | extern int io_apic_set_pci_routing(struct device *dev, int irq, | 148 | extern int io_apic_set_pci_routing(struct device *dev, int irq, |
| 148 | struct io_apic_irq_attr *irq_attr); | 149 | struct io_apic_irq_attr *irq_attr); |
| 149 | void setup_IO_APIC_irq_extra(u32 gsi); | 150 | void setup_IO_APIC_irq_extra(u32 gsi); |
| 150 | extern void ioapic_insert_resources(void); | 151 | extern void ioapic_insert_resources(void); |
| 151 | 152 | ||
| 153 | extern int native_setup_ioapic_entry(int, struct IO_APIC_route_entry *, | ||
| 154 | unsigned int, int, | ||
| 155 | struct io_apic_irq_attr *); | ||
| 156 | extern int native_setup_ioapic_entry(int, struct IO_APIC_route_entry *, | ||
| 157 | unsigned int, int, | ||
| 158 | struct io_apic_irq_attr *); | ||
| 159 | extern void eoi_ioapic_irq(unsigned int irq, struct irq_cfg *cfg); | ||
| 160 | |||
| 161 | extern void native_compose_msi_msg(struct pci_dev *pdev, | ||
| 162 | unsigned int irq, unsigned int dest, | ||
| 163 | struct msi_msg *msg, u8 hpet_id); | ||
| 164 | extern void native_eoi_ioapic_pin(int apic, int pin, int vector); | ||
| 152 | int io_apic_setup_irq_pin_once(unsigned int irq, int node, struct io_apic_irq_attr *attr); | 165 | int io_apic_setup_irq_pin_once(unsigned int irq, int node, struct io_apic_irq_attr *attr); |
| 153 | 166 | ||
| 154 | extern int save_ioapic_entries(void); | 167 | extern int save_ioapic_entries(void); |
| @@ -179,6 +192,12 @@ extern void __init native_io_apic_init_mappings(void); | |||
| 179 | extern unsigned int native_io_apic_read(unsigned int apic, unsigned int reg); | 192 | extern unsigned int native_io_apic_read(unsigned int apic, unsigned int reg); |
| 180 | extern void native_io_apic_write(unsigned int apic, unsigned int reg, unsigned int val); | 193 | extern void native_io_apic_write(unsigned int apic, unsigned int reg, unsigned int val); |
| 181 | extern void native_io_apic_modify(unsigned int apic, unsigned int reg, unsigned int val); | 194 | extern void native_io_apic_modify(unsigned int apic, unsigned int reg, unsigned int val); |
| 195 | extern void native_disable_io_apic(void); | ||
| 196 | extern void native_io_apic_print_entries(unsigned int apic, unsigned int nr_entries); | ||
| 197 | extern void intel_ir_io_apic_print_entries(unsigned int apic, unsigned int nr_entries); | ||
| 198 | extern int native_ioapic_set_affinity(struct irq_data *, | ||
| 199 | const struct cpumask *, | ||
| 200 | bool); | ||
| 182 | 201 | ||
| 183 | static inline unsigned int io_apic_read(unsigned int apic, unsigned int reg) | 202 | static inline unsigned int io_apic_read(unsigned int apic, unsigned int reg) |
| 184 | { | 203 | { |
| @@ -193,6 +212,9 @@ static inline void io_apic_modify(unsigned int apic, unsigned int reg, unsigned | |||
| 193 | { | 212 | { |
| 194 | x86_io_apic_ops.modify(apic, reg, value); | 213 | x86_io_apic_ops.modify(apic, reg, value); |
| 195 | } | 214 | } |
| 215 | |||
| 216 | extern void io_apic_eoi(unsigned int apic, unsigned int vector); | ||
| 217 | |||
| 196 | #else /* !CONFIG_X86_IO_APIC */ | 218 | #else /* !CONFIG_X86_IO_APIC */ |
| 197 | 219 | ||
| 198 | #define io_apic_assign_pci_irqs 0 | 220 | #define io_apic_assign_pci_irqs 0 |
| @@ -223,6 +245,12 @@ static inline void disable_ioapic_support(void) { } | |||
| 223 | #define native_io_apic_read NULL | 245 | #define native_io_apic_read NULL |
| 224 | #define native_io_apic_write NULL | 246 | #define native_io_apic_write NULL |
| 225 | #define native_io_apic_modify NULL | 247 | #define native_io_apic_modify NULL |
| 248 | #define native_disable_io_apic NULL | ||
| 249 | #define native_io_apic_print_entries NULL | ||
| 250 | #define native_ioapic_set_affinity NULL | ||
| 251 | #define native_setup_ioapic_entry NULL | ||
| 252 | #define native_compose_msi_msg NULL | ||
| 253 | #define native_eoi_ioapic_pin NULL | ||
| 226 | #endif | 254 | #endif |
| 227 | 255 | ||
| 228 | #endif /* _ASM_X86_IO_APIC_H */ | 256 | #endif /* _ASM_X86_IO_APIC_H */ |
diff --git a/arch/x86/include/asm/irq_remapping.h b/arch/x86/include/asm/irq_remapping.h index 5fb9bbbd2f14..95fd3527f632 100644 --- a/arch/x86/include/asm/irq_remapping.h +++ b/arch/x86/include/asm/irq_remapping.h | |||
| @@ -26,8 +26,6 @@ | |||
| 26 | 26 | ||
| 27 | #ifdef CONFIG_IRQ_REMAP | 27 | #ifdef CONFIG_IRQ_REMAP |
| 28 | 28 | ||
| 29 | extern int irq_remapping_enabled; | ||
| 30 | |||
| 31 | extern void setup_irq_remapping_ops(void); | 29 | extern void setup_irq_remapping_ops(void); |
| 32 | extern int irq_remapping_supported(void); | 30 | extern int irq_remapping_supported(void); |
| 33 | extern int irq_remapping_prepare(void); | 31 | extern int irq_remapping_prepare(void); |
| @@ -40,21 +38,19 @@ extern int setup_ioapic_remapped_entry(int irq, | |||
| 40 | unsigned int destination, | 38 | unsigned int destination, |
| 41 | int vector, | 39 | int vector, |
| 42 | struct io_apic_irq_attr *attr); | 40 | struct io_apic_irq_attr *attr); |
| 43 | extern int set_remapped_irq_affinity(struct irq_data *data, | ||
| 44 | const struct cpumask *mask, | ||
| 45 | bool force); | ||
| 46 | extern void free_remapped_irq(int irq); | 41 | extern void free_remapped_irq(int irq); |
| 47 | extern void compose_remapped_msi_msg(struct pci_dev *pdev, | 42 | extern void compose_remapped_msi_msg(struct pci_dev *pdev, |
| 48 | unsigned int irq, unsigned int dest, | 43 | unsigned int irq, unsigned int dest, |
| 49 | struct msi_msg *msg, u8 hpet_id); | 44 | struct msi_msg *msg, u8 hpet_id); |
| 50 | extern int msi_alloc_remapped_irq(struct pci_dev *pdev, int irq, int nvec); | ||
| 51 | extern int msi_setup_remapped_irq(struct pci_dev *pdev, unsigned int irq, | ||
| 52 | int index, int sub_handle); | ||
| 53 | extern int setup_hpet_msi_remapped(unsigned int irq, unsigned int id); | 45 | extern int setup_hpet_msi_remapped(unsigned int irq, unsigned int id); |
| 46 | extern void panic_if_irq_remap(const char *msg); | ||
| 47 | extern bool setup_remapped_irq(int irq, | ||
| 48 | struct irq_cfg *cfg, | ||
| 49 | struct irq_chip *chip); | ||
| 54 | 50 | ||
| 55 | #else /* CONFIG_IRQ_REMAP */ | 51 | void irq_remap_modify_chip_defaults(struct irq_chip *chip); |
| 56 | 52 | ||
| 57 | #define irq_remapping_enabled 0 | 53 | #else /* CONFIG_IRQ_REMAP */ |
| 58 | 54 | ||
| 59 | static inline void setup_irq_remapping_ops(void) { } | 55 | static inline void setup_irq_remapping_ops(void) { } |
| 60 | static inline int irq_remapping_supported(void) { return 0; } | 56 | static inline int irq_remapping_supported(void) { return 0; } |
| @@ -71,30 +67,30 @@ static inline int setup_ioapic_remapped_entry(int irq, | |||
| 71 | { | 67 | { |
| 72 | return -ENODEV; | 68 | return -ENODEV; |
| 73 | } | 69 | } |
| 74 | static inline int set_remapped_irq_affinity(struct irq_data *data, | ||
| 75 | const struct cpumask *mask, | ||
| 76 | bool force) | ||
| 77 | { | ||
| 78 | return 0; | ||
| 79 | } | ||
| 80 | static inline void free_remapped_irq(int irq) { } | 70 | static inline void free_remapped_irq(int irq) { } |
| 81 | static inline void compose_remapped_msi_msg(struct pci_dev *pdev, | 71 | static inline void compose_remapped_msi_msg(struct pci_dev *pdev, |
| 82 | unsigned int irq, unsigned int dest, | 72 | unsigned int irq, unsigned int dest, |
| 83 | struct msi_msg *msg, u8 hpet_id) | 73 | struct msi_msg *msg, u8 hpet_id) |
| 84 | { | 74 | { |
| 85 | } | 75 | } |
| 86 | static inline int msi_alloc_remapped_irq(struct pci_dev *pdev, int irq, int nvec) | 76 | static inline int setup_hpet_msi_remapped(unsigned int irq, unsigned int id) |
| 87 | { | 77 | { |
| 88 | return -ENODEV; | 78 | return -ENODEV; |
| 89 | } | 79 | } |
| 90 | static inline int msi_setup_remapped_irq(struct pci_dev *pdev, unsigned int irq, | 80 | |
| 91 | int index, int sub_handle) | 81 | static inline void panic_if_irq_remap(const char *msg) |
| 82 | { | ||
| 83 | } | ||
| 84 | |||
| 85 | static inline void irq_remap_modify_chip_defaults(struct irq_chip *chip) | ||
| 92 | { | 86 | { |
| 93 | return -ENODEV; | ||
| 94 | } | 87 | } |
| 95 | static inline int setup_hpet_msi_remapped(unsigned int irq, unsigned int id) | 88 | |
| 89 | static inline bool setup_remapped_irq(int irq, | ||
| 90 | struct irq_cfg *cfg, | ||
| 91 | struct irq_chip *chip) | ||
| 96 | { | 92 | { |
| 97 | return -ENODEV; | 93 | return false; |
| 98 | } | 94 | } |
| 99 | #endif /* CONFIG_IRQ_REMAP */ | 95 | #endif /* CONFIG_IRQ_REMAP */ |
| 100 | 96 | ||
diff --git a/arch/x86/include/asm/pci.h b/arch/x86/include/asm/pci.h index dba7805176bf..c28fd02f4bf7 100644 --- a/arch/x86/include/asm/pci.h +++ b/arch/x86/include/asm/pci.h | |||
| @@ -121,9 +121,12 @@ static inline void x86_restore_msi_irqs(struct pci_dev *dev, int irq) | |||
| 121 | #define arch_teardown_msi_irq x86_teardown_msi_irq | 121 | #define arch_teardown_msi_irq x86_teardown_msi_irq |
| 122 | #define arch_restore_msi_irqs x86_restore_msi_irqs | 122 | #define arch_restore_msi_irqs x86_restore_msi_irqs |
| 123 | /* implemented in arch/x86/kernel/apic/io_apic. */ | 123 | /* implemented in arch/x86/kernel/apic/io_apic. */ |
| 124 | struct msi_desc; | ||
| 124 | int native_setup_msi_irqs(struct pci_dev *dev, int nvec, int type); | 125 | int native_setup_msi_irqs(struct pci_dev *dev, int nvec, int type); |
| 125 | void native_teardown_msi_irq(unsigned int irq); | 126 | void native_teardown_msi_irq(unsigned int irq); |
| 126 | void native_restore_msi_irqs(struct pci_dev *dev, int irq); | 127 | void native_restore_msi_irqs(struct pci_dev *dev, int irq); |
| 128 | int setup_msi_irq(struct pci_dev *dev, struct msi_desc *msidesc, | ||
| 129 | unsigned int irq_base, unsigned int irq_offset); | ||
| 127 | /* default to the implementation in drivers/lib/msi.c */ | 130 | /* default to the implementation in drivers/lib/msi.c */ |
| 128 | #define HAVE_DEFAULT_MSI_TEARDOWN_IRQS | 131 | #define HAVE_DEFAULT_MSI_TEARDOWN_IRQS |
| 129 | #define HAVE_DEFAULT_MSI_RESTORE_IRQS | 132 | #define HAVE_DEFAULT_MSI_RESTORE_IRQS |
diff --git a/arch/x86/include/asm/x86_init.h b/arch/x86/include/asm/x86_init.h index 57693498519c..7669941cc9d2 100644 --- a/arch/x86/include/asm/x86_init.h +++ b/arch/x86/include/asm/x86_init.h | |||
| @@ -181,19 +181,38 @@ struct x86_platform_ops { | |||
| 181 | }; | 181 | }; |
| 182 | 182 | ||
| 183 | struct pci_dev; | 183 | struct pci_dev; |
| 184 | struct msi_msg; | ||
| 184 | 185 | ||
| 185 | struct x86_msi_ops { | 186 | struct x86_msi_ops { |
| 186 | int (*setup_msi_irqs)(struct pci_dev *dev, int nvec, int type); | 187 | int (*setup_msi_irqs)(struct pci_dev *dev, int nvec, int type); |
| 188 | void (*compose_msi_msg)(struct pci_dev *dev, unsigned int irq, | ||
| 189 | unsigned int dest, struct msi_msg *msg, | ||
| 190 | u8 hpet_id); | ||
| 187 | void (*teardown_msi_irq)(unsigned int irq); | 191 | void (*teardown_msi_irq)(unsigned int irq); |
| 188 | void (*teardown_msi_irqs)(struct pci_dev *dev); | 192 | void (*teardown_msi_irqs)(struct pci_dev *dev); |
| 189 | void (*restore_msi_irqs)(struct pci_dev *dev, int irq); | 193 | void (*restore_msi_irqs)(struct pci_dev *dev, int irq); |
| 194 | int (*setup_hpet_msi)(unsigned int irq, unsigned int id); | ||
| 190 | }; | 195 | }; |
| 191 | 196 | ||
| 197 | struct IO_APIC_route_entry; | ||
| 198 | struct io_apic_irq_attr; | ||
| 199 | struct irq_data; | ||
| 200 | struct cpumask; | ||
| 201 | |||
| 192 | struct x86_io_apic_ops { | 202 | struct x86_io_apic_ops { |
| 193 | void (*init) (void); | 203 | void (*init) (void); |
| 194 | unsigned int (*read) (unsigned int apic, unsigned int reg); | 204 | unsigned int (*read) (unsigned int apic, unsigned int reg); |
| 195 | void (*write) (unsigned int apic, unsigned int reg, unsigned int value); | 205 | void (*write) (unsigned int apic, unsigned int reg, unsigned int value); |
| 196 | void (*modify)(unsigned int apic, unsigned int reg, unsigned int value); | 206 | void (*modify) (unsigned int apic, unsigned int reg, unsigned int value); |
| 207 | void (*disable)(void); | ||
| 208 | void (*print_entries)(unsigned int apic, unsigned int nr_entries); | ||
| 209 | int (*set_affinity)(struct irq_data *data, | ||
| 210 | const struct cpumask *mask, | ||
| 211 | bool force); | ||
| 212 | int (*setup_entry)(int irq, struct IO_APIC_route_entry *entry, | ||
| 213 | unsigned int destination, int vector, | ||
| 214 | struct io_apic_irq_attr *attr); | ||
| 215 | void (*eoi_ioapic_pin)(int apic, int pin, int vector); | ||
| 197 | }; | 216 | }; |
| 198 | 217 | ||
| 199 | extern struct x86_init_ops x86_init; | 218 | extern struct x86_init_ops x86_init; |
diff --git a/arch/x86/kernel/apic/apic.c b/arch/x86/kernel/apic/apic.c index b994cc84aa7e..a5b4dce1b7ac 100644 --- a/arch/x86/kernel/apic/apic.c +++ b/arch/x86/kernel/apic/apic.c | |||
| @@ -1477,8 +1477,7 @@ void __init bsp_end_local_APIC_setup(void) | |||
| 1477 | * Now that local APIC setup is completed for BP, configure the fault | 1477 | * Now that local APIC setup is completed for BP, configure the fault |
| 1478 | * handling for interrupt remapping. | 1478 | * handling for interrupt remapping. |
| 1479 | */ | 1479 | */ |
| 1480 | if (irq_remapping_enabled) | 1480 | irq_remap_enable_fault_handling(); |
| 1481 | irq_remap_enable_fault_handling(); | ||
| 1482 | 1481 | ||
| 1483 | } | 1482 | } |
| 1484 | 1483 | ||
| @@ -2251,8 +2250,7 @@ static int lapic_suspend(void) | |||
| 2251 | local_irq_save(flags); | 2250 | local_irq_save(flags); |
| 2252 | disable_local_APIC(); | 2251 | disable_local_APIC(); |
| 2253 | 2252 | ||
| 2254 | if (irq_remapping_enabled) | 2253 | irq_remapping_disable(); |
| 2255 | irq_remapping_disable(); | ||
| 2256 | 2254 | ||
| 2257 | local_irq_restore(flags); | 2255 | local_irq_restore(flags); |
| 2258 | return 0; | 2256 | return 0; |
| @@ -2268,16 +2266,15 @@ static void lapic_resume(void) | |||
| 2268 | return; | 2266 | return; |
| 2269 | 2267 | ||
| 2270 | local_irq_save(flags); | 2268 | local_irq_save(flags); |
| 2271 | if (irq_remapping_enabled) { | 2269 | |
| 2272 | /* | 2270 | /* |
| 2273 | * IO-APIC and PIC have their own resume routines. | 2271 | * IO-APIC and PIC have their own resume routines. |
| 2274 | * We just mask them here to make sure the interrupt | 2272 | * We just mask them here to make sure the interrupt |
| 2275 | * subsystem is completely quiet while we enable x2apic | 2273 | * subsystem is completely quiet while we enable x2apic |
| 2276 | * and interrupt-remapping. | 2274 | * and interrupt-remapping. |
| 2277 | */ | 2275 | */ |
| 2278 | mask_ioapic_entries(); | 2276 | mask_ioapic_entries(); |
| 2279 | legacy_pic->mask_all(); | 2277 | legacy_pic->mask_all(); |
| 2280 | } | ||
| 2281 | 2278 | ||
| 2282 | if (x2apic_mode) | 2279 | if (x2apic_mode) |
| 2283 | enable_x2apic(); | 2280 | enable_x2apic(); |
| @@ -2320,8 +2317,7 @@ static void lapic_resume(void) | |||
| 2320 | apic_write(APIC_ESR, 0); | 2317 | apic_write(APIC_ESR, 0); |
| 2321 | apic_read(APIC_ESR); | 2318 | apic_read(APIC_ESR); |
| 2322 | 2319 | ||
| 2323 | if (irq_remapping_enabled) | 2320 | irq_remapping_reenable(x2apic_mode); |
| 2324 | irq_remapping_reenable(x2apic_mode); | ||
| 2325 | 2321 | ||
| 2326 | local_irq_restore(flags); | 2322 | local_irq_restore(flags); |
| 2327 | } | 2323 | } |
diff --git a/arch/x86/kernel/apic/io_apic.c b/arch/x86/kernel/apic/io_apic.c index 2016f9dabd72..9ed796ccc32c 100644 --- a/arch/x86/kernel/apic/io_apic.c +++ b/arch/x86/kernel/apic/io_apic.c | |||
| @@ -68,22 +68,6 @@ | |||
| 68 | #define for_each_irq_pin(entry, head) \ | 68 | #define for_each_irq_pin(entry, head) \ |
| 69 | for (entry = head; entry; entry = entry->next) | 69 | for (entry = head; entry; entry = entry->next) |
| 70 | 70 | ||
| 71 | #ifdef CONFIG_IRQ_REMAP | ||
| 72 | static void irq_remap_modify_chip_defaults(struct irq_chip *chip); | ||
| 73 | static inline bool irq_remapped(struct irq_cfg *cfg) | ||
| 74 | { | ||
| 75 | return cfg->irq_2_iommu.iommu != NULL; | ||
| 76 | } | ||
| 77 | #else | ||
| 78 | static inline bool irq_remapped(struct irq_cfg *cfg) | ||
| 79 | { | ||
| 80 | return false; | ||
| 81 | } | ||
| 82 | static inline void irq_remap_modify_chip_defaults(struct irq_chip *chip) | ||
| 83 | { | ||
| 84 | } | ||
| 85 | #endif | ||
| 86 | |||
| 87 | /* | 71 | /* |
| 88 | * Is the SiS APIC rmw bug present ? | 72 | * Is the SiS APIC rmw bug present ? |
| 89 | * -1 = don't know, 0 = no, 1 = yes | 73 | * -1 = don't know, 0 = no, 1 = yes |
| @@ -326,7 +310,7 @@ static __attribute_const__ struct io_apic __iomem *io_apic_base(int idx) | |||
| 326 | + (mpc_ioapic_addr(idx) & ~PAGE_MASK); | 310 | + (mpc_ioapic_addr(idx) & ~PAGE_MASK); |
| 327 | } | 311 | } |
| 328 | 312 | ||
| 329 | static inline void io_apic_eoi(unsigned int apic, unsigned int vector) | 313 | void io_apic_eoi(unsigned int apic, unsigned int vector) |
| 330 | { | 314 | { |
| 331 | struct io_apic __iomem *io_apic = io_apic_base(apic); | 315 | struct io_apic __iomem *io_apic = io_apic_base(apic); |
| 332 | writel(vector, &io_apic->eoi); | 316 | writel(vector, &io_apic->eoi); |
| @@ -573,19 +557,10 @@ static void unmask_ioapic_irq(struct irq_data *data) | |||
| 573 | * Otherwise, we simulate the EOI message manually by changing the trigger | 557 | * Otherwise, we simulate the EOI message manually by changing the trigger |
| 574 | * mode to edge and then back to level, with RTE being masked during this. | 558 | * mode to edge and then back to level, with RTE being masked during this. |
| 575 | */ | 559 | */ |
| 576 | static void __eoi_ioapic_pin(int apic, int pin, int vector, struct irq_cfg *cfg) | 560 | void native_eoi_ioapic_pin(int apic, int pin, int vector) |
| 577 | { | 561 | { |
| 578 | if (mpc_ioapic_ver(apic) >= 0x20) { | 562 | if (mpc_ioapic_ver(apic) >= 0x20) { |
| 579 | /* | 563 | io_apic_eoi(apic, vector); |
| 580 | * Intr-remapping uses pin number as the virtual vector | ||
| 581 | * in the RTE. Actual vector is programmed in | ||
| 582 | * intr-remapping table entry. Hence for the io-apic | ||
| 583 | * EOI we use the pin number. | ||
| 584 | */ | ||
| 585 | if (cfg && irq_remapped(cfg)) | ||
| 586 | io_apic_eoi(apic, pin); | ||
| 587 | else | ||
| 588 | io_apic_eoi(apic, vector); | ||
| 589 | } else { | 564 | } else { |
| 590 | struct IO_APIC_route_entry entry, entry1; | 565 | struct IO_APIC_route_entry entry, entry1; |
| 591 | 566 | ||
| @@ -606,14 +581,15 @@ static void __eoi_ioapic_pin(int apic, int pin, int vector, struct irq_cfg *cfg) | |||
| 606 | } | 581 | } |
| 607 | } | 582 | } |
| 608 | 583 | ||
| 609 | static void eoi_ioapic_irq(unsigned int irq, struct irq_cfg *cfg) | 584 | void eoi_ioapic_irq(unsigned int irq, struct irq_cfg *cfg) |
| 610 | { | 585 | { |
| 611 | struct irq_pin_list *entry; | 586 | struct irq_pin_list *entry; |
| 612 | unsigned long flags; | 587 | unsigned long flags; |
| 613 | 588 | ||
| 614 | raw_spin_lock_irqsave(&ioapic_lock, flags); | 589 | raw_spin_lock_irqsave(&ioapic_lock, flags); |
| 615 | for_each_irq_pin(entry, cfg->irq_2_pin) | 590 | for_each_irq_pin(entry, cfg->irq_2_pin) |
| 616 | __eoi_ioapic_pin(entry->apic, entry->pin, cfg->vector, cfg); | 591 | x86_io_apic_ops.eoi_ioapic_pin(entry->apic, entry->pin, |
| 592 | cfg->vector); | ||
| 617 | raw_spin_unlock_irqrestore(&ioapic_lock, flags); | 593 | raw_spin_unlock_irqrestore(&ioapic_lock, flags); |
| 618 | } | 594 | } |
| 619 | 595 | ||
| @@ -650,7 +626,7 @@ static void clear_IO_APIC_pin(unsigned int apic, unsigned int pin) | |||
| 650 | } | 626 | } |
| 651 | 627 | ||
| 652 | raw_spin_lock_irqsave(&ioapic_lock, flags); | 628 | raw_spin_lock_irqsave(&ioapic_lock, flags); |
| 653 | __eoi_ioapic_pin(apic, pin, entry.vector, NULL); | 629 | x86_io_apic_ops.eoi_ioapic_pin(apic, pin, entry.vector); |
| 654 | raw_spin_unlock_irqrestore(&ioapic_lock, flags); | 630 | raw_spin_unlock_irqrestore(&ioapic_lock, flags); |
| 655 | } | 631 | } |
| 656 | 632 | ||
| @@ -1304,25 +1280,18 @@ static void ioapic_register_intr(unsigned int irq, struct irq_cfg *cfg, | |||
| 1304 | fasteoi = false; | 1280 | fasteoi = false; |
| 1305 | } | 1281 | } |
| 1306 | 1282 | ||
| 1307 | if (irq_remapped(cfg)) { | 1283 | if (setup_remapped_irq(irq, cfg, chip)) |
| 1308 | irq_set_status_flags(irq, IRQ_MOVE_PCNTXT); | ||
| 1309 | irq_remap_modify_chip_defaults(chip); | ||
| 1310 | fasteoi = trigger != 0; | 1284 | fasteoi = trigger != 0; |
| 1311 | } | ||
| 1312 | 1285 | ||
| 1313 | hdl = fasteoi ? handle_fasteoi_irq : handle_edge_irq; | 1286 | hdl = fasteoi ? handle_fasteoi_irq : handle_edge_irq; |
| 1314 | irq_set_chip_and_handler_name(irq, chip, hdl, | 1287 | irq_set_chip_and_handler_name(irq, chip, hdl, |
| 1315 | fasteoi ? "fasteoi" : "edge"); | 1288 | fasteoi ? "fasteoi" : "edge"); |
| 1316 | } | 1289 | } |
| 1317 | 1290 | ||
| 1318 | static int setup_ioapic_entry(int irq, struct IO_APIC_route_entry *entry, | 1291 | int native_setup_ioapic_entry(int irq, struct IO_APIC_route_entry *entry, |
| 1319 | unsigned int destination, int vector, | 1292 | unsigned int destination, int vector, |
| 1320 | struct io_apic_irq_attr *attr) | 1293 | struct io_apic_irq_attr *attr) |
| 1321 | { | 1294 | { |
| 1322 | if (irq_remapping_enabled) | ||
| 1323 | return setup_ioapic_remapped_entry(irq, entry, destination, | ||
| 1324 | vector, attr); | ||
| 1325 | |||
| 1326 | memset(entry, 0, sizeof(*entry)); | 1295 | memset(entry, 0, sizeof(*entry)); |
| 1327 | 1296 | ||
| 1328 | entry->delivery_mode = apic->irq_delivery_mode; | 1297 | entry->delivery_mode = apic->irq_delivery_mode; |
| @@ -1370,8 +1339,8 @@ static void setup_ioapic_irq(unsigned int irq, struct irq_cfg *cfg, | |||
| 1370 | attr->ioapic, mpc_ioapic_id(attr->ioapic), attr->ioapic_pin, | 1339 | attr->ioapic, mpc_ioapic_id(attr->ioapic), attr->ioapic_pin, |
| 1371 | cfg->vector, irq, attr->trigger, attr->polarity, dest); | 1340 | cfg->vector, irq, attr->trigger, attr->polarity, dest); |
| 1372 | 1341 | ||
| 1373 | if (setup_ioapic_entry(irq, &entry, dest, cfg->vector, attr)) { | 1342 | if (x86_io_apic_ops.setup_entry(irq, &entry, dest, cfg->vector, attr)) { |
| 1374 | pr_warn("Failed to setup ioapic entry for ioapic %d, pin %d\n", | 1343 | pr_warn("Failed to setup ioapic entry for ioapic %d, pin %d\n", |
| 1375 | mpc_ioapic_id(attr->ioapic), attr->ioapic_pin); | 1344 | mpc_ioapic_id(attr->ioapic), attr->ioapic_pin); |
| 1376 | __clear_irq_vector(irq, cfg); | 1345 | __clear_irq_vector(irq, cfg); |
| 1377 | 1346 | ||
| @@ -1479,9 +1448,6 @@ static void __init setup_timer_IRQ0_pin(unsigned int ioapic_idx, | |||
| 1479 | struct IO_APIC_route_entry entry; | 1448 | struct IO_APIC_route_entry entry; |
| 1480 | unsigned int dest; | 1449 | unsigned int dest; |
| 1481 | 1450 | ||
| 1482 | if (irq_remapping_enabled) | ||
| 1483 | return; | ||
| 1484 | |||
| 1485 | memset(&entry, 0, sizeof(entry)); | 1451 | memset(&entry, 0, sizeof(entry)); |
| 1486 | 1452 | ||
| 1487 | /* | 1453 | /* |
| @@ -1513,9 +1479,63 @@ static void __init setup_timer_IRQ0_pin(unsigned int ioapic_idx, | |||
| 1513 | ioapic_write_entry(ioapic_idx, pin, entry); | 1479 | ioapic_write_entry(ioapic_idx, pin, entry); |
| 1514 | } | 1480 | } |
| 1515 | 1481 | ||
| 1516 | __apicdebuginit(void) print_IO_APIC(int ioapic_idx) | 1482 | void native_io_apic_print_entries(unsigned int apic, unsigned int nr_entries) |
| 1517 | { | 1483 | { |
| 1518 | int i; | 1484 | int i; |
| 1485 | |||
| 1486 | pr_debug(" NR Dst Mask Trig IRR Pol Stat Dmod Deli Vect:\n"); | ||
| 1487 | |||
| 1488 | for (i = 0; i <= nr_entries; i++) { | ||
| 1489 | struct IO_APIC_route_entry entry; | ||
| 1490 | |||
| 1491 | entry = ioapic_read_entry(apic, i); | ||
| 1492 | |||
| 1493 | pr_debug(" %02x %02X ", i, entry.dest); | ||
| 1494 | pr_cont("%1d %1d %1d %1d %1d " | ||
| 1495 | "%1d %1d %02X\n", | ||
| 1496 | entry.mask, | ||
| 1497 | entry.trigger, | ||
| 1498 | entry.irr, | ||
| 1499 | entry.polarity, | ||
| 1500 | entry.delivery_status, | ||
| 1501 | entry.dest_mode, | ||
| 1502 | entry.delivery_mode, | ||
| 1503 | entry.vector); | ||
| 1504 | } | ||
| 1505 | } | ||
| 1506 | |||
| 1507 | void intel_ir_io_apic_print_entries(unsigned int apic, | ||
| 1508 | unsigned int nr_entries) | ||
| 1509 | { | ||
| 1510 | int i; | ||
| 1511 | |||
| 1512 | pr_debug(" NR Indx Fmt Mask Trig IRR Pol Stat Indx2 Zero Vect:\n"); | ||
| 1513 | |||
| 1514 | for (i = 0; i <= nr_entries; i++) { | ||
| 1515 | struct IR_IO_APIC_route_entry *ir_entry; | ||
| 1516 | struct IO_APIC_route_entry entry; | ||
| 1517 | |||
| 1518 | entry = ioapic_read_entry(apic, i); | ||
| 1519 | |||
| 1520 | ir_entry = (struct IR_IO_APIC_route_entry *)&entry; | ||
| 1521 | |||
| 1522 | pr_debug(" %02x %04X ", i, ir_entry->index); | ||
| 1523 | pr_cont("%1d %1d %1d %1d %1d " | ||
| 1524 | "%1d %1d %X %02X\n", | ||
| 1525 | ir_entry->format, | ||
| 1526 | ir_entry->mask, | ||
| 1527 | ir_entry->trigger, | ||
| 1528 | ir_entry->irr, | ||
| 1529 | ir_entry->polarity, | ||
| 1530 | ir_entry->delivery_status, | ||
| 1531 | ir_entry->index2, | ||
| 1532 | ir_entry->zero, | ||
| 1533 | ir_entry->vector); | ||
| 1534 | } | ||
| 1535 | } | ||
| 1536 | |||
| 1537 | __apicdebuginit(void) print_IO_APIC(int ioapic_idx) | ||
| 1538 | { | ||
| 1519 | union IO_APIC_reg_00 reg_00; | 1539 | union IO_APIC_reg_00 reg_00; |
| 1520 | union IO_APIC_reg_01 reg_01; | 1540 | union IO_APIC_reg_01 reg_01; |
| 1521 | union IO_APIC_reg_02 reg_02; | 1541 | union IO_APIC_reg_02 reg_02; |
| @@ -1568,58 +1588,7 @@ __apicdebuginit(void) print_IO_APIC(int ioapic_idx) | |||
| 1568 | 1588 | ||
| 1569 | printk(KERN_DEBUG ".... IRQ redirection table:\n"); | 1589 | printk(KERN_DEBUG ".... IRQ redirection table:\n"); |
| 1570 | 1590 | ||
| 1571 | if (irq_remapping_enabled) { | 1591 | x86_io_apic_ops.print_entries(ioapic_idx, reg_01.bits.entries); |
| 1572 | printk(KERN_DEBUG " NR Indx Fmt Mask Trig IRR" | ||
| 1573 | " Pol Stat Indx2 Zero Vect:\n"); | ||
| 1574 | } else { | ||
| 1575 | printk(KERN_DEBUG " NR Dst Mask Trig IRR Pol" | ||
| 1576 | " Stat Dmod Deli Vect:\n"); | ||
| 1577 | } | ||
| 1578 | |||
| 1579 | for (i = 0; i <= reg_01.bits.entries; i++) { | ||
| 1580 | if (irq_remapping_enabled) { | ||
| 1581 | struct IO_APIC_route_entry entry; | ||
| 1582 | struct IR_IO_APIC_route_entry *ir_entry; | ||
| 1583 | |||
| 1584 | entry = ioapic_read_entry(ioapic_idx, i); | ||
| 1585 | ir_entry = (struct IR_IO_APIC_route_entry *) &entry; | ||
| 1586 | printk(KERN_DEBUG " %02x %04X ", | ||
| 1587 | i, | ||
| 1588 | ir_entry->index | ||
| 1589 | ); | ||
| 1590 | pr_cont("%1d %1d %1d %1d %1d " | ||
| 1591 | "%1d %1d %X %02X\n", | ||
| 1592 | ir_entry->format, | ||
| 1593 | ir_entry->mask, | ||
| 1594 | ir_entry->trigger, | ||
| 1595 | ir_entry->irr, | ||
| 1596 | ir_entry->polarity, | ||
| 1597 | ir_entry->delivery_status, | ||
| 1598 | ir_entry->index2, | ||
| 1599 | ir_entry->zero, | ||
| 1600 | ir_entry->vector | ||
| 1601 | ); | ||
| 1602 | } else { | ||
| 1603 | struct IO_APIC_route_entry entry; | ||
| 1604 | |||
| 1605 | entry = ioapic_read_entry(ioapic_idx, i); | ||
| 1606 | printk(KERN_DEBUG " %02x %02X ", | ||
| 1607 | i, | ||
| 1608 | entry.dest | ||
| 1609 | ); | ||
| 1610 | pr_cont("%1d %1d %1d %1d %1d " | ||
| 1611 | "%1d %1d %02X\n", | ||
| 1612 | entry.mask, | ||
| 1613 | entry.trigger, | ||
| 1614 | entry.irr, | ||
| 1615 | entry.polarity, | ||
| 1616 | entry.delivery_status, | ||
| 1617 | entry.dest_mode, | ||
| 1618 | entry.delivery_mode, | ||
| 1619 | entry.vector | ||
| 1620 | ); | ||
| 1621 | } | ||
| 1622 | } | ||
| 1623 | } | 1592 | } |
| 1624 | 1593 | ||
| 1625 | __apicdebuginit(void) print_IO_APICs(void) | 1594 | __apicdebuginit(void) print_IO_APICs(void) |
| @@ -1921,30 +1890,14 @@ void __init enable_IO_APIC(void) | |||
| 1921 | clear_IO_APIC(); | 1890 | clear_IO_APIC(); |
| 1922 | } | 1891 | } |
| 1923 | 1892 | ||
| 1924 | /* | 1893 | void native_disable_io_apic(void) |
| 1925 | * Not an __init, needed by the reboot code | ||
| 1926 | */ | ||
| 1927 | void disable_IO_APIC(void) | ||
| 1928 | { | 1894 | { |
| 1929 | /* | 1895 | /* |
| 1930 | * Clear the IO-APIC before rebooting: | ||
| 1931 | */ | ||
| 1932 | clear_IO_APIC(); | ||
| 1933 | |||
| 1934 | if (!legacy_pic->nr_legacy_irqs) | ||
| 1935 | return; | ||
| 1936 | |||
| 1937 | /* | ||
| 1938 | * If the i8259 is routed through an IOAPIC | 1896 | * If the i8259 is routed through an IOAPIC |
| 1939 | * Put that IOAPIC in virtual wire mode | 1897 | * Put that IOAPIC in virtual wire mode |
| 1940 | * so legacy interrupts can be delivered. | 1898 | * so legacy interrupts can be delivered. |
| 1941 | * | ||
| 1942 | * With interrupt-remapping, for now we will use virtual wire A mode, | ||
| 1943 | * as virtual wire B is little complex (need to configure both | ||
| 1944 | * IOAPIC RTE as well as interrupt-remapping table entry). | ||
| 1945 | * As this gets called during crash dump, keep this simple for now. | ||
| 1946 | */ | 1899 | */ |
| 1947 | if (ioapic_i8259.pin != -1 && !irq_remapping_enabled) { | 1900 | if (ioapic_i8259.pin != -1) { |
| 1948 | struct IO_APIC_route_entry entry; | 1901 | struct IO_APIC_route_entry entry; |
| 1949 | 1902 | ||
| 1950 | memset(&entry, 0, sizeof(entry)); | 1903 | memset(&entry, 0, sizeof(entry)); |
| @@ -1964,12 +1917,25 @@ void disable_IO_APIC(void) | |||
| 1964 | ioapic_write_entry(ioapic_i8259.apic, ioapic_i8259.pin, entry); | 1917 | ioapic_write_entry(ioapic_i8259.apic, ioapic_i8259.pin, entry); |
| 1965 | } | 1918 | } |
| 1966 | 1919 | ||
| 1920 | if (cpu_has_apic || apic_from_smp_config()) | ||
| 1921 | disconnect_bsp_APIC(ioapic_i8259.pin != -1); | ||
| 1922 | |||
| 1923 | } | ||
| 1924 | |||
| 1925 | /* | ||
| 1926 | * Not an __init, needed by the reboot code | ||
| 1927 | */ | ||
| 1928 | void disable_IO_APIC(void) | ||
| 1929 | { | ||
| 1967 | /* | 1930 | /* |
| 1968 | * Use virtual wire A mode when interrupt remapping is enabled. | 1931 | * Clear the IO-APIC before rebooting: |
| 1969 | */ | 1932 | */ |
| 1970 | if (cpu_has_apic || apic_from_smp_config()) | 1933 | clear_IO_APIC(); |
| 1971 | disconnect_bsp_APIC(!irq_remapping_enabled && | 1934 | |
| 1972 | ioapic_i8259.pin != -1); | 1935 | if (!legacy_pic->nr_legacy_irqs) |
| 1936 | return; | ||
| 1937 | |||
| 1938 | x86_io_apic_ops.disable(); | ||
| 1973 | } | 1939 | } |
| 1974 | 1940 | ||
| 1975 | #ifdef CONFIG_X86_32 | 1941 | #ifdef CONFIG_X86_32 |
| @@ -2322,12 +2288,8 @@ static void __target_IO_APIC_irq(unsigned int irq, unsigned int dest, struct irq | |||
| 2322 | 2288 | ||
| 2323 | apic = entry->apic; | 2289 | apic = entry->apic; |
| 2324 | pin = entry->pin; | 2290 | pin = entry->pin; |
| 2325 | /* | 2291 | |
| 2326 | * With interrupt-remapping, destination information comes | 2292 | io_apic_write(apic, 0x11 + pin*2, dest); |
| 2327 | * from interrupt-remapping table entry. | ||
| 2328 | */ | ||
| 2329 | if (!irq_remapped(cfg)) | ||
| 2330 | io_apic_write(apic, 0x11 + pin*2, dest); | ||
| 2331 | reg = io_apic_read(apic, 0x10 + pin*2); | 2293 | reg = io_apic_read(apic, 0x10 + pin*2); |
| 2332 | reg &= ~IO_APIC_REDIR_VECTOR_MASK; | 2294 | reg &= ~IO_APIC_REDIR_VECTOR_MASK; |
| 2333 | reg |= vector; | 2295 | reg |= vector; |
| @@ -2369,9 +2331,10 @@ int __ioapic_set_affinity(struct irq_data *data, const struct cpumask *mask, | |||
| 2369 | return 0; | 2331 | return 0; |
| 2370 | } | 2332 | } |
| 2371 | 2333 | ||
| 2372 | static int | 2334 | |
| 2373 | ioapic_set_affinity(struct irq_data *data, const struct cpumask *mask, | 2335 | int native_ioapic_set_affinity(struct irq_data *data, |
| 2374 | bool force) | 2336 | const struct cpumask *mask, |
| 2337 | bool force) | ||
| 2375 | { | 2338 | { |
| 2376 | unsigned int dest, irq = data->irq; | 2339 | unsigned int dest, irq = data->irq; |
| 2377 | unsigned long flags; | 2340 | unsigned long flags; |
| @@ -2548,33 +2511,6 @@ static void ack_apic_level(struct irq_data *data) | |||
| 2548 | ioapic_irqd_unmask(data, cfg, masked); | 2511 | ioapic_irqd_unmask(data, cfg, masked); |
| 2549 | } | 2512 | } |
| 2550 | 2513 | ||
| 2551 | #ifdef CONFIG_IRQ_REMAP | ||
| 2552 | static void ir_ack_apic_edge(struct irq_data *data) | ||
| 2553 | { | ||
| 2554 | ack_APIC_irq(); | ||
| 2555 | } | ||
| 2556 | |||
| 2557 | static void ir_ack_apic_level(struct irq_data *data) | ||
| 2558 | { | ||
| 2559 | ack_APIC_irq(); | ||
| 2560 | eoi_ioapic_irq(data->irq, data->chip_data); | ||
| 2561 | } | ||
| 2562 | |||
| 2563 | static void ir_print_prefix(struct irq_data *data, struct seq_file *p) | ||
| 2564 | { | ||
| 2565 | seq_printf(p, " IR-%s", data->chip->name); | ||
| 2566 | } | ||
| 2567 | |||
| 2568 | static void irq_remap_modify_chip_defaults(struct irq_chip *chip) | ||
| 2569 | { | ||
| 2570 | chip->irq_print_chip = ir_print_prefix; | ||
| 2571 | chip->irq_ack = ir_ack_apic_edge; | ||
| 2572 | chip->irq_eoi = ir_ack_apic_level; | ||
| 2573 | |||
| 2574 | chip->irq_set_affinity = set_remapped_irq_affinity; | ||
| 2575 | } | ||
| 2576 | #endif /* CONFIG_IRQ_REMAP */ | ||
| 2577 | |||
| 2578 | static struct irq_chip ioapic_chip __read_mostly = { | 2514 | static struct irq_chip ioapic_chip __read_mostly = { |
| 2579 | .name = "IO-APIC", | 2515 | .name = "IO-APIC", |
| 2580 | .irq_startup = startup_ioapic_irq, | 2516 | .irq_startup = startup_ioapic_irq, |
| @@ -2582,7 +2518,7 @@ static struct irq_chip ioapic_chip __read_mostly = { | |||
| 2582 | .irq_unmask = unmask_ioapic_irq, | 2518 | .irq_unmask = unmask_ioapic_irq, |
| 2583 | .irq_ack = ack_apic_edge, | 2519 | .irq_ack = ack_apic_edge, |
| 2584 | .irq_eoi = ack_apic_level, | 2520 | .irq_eoi = ack_apic_level, |
| 2585 | .irq_set_affinity = ioapic_set_affinity, | 2521 | .irq_set_affinity = native_ioapic_set_affinity, |
| 2586 | .irq_retrigger = ioapic_retrigger_irq, | 2522 | .irq_retrigger = ioapic_retrigger_irq, |
| 2587 | }; | 2523 | }; |
| 2588 | 2524 | ||
| @@ -2781,8 +2717,7 @@ static inline void __init check_timer(void) | |||
| 2781 | * 8259A. | 2717 | * 8259A. |
| 2782 | */ | 2718 | */ |
| 2783 | if (pin1 == -1) { | 2719 | if (pin1 == -1) { |
| 2784 | if (irq_remapping_enabled) | 2720 | panic_if_irq_remap("BIOS bug: timer not connected to IO-APIC"); |
| 2785 | panic("BIOS bug: timer not connected to IO-APIC"); | ||
| 2786 | pin1 = pin2; | 2721 | pin1 = pin2; |
| 2787 | apic1 = apic2; | 2722 | apic1 = apic2; |
| 2788 | no_pin1 = 1; | 2723 | no_pin1 = 1; |
| @@ -2814,8 +2749,7 @@ static inline void __init check_timer(void) | |||
| 2814 | clear_IO_APIC_pin(0, pin1); | 2749 | clear_IO_APIC_pin(0, pin1); |
| 2815 | goto out; | 2750 | goto out; |
| 2816 | } | 2751 | } |
| 2817 | if (irq_remapping_enabled) | 2752 | panic_if_irq_remap("timer doesn't work through Interrupt-remapped IO-APIC"); |
| 2818 | panic("timer doesn't work through Interrupt-remapped IO-APIC"); | ||
| 2819 | local_irq_disable(); | 2753 | local_irq_disable(); |
| 2820 | clear_IO_APIC_pin(apic1, pin1); | 2754 | clear_IO_APIC_pin(apic1, pin1); |
| 2821 | if (!no_pin1) | 2755 | if (!no_pin1) |
| @@ -3058,15 +2992,15 @@ void destroy_irq(unsigned int irq) | |||
| 3058 | 2992 | ||
| 3059 | irq_set_status_flags(irq, IRQ_NOREQUEST|IRQ_NOPROBE); | 2993 | irq_set_status_flags(irq, IRQ_NOREQUEST|IRQ_NOPROBE); |
| 3060 | 2994 | ||
| 3061 | if (irq_remapped(cfg)) | 2995 | free_remapped_irq(irq); |
| 3062 | free_remapped_irq(irq); | 2996 | |
| 3063 | raw_spin_lock_irqsave(&vector_lock, flags); | 2997 | raw_spin_lock_irqsave(&vector_lock, flags); |
| 3064 | __clear_irq_vector(irq, cfg); | 2998 | __clear_irq_vector(irq, cfg); |
| 3065 | raw_spin_unlock_irqrestore(&vector_lock, flags); | 2999 | raw_spin_unlock_irqrestore(&vector_lock, flags); |
| 3066 | free_irq_at(irq, cfg); | 3000 | free_irq_at(irq, cfg); |
| 3067 | } | 3001 | } |
| 3068 | 3002 | ||
| 3069 | static inline void destroy_irqs(unsigned int irq, unsigned int count) | 3003 | void destroy_irqs(unsigned int irq, unsigned int count) |
| 3070 | { | 3004 | { |
| 3071 | unsigned int i; | 3005 | unsigned int i; |
| 3072 | 3006 | ||
| @@ -3077,37 +3011,16 @@ static inline void destroy_irqs(unsigned int irq, unsigned int count) | |||
| 3077 | /* | 3011 | /* |
| 3078 | * MSI message composition | 3012 | * MSI message composition |
| 3079 | */ | 3013 | */ |
| 3080 | #ifdef CONFIG_PCI_MSI | 3014 | void native_compose_msi_msg(struct pci_dev *pdev, |
| 3081 | static int msi_compose_msg(struct pci_dev *pdev, unsigned int irq, | 3015 | unsigned int irq, unsigned int dest, |
| 3082 | struct msi_msg *msg, u8 hpet_id) | 3016 | struct msi_msg *msg, u8 hpet_id) |
| 3083 | { | 3017 | { |
| 3084 | struct irq_cfg *cfg; | 3018 | struct irq_cfg *cfg = irq_cfg(irq); |
| 3085 | int err; | ||
| 3086 | unsigned dest; | ||
| 3087 | 3019 | ||
| 3088 | if (disable_apic) | 3020 | msg->address_hi = MSI_ADDR_BASE_HI; |
| 3089 | return -ENXIO; | ||
| 3090 | |||
| 3091 | cfg = irq_cfg(irq); | ||
| 3092 | err = assign_irq_vector(irq, cfg, apic->target_cpus()); | ||
| 3093 | if (err) | ||
| 3094 | return err; | ||
| 3095 | |||
| 3096 | err = apic->cpu_mask_to_apicid_and(cfg->domain, | ||
| 3097 | apic->target_cpus(), &dest); | ||
| 3098 | if (err) | ||
| 3099 | return err; | ||
| 3100 | |||
| 3101 | if (irq_remapped(cfg)) { | ||
| 3102 | compose_remapped_msi_msg(pdev, irq, dest, msg, hpet_id); | ||
| 3103 | return 0; | ||
| 3104 | } | ||
| 3105 | 3021 | ||
| 3106 | if (x2apic_enabled()) | 3022 | if (x2apic_enabled()) |
| 3107 | msg->address_hi = MSI_ADDR_BASE_HI | | 3023 | msg->address_hi |= MSI_ADDR_EXT_DEST_ID(dest); |
| 3108 | MSI_ADDR_EXT_DEST_ID(dest); | ||
| 3109 | else | ||
| 3110 | msg->address_hi = MSI_ADDR_BASE_HI; | ||
| 3111 | 3024 | ||
| 3112 | msg->address_lo = | 3025 | msg->address_lo = |
| 3113 | MSI_ADDR_BASE_LO | | 3026 | MSI_ADDR_BASE_LO | |
| @@ -3126,6 +3039,30 @@ static int msi_compose_msg(struct pci_dev *pdev, unsigned int irq, | |||
| 3126 | MSI_DATA_DELIVERY_FIXED: | 3039 | MSI_DATA_DELIVERY_FIXED: |
| 3127 | MSI_DATA_DELIVERY_LOWPRI) | | 3040 | MSI_DATA_DELIVERY_LOWPRI) | |
| 3128 | MSI_DATA_VECTOR(cfg->vector); | 3041 | MSI_DATA_VECTOR(cfg->vector); |
| 3042 | } | ||
| 3043 | |||
| 3044 | #ifdef CONFIG_PCI_MSI | ||
| 3045 | static int msi_compose_msg(struct pci_dev *pdev, unsigned int irq, | ||
| 3046 | struct msi_msg *msg, u8 hpet_id) | ||
| 3047 | { | ||
| 3048 | struct irq_cfg *cfg; | ||
| 3049 | int err; | ||
| 3050 | unsigned dest; | ||
| 3051 | |||
| 3052 | if (disable_apic) | ||
| 3053 | return -ENXIO; | ||
| 3054 | |||
| 3055 | cfg = irq_cfg(irq); | ||
| 3056 | err = assign_irq_vector(irq, cfg, apic->target_cpus()); | ||
| 3057 | if (err) | ||
| 3058 | return err; | ||
| 3059 | |||
| 3060 | err = apic->cpu_mask_to_apicid_and(cfg->domain, | ||
| 3061 | apic->target_cpus(), &dest); | ||
| 3062 | if (err) | ||
| 3063 | return err; | ||
| 3064 | |||
| 3065 | x86_msi.compose_msi_msg(pdev, irq, dest, msg, hpet_id); | ||
| 3129 | 3066 | ||
| 3130 | return 0; | 3067 | return 0; |
| 3131 | } | 3068 | } |
| @@ -3165,8 +3102,8 @@ static struct irq_chip msi_chip = { | |||
| 3165 | .irq_retrigger = ioapic_retrigger_irq, | 3102 | .irq_retrigger = ioapic_retrigger_irq, |
| 3166 | }; | 3103 | }; |
| 3167 | 3104 | ||
| 3168 | static int setup_msi_irq(struct pci_dev *dev, struct msi_desc *msidesc, | 3105 | int setup_msi_irq(struct pci_dev *dev, struct msi_desc *msidesc, |
| 3169 | unsigned int irq_base, unsigned int irq_offset) | 3106 | unsigned int irq_base, unsigned int irq_offset) |
| 3170 | { | 3107 | { |
| 3171 | struct irq_chip *chip = &msi_chip; | 3108 | struct irq_chip *chip = &msi_chip; |
| 3172 | struct msi_msg msg; | 3109 | struct msi_msg msg; |
| @@ -3186,10 +3123,7 @@ static int setup_msi_irq(struct pci_dev *dev, struct msi_desc *msidesc, | |||
| 3186 | if (!irq_offset) | 3123 | if (!irq_offset) |
| 3187 | write_msi_msg(irq, &msg); | 3124 | write_msi_msg(irq, &msg); |
| 3188 | 3125 | ||
| 3189 | if (irq_remapped(irq_get_chip_data(irq))) { | 3126 | setup_remapped_irq(irq, irq_get_chip_data(irq), chip); |
| 3190 | irq_set_status_flags(irq, IRQ_MOVE_PCNTXT); | ||
| 3191 | irq_remap_modify_chip_defaults(chip); | ||
| 3192 | } | ||
| 3193 | 3127 | ||
| 3194 | irq_set_chip_and_handler_name(irq, chip, handle_edge_irq, "edge"); | 3128 | irq_set_chip_and_handler_name(irq, chip, handle_edge_irq, "edge"); |
| 3195 | 3129 | ||
| @@ -3198,44 +3132,28 @@ static int setup_msi_irq(struct pci_dev *dev, struct msi_desc *msidesc, | |||
| 3198 | return 0; | 3132 | return 0; |
| 3199 | } | 3133 | } |
| 3200 | 3134 | ||
| 3201 | int setup_msix_irqs(struct pci_dev *dev, int nvec) | 3135 | int native_setup_msi_irqs(struct pci_dev *dev, int nvec, int type) |
| 3202 | { | 3136 | { |
| 3203 | int node, ret, sub_handle, index = 0; | ||
| 3204 | unsigned int irq, irq_want; | 3137 | unsigned int irq, irq_want; |
| 3205 | struct msi_desc *msidesc; | 3138 | struct msi_desc *msidesc; |
| 3139 | int node, ret; | ||
| 3140 | |||
| 3141 | /* Multiple MSI vectors only supported with interrupt remapping */ | ||
| 3142 | if (type == PCI_CAP_ID_MSI && nvec > 1) | ||
| 3143 | return 1; | ||
| 3206 | 3144 | ||
| 3207 | node = dev_to_node(&dev->dev); | 3145 | node = dev_to_node(&dev->dev); |
| 3208 | irq_want = nr_irqs_gsi; | 3146 | irq_want = nr_irqs_gsi; |
| 3209 | sub_handle = 0; | ||
| 3210 | list_for_each_entry(msidesc, &dev->msi_list, list) { | 3147 | list_for_each_entry(msidesc, &dev->msi_list, list) { |
| 3211 | irq = create_irq_nr(irq_want, node); | 3148 | irq = create_irq_nr(irq_want, node); |
| 3212 | if (irq == 0) | 3149 | if (irq == 0) |
| 3213 | return -ENOSPC; | 3150 | return -ENOSPC; |
| 3151 | |||
| 3214 | irq_want = irq + 1; | 3152 | irq_want = irq + 1; |
| 3215 | if (!irq_remapping_enabled) | ||
| 3216 | goto no_ir; | ||
| 3217 | 3153 | ||
| 3218 | if (!sub_handle) { | ||
| 3219 | /* | ||
| 3220 | * allocate the consecutive block of IRTE's | ||
| 3221 | * for 'nvec' | ||
| 3222 | */ | ||
| 3223 | index = msi_alloc_remapped_irq(dev, irq, nvec); | ||
| 3224 | if (index < 0) { | ||
| 3225 | ret = index; | ||
| 3226 | goto error; | ||
| 3227 | } | ||
| 3228 | } else { | ||
| 3229 | ret = msi_setup_remapped_irq(dev, irq, index, | ||
| 3230 | sub_handle); | ||
| 3231 | if (ret < 0) | ||
| 3232 | goto error; | ||
| 3233 | } | ||
| 3234 | no_ir: | ||
| 3235 | ret = setup_msi_irq(dev, msidesc, irq, 0); | 3154 | ret = setup_msi_irq(dev, msidesc, irq, 0); |
| 3236 | if (ret < 0) | 3155 | if (ret < 0) |
| 3237 | goto error; | 3156 | goto error; |
| 3238 | sub_handle++; | ||
| 3239 | } | 3157 | } |
| 3240 | return 0; | 3158 | return 0; |
| 3241 | 3159 | ||
| @@ -3244,74 +3162,6 @@ error: | |||
| 3244 | return ret; | 3162 | return ret; |
| 3245 | } | 3163 | } |
| 3246 | 3164 | ||
| 3247 | int setup_msi_irqs(struct pci_dev *dev, int nvec) | ||
| 3248 | { | ||
| 3249 | int node, ret, sub_handle, index = 0; | ||
| 3250 | unsigned int irq; | ||
| 3251 | struct msi_desc *msidesc; | ||
| 3252 | |||
| 3253 | if (nvec > 1 && !irq_remapping_enabled) | ||
| 3254 | return 1; | ||
| 3255 | |||
| 3256 | nvec = __roundup_pow_of_two(nvec); | ||
| 3257 | |||
| 3258 | WARN_ON(!list_is_singular(&dev->msi_list)); | ||
| 3259 | msidesc = list_entry(dev->msi_list.next, struct msi_desc, list); | ||
| 3260 | WARN_ON(msidesc->irq); | ||
| 3261 | WARN_ON(msidesc->msi_attrib.multiple); | ||
| 3262 | |||
| 3263 | node = dev_to_node(&dev->dev); | ||
| 3264 | irq = __create_irqs(nr_irqs_gsi, nvec, node); | ||
| 3265 | if (irq == 0) | ||
| 3266 | return -ENOSPC; | ||
| 3267 | |||
| 3268 | if (!irq_remapping_enabled) { | ||
| 3269 | ret = setup_msi_irq(dev, msidesc, irq, 0); | ||
| 3270 | if (ret < 0) | ||
| 3271 | goto error; | ||
| 3272 | return 0; | ||
| 3273 | } | ||
| 3274 | |||
| 3275 | msidesc->msi_attrib.multiple = ilog2(nvec); | ||
| 3276 | for (sub_handle = 0; sub_handle < nvec; sub_handle++) { | ||
| 3277 | if (!sub_handle) { | ||
| 3278 | index = msi_alloc_remapped_irq(dev, irq, nvec); | ||
| 3279 | if (index < 0) { | ||
| 3280 | ret = index; | ||
| 3281 | goto error; | ||
| 3282 | } | ||
| 3283 | } else { | ||
| 3284 | ret = msi_setup_remapped_irq(dev, irq + sub_handle, | ||
| 3285 | index, sub_handle); | ||
| 3286 | if (ret < 0) | ||
| 3287 | goto error; | ||
| 3288 | } | ||
| 3289 | ret = setup_msi_irq(dev, msidesc, irq, sub_handle); | ||
| 3290 | if (ret < 0) | ||
| 3291 | goto error; | ||
| 3292 | } | ||
| 3293 | return 0; | ||
| 3294 | |||
| 3295 | error: | ||
| 3296 | destroy_irqs(irq, nvec); | ||
| 3297 | |||
| 3298 | /* | ||
| 3299 | * Restore altered MSI descriptor fields and prevent just destroyed | ||
| 3300 | * IRQs from tearing down again in default_teardown_msi_irqs() | ||
| 3301 | */ | ||
| 3302 | msidesc->irq = 0; | ||
| 3303 | msidesc->msi_attrib.multiple = 0; | ||
| 3304 | |||
| 3305 | return ret; | ||
| 3306 | } | ||
| 3307 | |||
| 3308 | int native_setup_msi_irqs(struct pci_dev *dev, int nvec, int type) | ||
| 3309 | { | ||
| 3310 | if (type == PCI_CAP_ID_MSI) | ||
| 3311 | return setup_msi_irqs(dev, nvec); | ||
| 3312 | return setup_msix_irqs(dev, nvec); | ||
| 3313 | } | ||
| 3314 | |||
| 3315 | void native_teardown_msi_irq(unsigned int irq) | 3165 | void native_teardown_msi_irq(unsigned int irq) |
| 3316 | { | 3166 | { |
| 3317 | destroy_irq(irq); | 3167 | destroy_irq(irq); |
| @@ -3399,26 +3249,19 @@ static struct irq_chip hpet_msi_type = { | |||
| 3399 | .irq_retrigger = ioapic_retrigger_irq, | 3249 | .irq_retrigger = ioapic_retrigger_irq, |
| 3400 | }; | 3250 | }; |
| 3401 | 3251 | ||
| 3402 | int arch_setup_hpet_msi(unsigned int irq, unsigned int id) | 3252 | int default_setup_hpet_msi(unsigned int irq, unsigned int id) |
| 3403 | { | 3253 | { |
| 3404 | struct irq_chip *chip = &hpet_msi_type; | 3254 | struct irq_chip *chip = &hpet_msi_type; |
| 3405 | struct msi_msg msg; | 3255 | struct msi_msg msg; |
| 3406 | int ret; | 3256 | int ret; |
| 3407 | 3257 | ||
| 3408 | if (irq_remapping_enabled) { | ||
| 3409 | ret = setup_hpet_msi_remapped(irq, id); | ||
| 3410 | if (ret) | ||
| 3411 | return ret; | ||
| 3412 | } | ||
| 3413 | |||
| 3414 | ret = msi_compose_msg(NULL, irq, &msg, id); | 3258 | ret = msi_compose_msg(NULL, irq, &msg, id); |
| 3415 | if (ret < 0) | 3259 | if (ret < 0) |
| 3416 | return ret; | 3260 | return ret; |
| 3417 | 3261 | ||
| 3418 | hpet_msi_write(irq_get_handler_data(irq), &msg); | 3262 | hpet_msi_write(irq_get_handler_data(irq), &msg); |
| 3419 | irq_set_status_flags(irq, IRQ_MOVE_PCNTXT); | 3263 | irq_set_status_flags(irq, IRQ_MOVE_PCNTXT); |
| 3420 | if (irq_remapped(irq_get_chip_data(irq))) | 3264 | setup_remapped_irq(irq, irq_get_chip_data(irq), chip); |
| 3421 | irq_remap_modify_chip_defaults(chip); | ||
| 3422 | 3265 | ||
| 3423 | irq_set_chip_and_handler_name(irq, chip, handle_edge_irq, "edge"); | 3266 | irq_set_chip_and_handler_name(irq, chip, handle_edge_irq, "edge"); |
| 3424 | return 0; | 3267 | return 0; |
| @@ -3784,10 +3627,7 @@ void __init setup_ioapic_dest(void) | |||
| 3784 | else | 3627 | else |
| 3785 | mask = apic->target_cpus(); | 3628 | mask = apic->target_cpus(); |
| 3786 | 3629 | ||
| 3787 | if (irq_remapping_enabled) | 3630 | x86_io_apic_ops.set_affinity(idata, mask, false); |
| 3788 | set_remapped_irq_affinity(idata, mask, false); | ||
| 3789 | else | ||
| 3790 | ioapic_set_affinity(idata, mask, false); | ||
| 3791 | } | 3631 | } |
| 3792 | 3632 | ||
| 3793 | } | 3633 | } |
diff --git a/arch/x86/kernel/hpet.c b/arch/x86/kernel/hpet.c index e28670f9a589..da85a8e830a1 100644 --- a/arch/x86/kernel/hpet.c +++ b/arch/x86/kernel/hpet.c | |||
| @@ -478,7 +478,7 @@ static int hpet_msi_next_event(unsigned long delta, | |||
| 478 | 478 | ||
| 479 | static int hpet_setup_msi_irq(unsigned int irq) | 479 | static int hpet_setup_msi_irq(unsigned int irq) |
| 480 | { | 480 | { |
| 481 | if (arch_setup_hpet_msi(irq, hpet_blockid)) { | 481 | if (x86_msi.setup_hpet_msi(irq, hpet_blockid)) { |
| 482 | destroy_irq(irq); | 482 | destroy_irq(irq); |
| 483 | return -EINVAL; | 483 | return -EINVAL; |
| 484 | } | 484 | } |
diff --git a/arch/x86/kernel/x86_init.c b/arch/x86/kernel/x86_init.c index 7a3d075a814a..d065d67c2672 100644 --- a/arch/x86/kernel/x86_init.c +++ b/arch/x86/kernel/x86_init.c | |||
| @@ -19,6 +19,7 @@ | |||
| 19 | #include <asm/time.h> | 19 | #include <asm/time.h> |
| 20 | #include <asm/irq.h> | 20 | #include <asm/irq.h> |
| 21 | #include <asm/io_apic.h> | 21 | #include <asm/io_apic.h> |
| 22 | #include <asm/hpet.h> | ||
| 22 | #include <asm/pat.h> | 23 | #include <asm/pat.h> |
| 23 | #include <asm/tsc.h> | 24 | #include <asm/tsc.h> |
| 24 | #include <asm/iommu.h> | 25 | #include <asm/iommu.h> |
| @@ -111,15 +112,22 @@ struct x86_platform_ops x86_platform = { | |||
| 111 | 112 | ||
| 112 | EXPORT_SYMBOL_GPL(x86_platform); | 113 | EXPORT_SYMBOL_GPL(x86_platform); |
| 113 | struct x86_msi_ops x86_msi = { | 114 | struct x86_msi_ops x86_msi = { |
| 114 | .setup_msi_irqs = native_setup_msi_irqs, | 115 | .setup_msi_irqs = native_setup_msi_irqs, |
| 115 | .teardown_msi_irq = native_teardown_msi_irq, | 116 | .compose_msi_msg = native_compose_msi_msg, |
| 116 | .teardown_msi_irqs = default_teardown_msi_irqs, | 117 | .teardown_msi_irq = native_teardown_msi_irq, |
| 117 | .restore_msi_irqs = default_restore_msi_irqs, | 118 | .teardown_msi_irqs = default_teardown_msi_irqs, |
| 119 | .restore_msi_irqs = default_restore_msi_irqs, | ||
| 120 | .setup_hpet_msi = default_setup_hpet_msi, | ||
| 118 | }; | 121 | }; |
| 119 | 122 | ||
| 120 | struct x86_io_apic_ops x86_io_apic_ops = { | 123 | struct x86_io_apic_ops x86_io_apic_ops = { |
| 121 | .init = native_io_apic_init_mappings, | 124 | .init = native_io_apic_init_mappings, |
| 122 | .read = native_io_apic_read, | 125 | .read = native_io_apic_read, |
| 123 | .write = native_io_apic_write, | 126 | .write = native_io_apic_write, |
| 124 | .modify = native_io_apic_modify, | 127 | .modify = native_io_apic_modify, |
| 128 | .disable = native_disable_io_apic, | ||
| 129 | .print_entries = native_io_apic_print_entries, | ||
| 130 | .set_affinity = native_ioapic_set_affinity, | ||
| 131 | .setup_entry = native_setup_ioapic_entry, | ||
| 132 | .eoi_ioapic_pin = native_eoi_ioapic_pin, | ||
| 125 | }; | 133 | }; |
diff --git a/drivers/iommu/amd_iommu.c b/drivers/iommu/amd_iommu.c index c1c74e030a58..d33eaaf783ad 100644 --- a/drivers/iommu/amd_iommu.c +++ b/drivers/iommu/amd_iommu.c | |||
| @@ -4017,10 +4017,10 @@ static int alloc_irq_index(struct irq_cfg *cfg, u16 devid, int count) | |||
| 4017 | 4017 | ||
| 4018 | index -= count - 1; | 4018 | index -= count - 1; |
| 4019 | 4019 | ||
| 4020 | cfg->remapped = 1; | ||
| 4020 | irte_info = &cfg->irq_2_iommu; | 4021 | irte_info = &cfg->irq_2_iommu; |
| 4021 | irte_info->sub_handle = devid; | 4022 | irte_info->sub_handle = devid; |
| 4022 | irte_info->irte_index = index; | 4023 | irte_info->irte_index = index; |
| 4023 | irte_info->iommu = (void *)cfg; | ||
| 4024 | 4024 | ||
| 4025 | goto out; | 4025 | goto out; |
| 4026 | } | 4026 | } |
| @@ -4127,9 +4127,9 @@ static int setup_ioapic_entry(int irq, struct IO_APIC_route_entry *entry, | |||
| 4127 | index = attr->ioapic_pin; | 4127 | index = attr->ioapic_pin; |
| 4128 | 4128 | ||
| 4129 | /* Setup IRQ remapping info */ | 4129 | /* Setup IRQ remapping info */ |
| 4130 | cfg->remapped = 1; | ||
| 4130 | irte_info->sub_handle = devid; | 4131 | irte_info->sub_handle = devid; |
| 4131 | irte_info->irte_index = index; | 4132 | irte_info->irte_index = index; |
| 4132 | irte_info->iommu = (void *)cfg; | ||
| 4133 | 4133 | ||
| 4134 | /* Setup IRTE for IOMMU */ | 4134 | /* Setup IRTE for IOMMU */ |
| 4135 | irte.val = 0; | 4135 | irte.val = 0; |
| @@ -4288,9 +4288,9 @@ static int msi_setup_irq(struct pci_dev *pdev, unsigned int irq, | |||
| 4288 | devid = get_device_id(&pdev->dev); | 4288 | devid = get_device_id(&pdev->dev); |
| 4289 | irte_info = &cfg->irq_2_iommu; | 4289 | irte_info = &cfg->irq_2_iommu; |
| 4290 | 4290 | ||
| 4291 | cfg->remapped = 1; | ||
| 4291 | irte_info->sub_handle = devid; | 4292 | irte_info->sub_handle = devid; |
| 4292 | irte_info->irte_index = index + offset; | 4293 | irte_info->irte_index = index + offset; |
| 4293 | irte_info->iommu = (void *)cfg; | ||
| 4294 | 4294 | ||
| 4295 | return 0; | 4295 | return 0; |
| 4296 | } | 4296 | } |
| @@ -4314,9 +4314,9 @@ static int setup_hpet_msi(unsigned int irq, unsigned int id) | |||
| 4314 | if (index < 0) | 4314 | if (index < 0) |
| 4315 | return index; | 4315 | return index; |
| 4316 | 4316 | ||
| 4317 | cfg->remapped = 1; | ||
| 4317 | irte_info->sub_handle = devid; | 4318 | irte_info->sub_handle = devid; |
| 4318 | irte_info->irte_index = index; | 4319 | irte_info->irte_index = index; |
| 4319 | irte_info->iommu = (void *)cfg; | ||
| 4320 | 4320 | ||
| 4321 | return 0; | 4321 | return 0; |
| 4322 | } | 4322 | } |
diff --git a/drivers/iommu/dmar.c b/drivers/iommu/dmar.c index 86e2f4a62b9a..174bb654453d 100644 --- a/drivers/iommu/dmar.c +++ b/drivers/iommu/dmar.c | |||
| @@ -41,6 +41,8 @@ | |||
| 41 | #include <asm/irq_remapping.h> | 41 | #include <asm/irq_remapping.h> |
| 42 | #include <asm/iommu_table.h> | 42 | #include <asm/iommu_table.h> |
| 43 | 43 | ||
| 44 | #include "irq_remapping.h" | ||
| 45 | |||
| 44 | /* No locks are needed as DMA remapping hardware unit | 46 | /* No locks are needed as DMA remapping hardware unit |
| 45 | * list is constructed at boot time and hotplug of | 47 | * list is constructed at boot time and hotplug of |
| 46 | * these units are not supported by the architecture. | 48 | * these units are not supported by the architecture. |
diff --git a/drivers/iommu/intel-iommu.c b/drivers/iommu/intel-iommu.c index b9d091157884..64ae948a3802 100644 --- a/drivers/iommu/intel-iommu.c +++ b/drivers/iommu/intel-iommu.c | |||
| @@ -46,6 +46,8 @@ | |||
| 46 | #include <asm/cacheflush.h> | 46 | #include <asm/cacheflush.h> |
| 47 | #include <asm/iommu.h> | 47 | #include <asm/iommu.h> |
| 48 | 48 | ||
| 49 | #include "irq_remapping.h" | ||
| 50 | |||
| 49 | #define ROOT_SIZE VTD_PAGE_SIZE | 51 | #define ROOT_SIZE VTD_PAGE_SIZE |
| 50 | #define CONTEXT_SIZE VTD_PAGE_SIZE | 52 | #define CONTEXT_SIZE VTD_PAGE_SIZE |
| 51 | 53 | ||
diff --git a/drivers/iommu/intel_irq_remapping.c b/drivers/iommu/intel_irq_remapping.c index af8904de1d44..7ca4947c3c10 100644 --- a/drivers/iommu/intel_irq_remapping.c +++ b/drivers/iommu/intel_irq_remapping.c | |||
| @@ -68,6 +68,7 @@ static int alloc_irte(struct intel_iommu *iommu, int irq, u16 count) | |||
| 68 | { | 68 | { |
| 69 | struct ir_table *table = iommu->ir_table; | 69 | struct ir_table *table = iommu->ir_table; |
| 70 | struct irq_2_iommu *irq_iommu = irq_2_iommu(irq); | 70 | struct irq_2_iommu *irq_iommu = irq_2_iommu(irq); |
| 71 | struct irq_cfg *cfg = irq_get_chip_data(irq); | ||
| 71 | u16 index, start_index; | 72 | u16 index, start_index; |
| 72 | unsigned int mask = 0; | 73 | unsigned int mask = 0; |
| 73 | unsigned long flags; | 74 | unsigned long flags; |
| @@ -115,6 +116,7 @@ static int alloc_irte(struct intel_iommu *iommu, int irq, u16 count) | |||
| 115 | for (i = index; i < index + count; i++) | 116 | for (i = index; i < index + count; i++) |
| 116 | table->base[i].present = 1; | 117 | table->base[i].present = 1; |
| 117 | 118 | ||
| 119 | cfg->remapped = 1; | ||
| 118 | irq_iommu->iommu = iommu; | 120 | irq_iommu->iommu = iommu; |
| 119 | irq_iommu->irte_index = index; | 121 | irq_iommu->irte_index = index; |
| 120 | irq_iommu->sub_handle = 0; | 122 | irq_iommu->sub_handle = 0; |
| @@ -155,6 +157,7 @@ static int map_irq_to_irte_handle(int irq, u16 *sub_handle) | |||
| 155 | static int set_irte_irq(int irq, struct intel_iommu *iommu, u16 index, u16 subhandle) | 157 | static int set_irte_irq(int irq, struct intel_iommu *iommu, u16 index, u16 subhandle) |
| 156 | { | 158 | { |
| 157 | struct irq_2_iommu *irq_iommu = irq_2_iommu(irq); | 159 | struct irq_2_iommu *irq_iommu = irq_2_iommu(irq); |
| 160 | struct irq_cfg *cfg = irq_get_chip_data(irq); | ||
| 158 | unsigned long flags; | 161 | unsigned long flags; |
| 159 | 162 | ||
| 160 | if (!irq_iommu) | 163 | if (!irq_iommu) |
| @@ -162,6 +165,7 @@ static int set_irte_irq(int irq, struct intel_iommu *iommu, u16 index, u16 subha | |||
| 162 | 165 | ||
| 163 | raw_spin_lock_irqsave(&irq_2_ir_lock, flags); | 166 | raw_spin_lock_irqsave(&irq_2_ir_lock, flags); |
| 164 | 167 | ||
| 168 | cfg->remapped = 1; | ||
| 165 | irq_iommu->iommu = iommu; | 169 | irq_iommu->iommu = iommu; |
| 166 | irq_iommu->irte_index = index; | 170 | irq_iommu->irte_index = index; |
| 167 | irq_iommu->sub_handle = subhandle; | 171 | irq_iommu->sub_handle = subhandle; |
| @@ -617,6 +621,14 @@ static int __init intel_enable_irq_remapping(void) | |||
| 617 | goto error; | 621 | goto error; |
| 618 | 622 | ||
| 619 | irq_remapping_enabled = 1; | 623 | irq_remapping_enabled = 1; |
| 624 | |||
| 625 | /* | ||
| 626 | * VT-d has a different layout for IO-APIC entries when | ||
| 627 | * interrupt remapping is enabled. So it needs a special routine | ||
| 628 | * to print IO-APIC entries for debugging purposes too. | ||
| 629 | */ | ||
| 630 | x86_io_apic_ops.print_entries = intel_ir_io_apic_print_entries; | ||
| 631 | |||
| 620 | pr_info("Enabled IRQ remapping in %s mode\n", eim ? "x2apic" : "xapic"); | 632 | pr_info("Enabled IRQ remapping in %s mode\n", eim ? "x2apic" : "xapic"); |
| 621 | 633 | ||
| 622 | return eim ? IRQ_REMAP_X2APIC_MODE : IRQ_REMAP_XAPIC_MODE; | 634 | return eim ? IRQ_REMAP_X2APIC_MODE : IRQ_REMAP_XAPIC_MODE; |
diff --git a/drivers/iommu/irq_remapping.c b/drivers/iommu/irq_remapping.c index faf85d6e33fe..d56f8c17c5fe 100644 --- a/drivers/iommu/irq_remapping.c +++ b/drivers/iommu/irq_remapping.c | |||
| @@ -1,11 +1,18 @@ | |||
| 1 | #include <linux/seq_file.h> | ||
| 2 | #include <linux/cpumask.h> | ||
| 1 | #include <linux/kernel.h> | 3 | #include <linux/kernel.h> |
| 2 | #include <linux/string.h> | 4 | #include <linux/string.h> |
| 3 | #include <linux/cpumask.h> | 5 | #include <linux/cpumask.h> |
| 4 | #include <linux/errno.h> | 6 | #include <linux/errno.h> |
| 5 | #include <linux/msi.h> | 7 | #include <linux/msi.h> |
| 8 | #include <linux/irq.h> | ||
| 9 | #include <linux/pci.h> | ||
| 6 | 10 | ||
| 7 | #include <asm/hw_irq.h> | 11 | #include <asm/hw_irq.h> |
| 8 | #include <asm/irq_remapping.h> | 12 | #include <asm/irq_remapping.h> |
| 13 | #include <asm/processor.h> | ||
| 14 | #include <asm/x86_init.h> | ||
| 15 | #include <asm/apic.h> | ||
| 9 | 16 | ||
| 10 | #include "irq_remapping.h" | 17 | #include "irq_remapping.h" |
| 11 | 18 | ||
| @@ -17,6 +24,152 @@ int no_x2apic_optout; | |||
| 17 | 24 | ||
| 18 | static struct irq_remap_ops *remap_ops; | 25 | static struct irq_remap_ops *remap_ops; |
| 19 | 26 | ||
| 27 | static int msi_alloc_remapped_irq(struct pci_dev *pdev, int irq, int nvec); | ||
| 28 | static int msi_setup_remapped_irq(struct pci_dev *pdev, unsigned int irq, | ||
| 29 | int index, int sub_handle); | ||
| 30 | static int set_remapped_irq_affinity(struct irq_data *data, | ||
| 31 | const struct cpumask *mask, | ||
| 32 | bool force); | ||
| 33 | |||
| 34 | static bool irq_remapped(struct irq_cfg *cfg) | ||
| 35 | { | ||
| 36 | return (cfg->remapped == 1); | ||
| 37 | } | ||
| 38 | |||
| 39 | static void irq_remapping_disable_io_apic(void) | ||
| 40 | { | ||
| 41 | /* | ||
| 42 | * With interrupt-remapping, for now we will use virtual wire A | ||
| 43 | * mode, as virtual wire B is little complex (need to configure | ||
| 44 | * both IOAPIC RTE as well as interrupt-remapping table entry). | ||
| 45 | * As this gets called during crash dump, keep this simple for | ||
| 46 | * now. | ||
| 47 | */ | ||
| 48 | if (cpu_has_apic || apic_from_smp_config()) | ||
| 49 | disconnect_bsp_APIC(0); | ||
| 50 | } | ||
| 51 | |||
| 52 | static int do_setup_msi_irqs(struct pci_dev *dev, int nvec) | ||
| 53 | { | ||
| 54 | int node, ret, sub_handle, index = 0; | ||
| 55 | unsigned int irq; | ||
| 56 | struct msi_desc *msidesc; | ||
| 57 | |||
| 58 | nvec = __roundup_pow_of_two(nvec); | ||
| 59 | |||
| 60 | WARN_ON(!list_is_singular(&dev->msi_list)); | ||
| 61 | msidesc = list_entry(dev->msi_list.next, struct msi_desc, list); | ||
| 62 | WARN_ON(msidesc->irq); | ||
| 63 | WARN_ON(msidesc->msi_attrib.multiple); | ||
| 64 | |||
| 65 | node = dev_to_node(&dev->dev); | ||
| 66 | irq = __create_irqs(get_nr_irqs_gsi(), nvec, node); | ||
| 67 | if (irq == 0) | ||
| 68 | return -ENOSPC; | ||
| 69 | |||
| 70 | msidesc->msi_attrib.multiple = ilog2(nvec); | ||
| 71 | for (sub_handle = 0; sub_handle < nvec; sub_handle++) { | ||
| 72 | if (!sub_handle) { | ||
| 73 | index = msi_alloc_remapped_irq(dev, irq, nvec); | ||
| 74 | if (index < 0) { | ||
| 75 | ret = index; | ||
| 76 | goto error; | ||
| 77 | } | ||
| 78 | } else { | ||
| 79 | ret = msi_setup_remapped_irq(dev, irq + sub_handle, | ||
| 80 | index, sub_handle); | ||
| 81 | if (ret < 0) | ||
| 82 | goto error; | ||
| 83 | } | ||
| 84 | ret = setup_msi_irq(dev, msidesc, irq, sub_handle); | ||
| 85 | if (ret < 0) | ||
| 86 | goto error; | ||
| 87 | } | ||
| 88 | return 0; | ||
| 89 | |||
| 90 | error: | ||
| 91 | destroy_irqs(irq, nvec); | ||
| 92 | |||
| 93 | /* | ||
| 94 | * Restore altered MSI descriptor fields and prevent just destroyed | ||
| 95 | * IRQs from tearing down again in default_teardown_msi_irqs() | ||
| 96 | */ | ||
| 97 | msidesc->irq = 0; | ||
| 98 | msidesc->msi_attrib.multiple = 0; | ||
| 99 | |||
| 100 | return ret; | ||
| 101 | } | ||
| 102 | |||
| 103 | static int do_setup_msix_irqs(struct pci_dev *dev, int nvec) | ||
| 104 | { | ||
| 105 | int node, ret, sub_handle, index = 0; | ||
| 106 | struct msi_desc *msidesc; | ||
| 107 | unsigned int irq; | ||
| 108 | |||
| 109 | node = dev_to_node(&dev->dev); | ||
| 110 | irq = get_nr_irqs_gsi(); | ||
| 111 | sub_handle = 0; | ||
| 112 | |||
| 113 | list_for_each_entry(msidesc, &dev->msi_list, list) { | ||
| 114 | |||
| 115 | irq = create_irq_nr(irq, node); | ||
| 116 | if (irq == 0) | ||
| 117 | return -1; | ||
| 118 | |||
| 119 | if (sub_handle == 0) | ||
| 120 | ret = index = msi_alloc_remapped_irq(dev, irq, nvec); | ||
| 121 | else | ||
| 122 | ret = msi_setup_remapped_irq(dev, irq, index, sub_handle); | ||
| 123 | |||
| 124 | if (ret < 0) | ||
| 125 | goto error; | ||
| 126 | |||
| 127 | ret = setup_msi_irq(dev, msidesc, irq, 0); | ||
| 128 | if (ret < 0) | ||
| 129 | goto error; | ||
| 130 | |||
| 131 | sub_handle += 1; | ||
| 132 | irq += 1; | ||
| 133 | } | ||
| 134 | |||
| 135 | return 0; | ||
| 136 | |||
| 137 | error: | ||
| 138 | destroy_irq(irq); | ||
| 139 | return ret; | ||
| 140 | } | ||
| 141 | |||
| 142 | static int irq_remapping_setup_msi_irqs(struct pci_dev *dev, | ||
| 143 | int nvec, int type) | ||
| 144 | { | ||
| 145 | if (type == PCI_CAP_ID_MSI) | ||
| 146 | return do_setup_msi_irqs(dev, nvec); | ||
| 147 | else | ||
| 148 | return do_setup_msix_irqs(dev, nvec); | ||
| 149 | } | ||
| 150 | |||
| 151 | void eoi_ioapic_pin_remapped(int apic, int pin, int vector) | ||
| 152 | { | ||
| 153 | /* | ||
| 154 | * Intr-remapping uses pin number as the virtual vector | ||
| 155 | * in the RTE. Actual vector is programmed in | ||
| 156 | * intr-remapping table entry. Hence for the io-apic | ||
| 157 | * EOI we use the pin number. | ||
| 158 | */ | ||
| 159 | io_apic_eoi(apic, pin); | ||
| 160 | } | ||
| 161 | |||
| 162 | static void __init irq_remapping_modify_x86_ops(void) | ||
| 163 | { | ||
| 164 | x86_io_apic_ops.disable = irq_remapping_disable_io_apic; | ||
| 165 | x86_io_apic_ops.set_affinity = set_remapped_irq_affinity; | ||
| 166 | x86_io_apic_ops.setup_entry = setup_ioapic_remapped_entry; | ||
| 167 | x86_io_apic_ops.eoi_ioapic_pin = eoi_ioapic_pin_remapped; | ||
| 168 | x86_msi.setup_msi_irqs = irq_remapping_setup_msi_irqs; | ||
| 169 | x86_msi.setup_hpet_msi = setup_hpet_msi_remapped; | ||
| 170 | x86_msi.compose_msi_msg = compose_remapped_msi_msg; | ||
| 171 | } | ||
| 172 | |||
| 20 | static __init int setup_nointremap(char *str) | 173 | static __init int setup_nointremap(char *str) |
| 21 | { | 174 | { |
| 22 | disable_irq_remap = 1; | 175 | disable_irq_remap = 1; |
| @@ -79,15 +232,24 @@ int __init irq_remapping_prepare(void) | |||
| 79 | 232 | ||
| 80 | int __init irq_remapping_enable(void) | 233 | int __init irq_remapping_enable(void) |
| 81 | { | 234 | { |
| 235 | int ret; | ||
| 236 | |||
| 82 | if (!remap_ops || !remap_ops->enable) | 237 | if (!remap_ops || !remap_ops->enable) |
| 83 | return -ENODEV; | 238 | return -ENODEV; |
| 84 | 239 | ||
| 85 | return remap_ops->enable(); | 240 | ret = remap_ops->enable(); |
| 241 | |||
| 242 | if (irq_remapping_enabled) | ||
| 243 | irq_remapping_modify_x86_ops(); | ||
| 244 | |||
| 245 | return ret; | ||
| 86 | } | 246 | } |
| 87 | 247 | ||
| 88 | void irq_remapping_disable(void) | 248 | void irq_remapping_disable(void) |
| 89 | { | 249 | { |
| 90 | if (!remap_ops || !remap_ops->disable) | 250 | if (!irq_remapping_enabled || |
| 251 | !remap_ops || | ||
| 252 | !remap_ops->disable) | ||
| 91 | return; | 253 | return; |
| 92 | 254 | ||
| 93 | remap_ops->disable(); | 255 | remap_ops->disable(); |
| @@ -95,7 +257,9 @@ void irq_remapping_disable(void) | |||
| 95 | 257 | ||
| 96 | int irq_remapping_reenable(int mode) | 258 | int irq_remapping_reenable(int mode) |
| 97 | { | 259 | { |
| 98 | if (!remap_ops || !remap_ops->reenable) | 260 | if (!irq_remapping_enabled || |
| 261 | !remap_ops || | ||
| 262 | !remap_ops->reenable) | ||
| 99 | return 0; | 263 | return 0; |
| 100 | 264 | ||
| 101 | return remap_ops->reenable(mode); | 265 | return remap_ops->reenable(mode); |
| @@ -103,6 +267,9 @@ int irq_remapping_reenable(int mode) | |||
| 103 | 267 | ||
| 104 | int __init irq_remap_enable_fault_handling(void) | 268 | int __init irq_remap_enable_fault_handling(void) |
| 105 | { | 269 | { |
| 270 | if (!irq_remapping_enabled) | ||
| 271 | return 0; | ||
| 272 | |||
| 106 | if (!remap_ops || !remap_ops->enable_faulting) | 273 | if (!remap_ops || !remap_ops->enable_faulting) |
| 107 | return -ENODEV; | 274 | return -ENODEV; |
| 108 | 275 | ||
| @@ -133,23 +300,28 @@ int set_remapped_irq_affinity(struct irq_data *data, const struct cpumask *mask, | |||
| 133 | 300 | ||
| 134 | void free_remapped_irq(int irq) | 301 | void free_remapped_irq(int irq) |
| 135 | { | 302 | { |
| 303 | struct irq_cfg *cfg = irq_get_chip_data(irq); | ||
| 304 | |||
| 136 | if (!remap_ops || !remap_ops->free_irq) | 305 | if (!remap_ops || !remap_ops->free_irq) |
| 137 | return; | 306 | return; |
| 138 | 307 | ||
| 139 | remap_ops->free_irq(irq); | 308 | if (irq_remapped(cfg)) |
| 309 | remap_ops->free_irq(irq); | ||
| 140 | } | 310 | } |
| 141 | 311 | ||
| 142 | void compose_remapped_msi_msg(struct pci_dev *pdev, | 312 | void compose_remapped_msi_msg(struct pci_dev *pdev, |
| 143 | unsigned int irq, unsigned int dest, | 313 | unsigned int irq, unsigned int dest, |
| 144 | struct msi_msg *msg, u8 hpet_id) | 314 | struct msi_msg *msg, u8 hpet_id) |
| 145 | { | 315 | { |
| 146 | if (!remap_ops || !remap_ops->compose_msi_msg) | 316 | struct irq_cfg *cfg = irq_get_chip_data(irq); |
| 147 | return; | ||
| 148 | 317 | ||
| 149 | remap_ops->compose_msi_msg(pdev, irq, dest, msg, hpet_id); | 318 | if (!irq_remapped(cfg)) |
| 319 | native_compose_msi_msg(pdev, irq, dest, msg, hpet_id); | ||
| 320 | else if (remap_ops && remap_ops->compose_msi_msg) | ||
| 321 | remap_ops->compose_msi_msg(pdev, irq, dest, msg, hpet_id); | ||
| 150 | } | 322 | } |
| 151 | 323 | ||
| 152 | int msi_alloc_remapped_irq(struct pci_dev *pdev, int irq, int nvec) | 324 | static int msi_alloc_remapped_irq(struct pci_dev *pdev, int irq, int nvec) |
| 153 | { | 325 | { |
| 154 | if (!remap_ops || !remap_ops->msi_alloc_irq) | 326 | if (!remap_ops || !remap_ops->msi_alloc_irq) |
| 155 | return -ENODEV; | 327 | return -ENODEV; |
| @@ -157,8 +329,8 @@ int msi_alloc_remapped_irq(struct pci_dev *pdev, int irq, int nvec) | |||
| 157 | return remap_ops->msi_alloc_irq(pdev, irq, nvec); | 329 | return remap_ops->msi_alloc_irq(pdev, irq, nvec); |
| 158 | } | 330 | } |
| 159 | 331 | ||
| 160 | int msi_setup_remapped_irq(struct pci_dev *pdev, unsigned int irq, | 332 | static int msi_setup_remapped_irq(struct pci_dev *pdev, unsigned int irq, |
| 161 | int index, int sub_handle) | 333 | int index, int sub_handle) |
| 162 | { | 334 | { |
| 163 | if (!remap_ops || !remap_ops->msi_setup_irq) | 335 | if (!remap_ops || !remap_ops->msi_setup_irq) |
| 164 | return -ENODEV; | 336 | return -ENODEV; |
| @@ -173,3 +345,42 @@ int setup_hpet_msi_remapped(unsigned int irq, unsigned int id) | |||
| 173 | 345 | ||
| 174 | return remap_ops->setup_hpet_msi(irq, id); | 346 | return remap_ops->setup_hpet_msi(irq, id); |
| 175 | } | 347 | } |
| 348 | |||
| 349 | void panic_if_irq_remap(const char *msg) | ||
| 350 | { | ||
| 351 | if (irq_remapping_enabled) | ||
| 352 | panic(msg); | ||
| 353 | } | ||
| 354 | |||
| 355 | static void ir_ack_apic_edge(struct irq_data *data) | ||
| 356 | { | ||
| 357 | ack_APIC_irq(); | ||
| 358 | } | ||
| 359 | |||
| 360 | static void ir_ack_apic_level(struct irq_data *data) | ||
| 361 | { | ||
| 362 | ack_APIC_irq(); | ||
| 363 | eoi_ioapic_irq(data->irq, data->chip_data); | ||
| 364 | } | ||
| 365 | |||
| 366 | static void ir_print_prefix(struct irq_data *data, struct seq_file *p) | ||
| 367 | { | ||
| 368 | seq_printf(p, " IR-%s", data->chip->name); | ||
| 369 | } | ||
| 370 | |||
| 371 | void irq_remap_modify_chip_defaults(struct irq_chip *chip) | ||
| 372 | { | ||
| 373 | chip->irq_print_chip = ir_print_prefix; | ||
| 374 | chip->irq_ack = ir_ack_apic_edge; | ||
| 375 | chip->irq_eoi = ir_ack_apic_level; | ||
| 376 | chip->irq_set_affinity = x86_io_apic_ops.set_affinity; | ||
| 377 | } | ||
| 378 | |||
| 379 | bool setup_remapped_irq(int irq, struct irq_cfg *cfg, struct irq_chip *chip) | ||
| 380 | { | ||
| 381 | if (!irq_remapped(cfg)) | ||
| 382 | return false; | ||
| 383 | irq_set_status_flags(irq, IRQ_MOVE_PCNTXT); | ||
| 384 | irq_remap_modify_chip_defaults(chip); | ||
| 385 | return true; | ||
| 386 | } | ||
diff --git a/drivers/iommu/irq_remapping.h b/drivers/iommu/irq_remapping.h index 95363acb583f..ecb637670405 100644 --- a/drivers/iommu/irq_remapping.h +++ b/drivers/iommu/irq_remapping.h | |||
| @@ -34,6 +34,7 @@ struct msi_msg; | |||
| 34 | extern int disable_irq_remap; | 34 | extern int disable_irq_remap; |
| 35 | extern int disable_sourceid_checking; | 35 | extern int disable_sourceid_checking; |
| 36 | extern int no_x2apic_optout; | 36 | extern int no_x2apic_optout; |
| 37 | extern int irq_remapping_enabled; | ||
| 37 | 38 | ||
| 38 | struct irq_remap_ops { | 39 | struct irq_remap_ops { |
| 39 | /* Check whether Interrupt Remapping is supported */ | 40 | /* Check whether Interrupt Remapping is supported */ |
diff --git a/include/linux/irq.h b/include/linux/irq.h index 1eab99111e94..bc4e06611958 100644 --- a/include/linux/irq.h +++ b/include/linux/irq.h | |||
| @@ -509,8 +509,11 @@ static inline void irq_set_percpu_devid_flags(unsigned int irq) | |||
| 509 | 509 | ||
| 510 | /* Handle dynamic irq creation and destruction */ | 510 | /* Handle dynamic irq creation and destruction */ |
| 511 | extern unsigned int create_irq_nr(unsigned int irq_want, int node); | 511 | extern unsigned int create_irq_nr(unsigned int irq_want, int node); |
| 512 | extern unsigned int __create_irqs(unsigned int from, unsigned int count, | ||
| 513 | int node); | ||
| 512 | extern int create_irq(void); | 514 | extern int create_irq(void); |
| 513 | extern void destroy_irq(unsigned int irq); | 515 | extern void destroy_irq(unsigned int irq); |
| 516 | extern void destroy_irqs(unsigned int irq, unsigned int count); | ||
| 514 | 517 | ||
| 515 | /* | 518 | /* |
| 516 | * Dynamic irq helper functions. Obsolete. Use irq_alloc_desc* and | 519 | * Dynamic irq helper functions. Obsolete. Use irq_alloc_desc* and |
