Merge tag 'irq-urgent-2020-11-29' of git://git.kernel.org/pub/scm/linux/kernel/git...
[linux-block.git] / drivers / irqchip / irq-tango.c
CommitLineData
2874c5fd 1// SPDX-License-Identifier: GPL-2.0-or-later
4bba6689
MR
2/*
3 * Copyright (C) 2014 Mans Rullgard <mans@mansr.com>
4bba6689
MR
4 */
5
6#include <linux/init.h>
7#include <linux/irq.h>
8#include <linux/irqchip.h>
9#include <linux/irqchip/chained_irq.h>
10#include <linux/ioport.h>
11#include <linux/io.h>
12#include <linux/of_address.h>
13#include <linux/of_irq.h>
14#include <linux/slab.h>
15
16#define IRQ0_CTL_BASE 0x0000
17#define IRQ1_CTL_BASE 0x0100
18#define EDGE_CTL_BASE 0x0200
19#define IRQ2_CTL_BASE 0x0300
20
21#define IRQ_CTL_HI 0x18
22#define EDGE_CTL_HI 0x20
23
24#define IRQ_STATUS 0x00
25#define IRQ_RAWSTAT 0x04
26#define IRQ_EN_SET 0x08
27#define IRQ_EN_CLR 0x0c
28#define IRQ_SOFT_SET 0x10
29#define IRQ_SOFT_CLR 0x14
30
31#define EDGE_STATUS 0x00
32#define EDGE_RAWSTAT 0x04
33#define EDGE_CFG_RISE 0x08
34#define EDGE_CFG_FALL 0x0c
35#define EDGE_CFG_RISE_SET 0x10
36#define EDGE_CFG_RISE_CLR 0x14
37#define EDGE_CFG_FALL_SET 0x18
38#define EDGE_CFG_FALL_CLR 0x1c
39
40struct tangox_irq_chip {
41 void __iomem *base;
42 unsigned long ctl;
43};
44
45static inline u32 intc_readl(struct tangox_irq_chip *chip, int reg)
46{
47 return readl_relaxed(chip->base + reg);
48}
49
50static inline void intc_writel(struct tangox_irq_chip *chip, int reg, u32 val)
51{
52 writel_relaxed(val, chip->base + reg);
53}
54
55static void tangox_dispatch_irqs(struct irq_domain *dom, unsigned int status,
56 int base)
57{
58 unsigned int hwirq;
59 unsigned int virq;
60
61 while (status) {
62 hwirq = __ffs(status);
63 virq = irq_find_mapping(dom, base + hwirq);
64 if (virq)
65 generic_handle_irq(virq);
66 status &= ~BIT(hwirq);
67 }
68}
69
70static void tangox_irq_handler(struct irq_desc *desc)
71{
72 struct irq_domain *dom = irq_desc_get_handler_data(desc);
73 struct irq_chip *host_chip = irq_desc_get_chip(desc);
74 struct tangox_irq_chip *chip = dom->host_data;
75 unsigned int status_lo, status_hi;
76
77 chained_irq_enter(host_chip, desc);
78
79 status_lo = intc_readl(chip, chip->ctl + IRQ_STATUS);
80 status_hi = intc_readl(chip, chip->ctl + IRQ_CTL_HI + IRQ_STATUS);
81
82 tangox_dispatch_irqs(dom, status_lo, 0);
83 tangox_dispatch_irqs(dom, status_hi, 32);
84
85 chained_irq_exit(host_chip, desc);
86}
87
88static int tangox_irq_set_type(struct irq_data *d, unsigned int flow_type)
89{
90 struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d);
91 struct tangox_irq_chip *chip = gc->domain->host_data;
92 struct irq_chip_regs *regs = &gc->chip_types[0].regs;
93
94 switch (flow_type & IRQ_TYPE_SENSE_MASK) {
95 case IRQ_TYPE_EDGE_RISING:
96 intc_writel(chip, regs->type + EDGE_CFG_RISE_SET, d->mask);
97 intc_writel(chip, regs->type + EDGE_CFG_FALL_CLR, d->mask);
98 break;
99
100 case IRQ_TYPE_EDGE_FALLING:
101 intc_writel(chip, regs->type + EDGE_CFG_RISE_CLR, d->mask);
102 intc_writel(chip, regs->type + EDGE_CFG_FALL_SET, d->mask);
103 break;
104
105 case IRQ_TYPE_LEVEL_HIGH:
106 intc_writel(chip, regs->type + EDGE_CFG_RISE_CLR, d->mask);
107 intc_writel(chip, regs->type + EDGE_CFG_FALL_CLR, d->mask);
108 break;
109
110 case IRQ_TYPE_LEVEL_LOW:
111 intc_writel(chip, regs->type + EDGE_CFG_RISE_SET, d->mask);
112 intc_writel(chip, regs->type + EDGE_CFG_FALL_SET, d->mask);
113 break;
114
115 default:
116 pr_err("Invalid trigger mode %x for IRQ %d\n",
117 flow_type, d->irq);
118 return -EINVAL;
119 }
120
121 return irq_setup_alt_chip(d, flow_type);
122}
123
124static void __init tangox_irq_init_chip(struct irq_chip_generic *gc,
125 unsigned long ctl_offs,
126 unsigned long edge_offs)
127{
128 struct tangox_irq_chip *chip = gc->domain->host_data;
129 struct irq_chip_type *ct = gc->chip_types;
130 unsigned long ctl_base = chip->ctl + ctl_offs;
131 unsigned long edge_base = EDGE_CTL_BASE + edge_offs;
132 int i;
133
134 gc->reg_base = chip->base;
135 gc->unused = 0;
136
137 for (i = 0; i < 2; i++) {
138 ct[i].chip.irq_ack = irq_gc_ack_set_bit;
139 ct[i].chip.irq_mask = irq_gc_mask_disable_reg;
16150904 140 ct[i].chip.irq_mask_ack = irq_gc_mask_disable_and_ack_set;
4bba6689
MR
141 ct[i].chip.irq_unmask = irq_gc_unmask_enable_reg;
142 ct[i].chip.irq_set_type = tangox_irq_set_type;
143 ct[i].chip.name = gc->domain->name;
144
145 ct[i].regs.enable = ctl_base + IRQ_EN_SET;
146 ct[i].regs.disable = ctl_base + IRQ_EN_CLR;
147 ct[i].regs.ack = edge_base + EDGE_RAWSTAT;
148 ct[i].regs.type = edge_base;
149 }
150
151 ct[0].type = IRQ_TYPE_LEVEL_MASK;
152 ct[0].handler = handle_level_irq;
153
154 ct[1].type = IRQ_TYPE_EDGE_BOTH;
155 ct[1].handler = handle_edge_irq;
156
157 intc_writel(chip, ct->regs.disable, 0xffffffff);
158 intc_writel(chip, ct->regs.ack, 0xffffffff);
159}
160
161static void __init tangox_irq_domain_init(struct irq_domain *dom)
162{
163 struct irq_chip_generic *gc;
164 int i;
165
166 for (i = 0; i < 2; i++) {
167 gc = irq_get_domain_generic_chip(dom, i * 32);
168 tangox_irq_init_chip(gc, i * IRQ_CTL_HI, i * EDGE_CTL_HI);
169 }
170}
171
172static int __init tangox_irq_init(void __iomem *base, struct resource *baseres,
173 struct device_node *node)
174{
175 struct tangox_irq_chip *chip;
176 struct irq_domain *dom;
177 struct resource res;
178 int irq;
179 int err;
180
181 irq = irq_of_parse_and_map(node, 0);
182 if (!irq)
f9c75bca 183 panic("%pOFn: failed to get IRQ", node);
4bba6689
MR
184
185 err = of_address_to_resource(node, 0, &res);
186 if (err)
f9c75bca 187 panic("%pOFn: failed to get address", node);
4bba6689
MR
188
189 chip = kzalloc(sizeof(*chip), GFP_KERNEL);
190 chip->ctl = res.start - baseres->start;
191 chip->base = base;
192
193 dom = irq_domain_add_linear(node, 64, &irq_generic_chip_ops, chip);
194 if (!dom)
f9c75bca 195 panic("%pOFn: failed to create irqdomain", node);
4bba6689
MR
196
197 err = irq_alloc_domain_generic_chips(dom, 32, 2, node->name,
198 handle_level_irq, 0, 0, 0);
199 if (err)
f9c75bca 200 panic("%pOFn: failed to allocate irqchip", node);
4bba6689
MR
201
202 tangox_irq_domain_init(dom);
203
44beda0c 204 irq_set_chained_handler_and_data(irq, tangox_irq_handler, dom);
4bba6689
MR
205
206 return 0;
207}
208
209static int __init tangox_of_irq_init(struct device_node *node,
210 struct device_node *parent)
211{
212 struct device_node *c;
213 struct resource res;
214 void __iomem *base;
215
216 base = of_iomap(node, 0);
217 if (!base)
f9c75bca 218 panic("%pOFn: of_iomap failed", node);
4bba6689
MR
219
220 of_address_to_resource(node, 0, &res);
221
222 for_each_child_of_node(node, c)
223 tangox_irq_init(base, &res, c);
224
225 return 0;
226}
227IRQCHIP_DECLARE(tangox_intc, "sigma,smp8642-intc", tangox_of_irq_init);