- 论坛徽章:
- 0
|
enable_irq时,如果发现还有IRQ_PENDING,就说明存在丢失的中断,需要挽救。
但关于IRQ_REPLAY,书中说是为了防止多次产生一个丢失的中断。
我的问题是:
即使不使用IRQ_REPLAY,在什么情况下会多次产生一个丢失的中断呢?
代码中,spin_lock已经保护了整段代码,并且hw_resend_irq是LAPIC发出的,会立即在本处理器上产生,也就是说,
执行到最后的spin_unlock(这里书中写错了,不是spin_lock)时,根本没有机会再次进入这段代码,那么怎么会产生“多次产生一个丢失的中断”的情况呢?
ULK 原文如下:
spin_lock_irqsave(&(irq_desc[irq].lock), flags);
if (--irq_desc[irq].depth == 0) {
irq_desc[irq].status &= ~IRQ_DISABLED;
if (irq_desc[irq].status & (IRQ_PENDING | IRQ_REPLAY))
== IRQ_PENDING) {
irq_desc[irq].status |= IRQ_REPLAY;
hw_resend_irq(irq_desc[irq].handler,irq);
}
irq_desc[irq].handler->enable(irq);
}
spin_lock_irqrestore(&(irq_desc[irq].lock), flags);
The function detects that an interrupt was lost by checking the value of the IRQ_PENDING flag. The flag is always cleared when leaving the interrupt handler; therefore, if the IRQ line is disabled and the flag is set, then an interrupt occurrence has been acknowledged but not yet serviced. In this case the hw_resend_irq( ) function raises a new interrupt. This is obtained by forcing the local APIC to generate a self-interrupt (see the later section "Interprocessor Interrupt Handling").
***The role of the IRQ_REPLAY flag is to ensure that exactly one self-interrupt is generated. ***
Remember that the _ _do_IRQ( ) function clears that flag when it starts handling the interrupt. |
|