// SPDX-License-Identifier: GPL-2.0 /* * Copyright (C) 1992, 1998-2006 Linus Torvalds, Ingo Molnar * Copyright (C) 2005-2006, Thomas Gleixner * * This file contains the IRQ-resend code * * If the interrupt is waiting to be processed, we try to re-run it. * We can't directly run it from here since the caller might be in an * interrupt-protected region. Not all irq controller chips can * retrigger interrupts at the hardware level, so in those cases * we allow the resending of IRQs via a tasklet.
*/
/* Tasklet to handle resend: */ static DECLARE_TASKLET(resend_tasklet, resend_irqs);
staticint irq_sw_resend(struct irq_desc *desc)
{ /* * Validate whether this interrupt can be safely injected from * non interrupt context
*/ if (irqd_is_handle_enforce_irqctx(&desc->irq_data)) return -EINVAL;
/* * If the interrupt is running in the thread context of the parent * irq we need to be careful, because we cannot trigger it * directly.
*/ if (irq_settings_is_nested_thread(desc)) { /* * If the parent_irq is valid, we retrigger the parent, * otherwise we do nothing.
*/ if (!desc->parent_irq) return -EINVAL;
desc = irq_to_desc(desc->parent_irq); if (!desc) return -EINVAL;
}
/* Add to resend_list and activate the softirq: */
scoped_guard(raw_spinlock, &irq_resend_lock) { if (hlist_unhashed(&desc->resend_node))
hlist_add_head(&desc->resend_node, &irq_resend_list);
}
tasklet_schedule(&resend_tasklet); return 0;
}
/* * IRQ resend * * Is called with interrupts disabled and desc->lock held.
*/ int check_irq_resend(struct irq_desc *desc, bool inject)
{ int err = 0;
/* * We do not resend level type interrupts. Level type interrupts * are resent by hardware when they are still active. Clear the * pending bit so suspend/resume does not get confused.
*/ if (irq_settings_is_level(desc)) {
desc->istate &= ~IRQS_PENDING; return -EINVAL;
}
if (desc->istate & IRQS_REPLAY) return -EBUSY;
if (!(desc->istate & IRQS_PENDING) && !inject) return 0;
desc->istate &= ~IRQS_PENDING;
if (!try_retrigger(desc))
err = irq_sw_resend(desc);
/* If the retrigger was successful, mark it with the REPLAY bit */ if (!err)
desc->istate |= IRQS_REPLAY; return err;
}
#ifdef CONFIG_GENERIC_IRQ_INJECTION /** * irq_inject_interrupt - Inject an interrupt for testing/error injection * @irq: The interrupt number * * This function must only be used for debug and testing purposes! * * Especially on x86 this can cause a premature completion of an interrupt * affinity change causing the interrupt line to become stale. Very * unlikely, but possible. * * The injection can fail for various reasons: * - Interrupt is not activated * - Interrupt is NMI type or currently replaying * - Interrupt is level type * - Interrupt does not support hardware retrigger and software resend is * either not enabled or not possible for the interrupt.
*/ int irq_inject_interrupt(unsignedint irq)
{ int err = -EINVAL;
/* Try the state injection hardware interface first */ if (!irq_set_irqchip_state(irq, IRQCHIP_STATE_PENDING, true)) return 0;
/* That failed, try via the resend mechanism */
scoped_irqdesc_get_and_buslock(irq, 0) { struct irq_desc *desc = scoped_irqdesc;
/* * Only try to inject when the interrupt is: * - not NMI type * - activated
*/ if (!irq_is_nmi(desc) && irqd_is_activated(&desc->irq_data))
err = check_irq_resend(desc, true);
} return err;
}
EXPORT_SYMBOL_GPL(irq_inject_interrupt); #endif
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.