2  * linux/arch/arm/mach-omap2/irq.c
 
   4  * Interrupt handler for OMAP2 boards.
 
   6  * Copyright (C) 2005 Nokia Corporation
 
   7  * Author: Paul Mundt <paul.mundt@nokia.com>
 
   9  * This file is subject to the terms and conditions of the GNU General Public
 
  10  * License. See the file "COPYING" in the main directory of this archive
 
  13 #include <linux/kernel.h>
 
  14 #include <linux/init.h>
 
  15 #include <linux/interrupt.h>
 
  17 #include <mach/hardware.h>
 
  18 #include <asm/mach/irq.h>
 
  21 /* selected INTC register offsets */
 
  23 #define INTC_REVISION           0x0000
 
  24 #define INTC_SYSCONFIG          0x0010
 
  25 #define INTC_SYSSTATUS          0x0014
 
  26 #define INTC_SIR                0x0040
 
  27 #define INTC_CONTROL            0x0048
 
  28 #define INTC_MIR_CLEAR0         0x0088
 
  29 #define INTC_MIR_SET0           0x008c
 
  30 #define INTC_PENDING_IRQ0       0x0098
 
  32 /* Number of IRQ state bits in each MIR register */
 
  33 #define IRQ_BITS_PER_REG        32
 
  36  * OMAP2 has a number of different interrupt controllers, each interrupt
 
  37  * controller is identified as its own "bank". Register definitions are
 
  38  * fairly consistent for each bank, but not all registers are implemented
 
  39  * for each bank.. when in doubt, consult the TRM.
 
  41 static struct omap_irq_bank {
 
  42         void __iomem *base_reg;
 
  44 } __attribute__ ((aligned(4))) irq_banks[] = {
 
  52 /* INTC bank register get/set */
 
  54 static void intc_bank_write_reg(u32 val, struct omap_irq_bank *bank, u16 reg)
 
  56         __raw_writel(val, bank->base_reg + reg);
 
  59 static u32 intc_bank_read_reg(struct omap_irq_bank *bank, u16 reg)
 
  61         return __raw_readl(bank->base_reg + reg);
 
  64 static int previous_irq;
 
  67  * On 34xx we can get occasional spurious interrupts if the ack from
 
  68  * an interrupt handler does not get posted before we unmask. Warn about
 
  69  * the interrupt handlers that need to flush posted writes.
 
  71 static int omap_check_spurious(unsigned int irq)
 
  75         sir = intc_bank_read_reg(&irq_banks[0], INTC_SIR);
 
  79                 printk(KERN_WARNING "Spurious irq %i: 0x%08x, please flush "
 
  80                                         "posted write for irq %i\n",
 
  81                                         irq, sir, previous_irq);
 
  88 /* XXX: FIQ and additional INTC support (only MPU at the moment) */
 
  89 static void omap_ack_irq(unsigned int irq)
 
  91         intc_bank_write_reg(0x1, &irq_banks[0], INTC_CONTROL);
 
  94 static void omap_mask_irq(unsigned int irq)
 
  96         int offset = irq & (~(IRQ_BITS_PER_REG - 1));
 
  98         if (cpu_is_omap34xx()) {
 
 102                  * INT_34XX_GPT12_IRQ is also the spurious irq. Maybe because
 
 103                  * it is the highest irq number?
 
 105                 if (irq == INT_34XX_GPT12_IRQ)
 
 106                         spurious = omap_check_spurious(irq);
 
 112         irq &= (IRQ_BITS_PER_REG - 1);
 
 114         intc_bank_write_reg(1 << irq, &irq_banks[0], INTC_MIR_SET0 + offset);
 
 117 static void omap_unmask_irq(unsigned int irq)
 
 119         int offset = irq & (~(IRQ_BITS_PER_REG - 1));
 
 121         irq &= (IRQ_BITS_PER_REG - 1);
 
 123         intc_bank_write_reg(1 << irq, &irq_banks[0], INTC_MIR_CLEAR0 + offset);
 
 126 static void omap_mask_ack_irq(unsigned int irq)
 
 132 static struct irq_chip omap_irq_chip = {
 
 134         .ack    = omap_mask_ack_irq,
 
 135         .mask   = omap_mask_irq,
 
 136         .unmask = omap_unmask_irq,
 
 137         .disable = omap_mask_irq,
 
 140 static void __init omap_irq_bank_init_one(struct omap_irq_bank *bank)
 
 144         tmp = intc_bank_read_reg(bank, INTC_REVISION) & 0xff;
 
 145         printk(KERN_INFO "IRQ: Found an INTC at 0x%p "
 
 146                          "(revision %ld.%ld) with %d interrupts\n",
 
 147                          bank->base_reg, tmp >> 4, tmp & 0xf, bank->nr_irqs);
 
 149         tmp = intc_bank_read_reg(bank, INTC_SYSCONFIG);
 
 150         tmp |= 1 << 1;  /* soft reset */
 
 151         intc_bank_write_reg(tmp, bank, INTC_SYSCONFIG);
 
 153         while (!(intc_bank_read_reg(bank, INTC_SYSSTATUS) & 0x1))
 
 154                 /* Wait for reset to complete */;
 
 156         /* Enable autoidle */
 
 157         intc_bank_write_reg(1 << 0, bank, INTC_SYSCONFIG);
 
 160 void __init omap_init_irq(void)
 
 162         unsigned long nr_of_irqs = 0;
 
 163         unsigned int nr_banks = 0;
 
 166         for (i = 0; i < ARRAY_SIZE(irq_banks); i++) {
 
 167                 struct omap_irq_bank *bank = irq_banks + i;
 
 169                 if (cpu_is_omap24xx())
 
 170                         bank->base_reg = OMAP2_IO_ADDRESS(OMAP24XX_IC_BASE);
 
 171                 else if (cpu_is_omap34xx())
 
 172                         bank->base_reg = OMAP2_IO_ADDRESS(OMAP34XX_IC_BASE);
 
 174                 omap_irq_bank_init_one(bank);
 
 176                 nr_of_irqs += bank->nr_irqs;
 
 180         printk(KERN_INFO "Total of %ld interrupts on %d active controller%s\n",
 
 181                nr_of_irqs, nr_banks, nr_banks > 1 ? "s" : "");
 
 183         for (i = 0; i < nr_of_irqs; i++) {
 
 184                 set_irq_chip(i, &omap_irq_chip);
 
 185                 set_irq_handler(i, handle_level_irq);
 
 186                 set_irq_flags(i, IRQF_VALID);