2005-04-16 15:20:36 -07:00
/*
* 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 .
*
* Code to handle x86 style IRQs plus some generic interrupt stuff .
*
* Copyright ( C ) 1992 Linus Torvalds
* Copyright ( C ) 1994 - 2000 Ralf Baechle
*/
# include <linux/kernel.h>
# include <linux/delay.h>
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/kernel_stat.h>
# include <linux/proc_fs.h>
# include <linux/mm.h>
# include <linux/random.h>
# include <linux/sched.h>
# include <linux/seq_file.h>
# include <linux/kallsyms.h>
2008-07-29 15:58:53 -05:00
# include <linux/kgdb.h>
2009-11-20 20:34:33 +08:00
# include <linux/ftrace.h>
2005-04-16 15:20:36 -07:00
2011-07-26 16:09:06 -07:00
# include <linux/atomic.h>
2005-04-16 15:20:36 -07:00
# include <asm/uaccess.h>
2008-07-29 15:58:53 -05:00
# ifdef CONFIG_KGDB
int kgdb_early_setup ;
# endif
2006-11-06 17:41:06 +00:00
static unsigned long irq_map [ NR_IRQS / BITS_PER_LONG ] ;
2007-05-10 01:20:30 +09:00
int allocate_irqno ( void )
2006-11-06 17:41:06 +00:00
{
int irq ;
again :
irq = find_first_zero_bit ( irq_map , NR_IRQS ) ;
if ( irq > = NR_IRQS )
return - ENOSPC ;
if ( test_and_set_bit ( irq , irq_map ) )
goto again ;
return irq ;
}
/*
* Allocate the 16 legacy interrupts for i8259 devices . This happens early
* in the kernel initialization so treating allocation failure as BUG ( ) is
* ok .
*/
void __init alloc_legacy_irqno ( void )
{
int i ;
for ( i = 0 ; i < = 16 ; i + + )
BUG_ON ( test_and_set_bit ( i , irq_map ) ) ;
}
2007-05-10 01:20:30 +09:00
void free_irqno ( unsigned int irq )
2006-11-06 17:41:06 +00:00
{
smp_mb__before_clear_bit ( ) ;
clear_bit ( irq , irq_map ) ;
smp_mb__after_clear_bit ( ) ;
}
2005-04-16 15:20:36 -07:00
/*
* ' what should we do if we get a hw irq event on an illegal vector ' .
* each architecture has to answer this themselves .
*/
void ack_bad_irq ( unsigned int irq )
{
2007-09-21 17:13:55 +01:00
smtc_im_ack_irq ( irq ) ;
2005-04-16 15:20:36 -07:00
printk ( " unexpected IRQ # %d \n " , irq ) ;
}
atomic_t irq_err_count ;
2011-03-23 21:09:05 +00:00
int arch_show_interrupts ( struct seq_file * p , int prec )
2005-04-16 15:20:36 -07:00
{
2011-03-23 21:09:05 +00:00
seq_printf ( p , " %*s: %10u \n " , prec , " ERR " , atomic_read ( & irq_err_count ) ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2006-10-07 19:44:33 +01:00
asmlinkage void spurious_interrupt ( void )
2006-04-01 21:17:45 +01:00
{
atomic_inc ( & irq_err_count ) ;
}
2005-04-16 15:20:36 -07:00
void __init init_IRQ ( void )
{
2008-02-08 04:22:02 -08:00
int i ;
2008-07-29 15:58:53 -05:00
# ifdef CONFIG_KGDB
if ( kgdb_early_setup )
return ;
# endif
2008-02-08 04:22:02 -08:00
for ( i = 0 ; i < NR_IRQS ; i + + )
2011-03-27 15:19:28 +02:00
irq_set_noprobe ( i ) ;
2008-02-08 04:22:02 -08:00
2005-04-16 15:20:36 -07:00
arch_init_irq ( ) ;
2008-07-29 15:58:53 -05:00
# ifdef CONFIG_KGDB
if ( ! kgdb_early_setup )
kgdb_early_setup = 1 ;
# endif
2005-04-16 15:20:36 -07:00
}
2009-11-20 20:34:33 +08:00
2010-08-27 18:32:06 +09:00
# ifdef DEBUG_STACKOVERFLOW
static inline void check_stack_overflow ( void )
{
unsigned long sp ;
__asm__ __volatile__ ( " move %0, $sp " : " =r " ( sp ) ) ;
sp & = THREAD_MASK ;
/*
* Check for stack overflow : is there less than STACK_WARN free ?
* STACK_WARN is defined as 1 / 8 of THREAD_SIZE by default .
*/
if ( unlikely ( sp < ( sizeof ( struct thread_info ) + STACK_WARN ) ) ) {
printk ( " do_IRQ: stack overflow: %ld \n " ,
sp - sizeof ( struct thread_info ) ) ;
dump_stack ( ) ;
}
}
# else
static inline void check_stack_overflow ( void ) { }
# endif
2009-11-20 20:34:33 +08:00
/*
* do_IRQ handles all normal device IRQ ' s ( the special
* SMP cross - CPU interrupts have their own specific
* handlers ) .
*/
void __irq_entry do_IRQ ( unsigned int irq )
{
irq_enter ( ) ;
2010-08-27 18:32:06 +09:00
check_stack_overflow ( ) ;
2011-03-23 21:09:04 +00:00
if ( ! smtc_handle_on_other_cpu ( irq ) )
generic_handle_irq ( irq ) ;
2009-11-20 20:34:33 +08:00
irq_exit ( ) ;
}
# ifdef CONFIG_MIPS_MT_SMTC_IRQAFF
/*
* To avoid inefficient and in some cases pathological re - checking of
* IRQ affinity , we have this variant that skips the affinity check .
*/
void __irq_entry do_IRQ_no_affinity ( unsigned int irq )
{
irq_enter ( ) ;
2011-03-23 21:09:04 +00:00
smtc_im_backstop ( irq ) ;
2009-11-20 20:34:33 +08:00
generic_handle_irq ( irq ) ;
irq_exit ( ) ;
}
# endif /* CONFIG_MIPS_MT_SMTC_IRQAFF */