2 * linux/arch/unicore32/kernel/irq.c
4 * Code specific to PKUnity SoC and UniCore ISA
6 * Copyright (C) 2001-2010 GUAN Xue-tao
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
12 #include <linux/kernel_stat.h>
13 #include <linux/module.h>
14 #include <linux/signal.h>
15 #include <linux/ioport.h>
16 #include <linux/interrupt.h>
17 #include <linux/irq.h>
18 #include <linux/random.h>
19 #include <linux/smp.h>
20 #include <linux/init.h>
21 #include <linux/seq_file.h>
22 #include <linux/errno.h>
23 #include <linux/list.h>
24 #include <linux/kallsyms.h>
25 #include <linux/proc_fs.h>
26 #include <linux/syscore_ops.h>
27 #include <linux/gpio.h>
29 #include <mach/hardware.h>
34 * PKUnity GPIO edge detection for IRQs:
35 * IRQs are generated on Falling-Edge, Rising-Edge, or both.
36 * Use this instead of directly setting GRER/GFER.
38 static int GPIO_IRQ_rising_edge
;
39 static int GPIO_IRQ_falling_edge
;
40 static int GPIO_IRQ_mask
= 0;
42 #define GPIO_MASK(irq) (1 << (irq - IRQ_GPIO0))
44 static int puv3_gpio_type(struct irq_data
*d
, unsigned int type
)
48 if (d
->irq
< IRQ_GPIOHIGH
)
51 mask
= GPIO_MASK(d
->irq
);
53 if (type
== IRQ_TYPE_PROBE
) {
54 if ((GPIO_IRQ_rising_edge
| GPIO_IRQ_falling_edge
) & mask
)
56 type
= IRQ_TYPE_EDGE_RISING
| IRQ_TYPE_EDGE_FALLING
;
59 if (type
& IRQ_TYPE_EDGE_RISING
)
60 GPIO_IRQ_rising_edge
|= mask
;
62 GPIO_IRQ_rising_edge
&= ~mask
;
63 if (type
& IRQ_TYPE_EDGE_FALLING
)
64 GPIO_IRQ_falling_edge
|= mask
;
66 GPIO_IRQ_falling_edge
&= ~mask
;
68 writel(GPIO_IRQ_rising_edge
& GPIO_IRQ_mask
, GPIO_GRER
);
69 writel(GPIO_IRQ_falling_edge
& GPIO_IRQ_mask
, GPIO_GFER
);
75 * GPIO IRQs must be acknowledged. This is for IRQs from 0 to 7.
77 static void puv3_low_gpio_ack(struct irq_data
*d
)
79 writel((1 << d
->irq
), GPIO_GEDR
);
82 static void puv3_low_gpio_mask(struct irq_data
*d
)
84 writel(readl(INTC_ICMR
) & ~(1 << d
->irq
), INTC_ICMR
);
87 static void puv3_low_gpio_unmask(struct irq_data
*d
)
89 writel(readl(INTC_ICMR
) | (1 << d
->irq
), INTC_ICMR
);
92 static int puv3_low_gpio_wake(struct irq_data
*d
, unsigned int on
)
95 writel(readl(PM_PWER
) | (1 << d
->irq
), PM_PWER
);
97 writel(readl(PM_PWER
) & ~(1 << d
->irq
), PM_PWER
);
101 static struct irq_chip puv3_low_gpio_chip
= {
103 .irq_ack
= puv3_low_gpio_ack
,
104 .irq_mask
= puv3_low_gpio_mask
,
105 .irq_unmask
= puv3_low_gpio_unmask
,
106 .irq_set_type
= puv3_gpio_type
,
107 .irq_set_wake
= puv3_low_gpio_wake
,
111 * IRQ8 (GPIO0 through 27) handler. We enter here with the
112 * irq_controller_lock held, and IRQs disabled. Decode the IRQ
113 * and call the handler.
116 puv3_gpio_handler(unsigned int irq
, struct irq_desc
*desc
)
120 mask
= readl(GPIO_GEDR
);
123 * clear down all currently active IRQ sources.
124 * We will be processing them all.
126 writel(mask
, GPIO_GEDR
);
131 generic_handle_irq(irq
);
135 mask
= readl(GPIO_GEDR
);
140 * GPIO0-27 edge IRQs need to be handled specially.
141 * In addition, the IRQs are all collected up into one bit in the
142 * interrupt controller registers.
144 static void puv3_high_gpio_ack(struct irq_data
*d
)
146 unsigned int mask
= GPIO_MASK(d
->irq
);
148 writel(mask
, GPIO_GEDR
);
151 static void puv3_high_gpio_mask(struct irq_data
*d
)
153 unsigned int mask
= GPIO_MASK(d
->irq
);
155 GPIO_IRQ_mask
&= ~mask
;
157 writel(readl(GPIO_GRER
) & ~mask
, GPIO_GRER
);
158 writel(readl(GPIO_GFER
) & ~mask
, GPIO_GFER
);
161 static void puv3_high_gpio_unmask(struct irq_data
*d
)
163 unsigned int mask
= GPIO_MASK(d
->irq
);
165 GPIO_IRQ_mask
|= mask
;
167 writel(GPIO_IRQ_rising_edge
& GPIO_IRQ_mask
, GPIO_GRER
);
168 writel(GPIO_IRQ_falling_edge
& GPIO_IRQ_mask
, GPIO_GFER
);
171 static int puv3_high_gpio_wake(struct irq_data
*d
, unsigned int on
)
174 writel(readl(PM_PWER
) | PM_PWER_GPIOHIGH
, PM_PWER
);
176 writel(readl(PM_PWER
) & ~PM_PWER_GPIOHIGH
, PM_PWER
);
180 static struct irq_chip puv3_high_gpio_chip
= {
182 .irq_ack
= puv3_high_gpio_ack
,
183 .irq_mask
= puv3_high_gpio_mask
,
184 .irq_unmask
= puv3_high_gpio_unmask
,
185 .irq_set_type
= puv3_gpio_type
,
186 .irq_set_wake
= puv3_high_gpio_wake
,
190 * We don't need to ACK IRQs on the PKUnity unless they're GPIOs
191 * this is for internal IRQs i.e. from 8 to 31.
193 static void puv3_mask_irq(struct irq_data
*d
)
195 writel(readl(INTC_ICMR
) & ~(1 << d
->irq
), INTC_ICMR
);
198 static void puv3_unmask_irq(struct irq_data
*d
)
200 writel(readl(INTC_ICMR
) | (1 << d
->irq
), INTC_ICMR
);
204 * Apart form GPIOs, only the RTC alarm can be a wakeup event.
206 static int puv3_set_wake(struct irq_data
*d
, unsigned int on
)
208 if (d
->irq
== IRQ_RTCAlarm
) {
210 writel(readl(PM_PWER
) | PM_PWER_RTC
, PM_PWER
);
212 writel(readl(PM_PWER
) & ~PM_PWER_RTC
, PM_PWER
);
218 static struct irq_chip puv3_normal_chip
= {
219 .name
= "PKUnity-v3",
220 .irq_ack
= puv3_mask_irq
,
221 .irq_mask
= puv3_mask_irq
,
222 .irq_unmask
= puv3_unmask_irq
,
223 .irq_set_wake
= puv3_set_wake
,
226 static struct resource irq_resource
= {
228 .start
= io_v2p(PKUNITY_INTC_BASE
),
229 .end
= io_v2p(PKUNITY_INTC_BASE
) + 0xFFFFF,
232 static struct puv3_irq_state
{
239 static int puv3_irq_suspend(void)
241 struct puv3_irq_state
*st
= &puv3_irq_state
;
244 st
->icmr
= readl(INTC_ICMR
);
245 st
->iclr
= readl(INTC_ICLR
);
246 st
->iccr
= readl(INTC_ICCR
);
249 * Disable all GPIO-based interrupts.
251 writel(readl(INTC_ICMR
) & ~(0x1ff), INTC_ICMR
);
254 * Set the appropriate edges for wakeup.
256 writel(readl(PM_PWER
) & GPIO_IRQ_rising_edge
, GPIO_GRER
);
257 writel(readl(PM_PWER
) & GPIO_IRQ_falling_edge
, GPIO_GFER
);
260 * Clear any pending GPIO interrupts.
262 writel(readl(GPIO_GEDR
), GPIO_GEDR
);
267 static void puv3_irq_resume(void)
269 struct puv3_irq_state
*st
= &puv3_irq_state
;
272 writel(st
->iccr
, INTC_ICCR
);
273 writel(st
->iclr
, INTC_ICLR
);
275 writel(GPIO_IRQ_rising_edge
& GPIO_IRQ_mask
, GPIO_GRER
);
276 writel(GPIO_IRQ_falling_edge
& GPIO_IRQ_mask
, GPIO_GFER
);
278 writel(st
->icmr
, INTC_ICMR
);
282 static struct syscore_ops puv3_irq_syscore_ops
= {
283 .suspend
= puv3_irq_suspend
,
284 .resume
= puv3_irq_resume
,
287 static int __init
puv3_irq_init_syscore(void)
289 register_syscore_ops(&puv3_irq_syscore_ops
);
293 device_initcall(puv3_irq_init_syscore
);
295 void __init
init_IRQ(void)
299 request_resource(&iomem_resource
, &irq_resource
);
301 /* disable all IRQs */
302 writel(0, INTC_ICMR
);
304 /* all IRQs are IRQ, not REAL */
305 writel(0, INTC_ICLR
);
307 /* clear all GPIO edge detects */
308 writel(FMASK(8, 0) & ~FIELD(1, 1, GPI_SOFF_REQ
), GPIO_GPIR
);
309 writel(0, GPIO_GFER
);
310 writel(0, GPIO_GRER
);
311 writel(0x0FFFFFFF, GPIO_GEDR
);
313 writel(1, INTC_ICCR
);
315 for (irq
= 0; irq
< IRQ_GPIOHIGH
; irq
++) {
316 irq_set_chip(irq
, &puv3_low_gpio_chip
);
317 irq_set_handler(irq
, handle_edge_irq
);
318 irq_modify_status(irq
,
319 IRQ_NOREQUEST
| IRQ_NOPROBE
| IRQ_NOAUTOEN
,
323 for (irq
= IRQ_GPIOHIGH
+ 1; irq
< IRQ_GPIO0
; irq
++) {
324 irq_set_chip(irq
, &puv3_normal_chip
);
325 irq_set_handler(irq
, handle_level_irq
);
326 irq_modify_status(irq
,
327 IRQ_NOREQUEST
| IRQ_NOAUTOEN
,
331 for (irq
= IRQ_GPIO0
; irq
<= IRQ_GPIO27
; irq
++) {
332 irq_set_chip(irq
, &puv3_high_gpio_chip
);
333 irq_set_handler(irq
, handle_edge_irq
);
334 irq_modify_status(irq
,
335 IRQ_NOREQUEST
| IRQ_NOPROBE
| IRQ_NOAUTOEN
,
340 * Install handler for GPIO 0-27 edge detect interrupts
342 irq_set_chip(IRQ_GPIOHIGH
, &puv3_normal_chip
);
343 irq_set_chained_handler(IRQ_GPIOHIGH
, puv3_gpio_handler
);
345 #ifdef CONFIG_PUV3_GPIO
351 * do_IRQ handles all hardware IRQ's. Decoded IRQs should not
352 * come via this function. Instead, they should provide their
355 asmlinkage
void asm_do_IRQ(unsigned int irq
, struct pt_regs
*regs
)
357 struct pt_regs
*old_regs
= set_irq_regs(regs
);
362 * Some hardware gives randomly wrong interrupts. Rather
363 * than crashing, do something sensible.
365 if (unlikely(irq
>= nr_irqs
)) {
366 if (printk_ratelimit())
367 printk(KERN_WARNING
"Bad IRQ%u\n", irq
);
370 generic_handle_irq(irq
);
374 set_irq_regs(old_regs
);