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(unsigned int irq
, struct irq_desc
*desc
)
98 struct megamod_cascade_data
*cascade
;
99 struct megamod_pic
*pic
;
103 cascade
= irq_desc_get_handler_data(desc
);
106 idx
= cascade
->index
;
108 while ((events
= soc_readl(&pic
->regs
->mevtflag
[idx
])) != 0) {
111 irq
= irq_linear_revmap(pic
->irqhost
, idx
* 32 + n
);
113 soc_writel(1 << n
, &pic
->regs
->evtclr
[idx
]);
115 generic_handle_irq(irq
);
119 static int megamod_map(struct irq_domain
*h
, unsigned int virq
,
122 struct megamod_pic
*pic
= h
->host_data
;
125 /* We shouldn't see a hwirq which is muxed to core controller */
126 for (i
= 0; i
< NR_MUX_OUTPUTS
; i
++)
127 if (pic
->output_to_irq
[i
] == hw
)
130 irq_set_chip_data(virq
, pic
);
131 irq_set_chip_and_handler(virq
, &megamod_chip
, handle_level_irq
);
133 /* Set default irq type */
134 irq_set_irq_type(virq
, IRQ_TYPE_NONE
);
139 static const struct irq_domain_ops megamod_domain_ops
= {
141 .xlate
= irq_domain_xlate_onecell
,
144 static void __init
set_megamod_mux(struct megamod_pic
*pic
, int src
, int output
)
149 if (src
< 0 || src
>= (NR_COMBINERS
* 32)) {
150 pic
->output_to_irq
[output
] = IRQ_UNMAPPED
;
154 /* four mappings per mux register */
156 offset
= (output
& 3) * 8;
158 val
= soc_readl(&pic
->regs
->intmux
[index
]);
159 val
&= ~(0xff << offset
);
160 val
|= src
<< offset
;
161 soc_writel(val
, &pic
->regs
->intmux
[index
]);
165 * Parse the MUX mapping, if one exists.
167 * The MUX map is an array of up to 12 cells; one for each usable core priority
168 * interrupt. The value of a given cell is the megamodule interrupt source
169 * which is to me MUXed to the output corresponding to the cell position
170 * withing the array. The first cell in the array corresponds to priority
171 * 4 and the last (12th) cell corresponds to priority 15. The allowed
172 * values are 4 - ((NR_COMBINERS * 32) - 1). Note that the combined interrupt
173 * sources (0 - 3) are not allowed to be mapped through this property. They
174 * are handled through the "interrupts" property. This allows us to use a
175 * value of zero as a "do not map" placeholder.
177 static void __init
parse_priority_map(struct megamod_pic
*pic
,
178 int *mapping
, int size
)
180 struct device_node
*np
= pic
->irqhost
->of_node
;
185 map
= of_get_property(np
, "ti,c64x+megamod-pic-mux", &maplen
);
191 for (i
= 0; i
< maplen
; i
++) {
192 val
= be32_to_cpup(map
);
200 static struct megamod_pic
* __init
init_megamod_pic(struct device_node
*np
)
202 struct megamod_pic
*pic
;
204 int mapping
[NR_MUX_OUTPUTS
];
206 pr_info("Initializing C64x+ Megamodule PIC\n");
208 pic
= kzalloc(sizeof(struct megamod_pic
), GFP_KERNEL
);
210 pr_err("%s: Could not alloc PIC structure.\n", np
->full_name
);
214 pic
->irqhost
= irq_domain_add_linear(np
, NR_COMBINERS
* 32,
215 &megamod_domain_ops
, pic
);
217 pr_err("%s: Could not alloc host.\n", np
->full_name
);
221 pic
->irqhost
->host_data
= pic
;
223 raw_spin_lock_init(&pic
->lock
);
225 pic
->regs
= of_iomap(np
, 0);
227 pr_err("%s: Could not map registers.\n", np
->full_name
);
231 /* Initialize MUX map */
232 for (i
= 0; i
< ARRAY_SIZE(mapping
); i
++)
233 mapping
[i
] = IRQ_UNMAPPED
;
235 parse_priority_map(pic
, mapping
, ARRAY_SIZE(mapping
));
238 * We can have up to 12 interrupts cascading to the core controller.
239 * These cascades can be from the combined interrupt sources or for
240 * individual interrupt sources. The "interrupts" property only
241 * deals with the cascaded combined interrupts. The individual
242 * interrupts muxed to the core controller use the core controller
243 * as their interrupt parent.
245 for (i
= 0; i
< NR_COMBINERS
; i
++) {
246 struct irq_data
*irq_data
;
247 irq_hw_number_t hwirq
;
249 irq
= irq_of_parse_and_map(np
, i
);
253 irq_data
= irq_get_irq_data(irq
);
255 pr_err("%s: combiner-%d no irq_data for virq %d!\n",
256 np
->full_name
, i
, irq
);
260 hwirq
= irq_data
->hwirq
;
263 * Check that device tree provided something in the range
264 * of the core priority interrupts (4 - 15).
266 if (hwirq
< 4 || hwirq
>= NR_PRIORITY_IRQS
) {
267 pr_err("%s: combiner-%d core irq %ld out of range!\n",
268 np
->full_name
, i
, hwirq
);
272 /* record the mapping */
273 mapping
[hwirq
- 4] = i
;
275 pr_debug("%s: combiner-%d cascading to hwirq %ld\n",
276 np
->full_name
, i
, hwirq
);
278 cascade_data
[i
].pic
= pic
;
279 cascade_data
[i
].index
= i
;
281 /* mask and clear all events in combiner */
282 soc_writel(~0, &pic
->regs
->evtmask
[i
]);
283 soc_writel(~0, &pic
->regs
->evtclr
[i
]);
285 irq_set_handler_data(irq
, &cascade_data
[i
]);
286 irq_set_chained_handler(irq
, megamod_irq_cascade
);
289 /* Finally, set up the MUX registers */
290 for (i
= 0; i
< NR_MUX_OUTPUTS
; i
++) {
291 if (mapping
[i
] != IRQ_UNMAPPED
) {
292 pr_debug("%s: setting mux %d to priority %d\n",
293 np
->full_name
, mapping
[i
], i
+ 4);
294 set_megamod_mux(pic
, mapping
[i
], i
);
307 * Return next active event after ACK'ing it.
308 * Return -1 if no events active.
310 static int get_exception(void)
315 for (i
= 0; i
< NR_COMBINERS
; i
++) {
316 mask
= soc_readl(&mm_pic
->regs
->mexpflag
[i
]);
319 soc_writel(1 << bit
, &mm_pic
->regs
->evtclr
[i
]);
320 return (i
* 32) + bit
;
326 static void assert_event(unsigned int val
)
328 soc_writel(val
, &mm_pic
->regs
->evtasrt
);
331 void __init
megamod_pic_init(void)
333 struct device_node
*np
;
335 np
= of_find_compatible_node(NULL
, NULL
, "ti,c64x+megamod-pic");
339 mm_pic
= init_megamod_pic(np
);
342 soc_ops
.get_exception
= get_exception
;
343 soc_ops
.assert_event
= assert_event
;