diff options
Diffstat (limited to 'kernel/irq/chip.c')
| -rw-r--r-- | kernel/irq/chip.c | 47 | 
1 files changed, 36 insertions, 11 deletions
| diff --git a/kernel/irq/chip.c b/kernel/irq/chip.c index f7c543a801d9..6080f6bc8c33 100644 --- a/kernel/irq/chip.c +++ b/kernel/irq/chip.c @@ -16,6 +16,8 @@  #include <linux/interrupt.h>  #include <linux/kernel_stat.h> +#include <trace/events/irq.h> +  #include "internals.h"  /** @@ -61,8 +63,7 @@ int irq_set_irq_type(unsigned int irq, unsigned int type)  		return -EINVAL;  	type &= IRQ_TYPE_SENSE_MASK; -	if (type != IRQ_TYPE_NONE) -		ret = __irq_set_trigger(desc, irq, type); +	ret = __irq_set_trigger(desc, irq, type);  	irq_put_desc_busunlock(desc, flags);  	return ret;  } @@ -157,19 +158,22 @@ static void irq_state_set_masked(struct irq_desc *desc)  	irqd_set(&desc->irq_data, IRQD_IRQ_MASKED);  } -int irq_startup(struct irq_desc *desc) +int irq_startup(struct irq_desc *desc, bool resend)  { +	int ret = 0; +  	irq_state_clr_disabled(desc);  	desc->depth = 0;  	if (desc->irq_data.chip->irq_startup) { -		int ret = desc->irq_data.chip->irq_startup(&desc->irq_data); +		ret = desc->irq_data.chip->irq_startup(&desc->irq_data);  		irq_state_clr_masked(desc); -		return ret; +	} else { +		irq_enable(desc);  	} - -	irq_enable(desc); -	return 0; +	if (resend) +		check_irq_resend(desc, desc->irq_data.irq); +	return ret;  }  void irq_shutdown(struct irq_desc *desc) @@ -330,6 +334,24 @@ out_unlock:  }  EXPORT_SYMBOL_GPL(handle_simple_irq); +/* + * Called unconditionally from handle_level_irq() and only for oneshot + * interrupts from handle_fasteoi_irq() + */ +static void cond_unmask_irq(struct irq_desc *desc) +{ +	/* +	 * We need to unmask in the following cases: +	 * - Standard level irq (IRQF_ONESHOT is not set) +	 * - Oneshot irq which did not wake the thread (caused by a +	 *   spurious interrupt or a primary handler handling it +	 *   completely). +	 */ +	if (!irqd_irq_disabled(&desc->irq_data) && +	    irqd_irq_masked(&desc->irq_data) && !desc->threads_oneshot) +		unmask_irq(desc); +} +  /**   *	handle_level_irq - Level type irq handler   *	@irq:	the interrupt number @@ -362,8 +384,8 @@ handle_level_irq(unsigned int irq, struct irq_desc *desc)  	handle_irq_event(desc); -	if (!irqd_irq_disabled(&desc->irq_data) && !(desc->istate & IRQS_ONESHOT)) -		unmask_irq(desc); +	cond_unmask_irq(desc); +  out_unlock:  	raw_spin_unlock(&desc->lock);  } @@ -417,6 +439,9 @@ handle_fasteoi_irq(unsigned int irq, struct irq_desc *desc)  	preflow_handler(desc);  	handle_irq_event(desc); +	if (desc->istate & IRQS_ONESHOT) +		cond_unmask_irq(desc); +  out_eoi:  	desc->irq_data.chip->irq_eoi(&desc->irq_data);  out_unlock: @@ -625,7 +650,7 @@ __irq_set_handler(unsigned int irq, irq_flow_handler_t handle, int is_chained,  		irq_settings_set_noprobe(desc);  		irq_settings_set_norequest(desc);  		irq_settings_set_nothread(desc); -		irq_startup(desc); +		irq_startup(desc, true);  	}  out:  	irq_put_desc_busunlock(desc, flags); | 
