aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/common/gic.c
diff options
context:
space:
mode:
authorMarc Zyngier <marc.zyngier@arm.com>2011-11-12 11:09:49 -0500
committerMarc Zyngier <marc.zyngier@arm.com>2011-11-15 13:13:03 -0500
commitdb0d4db22a78d31c59087f7057b8f1612fecc35d (patch)
treee2d84ee2c816d4bef066806863a3937792a08581 /arch/arm/common/gic.c
parentcfcfc9eca2bcbd26a8e206baeb005b055dbf8e37 (diff)
ARM: gic: allow GIC to support non-banked setups
The GIC support code is heavily using the fact that hardware implementations are exposing banked registers. Unfortunately, it looks like at least one GIC implementation (EXYNOS) offers both the distributor and the CPU interfaces at different addresses, depending on the CPU. This problem is solved by allowing the distributor and CPU interface addresses to be per-cpu variables for the platforms that require it. The EXYNOS code is updated not to mess with the GIC internals while handling interrupts, and struct gic_chip_data is back to being private. The DT binding for the gic is updated to allow an optional "cpu-offset" value, which is used to compute the various base addresses. Finally, a new config option (GIC_NON_BANKED) is used to control this feature, so the overhead is only present on kernels compiled with support for EXYNOS. Tested on Origen (EXYNOS4) and Panda (OMAP4). Cc: Kukjin Kim <kgene.kim@samsung.com> Cc: Will Deacon <will.deacon@arm.com> Cc: Thomas Abraham <thomas.abraham@linaro.org> Acked-by: Rob Herring <rob.herring@calxeda.com> Signed-off-by: Marc Zyngier <marc.zyngier@arm.com>
Diffstat (limited to 'arch/arm/common/gic.c')
-rw-r--r--arch/arm/common/gic.c133
1 files changed, 114 insertions, 19 deletions
diff --git a/arch/arm/common/gic.c b/arch/arm/common/gic.c
index 0e6ae470c94f..43cb6f1a7cf2 100644
--- a/arch/arm/common/gic.c
+++ b/arch/arm/common/gic.c
@@ -43,6 +43,31 @@
43#include <asm/mach/irq.h> 43#include <asm/mach/irq.h>
44#include <asm/hardware/gic.h> 44#include <asm/hardware/gic.h>
45 45
46union gic_base {
47 void __iomem *common_base;
48 void __percpu __iomem **percpu_base;
49};
50
51struct gic_chip_data {
52 unsigned int irq_offset;
53 union gic_base dist_base;
54 union gic_base cpu_base;
55#ifdef CONFIG_CPU_PM
56 u32 saved_spi_enable[DIV_ROUND_UP(1020, 32)];
57 u32 saved_spi_conf[DIV_ROUND_UP(1020, 16)];
58 u32 saved_spi_target[DIV_ROUND_UP(1020, 4)];
59 u32 __percpu *saved_ppi_enable;
60 u32 __percpu *saved_ppi_conf;
61#endif
62#ifdef CONFIG_IRQ_DOMAIN
63 struct irq_domain domain;
64#endif
65 unsigned int gic_irqs;
66#ifdef CONFIG_GIC_NON_BANKED
67 void __iomem *(*get_base)(union gic_base *);
68#endif
69};
70
46static DEFINE_RAW_SPINLOCK(irq_controller_lock); 71static DEFINE_RAW_SPINLOCK(irq_controller_lock);
47 72
48/* Address of GIC 0 CPU interface */ 73/* Address of GIC 0 CPU interface */
@@ -67,16 +92,48 @@ struct irq_chip gic_arch_extn = {
67 92
68static struct gic_chip_data gic_data[MAX_GIC_NR] __read_mostly; 93static struct gic_chip_data gic_data[MAX_GIC_NR] __read_mostly;
69 94
95#ifdef CONFIG_GIC_NON_BANKED
96static void __iomem *gic_get_percpu_base(union gic_base *base)
97{
98 return *__this_cpu_ptr(base->percpu_base);
99}
100
101static void __iomem *gic_get_common_base(union gic_base *base)
102{
103 return base->common_base;
104}
105
106static inline void __iomem *gic_data_dist_base(struct gic_chip_data *data)
107{
108 return data->get_base(&data->dist_base);
109}
110
111static inline void __iomem *gic_data_cpu_base(struct gic_chip_data *data)
112{
113 return data->get_base(&data->cpu_base);
114}
115
116static inline void gic_set_base_accessor(struct gic_chip_data *data,
117 void __iomem *(*f)(union gic_base *))
118{
119 data->get_base = f;
120}
121#else
122#define gic_data_dist_base(d) ((d)->dist_base.common_base)
123#define gic_data_cpu_base(d) ((d)->cpu_base.common_base)
124#define gic_set_base_accessor(d,f)
125#endif
126
70static inline void __iomem *gic_dist_base(struct irq_data *d) 127static inline void __iomem *gic_dist_base(struct irq_data *d)
71{ 128{
72 struct gic_chip_data *gic_data = irq_data_get_irq_chip_data(d); 129 struct gic_chip_data *gic_data = irq_data_get_irq_chip_data(d);
73 return gic_data->dist_base; 130 return gic_data_dist_base(gic_data);
74} 131}
75 132
76static inline void __iomem *gic_cpu_base(struct irq_data *d) 133static inline void __iomem *gic_cpu_base(struct irq_data *d)
77{ 134{
78 struct gic_chip_data *gic_data = irq_data_get_irq_chip_data(d); 135 struct gic_chip_data *gic_data = irq_data_get_irq_chip_data(d);
79 return gic_data->cpu_base; 136 return gic_data_cpu_base(gic_data);
80} 137}
81 138
82static inline unsigned int gic_irq(struct irq_data *d) 139static inline unsigned int gic_irq(struct irq_data *d)
@@ -225,7 +282,7 @@ static void gic_handle_cascade_irq(unsigned int irq, struct irq_desc *desc)
225 chained_irq_enter(chip, desc); 282 chained_irq_enter(chip, desc);
226 283
227 raw_spin_lock(&irq_controller_lock); 284 raw_spin_lock(&irq_controller_lock);
228 status = readl_relaxed(chip_data->cpu_base + GIC_CPU_INTACK); 285 status = readl_relaxed(gic_data_cpu_base(chip_data) + GIC_CPU_INTACK);
229 raw_spin_unlock(&irq_controller_lock); 286 raw_spin_unlock(&irq_controller_lock);
230 287
231 gic_irq = (status & 0x3ff); 288 gic_irq = (status & 0x3ff);
@@ -270,7 +327,7 @@ static void __init gic_dist_init(struct gic_chip_data *gic)
270 u32 cpumask; 327 u32 cpumask;
271 unsigned int gic_irqs = gic->gic_irqs; 328 unsigned int gic_irqs = gic->gic_irqs;
272 struct irq_domain *domain = &gic->domain; 329 struct irq_domain *domain = &gic->domain;
273 void __iomem *base = gic->dist_base; 330 void __iomem *base = gic_data_dist_base(gic);
274 u32 cpu = 0; 331 u32 cpu = 0;
275 332
276#ifdef CONFIG_SMP 333#ifdef CONFIG_SMP
@@ -330,8 +387,8 @@ static void __init gic_dist_init(struct gic_chip_data *gic)
330 387
331static void __cpuinit gic_cpu_init(struct gic_chip_data *gic) 388static void __cpuinit gic_cpu_init(struct gic_chip_data *gic)
332{ 389{
333 void __iomem *dist_base = gic->dist_base; 390 void __iomem *dist_base = gic_data_dist_base(gic);
334 void __iomem *base = gic->cpu_base; 391 void __iomem *base = gic_data_cpu_base(gic);
335 int i; 392 int i;
336 393
337 /* 394 /*
@@ -368,7 +425,7 @@ static void gic_dist_save(unsigned int gic_nr)
368 BUG(); 425 BUG();
369 426
370 gic_irqs = gic_data[gic_nr].gic_irqs; 427 gic_irqs = gic_data[gic_nr].gic_irqs;
371 dist_base = gic_data[gic_nr].dist_base; 428 dist_base = gic_data_dist_base(&gic_data[gic_nr]);
372 429
373 if (!dist_base) 430 if (!dist_base)
374 return; 431 return;
@@ -403,7 +460,7 @@ static void gic_dist_restore(unsigned int gic_nr)
403 BUG(); 460 BUG();
404 461
405 gic_irqs = gic_data[gic_nr].gic_irqs; 462 gic_irqs = gic_data[gic_nr].gic_irqs;
406 dist_base = gic_data[gic_nr].dist_base; 463 dist_base = gic_data_dist_base(&gic_data[gic_nr]);
407 464
408 if (!dist_base) 465 if (!dist_base)
409 return; 466 return;
@@ -439,8 +496,8 @@ static void gic_cpu_save(unsigned int gic_nr)
439 if (gic_nr >= MAX_GIC_NR) 496 if (gic_nr >= MAX_GIC_NR)
440 BUG(); 497 BUG();
441 498
442 dist_base = gic_data[gic_nr].dist_base; 499 dist_base = gic_data_dist_base(&gic_data[gic_nr]);
443 cpu_base = gic_data[gic_nr].cpu_base; 500 cpu_base = gic_data_cpu_base(&gic_data[gic_nr]);
444 501
445 if (!dist_base || !cpu_base) 502 if (!dist_base || !cpu_base)
446 return; 503 return;
@@ -465,8 +522,8 @@ static void gic_cpu_restore(unsigned int gic_nr)
465 if (gic_nr >= MAX_GIC_NR) 522 if (gic_nr >= MAX_GIC_NR)
466 BUG(); 523 BUG();
467 524
468 dist_base = gic_data[gic_nr].dist_base; 525 dist_base = gic_data_dist_base(&gic_data[gic_nr]);
469 cpu_base = gic_data[gic_nr].cpu_base; 526 cpu_base = gic_data_cpu_base(&gic_data[gic_nr]);
470 527
471 if (!dist_base || !cpu_base) 528 if (!dist_base || !cpu_base)
472 return; 529 return;
@@ -491,6 +548,11 @@ static int gic_notifier(struct notifier_block *self, unsigned long cmd, void *v)
491 int i; 548 int i;
492 549
493 for (i = 0; i < MAX_GIC_NR; i++) { 550 for (i = 0; i < MAX_GIC_NR; i++) {
551#ifdef CONFIG_GIC_NON_BANKED
552 /* Skip over unused GICs */
553 if (!gic_data[i].get_base)
554 continue;
555#endif
494 switch (cmd) { 556 switch (cmd) {
495 case CPU_PM_ENTER: 557 case CPU_PM_ENTER:
496 gic_cpu_save(i); 558 gic_cpu_save(i);
@@ -563,8 +625,9 @@ const struct irq_domain_ops gic_irq_domain_ops = {
563#endif 625#endif
564}; 626};
565 627
566void __init gic_init(unsigned int gic_nr, int irq_start, 628void __init gic_init_bases(unsigned int gic_nr, int irq_start,
567 void __iomem *dist_base, void __iomem *cpu_base) 629 void __iomem *dist_base, void __iomem *cpu_base,
630 u32 percpu_offset)
568{ 631{
569 struct gic_chip_data *gic; 632 struct gic_chip_data *gic;
570 struct irq_domain *domain; 633 struct irq_domain *domain;
@@ -574,8 +637,36 @@ void __init gic_init(unsigned int gic_nr, int irq_start,
574 637
575 gic = &gic_data[gic_nr]; 638 gic = &gic_data[gic_nr];
576 domain = &gic->domain; 639 domain = &gic->domain;
577 gic->dist_base = dist_base; 640#ifdef CONFIG_GIC_NON_BANKED
578 gic->cpu_base = cpu_base; 641 if (percpu_offset) { /* Frankein-GIC without banked registers... */
642 unsigned int cpu;
643
644 gic->dist_base.percpu_base = alloc_percpu(void __iomem *);
645 gic->cpu_base.percpu_base = alloc_percpu(void __iomem *);
646 if (WARN_ON(!gic->dist_base.percpu_base ||
647 !gic->cpu_base.percpu_base)) {
648 free_percpu(gic->dist_base.percpu_base);
649 free_percpu(gic->cpu_base.percpu_base);
650 return;
651 }
652
653 for_each_possible_cpu(cpu) {
654 unsigned long offset = percpu_offset * cpu_logical_map(cpu);
655 *per_cpu_ptr(gic->dist_base.percpu_base, cpu) = dist_base + offset;
656 *per_cpu_ptr(gic->cpu_base.percpu_base, cpu) = cpu_base + offset;
657 }
658
659 gic_set_base_accessor(gic, gic_get_percpu_base);
660 } else
661#endif
662 { /* Normal, sane GIC... */
663 WARN(percpu_offset,
664 "GIC_NON_BANKED not enabled, ignoring %08x offset!",
665 percpu_offset);
666 gic->dist_base.common_base = dist_base;
667 gic->cpu_base.common_base = cpu_base;
668 gic_set_base_accessor(gic, gic_get_common_base);
669 }
579 670
580 /* 671 /*
581 * For primary GICs, skip over SGIs. 672 * For primary GICs, skip over SGIs.
@@ -593,7 +684,7 @@ void __init gic_init(unsigned int gic_nr, int irq_start,
593 * Find out how many interrupts are supported. 684 * Find out how many interrupts are supported.
594 * The GIC only supports up to 1020 interrupt sources. 685 * The GIC only supports up to 1020 interrupt sources.
595 */ 686 */
596 gic_irqs = readl_relaxed(dist_base + GIC_DIST_CTR) & 0x1f; 687 gic_irqs = readl_relaxed(gic_data_dist_base(gic) + GIC_DIST_CTR) & 0x1f;
597 gic_irqs = (gic_irqs + 1) * 32; 688 gic_irqs = (gic_irqs + 1) * 32;
598 if (gic_irqs > 1020) 689 if (gic_irqs > 1020)
599 gic_irqs = 1020; 690 gic_irqs = 1020;
@@ -641,7 +732,7 @@ void gic_raise_softirq(const struct cpumask *mask, unsigned int irq)
641 dsb(); 732 dsb();
642 733
643 /* this always happens on GIC0 */ 734 /* this always happens on GIC0 */
644 writel_relaxed(map << 16 | irq, gic_data[0].dist_base + GIC_DIST_SOFTINT); 735 writel_relaxed(map << 16 | irq, gic_data_dist_base(&gic_data[0]) + GIC_DIST_SOFTINT);
645} 736}
646#endif 737#endif
647 738
@@ -652,6 +743,7 @@ int __init gic_of_init(struct device_node *node, struct device_node *parent)
652{ 743{
653 void __iomem *cpu_base; 744 void __iomem *cpu_base;
654 void __iomem *dist_base; 745 void __iomem *dist_base;
746 u32 percpu_offset;
655 int irq; 747 int irq;
656 struct irq_domain *domain = &gic_data[gic_cnt].domain; 748 struct irq_domain *domain = &gic_data[gic_cnt].domain;
657 749
@@ -664,9 +756,12 @@ int __init gic_of_init(struct device_node *node, struct device_node *parent)
664 cpu_base = of_iomap(node, 1); 756 cpu_base = of_iomap(node, 1);
665 WARN(!cpu_base, "unable to map gic cpu registers\n"); 757 WARN(!cpu_base, "unable to map gic cpu registers\n");
666 758
759 if (of_property_read_u32(node, "cpu-offset", &percpu_offset))
760 percpu_offset = 0;
761
667 domain->of_node = of_node_get(node); 762 domain->of_node = of_node_get(node);
668 763
669 gic_init(gic_cnt, -1, dist_base, cpu_base); 764 gic_init_bases(gic_cnt, -1, dist_base, cpu_base, percpu_offset);
670 765
671 if (parent) { 766 if (parent) {
672 irq = irq_of_parse_and_map(node, 0); 767 irq = irq_of_parse_and_map(node, 0);