2005-04-17 02:20:36 +04:00
/*
* Code to handle IP32 IRQs
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*
* Copyright ( C ) 2000 Harald Koerfgen
* Copyright ( C ) 2001 Keith M Wesolowski
*/
# include <linux/init.h>
# include <linux/kernel_stat.h>
# include <linux/types.h>
# include <linux/interrupt.h>
# include <linux/irq.h>
# include <linux/bitops.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/random.h>
# include <linux/sched.h>
2007-10-14 17:02:26 +04:00
# include <asm/irq_cpu.h>
2005-04-17 02:20:36 +04:00
# include <asm/mipsregs.h>
# include <asm/signal.h>
# include <asm/time.h>
# include <asm/ip32/crime.h>
# include <asm/ip32/mace.h>
# include <asm/ip32/ip32_ints.h>
/* issue a PIO read to make sure no PIO writes are pending */
static void inline flush_crime_bus ( void )
{
2006-05-30 05:13:16 +04:00
crime - > control ;
2005-04-17 02:20:36 +04:00
}
static void inline flush_mace_bus ( void )
{
2006-05-30 05:13:16 +04:00
mace - > perif . ctrl . misc ;
2005-04-17 02:20:36 +04:00
}
2007-10-14 17:02:26 +04:00
/*
* O2 irq map
2005-04-17 02:20:36 +04:00
*
* IP0 - > software ( ignored )
* IP1 - > software ( ignored )
* IP2 - > ( irq0 ) C crime 1.1 all interrupts ; crime 1.5 ? ? ?
* IP3 - > ( irq1 ) X unknown
* IP4 - > ( irq2 ) X unknown
* IP5 - > ( irq3 ) X unknown
* IP6 - > ( irq4 ) X unknown
2007-10-14 17:02:26 +04:00
* IP7 - > ( irq5 ) 7 CPU count / compare timer ( system timer )
2005-04-17 02:20:36 +04:00
*
* crime : ( C )
*
* CRIME_INT_STAT 31 : 0 :
*
2007-10-14 17:02:26 +04:00
* 0 - > 8 Video in 1
* 1 - > 9 Video in 2
* 2 - > 10 Video out
* 3 - > 11 Mace ethernet
2005-04-17 02:20:36 +04:00
* 4 - > S SuperIO sub - interrupt
* 5 - > M Miscellaneous sub - interrupt
* 6 - > A Audio sub - interrupt
2007-10-14 17:02:26 +04:00
* 7 - > 15 PCI bridge errors
* 8 - > 16 PCI SCSI aic7xxx 0
* 9 - > 17 PCI SCSI aic7xxx 1
* 10 - > 18 PCI slot 0
* 11 - > 19 unused ( PCI slot 1 )
* 12 - > 20 unused ( PCI slot 2 )
* 13 - > 21 unused ( PCI shared 0 )
* 14 - > 22 unused ( PCI shared 1 )
* 15 - > 23 unused ( PCI shared 2 )
* 16 - > 24 GBE0 ( E )
* 17 - > 25 GBE1 ( E )
* 18 - > 26 GBE2 ( E )
* 19 - > 27 GBE3 ( E )
* 20 - > 28 CPU errors
* 21 - > 29 Memory errors
* 22 - > 30 RE empty edge ( E )
* 23 - > 31 RE full edge ( E )
* 24 - > 32 RE idle edge ( E )
* 25 - > 33 RE empty level
* 26 - > 34 RE full level
* 27 - > 35 RE idle level
* 28 - > 36 unused ( software 0 ) ( E )
* 29 - > 37 unused ( software 1 ) ( E )
* 30 - > 38 unused ( software 2 ) - crime 1.5 CPU SysCorError ( E )
* 31 - > 39 VICE
2005-04-17 02:20:36 +04:00
*
* S , M , A : Use the MACE ISA interrupt register
* MACE_ISA_INT_STAT 31 : 0
*
2007-10-14 17:02:26 +04:00
* 0 - 7 - > 40 - 47 Audio
* 8 - > 48 RTC
* 9 - > 49 Keyboard
2005-04-17 02:20:36 +04:00
* 10 - > X Keyboard polled
2007-10-14 17:02:26 +04:00
* 11 - > 51 Mouse
2005-04-17 02:20:36 +04:00
* 12 - > X Mouse polled
2007-10-14 17:02:26 +04:00
* 13 - 15 - > 53 - 55 Count / compare timers
* 16 - 19 - > 56 - 59 Parallel ( 16 E )
* 20 - 25 - > 60 - 62 Serial 1 ( 22 E )
* 26 - 31 - > 66 - 71 Serial 2 ( 28 E )
2005-04-17 02:20:36 +04:00
*
2007-10-14 17:02:26 +04:00
* Note that this means IRQs 12 - 14 , 50 , and 52 do not exist . This is a
2005-04-17 02:20:36 +04:00
* different IRQ map than IRIX uses , but that ' s OK as Linux irq handling
* is quite different anyway .
*/
/* Some initial interrupts to set up */
2006-10-07 22:44:33 +04:00
extern irqreturn_t crime_memerr_intr ( int irq , void * dev_id ) ;
extern irqreturn_t crime_cpuerr_intr ( int irq , void * dev_id ) ;
2005-04-17 02:20:36 +04:00
2009-03-30 23:53:25 +04:00
static struct irqaction memerr_irq = {
2007-08-28 13:03:01 +04:00
. handler = crime_memerr_intr ,
. name = " CRIME memory error " ,
} ;
2007-10-28 21:46:39 +03:00
2009-03-30 23:53:25 +04:00
static struct irqaction cpuerr_irq = {
2007-08-28 13:03:01 +04:00
. handler = crime_cpuerr_intr ,
. name = " CRIME CPU error " ,
} ;
2005-04-17 02:20:36 +04:00
/*
* This is for pure CRIME interrupts - ie not MACE . The advantage ?
* We get to split the register in half and do faster lookups .
*/
static uint64_t crime_mask ;
2011-03-24 00:09:13 +03:00
static inline void crime_enable_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
2011-03-24 00:09:13 +03:00
unsigned int bit = d - > irq - CRIME_IRQ_BASE ;
2007-10-28 21:46:39 +03:00
crime_mask | = 1 < < bit ;
2005-04-17 02:20:36 +04:00
crime - > imask = crime_mask ;
}
2011-03-24 00:09:13 +03:00
static inline void crime_disable_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
2011-03-24 00:09:13 +03:00
unsigned int bit = d - > irq - CRIME_IRQ_BASE ;
2007-10-28 21:46:39 +03:00
crime_mask & = ~ ( 1 < < bit ) ;
2005-04-17 02:20:36 +04:00
crime - > imask = crime_mask ;
flush_crime_bus ( ) ;
}
2007-10-28 21:46:39 +03:00
static struct irq_chip crime_level_interrupt = {
. name = " IP32 CRIME " ,
2011-03-24 00:09:13 +03:00
. irq_mask = crime_disable_irq ,
. irq_unmask = crime_enable_irq ,
2007-10-28 21:46:39 +03:00
} ;
2011-03-24 00:09:13 +03:00
static void crime_edge_mask_and_ack_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
2011-03-24 00:09:13 +03:00
unsigned int bit = d - > irq - CRIME_IRQ_BASE ;
2007-10-28 21:46:39 +03:00
uint64_t crime_int ;
2005-04-17 02:20:36 +04:00
/* Edge triggered interrupts must be cleared. */
2007-10-28 21:46:39 +03:00
crime_int = crime - > hard_int ;
crime_int & = ~ ( 1 < < bit ) ;
crime - > hard_int = crime_int ;
2011-03-24 00:09:13 +03:00
crime_disable_irq ( d ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-28 21:46:39 +03:00
static struct irq_chip crime_edge_interrupt = {
. name = " IP32 CRIME " ,
2011-03-24 00:09:13 +03:00
. irq_ack = crime_edge_mask_and_ack_irq ,
. irq_mask = crime_disable_irq ,
. irq_mask_ack = crime_edge_mask_and_ack_irq ,
. irq_unmask = crime_enable_irq ,
2005-04-17 02:20:36 +04:00
} ;
/*
* This is for MACE PCI interrupts . We can decrease bus traffic by masking
2013-01-22 15:59:30 +04:00
* as close to the source as possible . This also means we can take the
2005-04-17 02:20:36 +04:00
* next chunk of the CRIME register in one piece .
*/
static unsigned long macepci_mask ;
2011-03-24 00:09:13 +03:00
static void enable_macepci_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
2011-03-24 00:09:13 +03:00
macepci_mask | = MACEPCI_CONTROL_INT ( d - > irq - MACEPCI_SCSI0_IRQ ) ;
2005-04-17 02:20:36 +04:00
mace - > pci . control = macepci_mask ;
2011-03-24 00:09:13 +03:00
crime_mask | = 1 < < ( d - > irq - CRIME_IRQ_BASE ) ;
2005-04-17 02:20:36 +04:00
crime - > imask = crime_mask ;
}
2011-03-24 00:09:13 +03:00
static void disable_macepci_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
2011-03-24 00:09:13 +03:00
crime_mask & = ~ ( 1 < < ( d - > irq - CRIME_IRQ_BASE ) ) ;
2005-04-17 02:20:36 +04:00
crime - > imask = crime_mask ;
flush_crime_bus ( ) ;
2011-03-24 00:09:13 +03:00
macepci_mask & = ~ MACEPCI_CONTROL_INT ( d - > irq - MACEPCI_SCSI0_IRQ ) ;
2005-04-17 02:20:36 +04:00
mace - > pci . control = macepci_mask ;
flush_mace_bus ( ) ;
}
2006-07-02 17:41:42 +04:00
static struct irq_chip ip32_macepci_interrupt = {
2007-01-14 18:07:25 +03:00
. name = " IP32 MACE PCI " ,
2011-03-24 00:09:13 +03:00
. irq_mask = disable_macepci_irq ,
. irq_unmask = enable_macepci_irq ,
2005-04-17 02:20:36 +04:00
} ;
/* This is used for MACE ISA interrupts. That means bits 4-6 in the
* CRIME register .
*/
# define MACEISA_AUDIO_INT (MACEISA_AUDIO_SW_INT | \
MACEISA_AUDIO_SC_INT | \
MACEISA_AUDIO1_DMAT_INT | \
MACEISA_AUDIO1_OF_INT | \
MACEISA_AUDIO2_DMAT_INT | \
MACEISA_AUDIO2_MERR_INT | \
MACEISA_AUDIO3_DMAT_INT | \
MACEISA_AUDIO3_MERR_INT )
# define MACEISA_MISC_INT (MACEISA_RTC_INT | \
MACEISA_KEYB_INT | \
MACEISA_KEYB_POLL_INT | \
MACEISA_MOUSE_INT | \
MACEISA_MOUSE_POLL_INT | \
2006-07-05 21:43:29 +04:00
MACEISA_TIMER0_INT | \
MACEISA_TIMER1_INT | \
MACEISA_TIMER2_INT )
2005-04-17 02:20:36 +04:00
# define MACEISA_SUPERIO_INT (MACEISA_PARALLEL_INT | \
MACEISA_PAR_CTXA_INT | \
MACEISA_PAR_CTXB_INT | \
MACEISA_PAR_MERR_INT | \
MACEISA_SERIAL1_INT | \
MACEISA_SERIAL1_TDMAT_INT | \
MACEISA_SERIAL1_TDMAPR_INT | \
MACEISA_SERIAL1_TDMAME_INT | \
MACEISA_SERIAL1_RDMAT_INT | \
MACEISA_SERIAL1_RDMAOR_INT | \
MACEISA_SERIAL2_INT | \
MACEISA_SERIAL2_TDMAT_INT | \
MACEISA_SERIAL2_TDMAPR_INT | \
MACEISA_SERIAL2_TDMAME_INT | \
MACEISA_SERIAL2_RDMAT_INT | \
MACEISA_SERIAL2_RDMAOR_INT )
static unsigned long maceisa_mask ;
2011-03-24 00:09:13 +03:00
static void enable_maceisa_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
unsigned int crime_int = 0 ;
2011-03-24 00:09:13 +03:00
pr_debug ( " maceisa enable: %u \n " , d - > irq ) ;
2005-04-17 02:20:36 +04:00
2011-03-24 00:09:13 +03:00
switch ( d - > irq ) {
2005-04-17 02:20:36 +04:00
case MACEISA_AUDIO_SW_IRQ . . . MACEISA_AUDIO3_MERR_IRQ :
crime_int = MACE_AUDIO_INT ;
break ;
2006-07-05 21:43:29 +04:00
case MACEISA_RTC_IRQ . . . MACEISA_TIMER2_IRQ :
2005-04-17 02:20:36 +04:00
crime_int = MACE_MISC_INT ;
break ;
case MACEISA_PARALLEL_IRQ . . . MACEISA_SERIAL2_RDMAOR_IRQ :
crime_int = MACE_SUPERIO_INT ;
break ;
}
2007-10-28 21:46:39 +03:00
pr_debug ( " crime_int %08x enabled \n " , crime_int ) ;
2005-04-17 02:20:36 +04:00
crime_mask | = crime_int ;
crime - > imask = crime_mask ;
2011-03-24 00:09:13 +03:00
maceisa_mask | = 1 < < ( d - > irq - MACEISA_AUDIO_SW_IRQ ) ;
2005-04-17 02:20:36 +04:00
mace - > perif . ctrl . imask = maceisa_mask ;
}
2011-03-24 00:09:13 +03:00
static void disable_maceisa_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
unsigned int crime_int = 0 ;
2011-03-24 00:09:13 +03:00
maceisa_mask & = ~ ( 1 < < ( d - > irq - MACEISA_AUDIO_SW_IRQ ) ) ;
2013-01-22 15:59:30 +04:00
if ( ! ( maceisa_mask & MACEISA_AUDIO_INT ) )
2005-04-17 02:20:36 +04:00
crime_int | = MACE_AUDIO_INT ;
2013-01-22 15:59:30 +04:00
if ( ! ( maceisa_mask & MACEISA_MISC_INT ) )
2005-04-17 02:20:36 +04:00
crime_int | = MACE_MISC_INT ;
2013-01-22 15:59:30 +04:00
if ( ! ( maceisa_mask & MACEISA_SUPERIO_INT ) )
2005-04-17 02:20:36 +04:00
crime_int | = MACE_SUPERIO_INT ;
crime_mask & = ~ crime_int ;
crime - > imask = crime_mask ;
flush_crime_bus ( ) ;
mace - > perif . ctrl . imask = maceisa_mask ;
flush_mace_bus ( ) ;
}
2011-03-24 00:09:13 +03:00
static void mask_and_ack_maceisa_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
2006-11-01 20:08:36 +03:00
unsigned long mace_int ;
2005-04-17 02:20:36 +04:00
2009-03-30 16:49:44 +04:00
/* edge triggered */
mace_int = mace - > perif . ctrl . istat ;
2011-03-24 00:09:13 +03:00
mace_int & = ~ ( 1 < < ( d - > irq - MACEISA_AUDIO_SW_IRQ ) ) ;
2009-03-30 16:49:44 +04:00
mace - > perif . ctrl . istat = mace_int ;
2011-03-24 00:09:13 +03:00
disable_maceisa_irq ( d ) ;
2005-04-17 02:20:36 +04:00
}
2009-03-30 16:49:44 +04:00
static struct irq_chip ip32_maceisa_level_interrupt = {
. name = " IP32 MACE ISA " ,
2011-03-24 00:09:13 +03:00
. irq_mask = disable_maceisa_irq ,
. irq_unmask = enable_maceisa_irq ,
2009-03-30 16:49:44 +04:00
} ;
static struct irq_chip ip32_maceisa_edge_interrupt = {
2007-10-28 21:46:39 +03:00
. name = " IP32 MACE ISA " ,
2011-03-24 00:09:13 +03:00
. irq_ack = mask_and_ack_maceisa_irq ,
. irq_mask = disable_maceisa_irq ,
. irq_mask_ack = mask_and_ack_maceisa_irq ,
. irq_unmask = enable_maceisa_irq ,
2005-04-17 02:20:36 +04:00
} ;
/* This is used for regular non-ISA, non-PCI MACE interrupts. That means
* bits 0 - 3 and 7 in the CRIME register .
*/
2011-03-24 00:09:13 +03:00
static void enable_mace_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
2011-03-24 00:09:13 +03:00
unsigned int bit = d - > irq - CRIME_IRQ_BASE ;
2007-10-30 18:43:44 +03:00
crime_mask | = ( 1 < < bit ) ;
2005-04-17 02:20:36 +04:00
crime - > imask = crime_mask ;
}
2011-03-24 00:09:13 +03:00
static void disable_mace_irq ( struct irq_data * d )
2005-04-17 02:20:36 +04:00
{
2011-03-24 00:09:13 +03:00
unsigned int bit = d - > irq - CRIME_IRQ_BASE ;
2007-10-30 18:43:44 +03:00
crime_mask & = ~ ( 1 < < bit ) ;
2005-04-17 02:20:36 +04:00
crime - > imask = crime_mask ;
flush_crime_bus ( ) ;
}
2006-07-02 17:41:42 +04:00
static struct irq_chip ip32_mace_interrupt = {
2007-01-14 18:07:25 +03:00
. name = " IP32 MACE " ,
2011-03-24 00:09:13 +03:00
. irq_mask = disable_mace_irq ,
. irq_unmask = enable_mace_irq ,
2005-04-17 02:20:36 +04:00
} ;
2006-10-07 22:44:33 +04:00
static void ip32_unknown_interrupt ( void )
2005-04-17 02:20:36 +04:00
{
2007-10-12 02:46:15 +04:00
printk ( " Unknown interrupt occurred! \n " ) ;
printk ( " cp0_status: %08x \n " , read_c0_status ( ) ) ;
printk ( " cp0_cause: %08x \n " , read_c0_cause ( ) ) ;
printk ( " CRIME intr mask: %016lx \n " , crime - > imask ) ;
printk ( " CRIME intr status: %016lx \n " , crime - > istat ) ;
printk ( " CRIME hardware intr register: %016lx \n " , crime - > hard_int ) ;
printk ( " MACE ISA intr mask: %08lx \n " , mace - > perif . ctrl . imask ) ;
printk ( " MACE ISA intr status: %08lx \n " , mace - > perif . ctrl . istat ) ;
printk ( " MACE PCI control register: %08x \n " , mace - > pci . control ) ;
2005-04-17 02:20:36 +04:00
printk ( " Register dump: \n " ) ;
2006-10-07 22:44:33 +04:00
show_regs ( get_irq_regs ( ) ) ;
2005-04-17 02:20:36 +04:00
printk ( " Please mail this report to linux-mips@linux-mips.org \n " ) ;
printk ( " Spinning... " ) ;
while ( 1 ) ;
}
/* CRIME 1.1 appears to deliver all interrupts to this one pin. */
/* change this to loop over all edge-triggered irqs, exception masked out ones */
2006-10-07 22:44:33 +04:00
static void ip32_irq0 ( void )
2005-04-17 02:20:36 +04:00
{
uint64_t crime_int ;
int irq = 0 ;
2007-10-14 17:02:26 +04:00
/*
* Sanity check interrupt numbering enum .
* MACE got 32 interrupts and there are 32 MACE ISA interrupts daisy
* chained .
*/
BUILD_BUG_ON ( CRIME_VICE_IRQ - MACE_VID_IN1_IRQ ! = 31 ) ;
BUILD_BUG_ON ( MACEISA_SERIAL2_RDMAOR_IRQ - MACEISA_AUDIO_SW_IRQ ! = 31 ) ;
2005-04-17 02:20:36 +04:00
crime_int = crime - > istat & crime_mask ;
2008-06-24 02:48:05 +04:00
/* crime sometime delivers spurious interrupts, ignore them */
if ( unlikely ( crime_int = = 0 ) )
return ;
2007-10-14 17:02:26 +04:00
irq = MACE_VID_IN1_IRQ + __ffs ( crime_int ) ;
2005-04-17 02:20:36 +04:00
if ( crime_int & CRIME_MACEISA_INT_MASK ) {
unsigned long mace_int = mace - > perif . ctrl . istat ;
2007-10-14 17:02:26 +04:00
irq = __ffs ( mace_int & maceisa_mask ) + MACEISA_AUDIO_SW_IRQ ;
2005-04-17 02:20:36 +04:00
}
2007-10-14 17:02:26 +04:00
2007-10-28 21:46:39 +03:00
pr_debug ( " *irq %u* \n " , irq ) ;
2006-10-07 22:44:33 +04:00
do_IRQ ( irq ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-07 22:44:33 +04:00
static void ip32_irq1 ( void )
2005-04-17 02:20:36 +04:00
{
2006-10-07 22:44:33 +04:00
ip32_unknown_interrupt ( ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-07 22:44:33 +04:00
static void ip32_irq2 ( void )
2005-04-17 02:20:36 +04:00
{
2006-10-07 22:44:33 +04:00
ip32_unknown_interrupt ( ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-07 22:44:33 +04:00
static void ip32_irq3 ( void )
2005-04-17 02:20:36 +04:00
{
2006-10-07 22:44:33 +04:00
ip32_unknown_interrupt ( ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-07 22:44:33 +04:00
static void ip32_irq4 ( void )
2005-04-17 02:20:36 +04:00
{
2006-10-07 22:44:33 +04:00
ip32_unknown_interrupt ( ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-07 22:44:33 +04:00
static void ip32_irq5 ( void )
2005-04-17 02:20:36 +04:00
{
2007-10-14 17:02:26 +04:00
do_IRQ ( MIPS_CPU_IRQ_BASE + 7 ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-07 22:44:33 +04:00
asmlinkage void plat_irq_dispatch ( void )
2006-04-03 20:56:36 +04:00
{
2007-03-19 03:13:37 +03:00
unsigned int pending = read_c0_status ( ) & read_c0_cause ( ) ;
2006-04-03 20:56:36 +04:00
if ( likely ( pending & IE_IRQ0 ) )
2006-10-07 22:44:33 +04:00
ip32_irq0 ( ) ;
2006-04-03 20:56:36 +04:00
else if ( unlikely ( pending & IE_IRQ1 ) )
2006-10-07 22:44:33 +04:00
ip32_irq1 ( ) ;
2006-04-03 20:56:36 +04:00
else if ( unlikely ( pending & IE_IRQ2 ) )
2006-10-07 22:44:33 +04:00
ip32_irq2 ( ) ;
2006-04-03 20:56:36 +04:00
else if ( unlikely ( pending & IE_IRQ3 ) )
2006-10-07 22:44:33 +04:00
ip32_irq3 ( ) ;
2006-04-03 20:56:36 +04:00
else if ( unlikely ( pending & IE_IRQ4 ) )
2006-10-07 22:44:33 +04:00
ip32_irq4 ( ) ;
2006-04-03 20:56:36 +04:00
else if ( likely ( pending & IE_IRQ5 ) )
2006-10-07 22:44:33 +04:00
ip32_irq5 ( ) ;
2006-04-03 20:56:36 +04:00
}
2005-04-17 02:20:36 +04:00
void __init arch_init_irq ( void )
{
unsigned int irq ;
/* Install our interrupt handler, then clear and disable all
* CRIME and MACE interrupts . */
crime - > imask = 0 ;
crime - > hard_int = 0 ;
crime - > soft_int = 0 ;
mace - > perif . ctrl . istat = 0 ;
mace - > perif . ctrl . imask = 0 ;
2007-10-14 17:02:26 +04:00
mips_cpu_irq_init ( ) ;
2007-10-30 18:43:44 +03:00
for ( irq = CRIME_IRQ_BASE ; irq < = IP32_IRQ_MAX ; irq + + ) {
2007-10-14 17:02:26 +04:00
switch ( irq ) {
case MACE_VID_IN1_IRQ . . . MACE_PCI_BRIDGE_IRQ :
2011-03-27 17:19:28 +04:00
irq_set_chip_and_handler_name ( irq ,
& ip32_mace_interrupt ,
handle_level_irq ,
" level " ) ;
2007-10-14 17:02:26 +04:00
break ;
2009-03-30 16:49:44 +04:00
2007-10-14 17:02:26 +04:00
case MACEPCI_SCSI0_IRQ . . . MACEPCI_SHARED2_IRQ :
2011-03-27 17:19:28 +04:00
irq_set_chip_and_handler_name ( irq ,
& ip32_macepci_interrupt ,
handle_level_irq ,
" level " ) ;
2007-10-28 21:46:39 +03:00
break ;
2009-03-30 16:49:44 +04:00
2007-10-28 21:46:39 +03:00
case CRIME_CPUERR_IRQ :
case CRIME_MEMERR_IRQ :
2011-03-27 17:19:28 +04:00
irq_set_chip_and_handler_name ( irq ,
& crime_level_interrupt ,
handle_level_irq ,
" level " ) ;
2007-10-14 17:02:26 +04:00
break ;
2009-03-30 16:49:44 +04:00
2010-01-20 02:59:27 +03:00
case CRIME_GBE0_IRQ . . . CRIME_GBE3_IRQ :
2007-10-28 21:46:39 +03:00
case CRIME_RE_EMPTY_E_IRQ . . . CRIME_RE_IDLE_E_IRQ :
case CRIME_SOFT0_IRQ . . . CRIME_SOFT2_IRQ :
case CRIME_VICE_IRQ :
2011-03-27 17:19:28 +04:00
irq_set_chip_and_handler_name ( irq ,
& crime_edge_interrupt ,
handle_edge_irq ,
" edge " ) ;
2009-03-30 16:49:44 +04:00
break ;
case MACEISA_PARALLEL_IRQ :
case MACEISA_SERIAL1_TDMAPR_IRQ :
case MACEISA_SERIAL2_TDMAPR_IRQ :
2011-03-27 17:19:28 +04:00
irq_set_chip_and_handler_name ( irq ,
& ip32_maceisa_edge_interrupt ,
handle_edge_irq ,
" edge " ) ;
2007-10-14 17:02:26 +04:00
break ;
2009-03-30 16:49:44 +04:00
2007-10-14 17:02:26 +04:00
default :
2011-03-27 17:19:28 +04:00
irq_set_chip_and_handler_name ( irq ,
& ip32_maceisa_level_interrupt ,
handle_level_irq ,
" level " ) ;
2007-10-28 21:46:39 +03:00
break ;
2007-10-14 17:02:26 +04:00
}
2005-04-17 02:20:36 +04:00
}
setup_irq ( CRIME_MEMERR_IRQ , & memerr_irq ) ;
setup_irq ( CRIME_CPUERR_IRQ , & cpuerr_irq ) ;
# define ALLINTS (IE_IRQ0 | IE_IRQ1 | IE_IRQ2 | IE_IRQ3 | IE_IRQ4 | IE_IRQ5)
change_c0_status ( ST0_IM , ALLINTS ) ;
}