2008-10-16 11:32:24 +02:00
/*
* Common interrupt code for 32 and 64 bit
*/
# include <linux/cpu.h>
# include <linux/interrupt.h>
# include <linux/kernel_stat.h>
2010-11-12 05:45:26 +00:00
# include <linux/of.h>
2008-10-16 11:32:24 +02:00
# include <linux/seq_file.h>
2009-01-04 16:22:17 +05:30
# include <linux/smp.h>
2009-02-06 14:09:41 -08:00
# include <linux/ftrace.h>
2011-03-25 15:20:14 +01:00
# include <linux/delay.h>
2011-05-26 12:22:53 -04:00
# include <linux/export.h>
2008-10-16 11:32:24 +02:00
2009-02-17 13:58:15 +01:00
# include <asm/apic.h>
2008-10-16 11:32:24 +02:00
# include <asm/io_apic.h>
2008-12-23 15:15:17 +01:00
# include <asm/irq.h>
2009-02-06 14:09:41 -08:00
# include <asm/idle.h>
2009-05-27 21:56:52 +02:00
# include <asm/mce.h>
2009-04-11 00:03:10 +05:30
# include <asm/hw_irq.h>
2008-10-16 11:32:24 +02:00
atomic_t irq_err_count ;
2009-03-04 12:56:05 -06:00
/* Function pointer for generic interrupt vector handling */
2009-10-14 09:22:57 -05:00
void ( * x86_platform_ipi_callback ) ( void ) = NULL ;
2009-03-04 12:56:05 -06:00
2008-10-16 12:18:50 +02: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 )
{
2009-04-12 20:47:39 +04:00
if ( printk_ratelimit ( ) )
pr_err ( " unexpected IRQ trap at vector %02x \n " , irq ) ;
2008-10-16 12:18:50 +02:00
/*
* Currently unexpected vectors happen only on SMP and APIC .
* We _must_ ack these because every local APIC has only N
* irq slots per priority level , and a ' hanging , unacked ' IRQ
* holds up an irq slot - in excessive cases ( when multiple
* unexpected vectors occur ) that might lock up the APIC
* completely .
* But only ack when the APIC is enabled - AK
*/
2009-04-12 20:47:41 +04:00
ack_APIC_irq ( ) ;
2008-10-16 12:18:50 +02:00
}
2009-01-19 00:38:57 +09:00
# define irq_stats(x) (&per_cpu(irq_stat, x))
2008-10-16 11:32:24 +02:00
/*
2010-12-16 17:59:57 +01:00
* / proc / interrupts printing for arch specific interrupts
2008-10-16 11:32:24 +02:00
*/
2010-12-16 17:59:57 +01:00
int arch_show_interrupts ( struct seq_file * p , int prec )
2008-10-16 11:32:24 +02:00
{
int j ;
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: " , prec , " NMI " ) ;
2008-10-16 11:32:24 +02:00
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , irq_stats ( j ) - > __nmi_count ) ;
seq_printf ( p , " Non-maskable interrupts \n " ) ;
# ifdef CONFIG_X86_LOCAL_APIC
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: " , prec , " LOC " ) ;
2008-10-16 11:32:24 +02:00
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , irq_stats ( j ) - > apic_timer_irqs ) ;
seq_printf ( p , " Local timer interrupts \n " ) ;
2009-03-23 02:08:34 +05:30
seq_printf ( p , " %*s: " , prec , " SPU " ) ;
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , irq_stats ( j ) - > irq_spurious_count ) ;
seq_printf ( p , " Spurious interrupts \n " ) ;
2009-10-14 18:50:39 +08:00
seq_printf ( p , " %*s: " , prec , " PMI " ) ;
2008-12-03 10:39:53 +01:00
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , irq_stats ( j ) - > apic_perf_irqs ) ;
2009-10-14 18:50:39 +08:00
seq_printf ( p , " Performance monitoring interrupts \n " ) ;
2010-10-14 14:01:34 +08:00
seq_printf ( p , " %*s: " , prec , " IWI " ) ;
2009-04-06 11:45:03 +02:00
for_each_online_cpu ( j )
2010-10-14 14:01:34 +08:00
seq_printf ( p , " %10u " , irq_stats ( j ) - > apic_irq_work_irqs ) ;
seq_printf ( p , " IRQ work interrupts \n " ) ;
2011-12-13 11:51:53 +09:00
seq_printf ( p , " %*s: " , prec , " RTR " ) ;
for_each_online_cpu ( j )
2011-12-15 11:32:24 +09:00
seq_printf ( p , " %10u " , irq_stats ( j ) - > icr_read_retry_count ) ;
2011-12-13 11:51:53 +09:00
seq_printf ( p , " APIC ICR read retries \n " ) ;
2008-10-16 11:32:24 +02:00
# endif
2009-10-14 09:22:57 -05:00
if ( x86_platform_ipi_callback ) {
2009-03-25 10:50:34 +09:00
seq_printf ( p , " %*s: " , prec , " PLT " ) ;
2009-03-04 12:56:05 -06:00
for_each_online_cpu ( j )
2009-10-14 09:22:57 -05:00
seq_printf ( p , " %10u " , irq_stats ( j ) - > x86_platform_ipis ) ;
2009-03-04 12:56:05 -06:00
seq_printf ( p , " Platform interrupts \n " ) ;
}
2008-10-16 11:32:24 +02:00
# ifdef CONFIG_SMP
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: " , prec , " RES " ) ;
2008-10-16 11:32:24 +02:00
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , irq_stats ( j ) - > irq_resched_count ) ;
seq_printf ( p , " Rescheduling interrupts \n " ) ;
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: " , prec , " CAL " ) ;
2008-10-16 11:32:24 +02:00
for_each_online_cpu ( j )
2012-09-26 11:11:28 +09:00
seq_printf ( p , " %10u " , irq_stats ( j ) - > irq_call_count -
irq_stats ( j ) - > irq_tlb_count ) ;
2008-10-16 11:32:24 +02:00
seq_printf ( p , " Function call interrupts \n " ) ;
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: " , prec , " TLB " ) ;
2008-10-16 11:32:24 +02:00
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , irq_stats ( j ) - > irq_tlb_count ) ;
seq_printf ( p , " TLB shootdowns \n " ) ;
# endif
2009-11-20 14:03:05 +00:00
# ifdef CONFIG_X86_THERMAL_VECTOR
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: " , prec , " TRM " ) ;
2008-10-16 11:32:24 +02:00
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , irq_stats ( j ) - > irq_thermal_count ) ;
seq_printf ( p , " Thermal event interrupts \n " ) ;
2009-11-20 14:03:05 +00:00
# endif
# ifdef CONFIG_X86_MCE_THRESHOLD
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: " , prec , " THR " ) ;
2008-10-16 11:32:24 +02:00
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , irq_stats ( j ) - > irq_threshold_count ) ;
seq_printf ( p , " Threshold APIC interrupts \n " ) ;
2009-05-27 21:56:52 +02:00
# endif
2009-07-09 00:31:41 +02:00
# ifdef CONFIG_X86_MCE
2009-05-27 21:56:52 +02:00
seq_printf ( p , " %*s: " , prec , " MCE " ) ;
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , per_cpu ( mce_exception_count , j ) ) ;
seq_printf ( p , " Machine check exceptions \n " ) ;
2009-05-27 21:56:57 +02:00
seq_printf ( p , " %*s: " , prec , " MCP " ) ;
for_each_online_cpu ( j )
seq_printf ( p , " %10u " , per_cpu ( mce_poll_count , j ) ) ;
seq_printf ( p , " Machine check polls \n " ) ;
2008-10-16 11:32:24 +02:00
# endif
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: %10u \n " , prec , " ERR " , atomic_read ( & irq_err_count ) ) ;
2008-10-16 11:32:24 +02:00
# if defined(CONFIG_X86_IO_APIC)
2009-03-12 12:45:15 +00:00
seq_printf ( p , " %*s: %10u \n " , prec , " MIS " , atomic_read ( & irq_mis_count ) ) ;
2008-10-16 11:32:24 +02:00
# endif
return 0 ;
}
/*
* / proc / stat helpers
*/
u64 arch_irq_stat_cpu ( unsigned int cpu )
{
u64 sum = irq_stats ( cpu ) - > __nmi_count ;
# ifdef CONFIG_X86_LOCAL_APIC
sum + = irq_stats ( cpu ) - > apic_timer_irqs ;
2009-03-23 02:08:34 +05:30
sum + = irq_stats ( cpu ) - > irq_spurious_count ;
2008-12-03 10:39:53 +01:00
sum + = irq_stats ( cpu ) - > apic_perf_irqs ;
2010-10-14 14:01:34 +08:00
sum + = irq_stats ( cpu ) - > apic_irq_work_irqs ;
2011-12-15 11:32:24 +09:00
sum + = irq_stats ( cpu ) - > icr_read_retry_count ;
2008-10-16 11:32:24 +02:00
# endif
2009-10-14 09:22:57 -05:00
if ( x86_platform_ipi_callback )
sum + = irq_stats ( cpu ) - > x86_platform_ipis ;
2008-10-16 11:32:24 +02:00
# ifdef CONFIG_SMP
sum + = irq_stats ( cpu ) - > irq_resched_count ;
sum + = irq_stats ( cpu ) - > irq_call_count ;
# endif
2009-11-20 14:03:05 +00:00
# ifdef CONFIG_X86_THERMAL_VECTOR
2008-10-16 11:32:24 +02:00
sum + = irq_stats ( cpu ) - > irq_thermal_count ;
2009-11-20 14:03:05 +00:00
# endif
# ifdef CONFIG_X86_MCE_THRESHOLD
2008-10-16 11:32:24 +02:00
sum + = irq_stats ( cpu ) - > irq_threshold_count ;
2009-06-02 16:53:23 +09:00
# endif
2009-07-09 00:31:41 +02:00
# ifdef CONFIG_X86_MCE
2009-06-02 16:53:23 +09:00
sum + = per_cpu ( mce_exception_count , cpu ) ;
sum + = per_cpu ( mce_poll_count , cpu ) ;
2008-10-16 11:32:24 +02:00
# endif
return sum ;
}
u64 arch_irq_stat ( void )
{
u64 sum = atomic_read ( & irq_err_count ) ;
return sum ;
}
2008-12-23 15:15:17 +01:00
2009-02-06 14:09:41 -08:00
/*
* do_IRQ handles all normal device IRQ ' s ( the special
* SMP cross - CPU interrupts have their own specific
* handlers ) .
*/
unsigned int __irq_entry do_IRQ ( struct pt_regs * regs )
{
struct pt_regs * old_regs = set_irq_regs ( regs ) ;
/* high bit used in ret_from_ code */
unsigned vector = ~ regs - > orig_ax ;
unsigned irq ;
irq_enter ( ) ;
2011-10-07 18:22:09 +02:00
exit_idle ( ) ;
2009-02-06 14:09:41 -08:00
2010-12-18 16:28:55 +01:00
irq = __this_cpu_read ( vector_irq [ vector ] ) ;
2009-02-06 14:09:41 -08:00
if ( ! handle_irq ( irq , regs ) ) {
2009-04-12 20:47:41 +04:00
ack_APIC_irq ( ) ;
2009-02-06 14:09:41 -08:00
if ( printk_ratelimit ( ) )
2009-04-12 20:47:39 +04:00
pr_emerg ( " %s: %d.%d No irq handler for vector (irq %d) \n " ,
__func__ , smp_processor_id ( ) , vector , irq ) ;
2009-02-06 14:09:41 -08:00
}
irq_exit ( ) ;
set_irq_regs ( old_regs ) ;
return 1 ;
}
2009-03-04 12:56:05 -06:00
/*
2009-10-14 09:22:57 -05:00
* Handler for X86_PLATFORM_IPI_VECTOR .
2009-03-04 12:56:05 -06:00
*/
x86, trace: Introduce entering/exiting_irq()
When implementing tracepoints in interrupt handers, if the tracepoints are
simply added in the performance sensitive path of interrupt handers,
it may cause potential performance problem due to the time penalty.
To solve the problem, an idea is to prepare non-trace/trace irq handers and
switch their IDTs at the enabling/disabling time.
So, let's introduce entering_irq()/exiting_irq() for pre/post-
processing of each irq handler.
A way to use them is as follows.
Non-trace irq handler:
smp_irq_handler()
{
entering_irq(); /* pre-processing of this handler */
__smp_irq_handler(); /*
* common logic between non-trace and trace handlers
* in a vector.
*/
exiting_irq(); /* post-processing of this handler */
}
Trace irq_handler:
smp_trace_irq_handler()
{
entering_irq(); /* pre-processing of this handler */
trace_irq_entry(); /* tracepoint for irq entry */
__smp_irq_handler(); /*
* common logic between non-trace and trace handlers
* in a vector.
*/
trace_irq_exit(); /* tracepoint for irq exit */
exiting_irq(); /* post-processing of this handler */
}
If tracepoints can place outside entering_irq()/exiting_irq() as follows,
it looks cleaner.
smp_trace_irq_handler()
{
trace_irq_entry();
smp_irq_handler();
trace_irq_exit();
}
But it doesn't work.
The problem is with irq_enter/exit() being called. They must be called before
trace_irq_enter/exit(), because of the rcu_irq_enter() must be called before
any tracepoints are used, as tracepoints use rcu to synchronize.
As a possible alternative, we may be able to call irq_enter() first as follows
if irq_enter() can nest.
smp_trace_irq_hander()
{
irq_entry();
trace_irq_entry();
smp_irq_handler();
trace_irq_exit();
irq_exit();
}
But it doesn't work, either.
If irq_enter() is nested, it may have a time penalty because it has to check if it
was already called or not. The time penalty is not desired in performance sensitive
paths even if it is tiny.
Signed-off-by: Seiji Aguchi <seiji.aguchi@hds.com>
Link: http://lkml.kernel.org/r/51C3238D.9040706@hds.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
2013-06-20 11:45:17 -04:00
void __smp_x86_platform_ipi ( void )
2009-03-04 12:56:05 -06:00
{
2009-10-14 09:22:57 -05:00
inc_irq_stat ( x86_platform_ipis ) ;
2009-03-04 12:56:05 -06:00
2009-10-14 09:22:57 -05:00
if ( x86_platform_ipi_callback )
x86_platform_ipi_callback ( ) ;
x86, trace: Introduce entering/exiting_irq()
When implementing tracepoints in interrupt handers, if the tracepoints are
simply added in the performance sensitive path of interrupt handers,
it may cause potential performance problem due to the time penalty.
To solve the problem, an idea is to prepare non-trace/trace irq handers and
switch their IDTs at the enabling/disabling time.
So, let's introduce entering_irq()/exiting_irq() for pre/post-
processing of each irq handler.
A way to use them is as follows.
Non-trace irq handler:
smp_irq_handler()
{
entering_irq(); /* pre-processing of this handler */
__smp_irq_handler(); /*
* common logic between non-trace and trace handlers
* in a vector.
*/
exiting_irq(); /* post-processing of this handler */
}
Trace irq_handler:
smp_trace_irq_handler()
{
entering_irq(); /* pre-processing of this handler */
trace_irq_entry(); /* tracepoint for irq entry */
__smp_irq_handler(); /*
* common logic between non-trace and trace handlers
* in a vector.
*/
trace_irq_exit(); /* tracepoint for irq exit */
exiting_irq(); /* post-processing of this handler */
}
If tracepoints can place outside entering_irq()/exiting_irq() as follows,
it looks cleaner.
smp_trace_irq_handler()
{
trace_irq_entry();
smp_irq_handler();
trace_irq_exit();
}
But it doesn't work.
The problem is with irq_enter/exit() being called. They must be called before
trace_irq_enter/exit(), because of the rcu_irq_enter() must be called before
any tracepoints are used, as tracepoints use rcu to synchronize.
As a possible alternative, we may be able to call irq_enter() first as follows
if irq_enter() can nest.
smp_trace_irq_hander()
{
irq_entry();
trace_irq_entry();
smp_irq_handler();
trace_irq_exit();
irq_exit();
}
But it doesn't work, either.
If irq_enter() is nested, it may have a time penalty because it has to check if it
was already called or not. The time penalty is not desired in performance sensitive
paths even if it is tiny.
Signed-off-by: Seiji Aguchi <seiji.aguchi@hds.com>
Link: http://lkml.kernel.org/r/51C3238D.9040706@hds.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
2013-06-20 11:45:17 -04:00
}
2009-03-04 12:56:05 -06:00
x86, trace: Introduce entering/exiting_irq()
When implementing tracepoints in interrupt handers, if the tracepoints are
simply added in the performance sensitive path of interrupt handers,
it may cause potential performance problem due to the time penalty.
To solve the problem, an idea is to prepare non-trace/trace irq handers and
switch their IDTs at the enabling/disabling time.
So, let's introduce entering_irq()/exiting_irq() for pre/post-
processing of each irq handler.
A way to use them is as follows.
Non-trace irq handler:
smp_irq_handler()
{
entering_irq(); /* pre-processing of this handler */
__smp_irq_handler(); /*
* common logic between non-trace and trace handlers
* in a vector.
*/
exiting_irq(); /* post-processing of this handler */
}
Trace irq_handler:
smp_trace_irq_handler()
{
entering_irq(); /* pre-processing of this handler */
trace_irq_entry(); /* tracepoint for irq entry */
__smp_irq_handler(); /*
* common logic between non-trace and trace handlers
* in a vector.
*/
trace_irq_exit(); /* tracepoint for irq exit */
exiting_irq(); /* post-processing of this handler */
}
If tracepoints can place outside entering_irq()/exiting_irq() as follows,
it looks cleaner.
smp_trace_irq_handler()
{
trace_irq_entry();
smp_irq_handler();
trace_irq_exit();
}
But it doesn't work.
The problem is with irq_enter/exit() being called. They must be called before
trace_irq_enter/exit(), because of the rcu_irq_enter() must be called before
any tracepoints are used, as tracepoints use rcu to synchronize.
As a possible alternative, we may be able to call irq_enter() first as follows
if irq_enter() can nest.
smp_trace_irq_hander()
{
irq_entry();
trace_irq_entry();
smp_irq_handler();
trace_irq_exit();
irq_exit();
}
But it doesn't work, either.
If irq_enter() is nested, it may have a time penalty because it has to check if it
was already called or not. The time penalty is not desired in performance sensitive
paths even if it is tiny.
Signed-off-by: Seiji Aguchi <seiji.aguchi@hds.com>
Link: http://lkml.kernel.org/r/51C3238D.9040706@hds.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
2013-06-20 11:45:17 -04:00
void smp_x86_platform_ipi ( struct pt_regs * regs )
{
struct pt_regs * old_regs = set_irq_regs ( regs ) ;
2009-03-04 12:56:05 -06:00
x86, trace: Introduce entering/exiting_irq()
When implementing tracepoints in interrupt handers, if the tracepoints are
simply added in the performance sensitive path of interrupt handers,
it may cause potential performance problem due to the time penalty.
To solve the problem, an idea is to prepare non-trace/trace irq handers and
switch their IDTs at the enabling/disabling time.
So, let's introduce entering_irq()/exiting_irq() for pre/post-
processing of each irq handler.
A way to use them is as follows.
Non-trace irq handler:
smp_irq_handler()
{
entering_irq(); /* pre-processing of this handler */
__smp_irq_handler(); /*
* common logic between non-trace and trace handlers
* in a vector.
*/
exiting_irq(); /* post-processing of this handler */
}
Trace irq_handler:
smp_trace_irq_handler()
{
entering_irq(); /* pre-processing of this handler */
trace_irq_entry(); /* tracepoint for irq entry */
__smp_irq_handler(); /*
* common logic between non-trace and trace handlers
* in a vector.
*/
trace_irq_exit(); /* tracepoint for irq exit */
exiting_irq(); /* post-processing of this handler */
}
If tracepoints can place outside entering_irq()/exiting_irq() as follows,
it looks cleaner.
smp_trace_irq_handler()
{
trace_irq_entry();
smp_irq_handler();
trace_irq_exit();
}
But it doesn't work.
The problem is with irq_enter/exit() being called. They must be called before
trace_irq_enter/exit(), because of the rcu_irq_enter() must be called before
any tracepoints are used, as tracepoints use rcu to synchronize.
As a possible alternative, we may be able to call irq_enter() first as follows
if irq_enter() can nest.
smp_trace_irq_hander()
{
irq_entry();
trace_irq_entry();
smp_irq_handler();
trace_irq_exit();
irq_exit();
}
But it doesn't work, either.
If irq_enter() is nested, it may have a time penalty because it has to check if it
was already called or not. The time penalty is not desired in performance sensitive
paths even if it is tiny.
Signed-off-by: Seiji Aguchi <seiji.aguchi@hds.com>
Link: http://lkml.kernel.org/r/51C3238D.9040706@hds.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
2013-06-20 11:45:17 -04:00
entering_ack_irq ( ) ;
__smp_x86_platform_ipi ( ) ;
exiting_irq ( ) ;
2009-03-04 12:56:05 -06:00
set_irq_regs ( old_regs ) ;
}
2013-04-11 19:25:11 +08:00
# ifdef CONFIG_HAVE_KVM
/*
* Handler for POSTED_INTERRUPT_VECTOR .
*/
void smp_kvm_posted_intr_ipi ( struct pt_regs * regs )
{
struct pt_regs * old_regs = set_irq_regs ( regs ) ;
ack_APIC_irq ( ) ;
irq_enter ( ) ;
exit_idle ( ) ;
inc_irq_stat ( kvm_posted_intr_ipis ) ;
irq_exit ( ) ;
set_irq_regs ( old_regs ) ;
}
# endif
2008-12-23 15:15:17 +01:00
EXPORT_SYMBOL_GPL ( vector_used_by_percpu_irq ) ;
2009-10-26 14:24:31 -08:00
# ifdef CONFIG_HOTPLUG_CPU
/* A cpu has been removed from cpu_online_mask. Reset irq affinities. */
void fixup_irqs ( void )
{
2009-10-26 14:24:36 -08:00
unsigned int irq , vector ;
2009-10-26 14:24:31 -08:00
static int warned ;
struct irq_desc * desc ;
2010-10-08 20:24:58 +02:00
struct irq_data * data ;
2011-02-10 21:40:36 +01:00
struct irq_chip * chip ;
2009-10-26 14:24:31 -08:00
for_each_irq_desc ( irq , desc ) {
int break_affinity = 0 ;
int set_affinity = 1 ;
const struct cpumask * affinity ;
if ( ! desc )
continue ;
if ( irq = = 2 )
continue ;
/* interrupt's are disabled at this point */
2009-11-17 16:46:45 +01:00
raw_spin_lock ( & desc - > lock ) ;
2009-10-26 14:24:31 -08:00
2011-02-10 21:40:36 +01:00
data = irq_desc_get_irq_data ( desc ) ;
2010-10-08 20:24:58 +02:00
affinity = data - > affinity ;
2011-05-06 14:43:36 +08:00
if ( ! irq_has_action ( irq ) | | irqd_is_per_cpu ( data ) | |
2011-02-17 15:54:26 +00:00
cpumask_subset ( affinity , cpu_online_mask ) ) {
2009-11-17 16:46:45 +01:00
raw_spin_unlock ( & desc - > lock ) ;
2009-10-26 14:24:31 -08:00
continue ;
}
2009-10-26 14:24:34 -08:00
/*
* Complete the irq move . This cpu is going down and for
* non intr - remapping case , we can ' t wait till this interrupt
* arrives at this cpu before completing the irq move .
*/
irq_force_complete_move ( irq ) ;
2009-10-26 14:24:31 -08:00
if ( cpumask_any_and ( affinity , cpu_online_mask ) > = nr_cpu_ids ) {
break_affinity = 1 ;
2012-08-14 06:55:01 +00:00
affinity = cpu_online_mask ;
2009-10-26 14:24:31 -08:00
}
2011-02-10 21:40:36 +01:00
chip = irq_data_get_irq_chip ( data ) ;
if ( ! irqd_can_move_in_process_context ( data ) & & chip - > irq_mask )
chip - > irq_mask ( data ) ;
2009-10-26 14:24:31 -08:00
2011-02-10 21:40:36 +01:00
if ( chip - > irq_set_affinity )
chip - > irq_set_affinity ( data , affinity , true ) ;
2009-10-26 14:24:31 -08:00
else if ( ! ( warned + + ) )
set_affinity = 0 ;
2012-03-26 07:11:50 +00:00
/*
* We unmask if the irq was not marked masked by the
* core code . That respects the lazy irq disable
* behaviour .
*/
2011-05-06 14:43:56 +08:00
if ( ! irqd_can_move_in_process_context ( data ) & &
2012-03-26 07:11:50 +00:00
! irqd_irq_masked ( data ) & & chip - > irq_unmask )
2011-02-10 21:40:36 +01:00
chip - > irq_unmask ( data ) ;
2009-10-26 14:24:31 -08:00
2009-11-17 16:46:45 +01:00
raw_spin_unlock ( & desc - > lock ) ;
2009-10-26 14:24:31 -08:00
if ( break_affinity & & set_affinity )
2012-05-21 19:50:07 -07:00
pr_notice ( " Broke affinity for irq %i \n " , irq ) ;
2009-10-26 14:24:31 -08:00
else if ( ! set_affinity )
2012-05-21 19:50:07 -07:00
pr_notice ( " Cannot set affinity for irq %i \n " , irq ) ;
2009-10-26 14:24:31 -08:00
}
2009-10-26 14:24:36 -08:00
/*
* We can remove mdelay ( ) and then send spuriuous interrupts to
* new cpu targets for all the irqs that were handled previously by
* this cpu . While it works , I have seen spurious interrupt messages
* ( nothing wrong but still . . . ) .
*
* So for now , retain mdelay ( 1 ) and check the IRR and then send those
* interrupts to new targets as this cpu is already offlined . . .
*/
2009-10-26 14:24:31 -08:00
mdelay ( 1 ) ;
2009-10-26 14:24:36 -08:00
for ( vector = FIRST_EXTERNAL_VECTOR ; vector < NR_VECTORS ; vector + + ) {
unsigned int irr ;
2010-12-18 16:28:55 +01:00
if ( __this_cpu_read ( vector_irq [ vector ] ) < 0 )
2009-10-26 14:24:36 -08:00
continue ;
irr = apic_read ( APIC_IRR + ( vector / 32 * 0x10 ) ) ;
if ( irr & ( 1 < < ( vector % 32 ) ) ) {
2010-12-18 16:28:55 +01:00
irq = __this_cpu_read ( vector_irq [ vector ] ) ;
2009-10-26 14:24:36 -08:00
2011-02-12 11:51:03 +01:00
desc = irq_to_desc ( irq ) ;
2011-02-10 21:40:36 +01:00
data = irq_desc_get_irq_data ( desc ) ;
chip = irq_data_get_irq_chip ( data ) ;
2009-11-17 16:46:45 +01:00
raw_spin_lock ( & desc - > lock ) ;
2011-02-10 21:40:36 +01:00
if ( chip - > irq_retrigger )
chip - > irq_retrigger ( data ) ;
2009-11-17 16:46:45 +01:00
raw_spin_unlock ( & desc - > lock ) ;
2009-10-26 14:24:36 -08:00
}
2012-07-26 19:47:32 +09:00
__this_cpu_write ( vector_irq [ vector ] , - 1 ) ;
2009-10-26 14:24:36 -08:00
}
2009-10-26 14:24:31 -08:00
}
# endif