diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2014-08-05 20:38:45 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2014-08-05 20:38:45 -0400 |
commit | 08d69a25714429850cf9ef71f22d8cdc9189d93f (patch) | |
tree | c03f4e91834b5280a4695f0c805b58c599e895cc /drivers/irqchip | |
parent | ed5c41d30ef2ce578fd6b6e2f7ec23f2a58b1eba (diff) | |
parent | c6f1224573c3b609bd8073b39f496637a16cc06f (diff) |
Merge branch 'irq-core-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip
Pull irq updates from Thomas Gleixner:
"Nothing spectacular from the irq department this time:
- overhaul of the crossbar chip driver
- overhaul of the spear shirq chip driver
- support for the atmel-aic chip
- code move from arch to drivers
- the usual tiny fixlets
- two reverts worth to mention which undo the too simple attempt of
supporting wakeup interrupts on shared interrupt lines"
* 'irq-core-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: (41 commits)
Revert "irq: Warn when shared interrupts do not match on NO_SUSPEND"
Revert "PM / sleep / irq: Do not suspend wakeup interrupts"
irq: Warn when shared interrupts do not match on NO_SUSPEND
irqchip: atmel-aic: Define irq fixups for atmel SoCs
irqchip: atmel-aic: Implement RTC irq fixup
irqchip: atmel-aic: Add irq fixup infrastructure
irqchip: atmel-aic: Add atmel AIC/AIC5 drivers
irqchip: atmel-aic: Move binding doc to interrupt-controller directory
genirq: generic chip: Export irq_map_generic_chip function
PM / sleep / irq: Do not suspend wakeup interrupts
irqchip: or1k-pic: Migrate from arch/openrisc/
irqchip: crossbar: Allow for quirky hardware with direct hardwiring of GIC
documentation: dt: omap: crossbar: Add description for interrupt consumer
irqchip: crossbar: Introduce centralized check for crossbar write
irqchip: crossbar: Introduce ti, max-crossbar-sources to identify valid crossbar mapping
irqchip: crossbar: Add kerneldoc for crossbar_domain_unmap callback
irqchip: crossbar: Set cb pointer to null in case of error
irqchip: crossbar: Change the goto naming
irqchip: crossbar: Return proper error value
irqchip: crossbar: Fix kerneldoc warning
...
Diffstat (limited to 'drivers/irqchip')
-rw-r--r-- | drivers/irqchip/Kconfig | 18 | ||||
-rw-r--r-- | drivers/irqchip/Makefile | 3 | ||||
-rw-r--r-- | drivers/irqchip/irq-atmel-aic-common.c | 254 | ||||
-rw-r--r-- | drivers/irqchip/irq-atmel-aic-common.h | 39 | ||||
-rw-r--r-- | drivers/irqchip/irq-atmel-aic.c | 262 | ||||
-rw-r--r-- | drivers/irqchip/irq-atmel-aic5.c | 353 | ||||
-rw-r--r-- | drivers/irqchip/irq-crossbar.c | 168 | ||||
-rw-r--r-- | drivers/irqchip/irq-nvic.c | 13 | ||||
-rw-r--r-- | drivers/irqchip/irq-or1k-pic.c | 182 | ||||
-rw-r--r-- | drivers/irqchip/spear-shirq.c | 304 |
10 files changed, 1395 insertions, 201 deletions
diff --git a/drivers/irqchip/Kconfig b/drivers/irqchip/Kconfig index 7f0c2a30267b..4e230e7c76ee 100644 --- a/drivers/irqchip/Kconfig +++ b/drivers/irqchip/Kconfig | |||
@@ -35,6 +35,20 @@ config ARM_VIC_NR | |||
35 | The maximum number of VICs available in the system, for | 35 | The maximum number of VICs available in the system, for |
36 | power management. | 36 | power management. |
37 | 37 | ||
38 | config ATMEL_AIC_IRQ | ||
39 | bool | ||
40 | select GENERIC_IRQ_CHIP | ||
41 | select IRQ_DOMAIN | ||
42 | select MULTI_IRQ_HANDLER | ||
43 | select SPARSE_IRQ | ||
44 | |||
45 | config ATMEL_AIC5_IRQ | ||
46 | bool | ||
47 | select GENERIC_IRQ_CHIP | ||
48 | select IRQ_DOMAIN | ||
49 | select MULTI_IRQ_HANDLER | ||
50 | select SPARSE_IRQ | ||
51 | |||
38 | config BRCMSTB_L2_IRQ | 52 | config BRCMSTB_L2_IRQ |
39 | bool | 53 | bool |
40 | depends on ARM | 54 | depends on ARM |
@@ -58,6 +72,10 @@ config CLPS711X_IRQCHIP | |||
58 | select SPARSE_IRQ | 72 | select SPARSE_IRQ |
59 | default y | 73 | default y |
60 | 74 | ||
75 | config OR1K_PIC | ||
76 | bool | ||
77 | select IRQ_DOMAIN | ||
78 | |||
61 | config ORION_IRQCHIP | 79 | config ORION_IRQCHIP |
62 | bool | 80 | bool |
63 | select IRQ_DOMAIN | 81 | select IRQ_DOMAIN |
diff --git a/drivers/irqchip/Makefile b/drivers/irqchip/Makefile index c57e642700d4..73052ba9ca62 100644 --- a/drivers/irqchip/Makefile +++ b/drivers/irqchip/Makefile | |||
@@ -11,6 +11,7 @@ obj-$(CONFIG_METAG) += irq-metag-ext.o | |||
11 | obj-$(CONFIG_METAG_PERFCOUNTER_IRQS) += irq-metag.o | 11 | obj-$(CONFIG_METAG_PERFCOUNTER_IRQS) += irq-metag.o |
12 | obj-$(CONFIG_ARCH_MOXART) += irq-moxart.o | 12 | obj-$(CONFIG_ARCH_MOXART) += irq-moxart.o |
13 | obj-$(CONFIG_CLPS711X_IRQCHIP) += irq-clps711x.o | 13 | obj-$(CONFIG_CLPS711X_IRQCHIP) += irq-clps711x.o |
14 | obj-$(CONFIG_OR1K_PIC) += irq-or1k-pic.o | ||
14 | obj-$(CONFIG_ORION_IRQCHIP) += irq-orion.o | 15 | obj-$(CONFIG_ORION_IRQCHIP) += irq-orion.o |
15 | obj-$(CONFIG_ARCH_SUNXI) += irq-sun4i.o | 16 | obj-$(CONFIG_ARCH_SUNXI) += irq-sun4i.o |
16 | obj-$(CONFIG_ARCH_SUNXI) += irq-sunxi-nmi.o | 17 | obj-$(CONFIG_ARCH_SUNXI) += irq-sunxi-nmi.o |
@@ -19,6 +20,8 @@ obj-$(CONFIG_ARM_GIC) += irq-gic.o irq-gic-common.o | |||
19 | obj-$(CONFIG_ARM_GIC_V3) += irq-gic-v3.o irq-gic-common.o | 20 | obj-$(CONFIG_ARM_GIC_V3) += irq-gic-v3.o irq-gic-common.o |
20 | obj-$(CONFIG_ARM_NVIC) += irq-nvic.o | 21 | obj-$(CONFIG_ARM_NVIC) += irq-nvic.o |
21 | obj-$(CONFIG_ARM_VIC) += irq-vic.o | 22 | obj-$(CONFIG_ARM_VIC) += irq-vic.o |
23 | obj-$(CONFIG_ATMEL_AIC_IRQ) += irq-atmel-aic-common.o irq-atmel-aic.o | ||
24 | obj-$(CONFIG_ATMEL_AIC5_IRQ) += irq-atmel-aic-common.o irq-atmel-aic5.o | ||
22 | obj-$(CONFIG_IMGPDC_IRQ) += irq-imgpdc.o | 25 | obj-$(CONFIG_IMGPDC_IRQ) += irq-imgpdc.o |
23 | obj-$(CONFIG_SIRF_IRQ) += irq-sirfsoc.o | 26 | obj-$(CONFIG_SIRF_IRQ) += irq-sirfsoc.o |
24 | obj-$(CONFIG_RENESAS_INTC_IRQPIN) += irq-renesas-intc-irqpin.o | 27 | obj-$(CONFIG_RENESAS_INTC_IRQPIN) += irq-renesas-intc-irqpin.o |
diff --git a/drivers/irqchip/irq-atmel-aic-common.c b/drivers/irqchip/irq-atmel-aic-common.c new file mode 100644 index 000000000000..6ae3cdee0681 --- /dev/null +++ b/drivers/irqchip/irq-atmel-aic-common.c | |||
@@ -0,0 +1,254 @@ | |||
1 | /* | ||
2 | * Atmel AT91 common AIC (Advanced Interrupt Controller) code shared by | ||
3 | * irq-atmel-aic and irq-atmel-aic5 drivers | ||
4 | * | ||
5 | * Copyright (C) 2004 SAN People | ||
6 | * Copyright (C) 2004 ATMEL | ||
7 | * Copyright (C) Rick Bronson | ||
8 | * Copyright (C) 2014 Free Electrons | ||
9 | * | ||
10 | * Author: Boris BREZILLON <boris.brezillon@free-electrons.com> | ||
11 | * | ||
12 | * This file is licensed under the terms of the GNU General Public | ||
13 | * License version 2. This program is licensed "as is" without any | ||
14 | * warranty of any kind, whether express or implied. | ||
15 | */ | ||
16 | |||
17 | #include <linux/errno.h> | ||
18 | #include <linux/io.h> | ||
19 | #include <linux/irq.h> | ||
20 | #include <linux/irqdomain.h> | ||
21 | #include <linux/of.h> | ||
22 | #include <linux/of_address.h> | ||
23 | #include <linux/slab.h> | ||
24 | |||
25 | #include "irq-atmel-aic-common.h" | ||
26 | |||
27 | #define AT91_AIC_PRIOR GENMASK(2, 0) | ||
28 | #define AT91_AIC_IRQ_MIN_PRIORITY 0 | ||
29 | #define AT91_AIC_IRQ_MAX_PRIORITY 7 | ||
30 | |||
31 | #define AT91_AIC_SRCTYPE GENMASK(7, 6) | ||
32 | #define AT91_AIC_SRCTYPE_LOW (0 << 5) | ||
33 | #define AT91_AIC_SRCTYPE_FALLING (1 << 5) | ||
34 | #define AT91_AIC_SRCTYPE_HIGH (2 << 5) | ||
35 | #define AT91_AIC_SRCTYPE_RISING (3 << 5) | ||
36 | |||
37 | struct aic_chip_data { | ||
38 | u32 ext_irqs; | ||
39 | }; | ||
40 | |||
41 | static void aic_common_shutdown(struct irq_data *d) | ||
42 | { | ||
43 | struct irq_chip_type *ct = irq_data_get_chip_type(d); | ||
44 | |||
45 | ct->chip.irq_mask(d); | ||
46 | } | ||
47 | |||
48 | int aic_common_set_type(struct irq_data *d, unsigned type, unsigned *val) | ||
49 | { | ||
50 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
51 | struct aic_chip_data *aic = gc->private; | ||
52 | unsigned aic_type; | ||
53 | |||
54 | switch (type) { | ||
55 | case IRQ_TYPE_LEVEL_HIGH: | ||
56 | aic_type = AT91_AIC_SRCTYPE_HIGH; | ||
57 | break; | ||
58 | case IRQ_TYPE_EDGE_RISING: | ||
59 | aic_type = AT91_AIC_SRCTYPE_RISING; | ||
60 | break; | ||
61 | case IRQ_TYPE_LEVEL_LOW: | ||
62 | if (!(d->mask & aic->ext_irqs)) | ||
63 | return -EINVAL; | ||
64 | |||
65 | aic_type = AT91_AIC_SRCTYPE_LOW; | ||
66 | break; | ||
67 | case IRQ_TYPE_EDGE_FALLING: | ||
68 | if (!(d->mask & aic->ext_irqs)) | ||
69 | return -EINVAL; | ||
70 | |||
71 | aic_type = AT91_AIC_SRCTYPE_FALLING; | ||
72 | break; | ||
73 | default: | ||
74 | return -EINVAL; | ||
75 | } | ||
76 | |||
77 | *val &= AT91_AIC_SRCTYPE; | ||
78 | *val |= aic_type; | ||
79 | |||
80 | return 0; | ||
81 | } | ||
82 | |||
83 | int aic_common_set_priority(int priority, unsigned *val) | ||
84 | { | ||
85 | if (priority < AT91_AIC_IRQ_MIN_PRIORITY || | ||
86 | priority > AT91_AIC_IRQ_MAX_PRIORITY) | ||
87 | return -EINVAL; | ||
88 | |||
89 | *val &= AT91_AIC_PRIOR; | ||
90 | *val |= priority; | ||
91 | |||
92 | return 0; | ||
93 | } | ||
94 | |||
95 | int aic_common_irq_domain_xlate(struct irq_domain *d, | ||
96 | struct device_node *ctrlr, | ||
97 | const u32 *intspec, | ||
98 | unsigned int intsize, | ||
99 | irq_hw_number_t *out_hwirq, | ||
100 | unsigned int *out_type) | ||
101 | { | ||
102 | if (WARN_ON(intsize < 3)) | ||
103 | return -EINVAL; | ||
104 | |||
105 | if (WARN_ON((intspec[2] < AT91_AIC_IRQ_MIN_PRIORITY) || | ||
106 | (intspec[2] > AT91_AIC_IRQ_MAX_PRIORITY))) | ||
107 | return -EINVAL; | ||
108 | |||
109 | *out_hwirq = intspec[0]; | ||
110 | *out_type = intspec[1] & IRQ_TYPE_SENSE_MASK; | ||
111 | |||
112 | return 0; | ||
113 | } | ||
114 | |||
115 | static void __init aic_common_ext_irq_of_init(struct irq_domain *domain) | ||
116 | { | ||
117 | struct device_node *node = domain->of_node; | ||
118 | struct irq_chip_generic *gc; | ||
119 | struct aic_chip_data *aic; | ||
120 | struct property *prop; | ||
121 | const __be32 *p; | ||
122 | u32 hwirq; | ||
123 | |||
124 | gc = irq_get_domain_generic_chip(domain, 0); | ||
125 | |||
126 | aic = gc->private; | ||
127 | aic->ext_irqs |= 1; | ||
128 | |||
129 | of_property_for_each_u32(node, "atmel,external-irqs", prop, p, hwirq) { | ||
130 | gc = irq_get_domain_generic_chip(domain, hwirq); | ||
131 | if (!gc) { | ||
132 | pr_warn("AIC: external irq %d >= %d skip it\n", | ||
133 | hwirq, domain->revmap_size); | ||
134 | continue; | ||
135 | } | ||
136 | |||
137 | aic = gc->private; | ||
138 | aic->ext_irqs |= (1 << (hwirq % 32)); | ||
139 | } | ||
140 | } | ||
141 | |||
142 | #define AT91_RTC_IDR 0x24 | ||
143 | #define AT91_RTC_IMR 0x28 | ||
144 | #define AT91_RTC_IRQ_MASK 0x1f | ||
145 | |||
146 | void __init aic_common_rtc_irq_fixup(struct device_node *root) | ||
147 | { | ||
148 | struct device_node *np; | ||
149 | void __iomem *regs; | ||
150 | |||
151 | np = of_find_compatible_node(root, NULL, "atmel,at91rm9200-rtc"); | ||
152 | if (!np) | ||
153 | np = of_find_compatible_node(root, NULL, | ||
154 | "atmel,at91sam9x5-rtc"); | ||
155 | |||
156 | if (!np) | ||
157 | return; | ||
158 | |||
159 | regs = of_iomap(np, 0); | ||
160 | of_node_put(np); | ||
161 | |||
162 | if (!regs) | ||
163 | return; | ||
164 | |||
165 | writel(AT91_RTC_IRQ_MASK, regs + AT91_RTC_IDR); | ||
166 | |||
167 | iounmap(regs); | ||
168 | } | ||
169 | |||
170 | void __init aic_common_irq_fixup(const struct of_device_id *matches) | ||
171 | { | ||
172 | struct device_node *root = of_find_node_by_path("/"); | ||
173 | const struct of_device_id *match; | ||
174 | |||
175 | if (!root) | ||
176 | return; | ||
177 | |||
178 | match = of_match_node(matches, root); | ||
179 | of_node_put(root); | ||
180 | |||
181 | if (match) { | ||
182 | void (*fixup)(struct device_node *) = match->data; | ||
183 | fixup(root); | ||
184 | } | ||
185 | |||
186 | of_node_put(root); | ||
187 | } | ||
188 | |||
189 | struct irq_domain *__init aic_common_of_init(struct device_node *node, | ||
190 | const struct irq_domain_ops *ops, | ||
191 | const char *name, int nirqs) | ||
192 | { | ||
193 | struct irq_chip_generic *gc; | ||
194 | struct irq_domain *domain; | ||
195 | struct aic_chip_data *aic; | ||
196 | void __iomem *reg_base; | ||
197 | int nchips; | ||
198 | int ret; | ||
199 | int i; | ||
200 | |||
201 | nchips = DIV_ROUND_UP(nirqs, 32); | ||
202 | |||
203 | reg_base = of_iomap(node, 0); | ||
204 | if (!reg_base) | ||
205 | return ERR_PTR(-ENOMEM); | ||
206 | |||
207 | aic = kcalloc(nchips, sizeof(*aic), GFP_KERNEL); | ||
208 | if (!aic) { | ||
209 | ret = -ENOMEM; | ||
210 | goto err_iounmap; | ||
211 | } | ||
212 | |||
213 | domain = irq_domain_add_linear(node, nchips * 32, ops, aic); | ||
214 | if (!domain) { | ||
215 | ret = -ENOMEM; | ||
216 | goto err_free_aic; | ||
217 | } | ||
218 | |||
219 | ret = irq_alloc_domain_generic_chips(domain, 32, 1, name, | ||
220 | handle_level_irq, 0, 0, | ||
221 | IRQCHIP_SKIP_SET_WAKE); | ||
222 | if (ret) | ||
223 | goto err_domain_remove; | ||
224 | |||
225 | for (i = 0; i < nchips; i++) { | ||
226 | gc = irq_get_domain_generic_chip(domain, i * 32); | ||
227 | |||
228 | gc->reg_base = reg_base; | ||
229 | |||
230 | gc->unused = 0; | ||
231 | gc->wake_enabled = ~0; | ||
232 | gc->chip_types[0].type = IRQ_TYPE_SENSE_MASK; | ||
233 | gc->chip_types[0].handler = handle_fasteoi_irq; | ||
234 | gc->chip_types[0].chip.irq_eoi = irq_gc_eoi; | ||
235 | gc->chip_types[0].chip.irq_set_wake = irq_gc_set_wake; | ||
236 | gc->chip_types[0].chip.irq_shutdown = aic_common_shutdown; | ||
237 | gc->private = &aic[i]; | ||
238 | } | ||
239 | |||
240 | aic_common_ext_irq_of_init(domain); | ||
241 | |||
242 | return domain; | ||
243 | |||
244 | err_domain_remove: | ||
245 | irq_domain_remove(domain); | ||
246 | |||
247 | err_free_aic: | ||
248 | kfree(aic); | ||
249 | |||
250 | err_iounmap: | ||
251 | iounmap(reg_base); | ||
252 | |||
253 | return ERR_PTR(ret); | ||
254 | } | ||
diff --git a/drivers/irqchip/irq-atmel-aic-common.h b/drivers/irqchip/irq-atmel-aic-common.h new file mode 100644 index 000000000000..90aa00e918d6 --- /dev/null +++ b/drivers/irqchip/irq-atmel-aic-common.h | |||
@@ -0,0 +1,39 @@ | |||
1 | /* | ||
2 | * Atmel AT91 common AIC (Advanced Interrupt Controller) header file | ||
3 | * | ||
4 | * Copyright (C) 2004 SAN People | ||
5 | * Copyright (C) 2004 ATMEL | ||
6 | * Copyright (C) Rick Bronson | ||
7 | * Copyright (C) 2014 Free Electrons | ||
8 | * | ||
9 | * Author: Boris BREZILLON <boris.brezillon@free-electrons.com> | ||
10 | * | ||
11 | * This file is licensed under the terms of the GNU General Public | ||
12 | * License version 2. This program is licensed "as is" without any | ||
13 | * warranty of any kind, whether express or implied. | ||
14 | */ | ||
15 | |||
16 | #ifndef __IRQ_ATMEL_AIC_COMMON_H | ||
17 | #define __IRQ_ATMEL_AIC_COMMON_H | ||
18 | |||
19 | |||
20 | int aic_common_set_type(struct irq_data *d, unsigned type, unsigned *val); | ||
21 | |||
22 | int aic_common_set_priority(int priority, unsigned *val); | ||
23 | |||
24 | int aic_common_irq_domain_xlate(struct irq_domain *d, | ||
25 | struct device_node *ctrlr, | ||
26 | const u32 *intspec, | ||
27 | unsigned int intsize, | ||
28 | irq_hw_number_t *out_hwirq, | ||
29 | unsigned int *out_type); | ||
30 | |||
31 | struct irq_domain *__init aic_common_of_init(struct device_node *node, | ||
32 | const struct irq_domain_ops *ops, | ||
33 | const char *name, int nirqs); | ||
34 | |||
35 | void __init aic_common_rtc_irq_fixup(struct device_node *root); | ||
36 | |||
37 | void __init aic_common_irq_fixup(const struct of_device_id *matches); | ||
38 | |||
39 | #endif /* __IRQ_ATMEL_AIC_COMMON_H */ | ||
diff --git a/drivers/irqchip/irq-atmel-aic.c b/drivers/irqchip/irq-atmel-aic.c new file mode 100644 index 000000000000..a82869e9fb26 --- /dev/null +++ b/drivers/irqchip/irq-atmel-aic.c | |||
@@ -0,0 +1,262 @@ | |||
1 | /* | ||
2 | * Atmel AT91 AIC (Advanced Interrupt Controller) driver | ||
3 | * | ||
4 | * Copyright (C) 2004 SAN People | ||
5 | * Copyright (C) 2004 ATMEL | ||
6 | * Copyright (C) Rick Bronson | ||
7 | * Copyright (C) 2014 Free Electrons | ||
8 | * | ||
9 | * Author: Boris BREZILLON <boris.brezillon@free-electrons.com> | ||
10 | * | ||
11 | * This file is licensed under the terms of the GNU General Public | ||
12 | * License version 2. This program is licensed "as is" without any | ||
13 | * warranty of any kind, whether express or implied. | ||
14 | */ | ||
15 | |||
16 | #include <linux/init.h> | ||
17 | #include <linux/module.h> | ||
18 | #include <linux/mm.h> | ||
19 | #include <linux/bitmap.h> | ||
20 | #include <linux/types.h> | ||
21 | #include <linux/irq.h> | ||
22 | #include <linux/of.h> | ||
23 | #include <linux/of_address.h> | ||
24 | #include <linux/of_irq.h> | ||
25 | #include <linux/irqdomain.h> | ||
26 | #include <linux/err.h> | ||
27 | #include <linux/slab.h> | ||
28 | #include <linux/io.h> | ||
29 | |||
30 | #include <asm/exception.h> | ||
31 | #include <asm/mach/irq.h> | ||
32 | |||
33 | #include "irq-atmel-aic-common.h" | ||
34 | #include "irqchip.h" | ||
35 | |||
36 | /* Number of irq lines managed by AIC */ | ||
37 | #define NR_AIC_IRQS 32 | ||
38 | |||
39 | #define AT91_AIC_SMR(n) ((n) * 4) | ||
40 | |||
41 | #define AT91_AIC_SVR(n) (0x80 + ((n) * 4)) | ||
42 | #define AT91_AIC_IVR 0x100 | ||
43 | #define AT91_AIC_FVR 0x104 | ||
44 | #define AT91_AIC_ISR 0x108 | ||
45 | |||
46 | #define AT91_AIC_IPR 0x10c | ||
47 | #define AT91_AIC_IMR 0x110 | ||
48 | #define AT91_AIC_CISR 0x114 | ||
49 | |||
50 | #define AT91_AIC_IECR 0x120 | ||
51 | #define AT91_AIC_IDCR 0x124 | ||
52 | #define AT91_AIC_ICCR 0x128 | ||
53 | #define AT91_AIC_ISCR 0x12c | ||
54 | #define AT91_AIC_EOICR 0x130 | ||
55 | #define AT91_AIC_SPU 0x134 | ||
56 | #define AT91_AIC_DCR 0x138 | ||
57 | |||
58 | static struct irq_domain *aic_domain; | ||
59 | |||
60 | static asmlinkage void __exception_irq_entry | ||
61 | aic_handle(struct pt_regs *regs) | ||
62 | { | ||
63 | struct irq_domain_chip_generic *dgc = aic_domain->gc; | ||
64 | struct irq_chip_generic *gc = dgc->gc[0]; | ||
65 | u32 irqnr; | ||
66 | u32 irqstat; | ||
67 | |||
68 | irqnr = irq_reg_readl(gc->reg_base + AT91_AIC_IVR); | ||
69 | irqstat = irq_reg_readl(gc->reg_base + AT91_AIC_ISR); | ||
70 | |||
71 | irqnr = irq_find_mapping(aic_domain, irqnr); | ||
72 | |||
73 | if (!irqstat) | ||
74 | irq_reg_writel(0, gc->reg_base + AT91_AIC_EOICR); | ||
75 | else | ||
76 | handle_IRQ(irqnr, regs); | ||
77 | } | ||
78 | |||
79 | static int aic_retrigger(struct irq_data *d) | ||
80 | { | ||
81 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
82 | |||
83 | /* Enable interrupt on AIC5 */ | ||
84 | irq_gc_lock(gc); | ||
85 | irq_reg_writel(d->mask, gc->reg_base + AT91_AIC_ISCR); | ||
86 | irq_gc_unlock(gc); | ||
87 | |||
88 | return 0; | ||
89 | } | ||
90 | |||
91 | static int aic_set_type(struct irq_data *d, unsigned type) | ||
92 | { | ||
93 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
94 | unsigned int smr; | ||
95 | int ret; | ||
96 | |||
97 | smr = irq_reg_readl(gc->reg_base + AT91_AIC_SMR(d->hwirq)); | ||
98 | ret = aic_common_set_type(d, type, &smr); | ||
99 | if (ret) | ||
100 | return ret; | ||
101 | |||
102 | irq_reg_writel(smr, gc->reg_base + AT91_AIC_SMR(d->hwirq)); | ||
103 | |||
104 | return 0; | ||
105 | } | ||
106 | |||
107 | #ifdef CONFIG_PM | ||
108 | static void aic_suspend(struct irq_data *d) | ||
109 | { | ||
110 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
111 | |||
112 | irq_gc_lock(gc); | ||
113 | irq_reg_writel(gc->mask_cache, gc->reg_base + AT91_AIC_IDCR); | ||
114 | irq_reg_writel(gc->wake_active, gc->reg_base + AT91_AIC_IECR); | ||
115 | irq_gc_unlock(gc); | ||
116 | } | ||
117 | |||
118 | static void aic_resume(struct irq_data *d) | ||
119 | { | ||
120 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
121 | |||
122 | irq_gc_lock(gc); | ||
123 | irq_reg_writel(gc->wake_active, gc->reg_base + AT91_AIC_IDCR); | ||
124 | irq_reg_writel(gc->mask_cache, gc->reg_base + AT91_AIC_IECR); | ||
125 | irq_gc_unlock(gc); | ||
126 | } | ||
127 | |||
128 | static void aic_pm_shutdown(struct irq_data *d) | ||
129 | { | ||
130 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
131 | |||
132 | irq_gc_lock(gc); | ||
133 | irq_reg_writel(0xffffffff, gc->reg_base + AT91_AIC_IDCR); | ||
134 | irq_reg_writel(0xffffffff, gc->reg_base + AT91_AIC_ICCR); | ||
135 | irq_gc_unlock(gc); | ||
136 | } | ||
137 | #else | ||
138 | #define aic_suspend NULL | ||
139 | #define aic_resume NULL | ||
140 | #define aic_pm_shutdown NULL | ||
141 | #endif /* CONFIG_PM */ | ||
142 | |||
143 | static void __init aic_hw_init(struct irq_domain *domain) | ||
144 | { | ||
145 | struct irq_chip_generic *gc = irq_get_domain_generic_chip(domain, 0); | ||
146 | int i; | ||
147 | |||
148 | /* | ||
149 | * Perform 8 End Of Interrupt Command to make sure AIC | ||
150 | * will not Lock out nIRQ | ||
151 | */ | ||
152 | for (i = 0; i < 8; i++) | ||
153 | irq_reg_writel(0, gc->reg_base + AT91_AIC_EOICR); | ||
154 | |||
155 | /* | ||
156 | * Spurious Interrupt ID in Spurious Vector Register. | ||
157 | * When there is no current interrupt, the IRQ Vector Register | ||
158 | * reads the value stored in AIC_SPU | ||
159 | */ | ||
160 | irq_reg_writel(0xffffffff, gc->reg_base + AT91_AIC_SPU); | ||
161 | |||
162 | /* No debugging in AIC: Debug (Protect) Control Register */ | ||
163 | irq_reg_writel(0, gc->reg_base + AT91_AIC_DCR); | ||
164 | |||
165 | /* Disable and clear all interrupts initially */ | ||
166 | irq_reg_writel(0xffffffff, gc->reg_base + AT91_AIC_IDCR); | ||
167 | irq_reg_writel(0xffffffff, gc->reg_base + AT91_AIC_ICCR); | ||
168 | |||
169 | for (i = 0; i < 32; i++) | ||
170 | irq_reg_writel(i, gc->reg_base + AT91_AIC_SVR(i)); | ||
171 | } | ||
172 | |||
173 | static int aic_irq_domain_xlate(struct irq_domain *d, | ||
174 | struct device_node *ctrlr, | ||
175 | const u32 *intspec, unsigned int intsize, | ||
176 | irq_hw_number_t *out_hwirq, | ||
177 | unsigned int *out_type) | ||
178 | { | ||
179 | struct irq_domain_chip_generic *dgc = d->gc; | ||
180 | struct irq_chip_generic *gc; | ||
181 | unsigned smr; | ||
182 | int idx; | ||
183 | int ret; | ||
184 | |||
185 | if (!dgc) | ||
186 | return -EINVAL; | ||
187 | |||
188 | ret = aic_common_irq_domain_xlate(d, ctrlr, intspec, intsize, | ||
189 | out_hwirq, out_type); | ||
190 | if (ret) | ||
191 | return ret; | ||
192 | |||
193 | idx = intspec[0] / dgc->irqs_per_chip; | ||
194 | if (idx >= dgc->num_chips) | ||
195 | return -EINVAL; | ||
196 | |||
197 | gc = dgc->gc[idx]; | ||
198 | |||
199 | irq_gc_lock(gc); | ||
200 | smr = irq_reg_readl(gc->reg_base + AT91_AIC_SMR(*out_hwirq)); | ||
201 | ret = aic_common_set_priority(intspec[2], &smr); | ||
202 | if (!ret) | ||
203 | irq_reg_writel(smr, gc->reg_base + AT91_AIC_SMR(*out_hwirq)); | ||
204 | irq_gc_unlock(gc); | ||
205 | |||
206 | return ret; | ||
207 | } | ||
208 | |||
209 | static const struct irq_domain_ops aic_irq_ops = { | ||
210 | .map = irq_map_generic_chip, | ||
211 | .xlate = aic_irq_domain_xlate, | ||
212 | }; | ||
213 | |||
214 | static void __init at91sam9_aic_irq_fixup(struct device_node *root) | ||
215 | { | ||
216 | aic_common_rtc_irq_fixup(root); | ||
217 | } | ||
218 | |||
219 | static const struct of_device_id __initdata aic_irq_fixups[] = { | ||
220 | { .compatible = "atmel,at91sam9g45", .data = at91sam9_aic_irq_fixup }, | ||
221 | { .compatible = "atmel,at91sam9n12", .data = at91sam9_aic_irq_fixup }, | ||
222 | { .compatible = "atmel,at91sam9rl", .data = at91sam9_aic_irq_fixup }, | ||
223 | { .compatible = "atmel,at91sam9x5", .data = at91sam9_aic_irq_fixup }, | ||
224 | { /* sentinel */ }, | ||
225 | }; | ||
226 | |||
227 | static int __init aic_of_init(struct device_node *node, | ||
228 | struct device_node *parent) | ||
229 | { | ||
230 | struct irq_chip_generic *gc; | ||
231 | struct irq_domain *domain; | ||
232 | |||
233 | if (aic_domain) | ||
234 | return -EEXIST; | ||
235 | |||
236 | domain = aic_common_of_init(node, &aic_irq_ops, "atmel-aic", | ||
237 | NR_AIC_IRQS); | ||
238 | if (IS_ERR(domain)) | ||
239 | return PTR_ERR(domain); | ||
240 | |||
241 | aic_common_irq_fixup(aic_irq_fixups); | ||
242 | |||
243 | aic_domain = domain; | ||
244 | gc = irq_get_domain_generic_chip(domain, 0); | ||
245 | |||
246 | gc->chip_types[0].regs.eoi = AT91_AIC_EOICR; | ||
247 | gc->chip_types[0].regs.enable = AT91_AIC_IECR; | ||
248 | gc->chip_types[0].regs.disable = AT91_AIC_IDCR; | ||
249 | gc->chip_types[0].chip.irq_mask = irq_gc_mask_disable_reg; | ||
250 | gc->chip_types[0].chip.irq_unmask = irq_gc_unmask_enable_reg; | ||
251 | gc->chip_types[0].chip.irq_retrigger = aic_retrigger; | ||
252 | gc->chip_types[0].chip.irq_set_type = aic_set_type; | ||
253 | gc->chip_types[0].chip.irq_suspend = aic_suspend; | ||
254 | gc->chip_types[0].chip.irq_resume = aic_resume; | ||
255 | gc->chip_types[0].chip.irq_pm_shutdown = aic_pm_shutdown; | ||
256 | |||
257 | aic_hw_init(domain); | ||
258 | set_handle_irq(aic_handle); | ||
259 | |||
260 | return 0; | ||
261 | } | ||
262 | IRQCHIP_DECLARE(at91rm9200_aic, "atmel,at91rm9200-aic", aic_of_init); | ||
diff --git a/drivers/irqchip/irq-atmel-aic5.c b/drivers/irqchip/irq-atmel-aic5.c new file mode 100644 index 000000000000..edb227081524 --- /dev/null +++ b/drivers/irqchip/irq-atmel-aic5.c | |||
@@ -0,0 +1,353 @@ | |||
1 | /* | ||
2 | * Atmel AT91 AIC5 (Advanced Interrupt Controller) driver | ||
3 | * | ||
4 | * Copyright (C) 2004 SAN People | ||
5 | * Copyright (C) 2004 ATMEL | ||
6 | * Copyright (C) Rick Bronson | ||
7 | * Copyright (C) 2014 Free Electrons | ||
8 | * | ||
9 | * Author: Boris BREZILLON <boris.brezillon@free-electrons.com> | ||
10 | * | ||
11 | * This file is licensed under the terms of the GNU General Public | ||
12 | * License version 2. This program is licensed "as is" without any | ||
13 | * warranty of any kind, whether express or implied. | ||
14 | */ | ||
15 | |||
16 | #include <linux/init.h> | ||
17 | #include <linux/module.h> | ||
18 | #include <linux/mm.h> | ||
19 | #include <linux/bitmap.h> | ||
20 | #include <linux/types.h> | ||
21 | #include <linux/irq.h> | ||
22 | #include <linux/of.h> | ||
23 | #include <linux/of_address.h> | ||
24 | #include <linux/of_irq.h> | ||
25 | #include <linux/irqdomain.h> | ||
26 | #include <linux/err.h> | ||
27 | #include <linux/slab.h> | ||
28 | #include <linux/io.h> | ||
29 | |||
30 | #include <asm/exception.h> | ||
31 | #include <asm/mach/irq.h> | ||
32 | |||
33 | #include "irq-atmel-aic-common.h" | ||
34 | #include "irqchip.h" | ||
35 | |||
36 | /* Number of irq lines managed by AIC */ | ||
37 | #define NR_AIC5_IRQS 128 | ||
38 | |||
39 | #define AT91_AIC5_SSR 0x0 | ||
40 | #define AT91_AIC5_INTSEL_MSK (0x7f << 0) | ||
41 | |||
42 | #define AT91_AIC5_SMR 0x4 | ||
43 | |||
44 | #define AT91_AIC5_SVR 0x8 | ||
45 | #define AT91_AIC5_IVR 0x10 | ||
46 | #define AT91_AIC5_FVR 0x14 | ||
47 | #define AT91_AIC5_ISR 0x18 | ||
48 | |||
49 | #define AT91_AIC5_IPR0 0x20 | ||
50 | #define AT91_AIC5_IPR1 0x24 | ||
51 | #define AT91_AIC5_IPR2 0x28 | ||
52 | #define AT91_AIC5_IPR3 0x2c | ||
53 | #define AT91_AIC5_IMR 0x30 | ||
54 | #define AT91_AIC5_CISR 0x34 | ||
55 | |||
56 | #define AT91_AIC5_IECR 0x40 | ||
57 | #define AT91_AIC5_IDCR 0x44 | ||
58 | #define AT91_AIC5_ICCR 0x48 | ||
59 | #define AT91_AIC5_ISCR 0x4c | ||
60 | #define AT91_AIC5_EOICR 0x38 | ||
61 | #define AT91_AIC5_SPU 0x3c | ||
62 | #define AT91_AIC5_DCR 0x6c | ||
63 | |||
64 | #define AT91_AIC5_FFER 0x50 | ||
65 | #define AT91_AIC5_FFDR 0x54 | ||
66 | #define AT91_AIC5_FFSR 0x58 | ||
67 | |||
68 | static struct irq_domain *aic5_domain; | ||
69 | |||
70 | static asmlinkage void __exception_irq_entry | ||
71 | aic5_handle(struct pt_regs *regs) | ||
72 | { | ||
73 | struct irq_domain_chip_generic *dgc = aic5_domain->gc; | ||
74 | struct irq_chip_generic *gc = dgc->gc[0]; | ||
75 | u32 irqnr; | ||
76 | u32 irqstat; | ||
77 | |||
78 | irqnr = irq_reg_readl(gc->reg_base + AT91_AIC5_IVR); | ||
79 | irqstat = irq_reg_readl(gc->reg_base + AT91_AIC5_ISR); | ||
80 | |||
81 | irqnr = irq_find_mapping(aic5_domain, irqnr); | ||
82 | |||
83 | if (!irqstat) | ||
84 | irq_reg_writel(0, gc->reg_base + AT91_AIC5_EOICR); | ||
85 | else | ||
86 | handle_IRQ(irqnr, regs); | ||
87 | } | ||
88 | |||
89 | static void aic5_mask(struct irq_data *d) | ||
90 | { | ||
91 | struct irq_domain *domain = d->domain; | ||
92 | struct irq_domain_chip_generic *dgc = domain->gc; | ||
93 | struct irq_chip_generic *gc = dgc->gc[0]; | ||
94 | |||
95 | /* Disable interrupt on AIC5 */ | ||
96 | irq_gc_lock(gc); | ||
97 | irq_reg_writel(d->hwirq, gc->reg_base + AT91_AIC5_SSR); | ||
98 | irq_reg_writel(1, gc->reg_base + AT91_AIC5_IDCR); | ||
99 | gc->mask_cache &= ~d->mask; | ||
100 | irq_gc_unlock(gc); | ||
101 | } | ||
102 | |||
103 | static void aic5_unmask(struct irq_data *d) | ||
104 | { | ||
105 | struct irq_domain *domain = d->domain; | ||
106 | struct irq_domain_chip_generic *dgc = domain->gc; | ||
107 | struct irq_chip_generic *gc = dgc->gc[0]; | ||
108 | |||
109 | /* Enable interrupt on AIC5 */ | ||
110 | irq_gc_lock(gc); | ||
111 | irq_reg_writel(d->hwirq, gc->reg_base + AT91_AIC5_SSR); | ||
112 | irq_reg_writel(1, gc->reg_base + AT91_AIC5_IECR); | ||
113 | gc->mask_cache |= d->mask; | ||
114 | irq_gc_unlock(gc); | ||
115 | } | ||
116 | |||
117 | static int aic5_retrigger(struct irq_data *d) | ||
118 | { | ||
119 | struct irq_domain *domain = d->domain; | ||
120 | struct irq_domain_chip_generic *dgc = domain->gc; | ||
121 | struct irq_chip_generic *gc = dgc->gc[0]; | ||
122 | |||
123 | /* Enable interrupt on AIC5 */ | ||
124 | irq_gc_lock(gc); | ||
125 | irq_reg_writel(d->hwirq, gc->reg_base + AT91_AIC5_SSR); | ||
126 | irq_reg_writel(1, gc->reg_base + AT91_AIC5_ISCR); | ||
127 | irq_gc_unlock(gc); | ||
128 | |||
129 | return 0; | ||
130 | } | ||
131 | |||
132 | static int aic5_set_type(struct irq_data *d, unsigned type) | ||
133 | { | ||
134 | struct irq_domain *domain = d->domain; | ||
135 | struct irq_domain_chip_generic *dgc = domain->gc; | ||
136 | struct irq_chip_generic *gc = dgc->gc[0]; | ||
137 | unsigned int smr; | ||
138 | int ret; | ||
139 | |||
140 | irq_gc_lock(gc); | ||
141 | irq_reg_writel(d->hwirq, gc->reg_base + AT91_AIC5_SSR); | ||
142 | smr = irq_reg_readl(gc->reg_base + AT91_AIC5_SMR); | ||
143 | ret = aic_common_set_type(d, type, &smr); | ||
144 | if (!ret) | ||
145 | irq_reg_writel(smr, gc->reg_base + AT91_AIC5_SMR); | ||
146 | irq_gc_unlock(gc); | ||
147 | |||
148 | return ret; | ||
149 | } | ||
150 | |||
151 | #ifdef CONFIG_PM | ||
152 | static void aic5_suspend(struct irq_data *d) | ||
153 | { | ||
154 | struct irq_domain *domain = d->domain; | ||
155 | struct irq_domain_chip_generic *dgc = domain->gc; | ||
156 | struct irq_chip_generic *bgc = dgc->gc[0]; | ||
157 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
158 | int i; | ||
159 | u32 mask; | ||
160 | |||
161 | irq_gc_lock(bgc); | ||
162 | for (i = 0; i < dgc->irqs_per_chip; i++) { | ||
163 | mask = 1 << i; | ||
164 | if ((mask & gc->mask_cache) == (mask & gc->wake_active)) | ||
165 | continue; | ||
166 | |||
167 | irq_reg_writel(i + gc->irq_base, | ||
168 | bgc->reg_base + AT91_AIC5_SSR); | ||
169 | if (mask & gc->wake_active) | ||
170 | irq_reg_writel(1, bgc->reg_base + AT91_AIC5_IECR); | ||
171 | else | ||
172 | irq_reg_writel(1, bgc->reg_base + AT91_AIC5_IDCR); | ||
173 | } | ||
174 | irq_gc_unlock(bgc); | ||
175 | } | ||
176 | |||
177 | static void aic5_resume(struct irq_data *d) | ||
178 | { | ||
179 | struct irq_domain *domain = d->domain; | ||
180 | struct irq_domain_chip_generic *dgc = domain->gc; | ||
181 | struct irq_chip_generic *bgc = dgc->gc[0]; | ||
182 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
183 | int i; | ||
184 | u32 mask; | ||
185 | |||
186 | irq_gc_lock(bgc); | ||
187 | for (i = 0; i < dgc->irqs_per_chip; i++) { | ||
188 | mask = 1 << i; | ||
189 | if ((mask & gc->mask_cache) == (mask & gc->wake_active)) | ||
190 | continue; | ||
191 | |||
192 | irq_reg_writel(i + gc->irq_base, | ||
193 | bgc->reg_base + AT91_AIC5_SSR); | ||
194 | if (mask & gc->mask_cache) | ||
195 | irq_reg_writel(1, bgc->reg_base + AT91_AIC5_IECR); | ||
196 | else | ||
197 | irq_reg_writel(1, bgc->reg_base + AT91_AIC5_IDCR); | ||
198 | } | ||
199 | irq_gc_unlock(bgc); | ||
200 | } | ||
201 | |||
202 | static void aic5_pm_shutdown(struct irq_data *d) | ||
203 | { | ||
204 | struct irq_domain *domain = d->domain; | ||
205 | struct irq_domain_chip_generic *dgc = domain->gc; | ||
206 | struct irq_chip_generic *bgc = dgc->gc[0]; | ||
207 | struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d); | ||
208 | int i; | ||
209 | |||
210 | irq_gc_lock(bgc); | ||
211 | for (i = 0; i < dgc->irqs_per_chip; i++) { | ||
212 | irq_reg_writel(i + gc->irq_base, | ||
213 | bgc->reg_base + AT91_AIC5_SSR); | ||
214 | irq_reg_writel(1, bgc->reg_base + AT91_AIC5_IDCR); | ||
215 | irq_reg_writel(1, bgc->reg_base + AT91_AIC5_ICCR); | ||
216 | } | ||
217 | irq_gc_unlock(bgc); | ||
218 | } | ||
219 | #else | ||
220 | #define aic5_suspend NULL | ||
221 | #define aic5_resume NULL | ||
222 | #define aic5_pm_shutdown NULL | ||
223 | #endif /* CONFIG_PM */ | ||
224 | |||
225 | static void __init aic5_hw_init(struct irq_domain *domain) | ||
226 | { | ||
227 | struct irq_chip_generic *gc = irq_get_domain_generic_chip(domain, 0); | ||
228 | int i; | ||
229 | |||
230 | /* | ||
231 | * Perform 8 End Of Interrupt Command to make sure AIC | ||
232 | * will not Lock out nIRQ | ||
233 | */ | ||
234 | for (i = 0; i < 8; i++) | ||
235 | irq_reg_writel(0, gc->reg_base + AT91_AIC5_EOICR); | ||
236 | |||
237 | /* | ||
238 | * Spurious Interrupt ID in Spurious Vector Register. | ||
239 | * When there is no current interrupt, the IRQ Vector Register | ||
240 | * reads the value stored in AIC_SPU | ||
241 | */ | ||
242 | irq_reg_writel(0xffffffff, gc->reg_base + AT91_AIC5_SPU); | ||
243 | |||
244 | /* No debugging in AIC: Debug (Protect) Control Register */ | ||
245 | irq_reg_writel(0, gc->reg_base + AT91_AIC5_DCR); | ||
246 | |||
247 | /* Disable and clear all interrupts initially */ | ||
248 | for (i = 0; i < domain->revmap_size; i++) { | ||
249 | irq_reg_writel(i, gc->reg_base + AT91_AIC5_SSR); | ||
250 | irq_reg_writel(i, gc->reg_base + AT91_AIC5_SVR); | ||
251 | irq_reg_writel(1, gc->reg_base + AT91_AIC5_IDCR); | ||
252 | irq_reg_writel(1, gc->reg_base + AT91_AIC5_ICCR); | ||
253 | } | ||
254 | } | ||
255 | |||
256 | static int aic5_irq_domain_xlate(struct irq_domain *d, | ||
257 | struct device_node *ctrlr, | ||
258 | const u32 *intspec, unsigned int intsize, | ||
259 | irq_hw_number_t *out_hwirq, | ||
260 | unsigned int *out_type) | ||
261 | { | ||
262 | struct irq_domain_chip_generic *dgc = d->gc; | ||
263 | struct irq_chip_generic *gc; | ||
264 | unsigned smr; | ||
265 | int ret; | ||
266 | |||
267 | if (!dgc) | ||
268 | return -EINVAL; | ||
269 | |||
270 | ret = aic_common_irq_domain_xlate(d, ctrlr, intspec, intsize, | ||
271 | out_hwirq, out_type); | ||
272 | if (ret) | ||
273 | return ret; | ||
274 | |||
275 | gc = dgc->gc[0]; | ||
276 | |||
277 | irq_gc_lock(gc); | ||
278 | irq_reg_writel(*out_hwirq, gc->reg_base + AT91_AIC5_SSR); | ||
279 | smr = irq_reg_readl(gc->reg_base + AT91_AIC5_SMR); | ||
280 | ret = aic_common_set_priority(intspec[2], &smr); | ||
281 | if (!ret) | ||
282 | irq_reg_writel(intspec[2] | smr, gc->reg_base + AT91_AIC5_SMR); | ||
283 | irq_gc_unlock(gc); | ||
284 | |||
285 | return ret; | ||
286 | } | ||
287 | |||
288 | static const struct irq_domain_ops aic5_irq_ops = { | ||
289 | .map = irq_map_generic_chip, | ||
290 | .xlate = aic5_irq_domain_xlate, | ||
291 | }; | ||
292 | |||
293 | static void __init sama5d3_aic_irq_fixup(struct device_node *root) | ||
294 | { | ||
295 | aic_common_rtc_irq_fixup(root); | ||
296 | } | ||
297 | |||
298 | static const struct of_device_id __initdata aic5_irq_fixups[] = { | ||
299 | { .compatible = "atmel,sama5d3", .data = sama5d3_aic_irq_fixup }, | ||
300 | { /* sentinel */ }, | ||
301 | }; | ||
302 | |||
303 | static int __init aic5_of_init(struct device_node *node, | ||
304 | struct device_node *parent, | ||
305 | int nirqs) | ||
306 | { | ||
307 | struct irq_chip_generic *gc; | ||
308 | struct irq_domain *domain; | ||
309 | int nchips; | ||
310 | int i; | ||
311 | |||
312 | if (nirqs > NR_AIC5_IRQS) | ||
313 | return -EINVAL; | ||
314 | |||
315 | if (aic5_domain) | ||
316 | return -EEXIST; | ||
317 | |||
318 | domain = aic_common_of_init(node, &aic5_irq_ops, "atmel-aic5", | ||
319 | nirqs); | ||
320 | if (IS_ERR(domain)) | ||
321 | return PTR_ERR(domain); | ||
322 | |||
323 | aic_common_irq_fixup(aic5_irq_fixups); | ||
324 | |||
325 | aic5_domain = domain; | ||
326 | nchips = aic5_domain->revmap_size / 32; | ||
327 | for (i = 0; i < nchips; i++) { | ||
328 | gc = irq_get_domain_generic_chip(domain, i * 32); | ||
329 | |||
330 | gc->chip_types[0].regs.eoi = AT91_AIC5_EOICR; | ||
331 | gc->chip_types[0].chip.irq_mask = aic5_mask; | ||
332 | gc->chip_types[0].chip.irq_unmask = aic5_unmask; | ||
333 | gc->chip_types[0].chip.irq_retrigger = aic5_retrigger; | ||
334 | gc->chip_types[0].chip.irq_set_type = aic5_set_type; | ||
335 | gc->chip_types[0].chip.irq_suspend = aic5_suspend; | ||
336 | gc->chip_types[0].chip.irq_resume = aic5_resume; | ||
337 | gc->chip_types[0].chip.irq_pm_shutdown = aic5_pm_shutdown; | ||
338 | } | ||
339 | |||
340 | aic5_hw_init(domain); | ||
341 | set_handle_irq(aic5_handle); | ||
342 | |||
343 | return 0; | ||
344 | } | ||
345 | |||
346 | #define NR_SAMA5D3_IRQS 50 | ||
347 | |||
348 | static int __init sama5d3_aic5_of_init(struct device_node *node, | ||
349 | struct device_node *parent) | ||
350 | { | ||
351 | return aic5_of_init(node, parent, NR_SAMA5D3_IRQS); | ||
352 | } | ||
353 | IRQCHIP_DECLARE(sama5d3_aic5, "atmel,sama5d3-aic", sama5d3_aic5_of_init); | ||
diff --git a/drivers/irqchip/irq-crossbar.c b/drivers/irqchip/irq-crossbar.c index 3d15d16a7088..85c2985d8bcb 100644 --- a/drivers/irqchip/irq-crossbar.c +++ b/drivers/irqchip/irq-crossbar.c | |||
@@ -15,22 +15,31 @@ | |||
15 | #include <linux/of_irq.h> | 15 | #include <linux/of_irq.h> |
16 | #include <linux/slab.h> | 16 | #include <linux/slab.h> |
17 | #include <linux/irqchip/arm-gic.h> | 17 | #include <linux/irqchip/arm-gic.h> |
18 | #include <linux/irqchip/irq-crossbar.h> | ||
18 | 19 | ||
19 | #define IRQ_FREE -1 | 20 | #define IRQ_FREE -1 |
21 | #define IRQ_RESERVED -2 | ||
22 | #define IRQ_SKIP -3 | ||
20 | #define GIC_IRQ_START 32 | 23 | #define GIC_IRQ_START 32 |
21 | 24 | ||
22 | /* | 25 | /** |
26 | * struct crossbar_device - crossbar device description | ||
23 | * @int_max: maximum number of supported interrupts | 27 | * @int_max: maximum number of supported interrupts |
28 | * @safe_map: safe default value to initialize the crossbar | ||
29 | * @max_crossbar_sources: Maximum number of crossbar sources | ||
24 | * @irq_map: array of interrupts to crossbar number mapping | 30 | * @irq_map: array of interrupts to crossbar number mapping |
25 | * @crossbar_base: crossbar base address | 31 | * @crossbar_base: crossbar base address |
26 | * @register_offsets: offsets for each irq number | 32 | * @register_offsets: offsets for each irq number |
33 | * @write: register write function pointer | ||
27 | */ | 34 | */ |
28 | struct crossbar_device { | 35 | struct crossbar_device { |
29 | uint int_max; | 36 | uint int_max; |
37 | uint safe_map; | ||
38 | uint max_crossbar_sources; | ||
30 | uint *irq_map; | 39 | uint *irq_map; |
31 | void __iomem *crossbar_base; | 40 | void __iomem *crossbar_base; |
32 | int *register_offsets; | 41 | int *register_offsets; |
33 | void (*write) (int, int); | 42 | void (*write)(int, int); |
34 | }; | 43 | }; |
35 | 44 | ||
36 | static struct crossbar_device *cb; | 45 | static struct crossbar_device *cb; |
@@ -50,11 +59,22 @@ static inline void crossbar_writeb(int irq_no, int cb_no) | |||
50 | writeb(cb_no, cb->crossbar_base + cb->register_offsets[irq_no]); | 59 | writeb(cb_no, cb->crossbar_base + cb->register_offsets[irq_no]); |
51 | } | 60 | } |
52 | 61 | ||
62 | static inline int get_prev_map_irq(int cb_no) | ||
63 | { | ||
64 | int i; | ||
65 | |||
66 | for (i = cb->int_max - 1; i >= 0; i--) | ||
67 | if (cb->irq_map[i] == cb_no) | ||
68 | return i; | ||
69 | |||
70 | return -ENODEV; | ||
71 | } | ||
72 | |||
53 | static inline int allocate_free_irq(int cb_no) | 73 | static inline int allocate_free_irq(int cb_no) |
54 | { | 74 | { |
55 | int i; | 75 | int i; |
56 | 76 | ||
57 | for (i = 0; i < cb->int_max; i++) { | 77 | for (i = cb->int_max - 1; i >= 0; i--) { |
58 | if (cb->irq_map[i] == IRQ_FREE) { | 78 | if (cb->irq_map[i] == IRQ_FREE) { |
59 | cb->irq_map[i] = cb_no; | 79 | cb->irq_map[i] = cb_no; |
60 | return i; | 80 | return i; |
@@ -64,19 +84,47 @@ static inline int allocate_free_irq(int cb_no) | |||
64 | return -ENODEV; | 84 | return -ENODEV; |
65 | } | 85 | } |
66 | 86 | ||
87 | static inline bool needs_crossbar_write(irq_hw_number_t hw) | ||
88 | { | ||
89 | int cb_no; | ||
90 | |||
91 | if (hw > GIC_IRQ_START) { | ||
92 | cb_no = cb->irq_map[hw - GIC_IRQ_START]; | ||
93 | if (cb_no != IRQ_RESERVED && cb_no != IRQ_SKIP) | ||
94 | return true; | ||
95 | } | ||
96 | |||
97 | return false; | ||
98 | } | ||
99 | |||
67 | static int crossbar_domain_map(struct irq_domain *d, unsigned int irq, | 100 | static int crossbar_domain_map(struct irq_domain *d, unsigned int irq, |
68 | irq_hw_number_t hw) | 101 | irq_hw_number_t hw) |
69 | { | 102 | { |
70 | cb->write(hw - GIC_IRQ_START, cb->irq_map[hw - GIC_IRQ_START]); | 103 | if (needs_crossbar_write(hw)) |
104 | cb->write(hw - GIC_IRQ_START, cb->irq_map[hw - GIC_IRQ_START]); | ||
105 | |||
71 | return 0; | 106 | return 0; |
72 | } | 107 | } |
73 | 108 | ||
109 | /** | ||
110 | * crossbar_domain_unmap - unmap a crossbar<->irq connection | ||
111 | * @d: domain of irq to unmap | ||
112 | * @irq: virq number | ||
113 | * | ||
114 | * We do not maintain a use count of total number of map/unmap | ||
115 | * calls for a particular irq to find out if a irq can be really | ||
116 | * unmapped. This is because unmap is called during irq_dispose_mapping(irq), | ||
117 | * after which irq is anyways unusable. So an explicit map has to be called | ||
118 | * after that. | ||
119 | */ | ||
74 | static void crossbar_domain_unmap(struct irq_domain *d, unsigned int irq) | 120 | static void crossbar_domain_unmap(struct irq_domain *d, unsigned int irq) |
75 | { | 121 | { |
76 | irq_hw_number_t hw = irq_get_irq_data(irq)->hwirq; | 122 | irq_hw_number_t hw = irq_get_irq_data(irq)->hwirq; |
77 | 123 | ||
78 | if (hw > GIC_IRQ_START) | 124 | if (needs_crossbar_write(hw)) { |
79 | cb->irq_map[hw - GIC_IRQ_START] = IRQ_FREE; | 125 | cb->irq_map[hw - GIC_IRQ_START] = IRQ_FREE; |
126 | cb->write(hw - GIC_IRQ_START, cb->safe_map); | ||
127 | } | ||
80 | } | 128 | } |
81 | 129 | ||
82 | static int crossbar_domain_xlate(struct irq_domain *d, | 130 | static int crossbar_domain_xlate(struct irq_domain *d, |
@@ -85,18 +133,41 @@ static int crossbar_domain_xlate(struct irq_domain *d, | |||
85 | unsigned long *out_hwirq, | 133 | unsigned long *out_hwirq, |
86 | unsigned int *out_type) | 134 | unsigned int *out_type) |
87 | { | 135 | { |
88 | unsigned long ret; | 136 | int ret; |
137 | int req_num = intspec[1]; | ||
138 | int direct_map_num; | ||
139 | |||
140 | if (req_num >= cb->max_crossbar_sources) { | ||
141 | direct_map_num = req_num - cb->max_crossbar_sources; | ||
142 | if (direct_map_num < cb->int_max) { | ||
143 | ret = cb->irq_map[direct_map_num]; | ||
144 | if (ret == IRQ_RESERVED || ret == IRQ_SKIP) { | ||
145 | /* We use the interrupt num as h/w irq num */ | ||
146 | ret = direct_map_num; | ||
147 | goto found; | ||
148 | } | ||
149 | } | ||
150 | |||
151 | pr_err("%s: requested crossbar number %d > max %d\n", | ||
152 | __func__, req_num, cb->max_crossbar_sources); | ||
153 | return -EINVAL; | ||
154 | } | ||
89 | 155 | ||
90 | ret = allocate_free_irq(intspec[1]); | 156 | ret = get_prev_map_irq(req_num); |
157 | if (ret >= 0) | ||
158 | goto found; | ||
91 | 159 | ||
92 | if (IS_ERR_VALUE(ret)) | 160 | ret = allocate_free_irq(req_num); |
161 | |||
162 | if (ret < 0) | ||
93 | return ret; | 163 | return ret; |
94 | 164 | ||
165 | found: | ||
95 | *out_hwirq = ret + GIC_IRQ_START; | 166 | *out_hwirq = ret + GIC_IRQ_START; |
96 | return 0; | 167 | return 0; |
97 | } | 168 | } |
98 | 169 | ||
99 | const struct irq_domain_ops routable_irq_domain_ops = { | 170 | static const struct irq_domain_ops routable_irq_domain_ops = { |
100 | .map = crossbar_domain_map, | 171 | .map = crossbar_domain_map, |
101 | .unmap = crossbar_domain_unmap, | 172 | .unmap = crossbar_domain_unmap, |
102 | .xlate = crossbar_domain_xlate | 173 | .xlate = crossbar_domain_xlate |
@@ -104,22 +175,36 @@ const struct irq_domain_ops routable_irq_domain_ops = { | |||
104 | 175 | ||
105 | static int __init crossbar_of_init(struct device_node *node) | 176 | static int __init crossbar_of_init(struct device_node *node) |
106 | { | 177 | { |
107 | int i, size, max, reserved = 0, entry; | 178 | int i, size, max = 0, reserved = 0, entry; |
108 | const __be32 *irqsr; | 179 | const __be32 *irqsr; |
180 | int ret = -ENOMEM; | ||
109 | 181 | ||
110 | cb = kzalloc(sizeof(*cb), GFP_KERNEL); | 182 | cb = kzalloc(sizeof(*cb), GFP_KERNEL); |
111 | 183 | ||
112 | if (!cb) | 184 | if (!cb) |
113 | return -ENOMEM; | 185 | return ret; |
114 | 186 | ||
115 | cb->crossbar_base = of_iomap(node, 0); | 187 | cb->crossbar_base = of_iomap(node, 0); |
116 | if (!cb->crossbar_base) | 188 | if (!cb->crossbar_base) |
117 | goto err1; | 189 | goto err_cb; |
190 | |||
191 | of_property_read_u32(node, "ti,max-crossbar-sources", | ||
192 | &cb->max_crossbar_sources); | ||
193 | if (!cb->max_crossbar_sources) { | ||
194 | pr_err("missing 'ti,max-crossbar-sources' property\n"); | ||
195 | ret = -EINVAL; | ||
196 | goto err_base; | ||
197 | } | ||
118 | 198 | ||
119 | of_property_read_u32(node, "ti,max-irqs", &max); | 199 | of_property_read_u32(node, "ti,max-irqs", &max); |
120 | cb->irq_map = kzalloc(max * sizeof(int), GFP_KERNEL); | 200 | if (!max) { |
201 | pr_err("missing 'ti,max-irqs' property\n"); | ||
202 | ret = -EINVAL; | ||
203 | goto err_base; | ||
204 | } | ||
205 | cb->irq_map = kcalloc(max, sizeof(int), GFP_KERNEL); | ||
121 | if (!cb->irq_map) | 206 | if (!cb->irq_map) |
122 | goto err2; | 207 | goto err_base; |
123 | 208 | ||
124 | cb->int_max = max; | 209 | cb->int_max = max; |
125 | 210 | ||
@@ -137,15 +222,35 @@ static int __init crossbar_of_init(struct device_node *node) | |||
137 | i, &entry); | 222 | i, &entry); |
138 | if (entry > max) { | 223 | if (entry > max) { |
139 | pr_err("Invalid reserved entry\n"); | 224 | pr_err("Invalid reserved entry\n"); |
140 | goto err3; | 225 | ret = -EINVAL; |
226 | goto err_irq_map; | ||
227 | } | ||
228 | cb->irq_map[entry] = IRQ_RESERVED; | ||
229 | } | ||
230 | } | ||
231 | |||
232 | /* Skip irqs hardwired to bypass the crossbar */ | ||
233 | irqsr = of_get_property(node, "ti,irqs-skip", &size); | ||
234 | if (irqsr) { | ||
235 | size /= sizeof(__be32); | ||
236 | |||
237 | for (i = 0; i < size; i++) { | ||
238 | of_property_read_u32_index(node, | ||
239 | "ti,irqs-skip", | ||
240 | i, &entry); | ||
241 | if (entry > max) { | ||
242 | pr_err("Invalid skip entry\n"); | ||
243 | ret = -EINVAL; | ||
244 | goto err_irq_map; | ||
141 | } | 245 | } |
142 | cb->irq_map[entry] = 0; | 246 | cb->irq_map[entry] = IRQ_SKIP; |
143 | } | 247 | } |
144 | } | 248 | } |
145 | 249 | ||
146 | cb->register_offsets = kzalloc(max * sizeof(int), GFP_KERNEL); | 250 | |
251 | cb->register_offsets = kcalloc(max, sizeof(int), GFP_KERNEL); | ||
147 | if (!cb->register_offsets) | 252 | if (!cb->register_offsets) |
148 | goto err3; | 253 | goto err_irq_map; |
149 | 254 | ||
150 | of_property_read_u32(node, "ti,reg-size", &size); | 255 | of_property_read_u32(node, "ti,reg-size", &size); |
151 | 256 | ||
@@ -161,7 +266,8 @@ static int __init crossbar_of_init(struct device_node *node) | |||
161 | break; | 266 | break; |
162 | default: | 267 | default: |
163 | pr_err("Invalid reg-size property\n"); | 268 | pr_err("Invalid reg-size property\n"); |
164 | goto err4; | 269 | ret = -EINVAL; |
270 | goto err_reg_offset; | ||
165 | break; | 271 | break; |
166 | } | 272 | } |
167 | 273 | ||
@@ -170,25 +276,37 @@ static int __init crossbar_of_init(struct device_node *node) | |||
170 | * reserved irqs. so find and store the offsets once. | 276 | * reserved irqs. so find and store the offsets once. |
171 | */ | 277 | */ |
172 | for (i = 0; i < max; i++) { | 278 | for (i = 0; i < max; i++) { |
173 | if (!cb->irq_map[i]) | 279 | if (cb->irq_map[i] == IRQ_RESERVED) |
174 | continue; | 280 | continue; |
175 | 281 | ||
176 | cb->register_offsets[i] = reserved; | 282 | cb->register_offsets[i] = reserved; |
177 | reserved += size; | 283 | reserved += size; |
178 | } | 284 | } |
179 | 285 | ||
286 | of_property_read_u32(node, "ti,irqs-safe-map", &cb->safe_map); | ||
287 | /* Initialize the crossbar with safe map to start with */ | ||
288 | for (i = 0; i < max; i++) { | ||
289 | if (cb->irq_map[i] == IRQ_RESERVED || | ||
290 | cb->irq_map[i] == IRQ_SKIP) | ||
291 | continue; | ||
292 | |||
293 | cb->write(i, cb->safe_map); | ||
294 | } | ||
295 | |||
180 | register_routable_domain_ops(&routable_irq_domain_ops); | 296 | register_routable_domain_ops(&routable_irq_domain_ops); |
181 | return 0; | 297 | return 0; |
182 | 298 | ||
183 | err4: | 299 | err_reg_offset: |
184 | kfree(cb->register_offsets); | 300 | kfree(cb->register_offsets); |
185 | err3: | 301 | err_irq_map: |
186 | kfree(cb->irq_map); | 302 | kfree(cb->irq_map); |
187 | err2: | 303 | err_base: |
188 | iounmap(cb->crossbar_base); | 304 | iounmap(cb->crossbar_base); |
189 | err1: | 305 | err_cb: |
190 | kfree(cb); | 306 | kfree(cb); |
191 | return -ENOMEM; | 307 | |
308 | cb = NULL; | ||
309 | return ret; | ||
192 | } | 310 | } |
193 | 311 | ||
194 | static const struct of_device_id crossbar_match[] __initconst = { | 312 | static const struct of_device_id crossbar_match[] __initconst = { |
diff --git a/drivers/irqchip/irq-nvic.c b/drivers/irqchip/irq-nvic.c index 70bdf6edb7bb..4ff0805fca01 100644 --- a/drivers/irqchip/irq-nvic.c +++ b/drivers/irqchip/irq-nvic.c | |||
@@ -49,14 +49,6 @@ nvic_handle_irq(irq_hw_number_t hwirq, struct pt_regs *regs) | |||
49 | handle_IRQ(irq, regs); | 49 | handle_IRQ(irq, regs); |
50 | } | 50 | } |
51 | 51 | ||
52 | static void nvic_eoi(struct irq_data *d) | ||
53 | { | ||
54 | /* | ||
55 | * This is a no-op as end of interrupt is signaled by the exception | ||
56 | * return sequence. | ||
57 | */ | ||
58 | } | ||
59 | |||
60 | static int __init nvic_of_init(struct device_node *node, | 52 | static int __init nvic_of_init(struct device_node *node, |
61 | struct device_node *parent) | 53 | struct device_node *parent) |
62 | { | 54 | { |
@@ -102,7 +94,10 @@ static int __init nvic_of_init(struct device_node *node, | |||
102 | gc->chip_types[0].regs.disable = NVIC_ICER; | 94 | gc->chip_types[0].regs.disable = NVIC_ICER; |
103 | gc->chip_types[0].chip.irq_mask = irq_gc_mask_disable_reg; | 95 | gc->chip_types[0].chip.irq_mask = irq_gc_mask_disable_reg; |
104 | gc->chip_types[0].chip.irq_unmask = irq_gc_unmask_enable_reg; | 96 | gc->chip_types[0].chip.irq_unmask = irq_gc_unmask_enable_reg; |
105 | gc->chip_types[0].chip.irq_eoi = nvic_eoi; | 97 | /* This is a no-op as end of interrupt is signaled by the |
98 | * exception return sequence. | ||
99 | */ | ||
100 | gc->chip_types[0].chip.irq_eoi = irq_gc_noop; | ||
106 | 101 | ||
107 | /* disable interrupts */ | 102 | /* disable interrupts */ |
108 | writel_relaxed(~0, gc->reg_base + NVIC_ICER); | 103 | writel_relaxed(~0, gc->reg_base + NVIC_ICER); |
diff --git a/drivers/irqchip/irq-or1k-pic.c b/drivers/irqchip/irq-or1k-pic.c new file mode 100644 index 000000000000..17ff033d9925 --- /dev/null +++ b/drivers/irqchip/irq-or1k-pic.c | |||
@@ -0,0 +1,182 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2010-2011 Jonas Bonn <jonas@southpole.se> | ||
3 | * Copyright (C) 2014 Stefan Kristansson <stefan.kristiansson@saunalahti.fi> | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or | ||
6 | * modify it under the terms of the GNU General Public License | ||
7 | * as published by the Free Software Foundation; either version | ||
8 | * 2 of the License, or (at your option) any later version. | ||
9 | */ | ||
10 | |||
11 | #include <linux/irq.h> | ||
12 | #include <linux/of.h> | ||
13 | #include <linux/of_irq.h> | ||
14 | #include <linux/of_address.h> | ||
15 | |||
16 | #include "irqchip.h" | ||
17 | |||
18 | /* OR1K PIC implementation */ | ||
19 | |||
20 | struct or1k_pic_dev { | ||
21 | struct irq_chip chip; | ||
22 | irq_flow_handler_t handle; | ||
23 | unsigned long flags; | ||
24 | }; | ||
25 | |||
26 | /* | ||
27 | * We're a couple of cycles faster than the generic implementations with | ||
28 | * these 'fast' versions. | ||
29 | */ | ||
30 | |||
31 | static void or1k_pic_mask(struct irq_data *data) | ||
32 | { | ||
33 | mtspr(SPR_PICMR, mfspr(SPR_PICMR) & ~(1UL << data->hwirq)); | ||
34 | } | ||
35 | |||
36 | static void or1k_pic_unmask(struct irq_data *data) | ||
37 | { | ||
38 | mtspr(SPR_PICMR, mfspr(SPR_PICMR) | (1UL << data->hwirq)); | ||
39 | } | ||
40 | |||
41 | static void or1k_pic_ack(struct irq_data *data) | ||
42 | { | ||
43 | mtspr(SPR_PICSR, (1UL << data->hwirq)); | ||
44 | } | ||
45 | |||
46 | static void or1k_pic_mask_ack(struct irq_data *data) | ||
47 | { | ||
48 | mtspr(SPR_PICMR, mfspr(SPR_PICMR) & ~(1UL << data->hwirq)); | ||
49 | mtspr(SPR_PICSR, (1UL << data->hwirq)); | ||
50 | } | ||
51 | |||
52 | /* | ||
53 | * There are two oddities with the OR1200 PIC implementation: | ||
54 | * i) LEVEL-triggered interrupts are latched and need to be cleared | ||
55 | * ii) the interrupt latch is cleared by writing a 0 to the bit, | ||
56 | * as opposed to a 1 as mandated by the spec | ||
57 | */ | ||
58 | static void or1k_pic_or1200_ack(struct irq_data *data) | ||
59 | { | ||
60 | mtspr(SPR_PICSR, mfspr(SPR_PICSR) & ~(1UL << data->hwirq)); | ||
61 | } | ||
62 | |||
63 | static void or1k_pic_or1200_mask_ack(struct irq_data *data) | ||
64 | { | ||
65 | mtspr(SPR_PICMR, mfspr(SPR_PICMR) & ~(1UL << data->hwirq)); | ||
66 | mtspr(SPR_PICSR, mfspr(SPR_PICSR) & ~(1UL << data->hwirq)); | ||
67 | } | ||
68 | |||
69 | static struct or1k_pic_dev or1k_pic_level = { | ||
70 | .chip = { | ||
71 | .name = "or1k-PIC-level", | ||
72 | .irq_unmask = or1k_pic_unmask, | ||
73 | .irq_mask = or1k_pic_mask, | ||
74 | .irq_mask_ack = or1k_pic_mask, | ||
75 | }, | ||
76 | .handle = handle_level_irq, | ||
77 | .flags = IRQ_LEVEL | IRQ_NOPROBE, | ||
78 | }; | ||
79 | |||
80 | static struct or1k_pic_dev or1k_pic_edge = { | ||
81 | .chip = { | ||
82 | .name = "or1k-PIC-edge", | ||
83 | .irq_unmask = or1k_pic_unmask, | ||
84 | .irq_mask = or1k_pic_mask, | ||
85 | .irq_ack = or1k_pic_ack, | ||
86 | .irq_mask_ack = or1k_pic_mask_ack, | ||
87 | }, | ||
88 | .handle = handle_edge_irq, | ||
89 | .flags = IRQ_LEVEL | IRQ_NOPROBE, | ||
90 | }; | ||
91 | |||
92 | static struct or1k_pic_dev or1k_pic_or1200 = { | ||
93 | .chip = { | ||
94 | .name = "or1200-PIC", | ||
95 | .irq_unmask = or1k_pic_unmask, | ||
96 | .irq_mask = or1k_pic_mask, | ||
97 | .irq_ack = or1k_pic_or1200_ack, | ||
98 | .irq_mask_ack = or1k_pic_or1200_mask_ack, | ||
99 | }, | ||
100 | .handle = handle_level_irq, | ||
101 | .flags = IRQ_LEVEL | IRQ_NOPROBE, | ||
102 | }; | ||
103 | |||
104 | static struct irq_domain *root_domain; | ||
105 | |||
106 | static inline int pic_get_irq(int first) | ||
107 | { | ||
108 | int hwirq; | ||
109 | |||
110 | hwirq = ffs(mfspr(SPR_PICSR) >> first); | ||
111 | if (!hwirq) | ||
112 | return NO_IRQ; | ||
113 | else | ||
114 | hwirq = hwirq + first - 1; | ||
115 | |||
116 | return irq_find_mapping(root_domain, hwirq); | ||
117 | } | ||
118 | |||
119 | static void or1k_pic_handle_irq(struct pt_regs *regs) | ||
120 | { | ||
121 | int irq = -1; | ||
122 | |||
123 | while ((irq = pic_get_irq(irq + 1)) != NO_IRQ) | ||
124 | handle_IRQ(irq, regs); | ||
125 | } | ||
126 | |||
127 | static int or1k_map(struct irq_domain *d, unsigned int irq, irq_hw_number_t hw) | ||
128 | { | ||
129 | struct or1k_pic_dev *pic = d->host_data; | ||
130 | |||
131 | irq_set_chip_and_handler(irq, &pic->chip, pic->handle); | ||
132 | irq_set_status_flags(irq, pic->flags); | ||
133 | |||
134 | return 0; | ||
135 | } | ||
136 | |||
137 | static const struct irq_domain_ops or1k_irq_domain_ops = { | ||
138 | .xlate = irq_domain_xlate_onecell, | ||
139 | .map = or1k_map, | ||
140 | }; | ||
141 | |||
142 | /* | ||
143 | * This sets up the IRQ domain for the PIC built in to the OpenRISC | ||
144 | * 1000 CPU. This is the "root" domain as these are the interrupts | ||
145 | * that directly trigger an exception in the CPU. | ||
146 | */ | ||
147 | static int __init or1k_pic_init(struct device_node *node, | ||
148 | struct or1k_pic_dev *pic) | ||
149 | { | ||
150 | /* Disable all interrupts until explicitly requested */ | ||
151 | mtspr(SPR_PICMR, (0UL)); | ||
152 | |||
153 | root_domain = irq_domain_add_linear(node, 32, &or1k_irq_domain_ops, | ||
154 | pic); | ||
155 | |||
156 | set_handle_irq(or1k_pic_handle_irq); | ||
157 | |||
158 | return 0; | ||
159 | } | ||
160 | |||
161 | static int __init or1k_pic_or1200_init(struct device_node *node, | ||
162 | struct device_node *parent) | ||
163 | { | ||
164 | return or1k_pic_init(node, &or1k_pic_or1200); | ||
165 | } | ||
166 | IRQCHIP_DECLARE(or1k_pic_or1200, "opencores,or1200-pic", or1k_pic_or1200_init); | ||
167 | IRQCHIP_DECLARE(or1k_pic, "opencores,or1k-pic", or1k_pic_or1200_init); | ||
168 | |||
169 | static int __init or1k_pic_level_init(struct device_node *node, | ||
170 | struct device_node *parent) | ||
171 | { | ||
172 | return or1k_pic_init(node, &or1k_pic_level); | ||
173 | } | ||
174 | IRQCHIP_DECLARE(or1k_pic_level, "opencores,or1k-pic-level", | ||
175 | or1k_pic_level_init); | ||
176 | |||
177 | static int __init or1k_pic_edge_init(struct device_node *node, | ||
178 | struct device_node *parent) | ||
179 | { | ||
180 | return or1k_pic_init(node, &or1k_pic_edge); | ||
181 | } | ||
182 | IRQCHIP_DECLARE(or1k_pic_edge, "opencores,or1k-pic-edge", or1k_pic_edge_init); | ||
diff --git a/drivers/irqchip/spear-shirq.c b/drivers/irqchip/spear-shirq.c index 6ce6bd3441bf..9c145a7cb056 100644 --- a/drivers/irqchip/spear-shirq.c +++ b/drivers/irqchip/spear-shirq.c | |||
@@ -19,7 +19,6 @@ | |||
19 | #include <linux/io.h> | 19 | #include <linux/io.h> |
20 | #include <linux/irq.h> | 20 | #include <linux/irq.h> |
21 | #include <linux/irqdomain.h> | 21 | #include <linux/irqdomain.h> |
22 | #include <linux/irqchip/spear-shirq.h> | ||
23 | #include <linux/of.h> | 22 | #include <linux/of.h> |
24 | #include <linux/of_address.h> | 23 | #include <linux/of_address.h> |
25 | #include <linux/of_irq.h> | 24 | #include <linux/of_irq.h> |
@@ -27,20 +26,73 @@ | |||
27 | 26 | ||
28 | #include "irqchip.h" | 27 | #include "irqchip.h" |
29 | 28 | ||
30 | static DEFINE_SPINLOCK(lock); | 29 | /* |
30 | * struct spear_shirq: shared irq structure | ||
31 | * | ||
32 | * base: Base register address | ||
33 | * status_reg: Status register offset for chained interrupt handler | ||
34 | * mask_reg: Mask register offset for irq chip | ||
35 | * mask: Mask to apply to the status register | ||
36 | * virq_base: Base virtual interrupt number | ||
37 | * nr_irqs: Number of interrupts handled by this block | ||
38 | * offset: Bit offset of the first interrupt | ||
39 | * irq_chip: Interrupt controller chip used for this instance, | ||
40 | * if NULL group is disabled, but accounted | ||
41 | */ | ||
42 | struct spear_shirq { | ||
43 | void __iomem *base; | ||
44 | u32 status_reg; | ||
45 | u32 mask_reg; | ||
46 | u32 mask; | ||
47 | u32 virq_base; | ||
48 | u32 nr_irqs; | ||
49 | u32 offset; | ||
50 | struct irq_chip *irq_chip; | ||
51 | }; | ||
31 | 52 | ||
32 | /* spear300 shared irq registers offsets and masks */ | 53 | /* spear300 shared irq registers offsets and masks */ |
33 | #define SPEAR300_INT_ENB_MASK_REG 0x54 | 54 | #define SPEAR300_INT_ENB_MASK_REG 0x54 |
34 | #define SPEAR300_INT_STS_MASK_REG 0x58 | 55 | #define SPEAR300_INT_STS_MASK_REG 0x58 |
35 | 56 | ||
57 | static DEFINE_RAW_SPINLOCK(shirq_lock); | ||
58 | |||
59 | static void shirq_irq_mask(struct irq_data *d) | ||
60 | { | ||
61 | struct spear_shirq *shirq = irq_data_get_irq_chip_data(d); | ||
62 | u32 val, shift = d->irq - shirq->virq_base + shirq->offset; | ||
63 | u32 __iomem *reg = shirq->base + shirq->mask_reg; | ||
64 | |||
65 | raw_spin_lock(&shirq_lock); | ||
66 | val = readl(reg) & ~(0x1 << shift); | ||
67 | writel(val, reg); | ||
68 | raw_spin_unlock(&shirq_lock); | ||
69 | } | ||
70 | |||
71 | static void shirq_irq_unmask(struct irq_data *d) | ||
72 | { | ||
73 | struct spear_shirq *shirq = irq_data_get_irq_chip_data(d); | ||
74 | u32 val, shift = d->irq - shirq->virq_base + shirq->offset; | ||
75 | u32 __iomem *reg = shirq->base + shirq->mask_reg; | ||
76 | |||
77 | raw_spin_lock(&shirq_lock); | ||
78 | val = readl(reg) | (0x1 << shift); | ||
79 | writel(val, reg); | ||
80 | raw_spin_unlock(&shirq_lock); | ||
81 | } | ||
82 | |||
83 | static struct irq_chip shirq_chip = { | ||
84 | .name = "spear-shirq", | ||
85 | .irq_mask = shirq_irq_mask, | ||
86 | .irq_unmask = shirq_irq_unmask, | ||
87 | }; | ||
88 | |||
36 | static struct spear_shirq spear300_shirq_ras1 = { | 89 | static struct spear_shirq spear300_shirq_ras1 = { |
37 | .irq_nr = 9, | 90 | .offset = 0, |
38 | .irq_bit_off = 0, | 91 | .nr_irqs = 9, |
39 | .regs = { | 92 | .mask = ((0x1 << 9) - 1) << 0, |
40 | .enb_reg = SPEAR300_INT_ENB_MASK_REG, | 93 | .irq_chip = &shirq_chip, |
41 | .status_reg = SPEAR300_INT_STS_MASK_REG, | 94 | .status_reg = SPEAR300_INT_STS_MASK_REG, |
42 | .clear_reg = -1, | 95 | .mask_reg = SPEAR300_INT_ENB_MASK_REG, |
43 | }, | ||
44 | }; | 96 | }; |
45 | 97 | ||
46 | static struct spear_shirq *spear300_shirq_blocks[] = { | 98 | static struct spear_shirq *spear300_shirq_blocks[] = { |
@@ -51,43 +103,35 @@ static struct spear_shirq *spear300_shirq_blocks[] = { | |||
51 | #define SPEAR310_INT_STS_MASK_REG 0x04 | 103 | #define SPEAR310_INT_STS_MASK_REG 0x04 |
52 | 104 | ||
53 | static struct spear_shirq spear310_shirq_ras1 = { | 105 | static struct spear_shirq spear310_shirq_ras1 = { |
54 | .irq_nr = 8, | 106 | .offset = 0, |
55 | .irq_bit_off = 0, | 107 | .nr_irqs = 8, |
56 | .regs = { | 108 | .mask = ((0x1 << 8) - 1) << 0, |
57 | .enb_reg = -1, | 109 | .irq_chip = &dummy_irq_chip, |
58 | .status_reg = SPEAR310_INT_STS_MASK_REG, | 110 | .status_reg = SPEAR310_INT_STS_MASK_REG, |
59 | .clear_reg = -1, | ||
60 | }, | ||
61 | }; | 111 | }; |
62 | 112 | ||
63 | static struct spear_shirq spear310_shirq_ras2 = { | 113 | static struct spear_shirq spear310_shirq_ras2 = { |
64 | .irq_nr = 5, | 114 | .offset = 8, |
65 | .irq_bit_off = 8, | 115 | .nr_irqs = 5, |
66 | .regs = { | 116 | .mask = ((0x1 << 5) - 1) << 8, |
67 | .enb_reg = -1, | 117 | .irq_chip = &dummy_irq_chip, |
68 | .status_reg = SPEAR310_INT_STS_MASK_REG, | 118 | .status_reg = SPEAR310_INT_STS_MASK_REG, |
69 | .clear_reg = -1, | ||
70 | }, | ||
71 | }; | 119 | }; |
72 | 120 | ||
73 | static struct spear_shirq spear310_shirq_ras3 = { | 121 | static struct spear_shirq spear310_shirq_ras3 = { |
74 | .irq_nr = 1, | 122 | .offset = 13, |
75 | .irq_bit_off = 13, | 123 | .nr_irqs = 1, |
76 | .regs = { | 124 | .mask = ((0x1 << 1) - 1) << 13, |
77 | .enb_reg = -1, | 125 | .irq_chip = &dummy_irq_chip, |
78 | .status_reg = SPEAR310_INT_STS_MASK_REG, | 126 | .status_reg = SPEAR310_INT_STS_MASK_REG, |
79 | .clear_reg = -1, | ||
80 | }, | ||
81 | }; | 127 | }; |
82 | 128 | ||
83 | static struct spear_shirq spear310_shirq_intrcomm_ras = { | 129 | static struct spear_shirq spear310_shirq_intrcomm_ras = { |
84 | .irq_nr = 3, | 130 | .offset = 14, |
85 | .irq_bit_off = 14, | 131 | .nr_irqs = 3, |
86 | .regs = { | 132 | .mask = ((0x1 << 3) - 1) << 14, |
87 | .enb_reg = -1, | 133 | .irq_chip = &dummy_irq_chip, |
88 | .status_reg = SPEAR310_INT_STS_MASK_REG, | 134 | .status_reg = SPEAR310_INT_STS_MASK_REG, |
89 | .clear_reg = -1, | ||
90 | }, | ||
91 | }; | 135 | }; |
92 | 136 | ||
93 | static struct spear_shirq *spear310_shirq_blocks[] = { | 137 | static struct spear_shirq *spear310_shirq_blocks[] = { |
@@ -102,50 +146,34 @@ static struct spear_shirq *spear310_shirq_blocks[] = { | |||
102 | #define SPEAR320_INT_CLR_MASK_REG 0x04 | 146 | #define SPEAR320_INT_CLR_MASK_REG 0x04 |
103 | #define SPEAR320_INT_ENB_MASK_REG 0x08 | 147 | #define SPEAR320_INT_ENB_MASK_REG 0x08 |
104 | 148 | ||
105 | static struct spear_shirq spear320_shirq_ras1 = { | 149 | static struct spear_shirq spear320_shirq_ras3 = { |
106 | .irq_nr = 3, | 150 | .offset = 0, |
107 | .irq_bit_off = 7, | 151 | .nr_irqs = 7, |
108 | .regs = { | 152 | .mask = ((0x1 << 7) - 1) << 0, |
109 | .enb_reg = -1, | ||
110 | .status_reg = SPEAR320_INT_STS_MASK_REG, | ||
111 | .clear_reg = SPEAR320_INT_CLR_MASK_REG, | ||
112 | .reset_to_clear = 1, | ||
113 | }, | ||
114 | }; | 153 | }; |
115 | 154 | ||
116 | static struct spear_shirq spear320_shirq_ras2 = { | 155 | static struct spear_shirq spear320_shirq_ras1 = { |
117 | .irq_nr = 1, | 156 | .offset = 7, |
118 | .irq_bit_off = 10, | 157 | .nr_irqs = 3, |
119 | .regs = { | 158 | .mask = ((0x1 << 3) - 1) << 7, |
120 | .enb_reg = -1, | 159 | .irq_chip = &dummy_irq_chip, |
121 | .status_reg = SPEAR320_INT_STS_MASK_REG, | 160 | .status_reg = SPEAR320_INT_STS_MASK_REG, |
122 | .clear_reg = SPEAR320_INT_CLR_MASK_REG, | ||
123 | .reset_to_clear = 1, | ||
124 | }, | ||
125 | }; | 161 | }; |
126 | 162 | ||
127 | static struct spear_shirq spear320_shirq_ras3 = { | 163 | static struct spear_shirq spear320_shirq_ras2 = { |
128 | .irq_nr = 7, | 164 | .offset = 10, |
129 | .irq_bit_off = 0, | 165 | .nr_irqs = 1, |
130 | .invalid_irq = 1, | 166 | .mask = ((0x1 << 1) - 1) << 10, |
131 | .regs = { | 167 | .irq_chip = &dummy_irq_chip, |
132 | .enb_reg = SPEAR320_INT_ENB_MASK_REG, | 168 | .status_reg = SPEAR320_INT_STS_MASK_REG, |
133 | .reset_to_enb = 1, | ||
134 | .status_reg = SPEAR320_INT_STS_MASK_REG, | ||
135 | .clear_reg = SPEAR320_INT_CLR_MASK_REG, | ||
136 | .reset_to_clear = 1, | ||
137 | }, | ||
138 | }; | 169 | }; |
139 | 170 | ||
140 | static struct spear_shirq spear320_shirq_intrcomm_ras = { | 171 | static struct spear_shirq spear320_shirq_intrcomm_ras = { |
141 | .irq_nr = 11, | 172 | .offset = 11, |
142 | .irq_bit_off = 11, | 173 | .nr_irqs = 11, |
143 | .regs = { | 174 | .mask = ((0x1 << 11) - 1) << 11, |
144 | .enb_reg = -1, | 175 | .irq_chip = &dummy_irq_chip, |
145 | .status_reg = SPEAR320_INT_STS_MASK_REG, | 176 | .status_reg = SPEAR320_INT_STS_MASK_REG, |
146 | .clear_reg = SPEAR320_INT_CLR_MASK_REG, | ||
147 | .reset_to_clear = 1, | ||
148 | }, | ||
149 | }; | 177 | }; |
150 | 178 | ||
151 | static struct spear_shirq *spear320_shirq_blocks[] = { | 179 | static struct spear_shirq *spear320_shirq_blocks[] = { |
@@ -155,104 +183,46 @@ static struct spear_shirq *spear320_shirq_blocks[] = { | |||
155 | &spear320_shirq_intrcomm_ras, | 183 | &spear320_shirq_intrcomm_ras, |
156 | }; | 184 | }; |
157 | 185 | ||
158 | static void shirq_irq_mask_unmask(struct irq_data *d, bool mask) | ||
159 | { | ||
160 | struct spear_shirq *shirq = irq_data_get_irq_chip_data(d); | ||
161 | u32 val, offset = d->irq - shirq->irq_base; | ||
162 | unsigned long flags; | ||
163 | |||
164 | if (shirq->regs.enb_reg == -1) | ||
165 | return; | ||
166 | |||
167 | spin_lock_irqsave(&lock, flags); | ||
168 | val = readl(shirq->base + shirq->regs.enb_reg); | ||
169 | |||
170 | if (mask ^ shirq->regs.reset_to_enb) | ||
171 | val &= ~(0x1 << shirq->irq_bit_off << offset); | ||
172 | else | ||
173 | val |= 0x1 << shirq->irq_bit_off << offset; | ||
174 | |||
175 | writel(val, shirq->base + shirq->regs.enb_reg); | ||
176 | spin_unlock_irqrestore(&lock, flags); | ||
177 | |||
178 | } | ||
179 | |||
180 | static void shirq_irq_mask(struct irq_data *d) | ||
181 | { | ||
182 | shirq_irq_mask_unmask(d, 1); | ||
183 | } | ||
184 | |||
185 | static void shirq_irq_unmask(struct irq_data *d) | ||
186 | { | ||
187 | shirq_irq_mask_unmask(d, 0); | ||
188 | } | ||
189 | |||
190 | static struct irq_chip shirq_chip = { | ||
191 | .name = "spear-shirq", | ||
192 | .irq_ack = shirq_irq_mask, | ||
193 | .irq_mask = shirq_irq_mask, | ||
194 | .irq_unmask = shirq_irq_unmask, | ||
195 | }; | ||
196 | |||
197 | static void shirq_handler(unsigned irq, struct irq_desc *desc) | 186 | static void shirq_handler(unsigned irq, struct irq_desc *desc) |
198 | { | 187 | { |
199 | u32 i, j, val, mask, tmp; | ||
200 | struct irq_chip *chip; | ||
201 | struct spear_shirq *shirq = irq_get_handler_data(irq); | 188 | struct spear_shirq *shirq = irq_get_handler_data(irq); |
189 | u32 pend; | ||
202 | 190 | ||
203 | chip = irq_get_chip(irq); | 191 | pend = readl(shirq->base + shirq->status_reg) & shirq->mask; |
204 | chip->irq_ack(&desc->irq_data); | 192 | pend >>= shirq->offset; |
205 | |||
206 | mask = ((0x1 << shirq->irq_nr) - 1) << shirq->irq_bit_off; | ||
207 | while ((val = readl(shirq->base + shirq->regs.status_reg) & | ||
208 | mask)) { | ||
209 | |||
210 | val >>= shirq->irq_bit_off; | ||
211 | for (i = 0, j = 1; i < shirq->irq_nr; i++, j <<= 1) { | ||
212 | |||
213 | if (!(j & val)) | ||
214 | continue; | ||
215 | 193 | ||
216 | generic_handle_irq(shirq->irq_base + i); | 194 | while (pend) { |
195 | int irq = __ffs(pend); | ||
217 | 196 | ||
218 | /* clear interrupt */ | 197 | pend &= ~(0x1 << irq); |
219 | if (shirq->regs.clear_reg == -1) | 198 | generic_handle_irq(shirq->virq_base + irq); |
220 | continue; | ||
221 | |||
222 | tmp = readl(shirq->base + shirq->regs.clear_reg); | ||
223 | if (shirq->regs.reset_to_clear) | ||
224 | tmp &= ~(j << shirq->irq_bit_off); | ||
225 | else | ||
226 | tmp |= (j << shirq->irq_bit_off); | ||
227 | writel(tmp, shirq->base + shirq->regs.clear_reg); | ||
228 | } | ||
229 | } | 199 | } |
230 | chip->irq_unmask(&desc->irq_data); | ||
231 | } | 200 | } |
232 | 201 | ||
233 | static void __init spear_shirq_register(struct spear_shirq *shirq) | 202 | static void __init spear_shirq_register(struct spear_shirq *shirq, |
203 | int parent_irq) | ||
234 | { | 204 | { |
235 | int i; | 205 | int i; |
236 | 206 | ||
237 | if (shirq->invalid_irq) | 207 | if (!shirq->irq_chip) |
238 | return; | 208 | return; |
239 | 209 | ||
240 | irq_set_chained_handler(shirq->irq, shirq_handler); | 210 | irq_set_chained_handler(parent_irq, shirq_handler); |
241 | for (i = 0; i < shirq->irq_nr; i++) { | 211 | irq_set_handler_data(parent_irq, shirq); |
242 | irq_set_chip_and_handler(shirq->irq_base + i, | ||
243 | &shirq_chip, handle_simple_irq); | ||
244 | set_irq_flags(shirq->irq_base + i, IRQF_VALID); | ||
245 | irq_set_chip_data(shirq->irq_base + i, shirq); | ||
246 | } | ||
247 | 212 | ||
248 | irq_set_handler_data(shirq->irq, shirq); | 213 | for (i = 0; i < shirq->nr_irqs; i++) { |
214 | irq_set_chip_and_handler(shirq->virq_base + i, | ||
215 | shirq->irq_chip, handle_simple_irq); | ||
216 | set_irq_flags(shirq->virq_base + i, IRQF_VALID); | ||
217 | irq_set_chip_data(shirq->virq_base + i, shirq); | ||
218 | } | ||
249 | } | 219 | } |
250 | 220 | ||
251 | static int __init shirq_init(struct spear_shirq **shirq_blocks, int block_nr, | 221 | static int __init shirq_init(struct spear_shirq **shirq_blocks, int block_nr, |
252 | struct device_node *np) | 222 | struct device_node *np) |
253 | { | 223 | { |
254 | int i, irq_base, hwirq = 0, irq_nr = 0; | 224 | int i, parent_irq, virq_base, hwirq = 0, nr_irqs = 0; |
255 | static struct irq_domain *shirq_domain; | 225 | struct irq_domain *shirq_domain; |
256 | void __iomem *base; | 226 | void __iomem *base; |
257 | 227 | ||
258 | base = of_iomap(np, 0); | 228 | base = of_iomap(np, 0); |
@@ -262,15 +232,15 @@ static int __init shirq_init(struct spear_shirq **shirq_blocks, int block_nr, | |||
262 | } | 232 | } |
263 | 233 | ||
264 | for (i = 0; i < block_nr; i++) | 234 | for (i = 0; i < block_nr; i++) |
265 | irq_nr += shirq_blocks[i]->irq_nr; | 235 | nr_irqs += shirq_blocks[i]->nr_irqs; |
266 | 236 | ||
267 | irq_base = irq_alloc_descs(-1, 0, irq_nr, 0); | 237 | virq_base = irq_alloc_descs(-1, 0, nr_irqs, 0); |
268 | if (IS_ERR_VALUE(irq_base)) { | 238 | if (IS_ERR_VALUE(virq_base)) { |
269 | pr_err("%s: irq desc alloc failed\n", __func__); | 239 | pr_err("%s: irq desc alloc failed\n", __func__); |
270 | goto err_unmap; | 240 | goto err_unmap; |
271 | } | 241 | } |
272 | 242 | ||
273 | shirq_domain = irq_domain_add_legacy(np, irq_nr, irq_base, 0, | 243 | shirq_domain = irq_domain_add_legacy(np, nr_irqs, virq_base, 0, |
274 | &irq_domain_simple_ops, NULL); | 244 | &irq_domain_simple_ops, NULL); |
275 | if (WARN_ON(!shirq_domain)) { | 245 | if (WARN_ON(!shirq_domain)) { |
276 | pr_warn("%s: irq domain init failed\n", __func__); | 246 | pr_warn("%s: irq domain init failed\n", __func__); |
@@ -279,41 +249,41 @@ static int __init shirq_init(struct spear_shirq **shirq_blocks, int block_nr, | |||
279 | 249 | ||
280 | for (i = 0; i < block_nr; i++) { | 250 | for (i = 0; i < block_nr; i++) { |
281 | shirq_blocks[i]->base = base; | 251 | shirq_blocks[i]->base = base; |
282 | shirq_blocks[i]->irq_base = irq_find_mapping(shirq_domain, | 252 | shirq_blocks[i]->virq_base = irq_find_mapping(shirq_domain, |
283 | hwirq); | 253 | hwirq); |
284 | shirq_blocks[i]->irq = irq_of_parse_and_map(np, i); | ||
285 | 254 | ||
286 | spear_shirq_register(shirq_blocks[i]); | 255 | parent_irq = irq_of_parse_and_map(np, i); |
287 | hwirq += shirq_blocks[i]->irq_nr; | 256 | spear_shirq_register(shirq_blocks[i], parent_irq); |
257 | hwirq += shirq_blocks[i]->nr_irqs; | ||
288 | } | 258 | } |
289 | 259 | ||
290 | return 0; | 260 | return 0; |
291 | 261 | ||
292 | err_free_desc: | 262 | err_free_desc: |
293 | irq_free_descs(irq_base, irq_nr); | 263 | irq_free_descs(virq_base, nr_irqs); |
294 | err_unmap: | 264 | err_unmap: |
295 | iounmap(base); | 265 | iounmap(base); |
296 | return -ENXIO; | 266 | return -ENXIO; |
297 | } | 267 | } |
298 | 268 | ||
299 | int __init spear300_shirq_of_init(struct device_node *np, | 269 | static int __init spear300_shirq_of_init(struct device_node *np, |
300 | struct device_node *parent) | 270 | struct device_node *parent) |
301 | { | 271 | { |
302 | return shirq_init(spear300_shirq_blocks, | 272 | return shirq_init(spear300_shirq_blocks, |
303 | ARRAY_SIZE(spear300_shirq_blocks), np); | 273 | ARRAY_SIZE(spear300_shirq_blocks), np); |
304 | } | 274 | } |
305 | IRQCHIP_DECLARE(spear300_shirq, "st,spear300-shirq", spear300_shirq_of_init); | 275 | IRQCHIP_DECLARE(spear300_shirq, "st,spear300-shirq", spear300_shirq_of_init); |
306 | 276 | ||
307 | int __init spear310_shirq_of_init(struct device_node *np, | 277 | static int __init spear310_shirq_of_init(struct device_node *np, |
308 | struct device_node *parent) | 278 | struct device_node *parent) |
309 | { | 279 | { |
310 | return shirq_init(spear310_shirq_blocks, | 280 | return shirq_init(spear310_shirq_blocks, |
311 | ARRAY_SIZE(spear310_shirq_blocks), np); | 281 | ARRAY_SIZE(spear310_shirq_blocks), np); |
312 | } | 282 | } |
313 | IRQCHIP_DECLARE(spear310_shirq, "st,spear310-shirq", spear310_shirq_of_init); | 283 | IRQCHIP_DECLARE(spear310_shirq, "st,spear310-shirq", spear310_shirq_of_init); |
314 | 284 | ||
315 | int __init spear320_shirq_of_init(struct device_node *np, | 285 | static int __init spear320_shirq_of_init(struct device_node *np, |
316 | struct device_node *parent) | 286 | struct device_node *parent) |
317 | { | 287 | { |
318 | return shirq_init(spear320_shirq_blocks, | 288 | return shirq_init(spear320_shirq_blocks, |
319 | ARRAY_SIZE(spear320_shirq_blocks), np); | 289 | ARRAY_SIZE(spear320_shirq_blocks), np); |