|
@@ -330,6 +330,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 +380,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 +435,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:
|