2 * Synopsys DW APB ICTL irqchip driver.
4 * Sebastian Hesselbarth <sebastian.hesselbarth@gmail.com>
6 * based on GPL'ed 2.6 kernel sources
7 * (c) Marvell International Ltd.
9 * This file is licensed under the terms of the GNU General Public
10 * License version 2. This program is licensed "as is" without any
11 * warranty of any kind, whether express or implied.
15 #include <linux/irq.h>
16 #include <linux/irqchip.h>
17 #include <linux/irqchip/chained_irq.h>
18 #include <linux/of_address.h>
19 #include <linux/of_irq.h>
20 #include <linux/interrupt.h>
22 #define APB_INT_ENABLE_L 0x00
23 #define APB_INT_ENABLE_H 0x04
24 #define APB_INT_MASK_L 0x08
25 #define APB_INT_MASK_H 0x0c
26 #define APB_INT_FINALSTATUS_L 0x30
27 #define APB_INT_FINALSTATUS_H 0x34
28 #define APB_INT_BASE_OFFSET 0x04
30 /* irq domain of the primary interrupt controller. */
31 static struct irq_domain
*dw_apb_ictl_irq_domain
;
33 static void __irq_entry
dw_apb_ictl_handle_irq(struct pt_regs
*regs
)
35 struct irq_domain
*d
= dw_apb_ictl_irq_domain
;
38 for (n
= 0; n
< d
->revmap_size
; n
+= 32) {
39 struct irq_chip_generic
*gc
= irq_get_domain_generic_chip(d
, n
);
40 u32 stat
= readl_relaxed(gc
->reg_base
+ APB_INT_FINALSTATUS_L
);
43 u32 hwirq
= ffs(stat
) - 1;
45 generic_handle_domain_irq(d
, hwirq
);
51 static void dw_apb_ictl_handle_irq_cascaded(struct irq_desc
*desc
)
53 struct irq_domain
*d
= irq_desc_get_handler_data(desc
);
54 struct irq_chip
*chip
= irq_desc_get_chip(desc
);
57 chained_irq_enter(chip
, desc
);
59 for (n
= 0; n
< d
->revmap_size
; n
+= 32) {
60 struct irq_chip_generic
*gc
= irq_get_domain_generic_chip(d
, n
);
61 u32 stat
= readl_relaxed(gc
->reg_base
+ APB_INT_FINALSTATUS_L
);
64 u32 hwirq
= ffs(stat
) - 1;
65 generic_handle_domain_irq(d
, gc
->irq_base
+ hwirq
);
71 chained_irq_exit(chip
, desc
);
74 static int dw_apb_ictl_irq_domain_alloc(struct irq_domain
*domain
, unsigned int virq
,
75 unsigned int nr_irqs
, void *arg
)
78 irq_hw_number_t hwirq
;
79 unsigned int type
= IRQ_TYPE_NONE
;
80 struct irq_fwspec
*fwspec
= arg
;
82 ret
= irq_domain_translate_onecell(domain
, fwspec
, &hwirq
, &type
);
86 for (i
= 0; i
< nr_irqs
; i
++)
87 irq_map_generic_chip(domain
, virq
+ i
, hwirq
+ i
);
92 static const struct irq_domain_ops dw_apb_ictl_irq_domain_ops
= {
93 .translate
= irq_domain_translate_onecell
,
94 .alloc
= dw_apb_ictl_irq_domain_alloc
,
95 .free
= irq_domain_free_irqs_top
,
99 static void dw_apb_ictl_resume(struct irq_data
*d
)
101 struct irq_chip_generic
*gc
= irq_data_get_irq_chip_data(d
);
102 struct irq_chip_type
*ct
= irq_data_get_chip_type(d
);
105 writel_relaxed(~0, gc
->reg_base
+ ct
->regs
.enable
);
106 writel_relaxed(*ct
->mask_cache
, gc
->reg_base
+ ct
->regs
.mask
);
110 #define dw_apb_ictl_resume NULL
111 #endif /* CONFIG_PM */
113 static int __init
dw_apb_ictl_init(struct device_node
*np
,
114 struct device_node
*parent
)
116 const struct irq_domain_ops
*domain_ops
;
117 unsigned int clr
= IRQ_NOREQUEST
| IRQ_NOPROBE
| IRQ_NOAUTOEN
;
119 struct irq_domain
*domain
;
120 struct irq_chip_generic
*gc
;
121 void __iomem
*iobase
;
122 int ret
, nrirqs
, parent_irq
, i
;
126 /* Used as the primary interrupt controller */
128 domain_ops
= &dw_apb_ictl_irq_domain_ops
;
130 /* Map the parent interrupt for the chained handler */
131 parent_irq
= irq_of_parse_and_map(np
, 0);
132 if (parent_irq
<= 0) {
133 pr_err("%pOF: unable to parse irq\n", np
);
136 domain_ops
= &irq_generic_chip_ops
;
139 ret
= of_address_to_resource(np
, 0, &r
);
141 pr_err("%pOF: unable to get resource\n", np
);
145 if (!request_mem_region(r
.start
, resource_size(&r
), np
->full_name
)) {
146 pr_err("%pOF: unable to request mem region\n", np
);
150 iobase
= ioremap(r
.start
, resource_size(&r
));
152 pr_err("%pOF: unable to map resource\n", np
);
158 * DW IP can be configured to allow 2-64 irqs. We can determine
159 * the number of irqs supported by writing into enable register
160 * and look for bits not set, as corresponding flip-flops will
161 * have been removed by synthesis tool.
164 /* mask and enable all interrupts */
165 writel_relaxed(~0, iobase
+ APB_INT_MASK_L
);
166 writel_relaxed(~0, iobase
+ APB_INT_MASK_H
);
167 writel_relaxed(~0, iobase
+ APB_INT_ENABLE_L
);
168 writel_relaxed(~0, iobase
+ APB_INT_ENABLE_H
);
170 reg
= readl_relaxed(iobase
+ APB_INT_ENABLE_H
);
172 nrirqs
= 32 + fls(reg
);
174 nrirqs
= fls(readl_relaxed(iobase
+ APB_INT_ENABLE_L
));
176 domain
= irq_domain_add_linear(np
, nrirqs
, domain_ops
, NULL
);
178 pr_err("%pOF: unable to add irq domain\n", np
);
183 ret
= irq_alloc_domain_generic_chips(domain
, 32, 1, np
->name
,
184 handle_level_irq
, clr
, 0,
185 IRQ_GC_INIT_MASK_CACHE
);
187 pr_err("%pOF: unable to alloc irq domain gc\n", np
);
191 for (i
= 0; i
< DIV_ROUND_UP(nrirqs
, 32); i
++) {
192 gc
= irq_get_domain_generic_chip(domain
, i
* 32);
193 gc
->reg_base
= iobase
+ i
* APB_INT_BASE_OFFSET
;
194 gc
->chip_types
[0].regs
.mask
= APB_INT_MASK_L
;
195 gc
->chip_types
[0].regs
.enable
= APB_INT_ENABLE_L
;
196 gc
->chip_types
[0].chip
.irq_mask
= irq_gc_mask_set_bit
;
197 gc
->chip_types
[0].chip
.irq_unmask
= irq_gc_mask_clr_bit
;
198 gc
->chip_types
[0].chip
.irq_resume
= dw_apb_ictl_resume
;
202 irq_set_chained_handler_and_data(parent_irq
,
203 dw_apb_ictl_handle_irq_cascaded
, domain
);
205 dw_apb_ictl_irq_domain
= domain
;
206 set_handle_irq(dw_apb_ictl_handle_irq
);
214 release_mem_region(r
.start
, resource_size(&r
));
217 IRQCHIP_DECLARE(dw_apb_ictl
,
218 "snps,dw-apb-ictl", dw_apb_ictl_init
);