1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
|
---
arch/avr32/mach-at32ap/pio.c | 73 ++++++++++++++++++++++++++++++++++++++++++-
1 file changed, 72 insertions(+), 1 deletion(-)
Index: linux-2.6.18-avr32/arch/avr32/mach-at32ap/pio.c
===================================================================
--- linux-2.6.18-avr32.orig/arch/avr32/mach-at32ap/pio.c 2006-11-29 18:17:15.000000000 +0100
+++ linux-2.6.18-avr32/arch/avr32/mach-at32ap/pio.c 2006-11-30 11:50:27.000000000 +0100
@@ -11,6 +11,7 @@
#include <linux/clk.h>
#include <linux/debugfs.h>
#include <linux/fs.h>
+#include <linux/irq.h>
#include <linux/platform_device.h>
#include <asm/gpio.h>
@@ -709,20 +710,90 @@ err_class_create:
late_initcall(pio_init_dev);
#endif /* CONFIG_PIO_DEV */
+/* Interrupt controller implementation */
+static void pio_mask_irq(unsigned int irq)
+{
+ struct pio_device *pio = get_irq_chip_data(irq);
+
+ pio_writel(pio, IDR, 1 << (irq & 0x1f));
+}
+
+static void pio_unmask_irq(unsigned int irq)
+{
+ struct pio_device *pio = get_irq_chip_data(irq);
+
+ pio_writel(pio, IER, 1 << (irq & 0x1f));
+}
+
+static int pio_set_irq_type(unsigned int irq, unsigned int flow_type)
+{
+ if (flow_type != IRQ_TYPE_NONE && flow_type != IRQ_TYPE_EDGE_BOTH)
+ return -EINVAL;
+
+ return 0;
+}
+
+static struct irq_chip pio_irq_chip = {
+ .name = "pio",
+ .mask = pio_mask_irq,
+ .unmask = pio_unmask_irq,
+ .set_type = pio_set_irq_type,
+};
+
+static void demux_pio_irq(unsigned int irq, struct irq_desc *desc,
+ struct pt_regs *regs)
+{
+ struct pio_device *pio = desc->handler_data;
+ struct irq_desc *ext_desc;
+ u32 status, pending;
+ unsigned int ext_irq;
+ unsigned int i;
+
+ status = pio_readl(pio, ISR);
+ pending = status & pio_readl(pio, IMR);
+
+ while (pending) {
+ i = fls(pending) - 1;
+ pending &= ~(1 << i);
+
+ ext_irq = gpio_to_irq(32 * pio_id(pio) + i);
+ ext_desc = irq_desc + ext_irq;
+ ext_desc->handle_irq(ext_irq, ext_desc, regs);
+ }
+}
+
static int __init pio_probe(struct platform_device *pdev)
{
struct pio_device *pio = NULL;
+ unsigned int i;
+ int int_irq, ext_irq;
BUG_ON(pdev->id >= MAX_NR_PIO_DEVICES);
pio = &pio_dev[pdev->id];
BUG_ON(!pio->regs);
- /* TODO: Interrupts */
+ /* Set up interrupts */
+ pio_writel(pio, IDR, ~0UL);
+ pio_readl(pio, ISR);
+
+ ext_irq = gpio_to_irq(32 * pdev->id);
+ for (i = 0; i < 32; i++, ext_irq++) {
+ set_irq_chip_and_handler(ext_irq, &pio_irq_chip,
+ handle_simple_irq);
+ set_irq_chip_data(ext_irq, pio);
+ }
+
+ int_irq = platform_get_irq(pdev, 0);
+ set_irq_chained_handler(int_irq, demux_pio_irq);
+ set_irq_data(int_irq, pio);
platform_set_drvdata(pdev, pio);
printk(KERN_INFO "%s: Atmel Port Multiplexer at 0x%p (irq %d)\n",
pio->name, pio->regs, platform_get_irq(pdev, 0));
+ printk(KERN_INFO
+ "%s: Handling 32 external IRQs, starting with IRQ %d\n",
+ pio->name, ext_irq - 32);
return 0;
}
|