2 * External interrupt handling for AT32AP CPUs
4 * Copyright (C) 2006 Atmel Corporation
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
11 #include <linux/errno.h>
12 #include <linux/init.h>
13 #include <linux/interrupt.h>
14 #include <linux/irq.h>
15 #include <linux/platform_device.h>
16 #include <linux/random.h>
20 #include <asm/arch/sm.h>
24 static void eim_ack_irq(unsigned int irq
)
26 struct at32_sm
*sm
= get_irq_chip_data(irq
);
27 sm_writel(sm
, EIM_ICR
, 1 << (irq
- sm
->eim_first_irq
));
30 static void eim_mask_irq(unsigned int irq
)
32 struct at32_sm
*sm
= get_irq_chip_data(irq
);
33 sm_writel(sm
, EIM_IDR
, 1 << (irq
- sm
->eim_first_irq
));
36 static void eim_mask_ack_irq(unsigned int irq
)
38 struct at32_sm
*sm
= get_irq_chip_data(irq
);
39 sm_writel(sm
, EIM_ICR
, 1 << (irq
- sm
->eim_first_irq
));
40 sm_writel(sm
, EIM_IDR
, 1 << (irq
- sm
->eim_first_irq
));
43 static void eim_unmask_irq(unsigned int irq
)
45 struct at32_sm
*sm
= get_irq_chip_data(irq
);
46 sm_writel(sm
, EIM_IER
, 1 << (irq
- sm
->eim_first_irq
));
49 static int eim_set_irq_type(unsigned int irq
, unsigned int flow_type
)
51 struct at32_sm
*sm
= get_irq_chip_data(irq
);
52 struct irq_desc
*desc
;
53 unsigned int i
= irq
- sm
->eim_first_irq
;
54 u32 mode
, edge
, level
;
58 flow_type
&= IRQ_TYPE_SENSE_MASK
;
59 if (flow_type
== IRQ_TYPE_NONE
)
60 flow_type
= IRQ_TYPE_LEVEL_LOW
;
62 desc
= &irq_desc
[irq
];
63 spin_lock_irqsave(&sm
->lock
, flags
);
65 mode
= sm_readl(sm
, EIM_MODE
);
66 edge
= sm_readl(sm
, EIM_EDGE
);
67 level
= sm_readl(sm
, EIM_LEVEL
);
70 case IRQ_TYPE_LEVEL_LOW
:
74 case IRQ_TYPE_LEVEL_HIGH
:
78 case IRQ_TYPE_EDGE_RISING
:
82 case IRQ_TYPE_EDGE_FALLING
:
92 sm_writel(sm
, EIM_MODE
, mode
);
93 sm_writel(sm
, EIM_EDGE
, edge
);
94 sm_writel(sm
, EIM_LEVEL
, level
);
96 if (flow_type
& (IRQ_TYPE_LEVEL_LOW
| IRQ_TYPE_LEVEL_HIGH
))
97 flow_type
|= IRQ_LEVEL
;
98 desc
->status
&= ~(IRQ_TYPE_SENSE_MASK
| IRQ_LEVEL
);
99 desc
->status
|= flow_type
;
102 spin_unlock_irqrestore(&sm
->lock
, flags
);
107 struct irq_chip eim_chip
= {
110 .mask
= eim_mask_irq
,
111 .mask_ack
= eim_mask_ack_irq
,
112 .unmask
= eim_unmask_irq
,
113 .set_type
= eim_set_irq_type
,
116 static void demux_eim_irq(unsigned int irq
, struct irq_desc
*desc
)
118 struct at32_sm
*sm
= desc
->handler_data
;
119 struct irq_desc
*ext_desc
;
120 unsigned long status
, pending
;
121 unsigned int i
, ext_irq
;
123 status
= sm_readl(sm
, EIM_ISR
);
124 pending
= status
& sm_readl(sm
, EIM_IMR
);
127 i
= fls(pending
) - 1;
128 pending
&= ~(1 << i
);
130 ext_irq
= i
+ sm
->eim_first_irq
;
131 ext_desc
= irq_desc
+ ext_irq
;
132 if (ext_desc
->status
& IRQ_LEVEL
)
133 handle_level_irq(ext_irq
, ext_desc
);
135 handle_edge_irq(ext_irq
, ext_desc
);
139 static int __init
eim_init(void)
141 struct at32_sm
*sm
= &system_manager
;
143 unsigned int nr_irqs
;
144 unsigned int int_irq
;
148 * The EIM is really the same module as SM, so register
149 * mapping, etc. has been taken care of already.
153 * Find out how many interrupt lines that are actually
154 * implemented in hardware.
156 sm_writel(sm
, EIM_IDR
, ~0UL);
157 sm_writel(sm
, EIM_MODE
, ~0UL);
158 pattern
= sm_readl(sm
, EIM_MODE
);
159 nr_irqs
= fls(pattern
);
161 /* Trigger on falling edge unless overridden by driver */
162 sm_writel(sm
, EIM_MODE
, 0UL);
163 sm_writel(sm
, EIM_EDGE
, 0UL);
165 sm
->eim_chip
= &eim_chip
;
167 for (i
= 0; i
< nr_irqs
; i
++) {
168 /* NOTE the handler we set here is ignored by the demux */
169 set_irq_chip_and_handler(sm
->eim_first_irq
+ i
, &eim_chip
,
171 set_irq_chip_data(sm
->eim_first_irq
+ i
, sm
);
174 int_irq
= platform_get_irq_byname(sm
->pdev
, "eim");
176 set_irq_chained_handler(int_irq
, demux_eim_irq
);
177 set_irq_data(int_irq
, sm
);
179 printk("EIM: External Interrupt Module at 0x%p, IRQ %u\n",
181 printk("EIM: Handling %u external IRQs, starting with IRQ %u\n",
182 nr_irqs
, sm
->eim_first_irq
);
186 arch_initcall(eim_init
);