2005-04-16 15:20:36 -07:00
/*
* linux / arch / alpha / kernel / irq . c
*
* Copyright ( C ) 1995 Linus Torvalds
*
* This file contains the code used by various IRQ handling routines :
* asking for different IRQ ' s should be done through these routines
* instead of just grabbing them . Thus setups with different IRQ numbers
* shouldn ' t result in any weird surprises , and installing new handlers
* should be easier .
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/errno.h>
# include <linux/kernel_stat.h>
# include <linux/signal.h>
# include <linux/sched.h>
# include <linux/ptrace.h>
# include <linux/interrupt.h>
# include <linux/random.h>
# include <linux/irq.h>
# include <linux/proc_fs.h>
# include <linux/seq_file.h>
# include <linux/profile.h>
# include <linux/bitops.h>
# include <asm/io.h>
# include <asm/uaccess.h>
volatile unsigned long irq_err_count ;
2010-08-09 17:20:05 -07:00
DEFINE_PER_CPU ( unsigned long , irq_pmi_count ) ;
2005-04-16 15:20:36 -07:00
2006-01-06 00:12:22 -08:00
void ack_bad_irq ( unsigned int irq )
2005-04-16 15:20:36 -07:00
{
irq_err_count + + ;
printk ( KERN_CRIT " Unexpected IRQ trap at vector %u \n " , irq ) ;
}
# ifdef CONFIG_SMP
static char irq_user_affinity [ NR_IRQS ] ;
2008-05-29 11:02:52 -07:00
int irq_select_affinity ( unsigned int irq )
2005-04-16 15:20:36 -07:00
{
2011-02-06 14:33:00 +00:00
struct irq_data * data = irq_get_irq_data ( irq ) ;
struct irq_chip * chip ;
2005-04-16 15:20:36 -07:00
static int last_cpu ;
int cpu = last_cpu + 1 ;
2011-02-06 14:33:00 +00:00
if ( ! data )
return 1 ;
chip = irq_data_get_irq_chip ( data ) ;
if ( ! chip - > irq_set_affinity | | irq_user_affinity [ irq ] )
2006-01-06 00:12:22 -08:00
return 1 ;
2005-04-16 15:20:36 -07:00
2009-01-01 10:12:26 +10:30
while ( ! cpu_possible ( cpu ) | |
! cpumask_test_cpu ( cpu , irq_default_affinity ) )
2005-04-16 15:20:36 -07:00
cpu = ( cpu < ( NR_CPUS - 1 ) ? cpu + 1 : 0 ) ;
last_cpu = cpu ;
2015-07-13 20:08:38 +00:00
cpumask_copy ( irq_data_get_affinity_mask ( data ) , cpumask_of ( cpu ) ) ;
2011-02-06 14:33:00 +00:00
chip - > irq_set_affinity ( data , cpumask_of ( cpu ) , false ) ;
2006-01-06 00:12:22 -08:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
# endif /* CONFIG_SMP */
2011-03-25 22:20:51 +01:00
int arch_show_interrupts ( struct seq_file * p , int prec )
2005-04-16 15:20:36 -07:00
{
int j ;
# ifdef CONFIG_SMP
2011-03-25 22:20:51 +01:00
seq_puts ( p , " IPI: " ) ;
for_each_online_cpu ( j )
seq_printf ( p , " %10lu " , cpu_data [ j ] . ipi_count ) ;
seq_putc ( p , ' \n ' ) ;
2005-04-16 15:20:36 -07:00
# endif
2011-03-25 22:20:51 +01:00
seq_puts ( p , " PMI: " ) ;
for_each_online_cpu ( j )
seq_printf ( p , " %10lu " , per_cpu ( irq_pmi_count , j ) ) ;
seq_puts ( p , " Performance Monitoring \n " ) ;
seq_printf ( p , " ERR: %10lu \n " , irq_err_count ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
/*
* handle_irq handles all normal device IRQ ' s ( the special
* SMP cross - CPU interrupts have their own specific
* handlers ) .
*/
# define MAX_ILLEGAL_IRQS 16
void
2006-10-08 14:37:32 +01:00
handle_irq ( int irq )
2005-04-16 15:20:36 -07:00
{
/*
* We ack quickly , we don ' t want the irq controller
* thinking we ' re snobs just because some other CPU has
* disabled global interrupts ( we have already done the
* INT_ACK cycles , it ' s too late to try to pretend to the
* controller that we aren ' t taking the interrupt ) .
*
* 0 return value means that this irq is already being
* handled by some other CPU . ( or is disabled )
*/
static unsigned int illegal_count = 0 ;
2010-10-14 22:31:25 -04:00
struct irq_desc * desc = irq_to_desc ( irq ) ;
2005-04-16 15:20:36 -07:00
2010-10-14 22:31:25 -04:00
if ( ! desc | | ( ( unsigned ) irq > ACTUAL_NR_IRQS & &
illegal_count < MAX_ILLEGAL_IRQS ) ) {
2005-04-16 15:20:36 -07:00
irq_err_count + + ;
illegal_count + + ;
printk ( KERN_CRIT " device_interrupt: invalid interrupt %d \n " ,
irq ) ;
return ;
}
2011-01-12 22:02:24 +00:00
irq_enter ( ) ;
2015-09-14 10:42:37 +02:00
generic_handle_irq_desc ( desc ) ;
2005-04-16 15:20:36 -07:00
irq_exit ( ) ;
}