2 * Support for C64x+ Megamodule Interrupt Controller
4 * Copyright (C) 2010, 2011 Texas Instruments Incorporated
5 * Contributed by: Mark Salter <msalter@redhat.com>
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License version 2 as
9 * published by the Free Software Foundation.
11 #include <linux/module.h>
12 #include <linux/interrupt.h>
15 #include <linux/of_irq.h>
16 #include <linux/of_address.h>
17 #include <linux/slab.h>
19 #include <asm/megamod-pic.h>
21 #define NR_COMBINERS 4
22 #define NR_MUX_OUTPUTS 12
24 #define IRQ_UNMAPPED 0xffff
27 * Megamodule Interrupt Controller register layout
51 struct irq_domain *irqhost;
52 struct megamod_regs __iomem *regs;
56 unsigned int output_to_irq[NR_MUX_OUTPUTS];
59 static struct megamod_pic *mm_pic;
61 struct megamod_cascade_data {
62 struct megamod_pic *pic;
66 static struct megamod_cascade_data cascade_data[NR_COMBINERS];
68 static void mask_megamod(struct irq_data *data)
70 struct megamod_pic *pic = irq_data_get_irq_chip_data(data);
71 irq_hw_number_t src = irqd_to_hwirq(data);
72 u32 __iomem *evtmask = &pic->regs->evtmask[src / 32];
74 raw_spin_lock(&pic->lock);
75 soc_writel(soc_readl(evtmask) | (1 << (src & 31)), evtmask);
76 raw_spin_unlock(&pic->lock);
79 static void unmask_megamod(struct irq_data *data)
81 struct megamod_pic *pic = irq_data_get_irq_chip_data(data);
82 irq_hw_number_t src = irqd_to_hwirq(data);
83 u32 __iomem *evtmask = &pic->regs->evtmask[src / 32];
85 raw_spin_lock(&pic->lock);
86 soc_writel(soc_readl(evtmask) & ~(1 << (src & 31)), evtmask);
87 raw_spin_unlock(&pic->lock);
90 static struct irq_chip megamod_chip = {
92 .irq_mask = mask_megamod,
93 .irq_unmask = unmask_megamod,
96 static void megamod_irq_cascade(struct irq_desc *desc)
98 struct megamod_cascade_data *cascade;
99 struct megamod_pic *pic;
104 cascade = irq_desc_get_handler_data(desc);
107 idx = cascade->index;
109 while ((events = soc_readl(&pic->regs->mevtflag[idx])) != 0) {
112 irq = irq_linear_revmap(pic->irqhost, idx * 32 + n);
114 soc_writel(1 << n, &pic->regs->evtclr[idx]);
116 generic_handle_irq(irq);
120 static int megamod_map(struct irq_domain *h, unsigned int virq,
123 struct megamod_pic *pic = h->host_data;
126 /* We shouldn't see a hwirq which is muxed to core controller */
127 for (i = 0; i < NR_MUX_OUTPUTS; i++)
128 if (pic->output_to_irq[i] == hw)
131 irq_set_chip_data(virq, pic);
132 irq_set_chip_and_handler(virq, &megamod_chip, handle_level_irq);
134 /* Set default irq type */
135 irq_set_irq_type(virq, IRQ_TYPE_NONE);
140 static const struct irq_domain_ops megamod_domain_ops = {
142 .xlate = irq_domain_xlate_onecell,
145 static void __init set_megamod_mux(struct megamod_pic *pic, int src, int output)
150 if (src < 0 || src >= (NR_COMBINERS * 32)) {
151 pic->output_to_irq[output] = IRQ_UNMAPPED;
155 /* four mappings per mux register */
157 offset = (output & 3) * 8;
159 val = soc_readl(&pic->regs->intmux[index]);
160 val &= ~(0xff << offset);
161 val |= src << offset;
162 soc_writel(val, &pic->regs->intmux[index]);
166 * Parse the MUX mapping, if one exists.
168 * The MUX map is an array of up to 12 cells; one for each usable core priority
169 * interrupt. The value of a given cell is the megamodule interrupt source
170 * which is to me MUXed to the output corresponding to the cell position
171 * withing the array. The first cell in the array corresponds to priority
172 * 4 and the last (12th) cell corresponds to priority 15. The allowed
173 * values are 4 - ((NR_COMBINERS * 32) - 1). Note that the combined interrupt
174 * sources (0 - 3) are not allowed to be mapped through this property. They
175 * are handled through the "interrupts" property. This allows us to use a
176 * value of zero as a "do not map" placeholder.
178 static void __init parse_priority_map(struct megamod_pic *pic,
179 int *mapping, int size)
181 struct device_node *np = irq_domain_get_of_node(pic->irqhost);
186 map = of_get_property(np, "ti,c64x+megamod-pic-mux", &maplen);
192 for (i = 0; i < maplen; i++) {
193 val = be32_to_cpup(map);
201 static struct megamod_pic * __init init_megamod_pic(struct device_node *np)
203 struct megamod_pic *pic;
205 int mapping[NR_MUX_OUTPUTS];
207 pr_info("Initializing C64x+ Megamodule PIC\n");
209 pic = kzalloc(sizeof(struct megamod_pic), GFP_KERNEL);
211 pr_err("%s: Could not alloc PIC structure.\n", np->full_name);
215 pic->irqhost = irq_domain_add_linear(np, NR_COMBINERS * 32,
216 &megamod_domain_ops, pic);
218 pr_err("%s: Could not alloc host.\n", np->full_name);
222 pic->irqhost->host_data = pic;
224 raw_spin_lock_init(&pic->lock);
226 pic->regs = of_iomap(np, 0);
228 pr_err("%s: Could not map registers.\n", np->full_name);
232 /* Initialize MUX map */
233 for (i = 0; i < ARRAY_SIZE(mapping); i++)
234 mapping[i] = IRQ_UNMAPPED;
236 parse_priority_map(pic, mapping, ARRAY_SIZE(mapping));
239 * We can have up to 12 interrupts cascading to the core controller.
240 * These cascades can be from the combined interrupt sources or for
241 * individual interrupt sources. The "interrupts" property only
242 * deals with the cascaded combined interrupts. The individual
243 * interrupts muxed to the core controller use the core controller
244 * as their interrupt parent.
246 for (i = 0; i < NR_COMBINERS; i++) {
247 struct irq_data *irq_data;
248 irq_hw_number_t hwirq;
250 irq = irq_of_parse_and_map(np, i);
254 irq_data = irq_get_irq_data(irq);
256 pr_err("%s: combiner-%d no irq_data for virq %d!\n",
257 np->full_name, i, irq);
261 hwirq = irq_data->hwirq;
264 * Check that device tree provided something in the range
265 * of the core priority interrupts (4 - 15).
267 if (hwirq < 4 || hwirq >= NR_PRIORITY_IRQS) {
268 pr_err("%s: combiner-%d core irq %ld out of range!\n",
269 np->full_name, i, hwirq);
273 /* record the mapping */
274 mapping[hwirq - 4] = i;
276 pr_debug("%s: combiner-%d cascading to hwirq %ld\n",
277 np->full_name, i, hwirq);
279 cascade_data[i].pic = pic;
280 cascade_data[i].index = i;
282 /* mask and clear all events in combiner */
283 soc_writel(~0, &pic->regs->evtmask[i]);
284 soc_writel(~0, &pic->regs->evtclr[i]);
286 irq_set_chained_handler_and_data(irq, megamod_irq_cascade,
290 /* Finally, set up the MUX registers */
291 for (i = 0; i < NR_MUX_OUTPUTS; i++) {
292 if (mapping[i] != IRQ_UNMAPPED) {
293 pr_debug("%s: setting mux %d to priority %d\n",
294 np->full_name, mapping[i], i + 4);
295 set_megamod_mux(pic, mapping[i], i);
308 * Return next active event after ACK'ing it.
309 * Return -1 if no events active.
311 static int get_exception(void)
316 for (i = 0; i < NR_COMBINERS; i++) {
317 mask = soc_readl(&mm_pic->regs->mexpflag[i]);
320 soc_writel(1 << bit, &mm_pic->regs->evtclr[i]);
321 return (i * 32) + bit;
327 static void assert_event(unsigned int val)
329 soc_writel(val, &mm_pic->regs->evtasrt);
332 void __init megamod_pic_init(void)
334 struct device_node *np;
336 np = of_find_compatible_node(NULL, NULL, "ti,c64x+megamod-pic");
340 mm_pic = init_megamod_pic(np);
343 soc_ops.get_exception = get_exception;
344 soc_ops.assert_event = assert_event;