diff options
Diffstat (limited to 'arch/ia64')
-rw-r--r-- | arch/ia64/kernel/iosapic.c | 60 |
1 files changed, 24 insertions, 36 deletions
diff --git a/arch/ia64/kernel/iosapic.c b/arch/ia64/kernel/iosapic.c index 522b13d0bde3..29ec86c24194 100644 --- a/arch/ia64/kernel/iosapic.c +++ b/arch/ia64/kernel/iosapic.c | |||
@@ -117,16 +117,22 @@ static DEFINE_SPINLOCK(iosapic_lock); | |||
117 | * These tables map IA-64 vectors to the IOSAPIC pin that generates this | 117 | * These tables map IA-64 vectors to the IOSAPIC pin that generates this |
118 | * vector. | 118 | * vector. |
119 | */ | 119 | */ |
120 | static struct iosapic { | ||
121 | char __iomem *addr; /* base address of IOSAPIC */ | ||
122 | unsigned int gsi_base; /* GSI base */ | ||
123 | unsigned short num_rte; /* # of RTEs on this IOSAPIC */ | ||
124 | int rtes_inuse; /* # of RTEs in use on this IOSAPIC */ | ||
125 | #ifdef CONFIG_NUMA | ||
126 | unsigned short node; /* numa node association via pxm */ | ||
127 | #endif | ||
128 | } iosapic_lists[NR_IOSAPICS]; | ||
120 | 129 | ||
121 | struct iosapic_rte_info { | 130 | struct iosapic_rte_info { |
122 | struct list_head rte_list; /* node in list of RTEs sharing the | 131 | struct list_head rte_list; /* RTEs sharing the same vector */ |
123 | * same vector */ | ||
124 | char __iomem *addr; /* base address of IOSAPIC */ | ||
125 | unsigned int gsi_base; /* first GSI assigned to this | ||
126 | * IOSAPIC */ | ||
127 | char rte_index; /* IOSAPIC RTE index */ | 132 | char rte_index; /* IOSAPIC RTE index */ |
128 | int refcnt; /* reference counter */ | 133 | int refcnt; /* reference counter */ |
129 | unsigned int flags; /* flags */ | 134 | unsigned int flags; /* flags */ |
135 | struct iosapic *iosapic; | ||
130 | } ____cacheline_aligned; | 136 | } ____cacheline_aligned; |
131 | 137 | ||
132 | static struct iosapic_intr_info { | 138 | static struct iosapic_intr_info { |
@@ -142,17 +148,6 @@ static struct iosapic_intr_info { | |||
142 | unsigned char trigger : 1; /* trigger mode (see iosapic.h) */ | 148 | unsigned char trigger : 1; /* trigger mode (see iosapic.h) */ |
143 | } iosapic_intr_info[IA64_NUM_VECTORS]; | 149 | } iosapic_intr_info[IA64_NUM_VECTORS]; |
144 | 150 | ||
145 | static struct iosapic { | ||
146 | char __iomem *addr; /* base address of IOSAPIC */ | ||
147 | unsigned int gsi_base; /* first GSI assigned to this | ||
148 | * IOSAPIC */ | ||
149 | unsigned short num_rte; /* # of RTEs on this IOSAPIC */ | ||
150 | int rtes_inuse; /* # of RTEs in use on this IOSAPIC */ | ||
151 | #ifdef CONFIG_NUMA | ||
152 | unsigned short node; /* numa node association via pxm */ | ||
153 | #endif | ||
154 | } iosapic_lists[NR_IOSAPICS]; | ||
155 | |||
156 | static unsigned char pcat_compat __devinitdata; /* 8259 compatibility flag */ | 151 | static unsigned char pcat_compat __devinitdata; /* 8259 compatibility flag */ |
157 | 152 | ||
158 | static int iosapic_kmalloc_ok; | 153 | static int iosapic_kmalloc_ok; |
@@ -184,7 +179,7 @@ _gsi_to_vector (unsigned int gsi) | |||
184 | for (info = iosapic_intr_info; info < | 179 | for (info = iosapic_intr_info; info < |
185 | iosapic_intr_info + IA64_NUM_VECTORS; ++info) | 180 | iosapic_intr_info + IA64_NUM_VECTORS; ++info) |
186 | list_for_each_entry(rte, &info->rtes, rte_list) | 181 | list_for_each_entry(rte, &info->rtes, rte_list) |
187 | if (rte->gsi_base + rte->rte_index == gsi) | 182 | if (rte->iosapic->gsi_base + rte->rte_index == gsi) |
188 | return info - iosapic_intr_info; | 183 | return info - iosapic_intr_info; |
189 | return -1; | 184 | return -1; |
190 | } | 185 | } |
@@ -221,7 +216,7 @@ static struct iosapic_rte_info *gsi_vector_to_rte(unsigned int gsi, | |||
221 | struct iosapic_rte_info *rte; | 216 | struct iosapic_rte_info *rte; |
222 | 217 | ||
223 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) | 218 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) |
224 | if (rte->gsi_base + rte->rte_index == gsi) | 219 | if (rte->iosapic->gsi_base + rte->rte_index == gsi) |
225 | return rte; | 220 | return rte; |
226 | return NULL; | 221 | return NULL; |
227 | } | 222 | } |
@@ -243,7 +238,7 @@ set_rte (unsigned int gsi, unsigned int vector, unsigned int dest, int mask) | |||
243 | return; /* not an IOSAPIC interrupt */ | 238 | return; /* not an IOSAPIC interrupt */ |
244 | 239 | ||
245 | rte_index = rte->rte_index; | 240 | rte_index = rte->rte_index; |
246 | addr = rte->addr; | 241 | addr = rte->iosapic->addr; |
247 | pol = iosapic_intr_info[vector].polarity; | 242 | pol = iosapic_intr_info[vector].polarity; |
248 | trigger = iosapic_intr_info[vector].trigger; | 243 | trigger = iosapic_intr_info[vector].trigger; |
249 | dmode = iosapic_intr_info[vector].dmode; | 244 | dmode = iosapic_intr_info[vector].dmode; |
@@ -297,10 +292,10 @@ kexec_disable_iosapic(void) | |||
297 | iosapic_intr_info + IA64_NUM_VECTORS; ++info, ++vec) { | 292 | iosapic_intr_info + IA64_NUM_VECTORS; ++info, ++vec) { |
298 | list_for_each_entry(rte, &info->rtes, | 293 | list_for_each_entry(rte, &info->rtes, |
299 | rte_list) { | 294 | rte_list) { |
300 | iosapic_write(rte->addr, | 295 | iosapic_write(rte->iosapic->addr, |
301 | IOSAPIC_RTE_LOW(rte->rte_index), | 296 | IOSAPIC_RTE_LOW(rte->rte_index), |
302 | IOSAPIC_MASK|vec); | 297 | IOSAPIC_MASK|vec); |
303 | iosapic_eoi(rte->addr, vec); | 298 | iosapic_eoi(rte->iosapic->addr, vec); |
304 | } | 299 | } |
305 | } | 300 | } |
306 | } | 301 | } |
@@ -323,7 +318,7 @@ mask_irq (unsigned int irq) | |||
323 | /* set only the mask bit */ | 318 | /* set only the mask bit */ |
324 | low32 = iosapic_intr_info[vec].low32 |= IOSAPIC_MASK; | 319 | low32 = iosapic_intr_info[vec].low32 |= IOSAPIC_MASK; |
325 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) { | 320 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) { |
326 | addr = rte->addr; | 321 | addr = rte->iosapic->addr; |
327 | rte_index = rte->rte_index; | 322 | rte_index = rte->rte_index; |
328 | iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32); | 323 | iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32); |
329 | } | 324 | } |
@@ -346,7 +341,7 @@ unmask_irq (unsigned int irq) | |||
346 | spin_lock_irqsave(&iosapic_lock, flags); | 341 | spin_lock_irqsave(&iosapic_lock, flags); |
347 | low32 = iosapic_intr_info[vec].low32 &= ~IOSAPIC_MASK; | 342 | low32 = iosapic_intr_info[vec].low32 &= ~IOSAPIC_MASK; |
348 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) { | 343 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) { |
349 | addr = rte->addr; | 344 | addr = rte->iosapic->addr; |
350 | rte_index = rte->rte_index; | 345 | rte_index = rte->rte_index; |
351 | iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32); | 346 | iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32); |
352 | } | 347 | } |
@@ -394,7 +389,7 @@ iosapic_set_affinity (unsigned int irq, cpumask_t mask) | |||
394 | iosapic_intr_info[vec].low32 = low32; | 389 | iosapic_intr_info[vec].low32 = low32; |
395 | iosapic_intr_info[vec].dest = dest; | 390 | iosapic_intr_info[vec].dest = dest; |
396 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) { | 391 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) { |
397 | addr = rte->addr; | 392 | addr = rte->iosapic->addr; |
398 | rte_index = rte->rte_index; | 393 | rte_index = rte->rte_index; |
399 | iosapic_write(addr, IOSAPIC_RTE_HIGH(rte_index), high32); | 394 | iosapic_write(addr, IOSAPIC_RTE_HIGH(rte_index), high32); |
400 | iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32); | 395 | iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32); |
@@ -422,7 +417,7 @@ iosapic_end_level_irq (unsigned int irq) | |||
422 | 417 | ||
423 | move_native_irq(irq); | 418 | move_native_irq(irq); |
424 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) | 419 | list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) |
425 | iosapic_eoi(rte->addr, vec); | 420 | iosapic_eoi(rte->iosapic->addr, vec); |
426 | } | 421 | } |
427 | 422 | ||
428 | #define iosapic_shutdown_level_irq mask_irq | 423 | #define iosapic_shutdown_level_irq mask_irq |
@@ -614,10 +609,7 @@ register_intr (unsigned int gsi, int vector, unsigned char delivery, | |||
614 | { | 609 | { |
615 | irq_desc_t *idesc; | 610 | irq_desc_t *idesc; |
616 | struct hw_interrupt_type *irq_type; | 611 | struct hw_interrupt_type *irq_type; |
617 | int rte_index; | ||
618 | int index; | 612 | int index; |
619 | unsigned long gsi_base; | ||
620 | void __iomem *iosapic_address; | ||
621 | struct iosapic_rte_info *rte; | 613 | struct iosapic_rte_info *rte; |
622 | 614 | ||
623 | index = find_iosapic(gsi); | 615 | index = find_iosapic(gsi); |
@@ -627,9 +619,6 @@ register_intr (unsigned int gsi, int vector, unsigned char delivery, | |||
627 | return -ENODEV; | 619 | return -ENODEV; |
628 | } | 620 | } |
629 | 621 | ||
630 | iosapic_address = iosapic_lists[index].addr; | ||
631 | gsi_base = iosapic_lists[index].gsi_base; | ||
632 | |||
633 | rte = gsi_vector_to_rte(gsi, vector); | 622 | rte = gsi_vector_to_rte(gsi, vector); |
634 | if (!rte) { | 623 | if (!rte) { |
635 | rte = iosapic_alloc_rte(); | 624 | rte = iosapic_alloc_rte(); |
@@ -639,10 +628,8 @@ register_intr (unsigned int gsi, int vector, unsigned char delivery, | |||
639 | return -ENOMEM; | 628 | return -ENOMEM; |
640 | } | 629 | } |
641 | 630 | ||
642 | rte_index = gsi - gsi_base; | 631 | rte->iosapic = &iosapic_lists[index]; |
643 | rte->rte_index = rte_index; | 632 | rte->rte_index = gsi - rte->iosapic->gsi_base; |
644 | rte->addr = iosapic_address; | ||
645 | rte->gsi_base = gsi_base; | ||
646 | rte->refcnt++; | 633 | rte->refcnt++; |
647 | list_add_tail(&rte->rte_list, &iosapic_intr_info[vector].rtes); | 634 | list_add_tail(&rte->rte_list, &iosapic_intr_info[vector].rtes); |
648 | iosapic_intr_info[vector].count++; | 635 | iosapic_intr_info[vector].count++; |
@@ -877,7 +864,8 @@ iosapic_unregister_intr (unsigned int gsi) | |||
877 | 864 | ||
878 | /* Mask the interrupt */ | 865 | /* Mask the interrupt */ |
879 | low32 = iosapic_intr_info[vector].low32 | IOSAPIC_MASK; | 866 | low32 = iosapic_intr_info[vector].low32 | IOSAPIC_MASK; |
880 | iosapic_write(rte->addr, IOSAPIC_RTE_LOW(rte->rte_index), low32); | 867 | iosapic_write(rte->iosapic->addr, |
868 | IOSAPIC_RTE_LOW(rte->rte_index), low32); | ||
881 | 869 | ||
882 | /* Remove the rte entry from the list */ | 870 | /* Remove the rte entry from the list */ |
883 | list_del(&rte->rte_list); | 871 | list_del(&rte->rte_list); |