2  * arch/mips/dec/int-handler.S
 
   4  * Copyright (C) 1995, 1996, 1997 Paul M. Antoine and Harald Koerfgen
 
   5  * Copyright (C) 2000, 2001, 2002, 2003  Maciej W. Rozycki
 
   7  * Written by Ralf Baechle and Andreas Busse, modified for DECStation
 
   8  * support by Paul Antoine and Harald Koerfgen.
 
  10  * completly rewritten:
 
  11  * Copyright (C) 1998 Harald Koerfgen
 
  13  * Rewritten extensively for controller-driven IRQ support
 
  14  * by Maciej W. Rozycki.
 
  16 #include <linux/config.h>
 
  18 #include <asm/regdef.h>
 
  19 #include <asm/mipsregs.h>
 
  20 #include <asm/stackframe.h>
 
  21 #include <asm/addrspace.h>
 
  23 #include <asm/dec/interrupts.h>
 
  24 #include <asm/dec/ioasic_addrs.h>
 
  25 #include <asm/dec/ioasic_ints.h>
 
  26 #include <asm/dec/kn01.h>
 
  27 #include <asm/dec/kn02.h>
 
  28 #include <asm/dec/kn02xa.h>
 
  29 #include <asm/dec/kn03.h>
 
  35  * decstation_handle_int: Interrupt handler for DECStations
 
  37  * We follow the model in the Indy interrupt code by David Miller, where he
 
  38  * says: a lot of complication here is taken away because:
 
  40  * 1) We handle one interrupt and return, sitting in a loop
 
  41  *    and moving across all the pending IRQ bits in the cause
 
  42  *    register is _NOT_ the answer, the common case is one
 
  43  *    pending IRQ so optimize in that direction.
 
  45  * 2) We need not check against bits in the status register
 
  46  *    IRQ mask, that would make this routine slow as hell.
 
  48  * 3) Linux only thinks in terms of all IRQs on or all IRQs
 
  49  *    off, nothing in between like BSD spl() brain-damage.
 
  51  * Furthermore, the IRQs on the DECStations look basically (barring
 
  52  * software IRQs which we don't use at all) like...
 
  54  * DS2100/3100's, aka kn01, aka Pmax:
 
  58  *             0        Software (ignored)
 
  59  *             1        Software (ignored)
 
  67  * DS5000/200, aka kn02, aka 3max:
 
  71  *             0        Software (ignored)
 
  72  *             1        Software (ignored)
 
  80  * DS5000/1xx's, aka kn02ba, aka 3min:
 
  84  *             0        Software (ignored)
 
  85  *             1        Software (ignored)
 
  86  *             2        TurboChannel Slot 0
 
  87  *             3        TurboChannel Slot 1
 
  88  *             4        TurboChannel Slot 2
 
  89  *             5        TurboChannel Slot 3 (ASIC)
 
  93  * DS5000/2x's, aka kn02ca, aka maxine:
 
  97  *             0        Software (ignored)
 
  98  *             1        Software (ignored)
 
  99  *             2        Periodic Interrupt (100usec)
 
 101  *             4        I/O write timeout
 
 102  *             5        TurboChannel (ASIC)
 
 103  *             6        Halt Keycode from Access.Bus keyboard (CTRL-ALT-ENTER)
 
 106  * DS5000/2xx's, aka kn03, aka 3maxplus:
 
 110  *             0        Software (ignored)
 
 111  *             1        Software (ignored)
 
 112  *             2        System Board (ASIC)
 
 119  * We handle the IRQ according to _our_ priority (see setup.c),
 
 120  * then we just return.  If multiple IRQs are pending then we will
 
 121  * just take another exception, big deal.
 
 124                 NESTED(decstation_handle_int, PT_SIZE, ra)
 
 127                 CLI                             # TEST: interrupts should be off
 
 132                  * Get pending Interrupts
 
 134                 mfc0    t0,CP0_CAUSE            # get pending interrupts
 
 139                 andi    t0,ST0_IM               # CAUSE.CE may be non-zero!
 
 140                 and     t0,t1                   # isolate allowed ones
 
 146                 bnez    t2,fpu                  # handle FPU immediately
 
 150                  * Find irq with highest priority
 
 152                  PTR_LA t1,cpu_mask_nr_tbl
 
 157                  addu   t1,2*PTRSIZE            # delay slot
 
 160                  * Do the low-level stuff
 
 164                 bgez    a0,handle_it            # irq_nr >= 0?
 
 165                                                 # irq_nr < 0: it is an address
 
 168                                                 # a trick to save a branch:
 
 169                  lui    t2,(KN03_IOASIC_BASE>>16)&0xffff
 
 170                                                 # upper part of IOASIC Address
 
 173  * Handle "IRQ Controller" Interrupts
 
 174  * Masked Interrupts are still visible and have to be masked "by hand".
 
 176                 FEXPORT(kn02_io_int)            # 3max
 
 177                 lui     t0,(KN02_CSR_BASE>>16)&0xffff
 
 178                                                 # get interrupt status and mask
 
 181                 andi    t1,t0,KN02_IRQ_ALL
 
 183                  srl    t0,16                   # shift interrupt mask
 
 185                 FEXPORT(kn02xa_io_int)          # 3min/maxine
 
 186                 lui     t2,(KN02XA_IOASIC_BASE>>16)&0xffff
 
 187                                                 # upper part of IOASIC Address
 
 189                 FEXPORT(kn03_io_int)            # 3max+ (t2 loaded earlier)
 
 190                 lw      t0,IO_REG_SIR(t2)       # get status: IOASIC sir
 
 191                 lw      t1,IO_REG_SIMR(t2)      # get mask:   IOASIC simr
 
 194 1:              and     t0,t1                   # mask out allowed ones
 
 199                  * Find irq with highest priority
 
 201                  PTR_LA t1,asic_mask_nr_tbl
 
 206                  addu   t1,2*PTRSIZE            # delay slot
 
 209                  * Do the low-level stuff
 
 211                 lw      a0,%lo(-PTRSIZE)(t1)
 
 213                 bgez    a0,handle_it            # irq_nr >= 0?
 
 214                                                 # irq_nr < 0: it is an address
 
 220  * Dispatch low-priority interrupts.  We reconsider all status
 
 221  * bits again, which looks like a lose, but it makes the code
 
 222  * simple and O(log n), so it gets compensated.
 
 224                 FEXPORT(cpu_all_int)            # HALT, timers, software junk
 
 225                 li      a0,DEC_CPU_IRQ_BASE
 
 227                 li      t1,CAUSEF_IP>>CAUSEB_IP # mask
 
 229                  li     t2,4                    # nr of bits / 2
 
 231                 FEXPORT(kn02_all_int)           # impossible ?
 
 233                 li      t1,KN02_IRQ_ALL         # mask
 
 235                  li     t2,4                    # nr of bits / 2
 
 237                 FEXPORT(asic_all_int)           # various I/O ASIC junk
 
 239                 li      t1,IO_IRQ_ALL           # mask
 
 241                  li     t2,8                    # nr of bits / 2
 
 244  * Dispatch DMA interrupts -- O(log n).
 
 246                 FEXPORT(asic_dma_int)           # I/O ASIC DMA events
 
 247                 li      a0,IO_IRQ_BASE+IO_INR_DMA
 
 249                 li      t1,IO_IRQ_DMA>>IO_INR_DMA # mask
 
 250                 li      t2,8                    # nr of bits / 2
 
 253                  * Find irq with highest priority.
 
 254                  * Highest irq number takes precedence.
 
 283                 END(decstation_handle_int)
 
 286  * Generic unimplemented interrupt routines -- cpu_mask_nr_tbl
 
 287  * and asic_mask_nr_tbl are initialized to point all interrupts here.
 
 288  * The tables are then filled in by machine-specific initialisation
 
 291                 FEXPORT(dec_intr_unimplemented)
 
 292                 move    a1,t0                   # cheats way of printing an arg!
 
 293                 PANIC("Unimplemented cpu interrupt! CP0_CAUSE: 0x%08x");
 
 295                 FEXPORT(asic_intr_unimplemented)
 
 296                 move    a1,t0                   # cheats way of printing an arg!
 
 297                 PANIC("Unimplemented asic interrupt! ASIC ISR: 0x%08x");