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 handle_domain_irq(d
, hwirq
, regs
);
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 u32 virq
= irq_find_mapping(d
, gc
->irq_base
+ hwirq
);
67 generic_handle_irq(virq
);
72 chained_irq_exit(chip
, desc
);
75 static int dw_apb_ictl_irq_domain_alloc(struct irq_domain
*domain
, unsigned int virq
,
76 unsigned int nr_irqs
, void *arg
)
79 irq_hw_number_t hwirq
;
80 unsigned int type
= IRQ_TYPE_NONE
;
81 struct irq_fwspec
*fwspec
= arg
;
83 ret
= irq_domain_translate_onecell(domain
, fwspec
, &hwirq
, &type
);
87 for (i
= 0; i
< nr_irqs
; i
++)
88 irq_map_generic_chip(domain
, virq
+ i
, hwirq
+ i
);
93 static const struct irq_domain_ops dw_apb_ictl_irq_domain_ops
= {
94 .translate
= irq_domain_translate_onecell
,
95 .alloc
= dw_apb_ictl_irq_domain_alloc
,
96 .free
= irq_domain_free_irqs_top
,
100 static void dw_apb_ictl_resume(struct irq_data
*d
)
102 struct irq_chip_generic
*gc
= irq_data_get_irq_chip_data(d
);
103 struct irq_chip_type
*ct
= irq_data_get_chip_type(d
);
106 writel_relaxed(~0, gc
->reg_base
+ ct
->regs
.enable
);
107 writel_relaxed(*ct
->mask_cache
, gc
->reg_base
+ ct
->regs
.mask
);
111 #define dw_apb_ictl_resume NULL
112 #endif /* CONFIG_PM */
114 static int __init
dw_apb_ictl_init(struct device_node
*np
,
115 struct device_node
*parent
)
117 const struct irq_domain_ops
*domain_ops
;
118 unsigned int clr
= IRQ_NOREQUEST
| IRQ_NOPROBE
| IRQ_NOAUTOEN
;
120 struct irq_domain
*domain
;
121 struct irq_chip_generic
*gc
;
122 void __iomem
*iobase
;
123 int ret
, nrirqs
, parent_irq
, i
;
127 /* Used as the primary interrupt controller */
129 domain_ops
= &dw_apb_ictl_irq_domain_ops
;
131 /* Map the parent interrupt for the chained handler */
132 parent_irq
= irq_of_parse_and_map(np
, 0);
133 if (parent_irq
<= 0) {
134 pr_err("%pOF: unable to parse irq\n", np
);
137 domain_ops
= &irq_generic_chip_ops
;
140 ret
= of_address_to_resource(np
, 0, &r
);
142 pr_err("%pOF: unable to get resource\n", np
);
146 if (!request_mem_region(r
.start
, resource_size(&r
), np
->full_name
)) {
147 pr_err("%pOF: unable to request mem region\n", np
);
151 iobase
= ioremap(r
.start
, resource_size(&r
));
153 pr_err("%pOF: unable to map resource\n", np
);
159 * DW IP can be configured to allow 2-64 irqs. We can determine
160 * the number of irqs supported by writing into enable register
161 * and look for bits not set, as corresponding flip-flops will
162 * have been removed by synthesis tool.
165 /* mask and enable all interrupts */
166 writel_relaxed(~0, iobase
+ APB_INT_MASK_L
);
167 writel_relaxed(~0, iobase
+ APB_INT_MASK_H
);
168 writel_relaxed(~0, iobase
+ APB_INT_ENABLE_L
);
169 writel_relaxed(~0, iobase
+ APB_INT_ENABLE_H
);
171 reg
= readl_relaxed(iobase
+ APB_INT_ENABLE_H
);
173 nrirqs
= 32 + fls(reg
);
175 nrirqs
= fls(readl_relaxed(iobase
+ APB_INT_ENABLE_L
));
177 domain
= irq_domain_add_linear(np
, nrirqs
, domain_ops
, NULL
);
179 pr_err("%pOF: unable to add irq domain\n", np
);
184 ret
= irq_alloc_domain_generic_chips(domain
, 32, 1, np
->name
,
185 handle_level_irq
, clr
, 0,
186 IRQ_GC_INIT_MASK_CACHE
);
188 pr_err("%pOF: unable to alloc irq domain gc\n", np
);
192 for (i
= 0; i
< DIV_ROUND_UP(nrirqs
, 32); i
++) {
193 gc
= irq_get_domain_generic_chip(domain
, i
* 32);
194 gc
->reg_base
= iobase
+ i
* APB_INT_BASE_OFFSET
;
195 gc
->chip_types
[0].regs
.mask
= APB_INT_MASK_L
;
196 gc
->chip_types
[0].regs
.enable
= APB_INT_ENABLE_L
;
197 gc
->chip_types
[0].chip
.irq_mask
= irq_gc_mask_set_bit
;
198 gc
->chip_types
[0].chip
.irq_unmask
= irq_gc_mask_clr_bit
;
199 gc
->chip_types
[0].chip
.irq_resume
= dw_apb_ictl_resume
;
203 irq_set_chained_handler_and_data(parent_irq
,
204 dw_apb_ictl_handle_irq_cascaded
, domain
);
206 dw_apb_ictl_irq_domain
= domain
;
207 set_handle_irq(dw_apb_ictl_handle_irq
);
215 release_mem_region(r
.start
, resource_size(&r
));
218 IRQCHIP_DECLARE(dw_apb_ictl
,
219 "snps,dw-apb-ictl", dw_apb_ictl_init
);