aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/mach-omap2/irq.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/arm/mach-omap2/irq.c')
-rw-r--r--arch/arm/mach-omap2/irq.c98
1 files changed, 96 insertions, 2 deletions
diff --git a/arch/arm/mach-omap2/irq.c b/arch/arm/mach-omap2/irq.c
index b82863887f10..26aeef560aa3 100644
--- a/arch/arm/mach-omap2/irq.c
+++ b/arch/arm/mach-omap2/irq.c
@@ -25,6 +25,10 @@
25#define INTC_SYSSTATUS 0x0014 25#define INTC_SYSSTATUS 0x0014
26#define INTC_SIR 0x0040 26#define INTC_SIR 0x0040
27#define INTC_CONTROL 0x0048 27#define INTC_CONTROL 0x0048
28#define INTC_PROTECTION 0x004C
29#define INTC_IDLE 0x0050
30#define INTC_THRESHOLD 0x0068
31#define INTC_MIR0 0x0084
28#define INTC_MIR_CLEAR0 0x0088 32#define INTC_MIR_CLEAR0 0x0088
29#define INTC_MIR_SET0 0x008c 33#define INTC_MIR_SET0 0x008c
30#define INTC_PENDING_IRQ0 0x0098 34#define INTC_PENDING_IRQ0 0x0098
@@ -48,6 +52,18 @@ static struct omap_irq_bank {
48 }, 52 },
49}; 53};
50 54
55/* Structure to save interrupt controller context */
56struct omap3_intc_regs {
57 u32 sysconfig;
58 u32 protection;
59 u32 idle;
60 u32 threshold;
61 u32 ilr[INTCPS_NR_IRQS];
62 u32 mir[INTCPS_NR_MIR_REGS];
63};
64
65static struct omap3_intc_regs intc_context[ARRAY_SIZE(irq_banks)];
66
51/* INTC bank register get/set */ 67/* INTC bank register get/set */
52 68
53static void intc_bank_write_reg(u32 val, struct omap_irq_bank *bank, u16 reg) 69static void intc_bank_write_reg(u32 val, struct omap_irq_bank *bank, u16 reg)
@@ -178,12 +194,22 @@ void __init omap_init_irq(void)
178 int i; 194 int i;
179 195
180 for (i = 0; i < ARRAY_SIZE(irq_banks); i++) { 196 for (i = 0; i < ARRAY_SIZE(irq_banks); i++) {
197 unsigned long base = 0;
181 struct omap_irq_bank *bank = irq_banks + i; 198 struct omap_irq_bank *bank = irq_banks + i;
182 199
183 if (cpu_is_omap24xx()) 200 if (cpu_is_omap24xx())
184 bank->base_reg = OMAP2_IO_ADDRESS(OMAP24XX_IC_BASE); 201 base = OMAP24XX_IC_BASE;
185 else if (cpu_is_omap34xx()) 202 else if (cpu_is_omap34xx())
186 bank->base_reg = OMAP2_IO_ADDRESS(OMAP34XX_IC_BASE); 203 base = OMAP34XX_IC_BASE;
204
205 BUG_ON(!base);
206
207 /* Static mapping, never released */
208 bank->base_reg = ioremap(base, SZ_4K);
209 if (!bank->base_reg) {
210 printk(KERN_ERR "Could not ioremap irq bank%i\n", i);
211 continue;
212 }
187 213
188 omap_irq_bank_init_one(bank); 214 omap_irq_bank_init_one(bank);
189 215
@@ -201,3 +227,71 @@ void __init omap_init_irq(void)
201 } 227 }
202} 228}
203 229
230#ifdef CONFIG_ARCH_OMAP3
231void omap_intc_save_context(void)
232{
233 int ind = 0, i = 0;
234 for (ind = 0; ind < ARRAY_SIZE(irq_banks); ind++) {
235 struct omap_irq_bank *bank = irq_banks + ind;
236 intc_context[ind].sysconfig =
237 intc_bank_read_reg(bank, INTC_SYSCONFIG);
238 intc_context[ind].protection =
239 intc_bank_read_reg(bank, INTC_PROTECTION);
240 intc_context[ind].idle =
241 intc_bank_read_reg(bank, INTC_IDLE);
242 intc_context[ind].threshold =
243 intc_bank_read_reg(bank, INTC_THRESHOLD);
244 for (i = 0; i < INTCPS_NR_IRQS; i++)
245 intc_context[ind].ilr[i] =
246 intc_bank_read_reg(bank, (0x100 + 0x4*i));
247 for (i = 0; i < INTCPS_NR_MIR_REGS; i++)
248 intc_context[ind].mir[i] =
249 intc_bank_read_reg(&irq_banks[0], INTC_MIR0 +
250 (0x20 * i));
251 }
252}
253
254void omap_intc_restore_context(void)
255{
256 int ind = 0, i = 0;
257
258 for (ind = 0; ind < ARRAY_SIZE(irq_banks); ind++) {
259 struct omap_irq_bank *bank = irq_banks + ind;
260 intc_bank_write_reg(intc_context[ind].sysconfig,
261 bank, INTC_SYSCONFIG);
262 intc_bank_write_reg(intc_context[ind].sysconfig,
263 bank, INTC_SYSCONFIG);
264 intc_bank_write_reg(intc_context[ind].protection,
265 bank, INTC_PROTECTION);
266 intc_bank_write_reg(intc_context[ind].idle,
267 bank, INTC_IDLE);
268 intc_bank_write_reg(intc_context[ind].threshold,
269 bank, INTC_THRESHOLD);
270 for (i = 0; i < INTCPS_NR_IRQS; i++)
271 intc_bank_write_reg(intc_context[ind].ilr[i],
272 bank, (0x100 + 0x4*i));
273 for (i = 0; i < INTCPS_NR_MIR_REGS; i++)
274 intc_bank_write_reg(intc_context[ind].mir[i],
275 &irq_banks[0], INTC_MIR0 + (0x20 * i));
276 }
277 /* MIRs are saved and restore with other PRCM registers */
278}
279
280void omap3_intc_suspend(void)
281{
282 /* A pending interrupt would prevent OMAP from entering suspend */
283 omap_ack_irq(0);
284}
285
286void omap3_intc_prepare_idle(void)
287{
288 /* Disable autoidle as it can stall interrupt controller */
289 intc_bank_write_reg(0, &irq_banks[0], INTC_SYSCONFIG);
290}
291
292void omap3_intc_resume_idle(void)
293{
294 /* Re-enable autoidle */
295 intc_bank_write_reg(1, &irq_banks[0], INTC_SYSCONFIG);
296}
297#endif /* CONFIG_ARCH_OMAP3 */