1、当发生中断时,PC指针通过异常向量表跳转到中断入口函数asm_do_IRQ
asm_do_IRQ是中断的C语言总入口函数,它在/arch/arm/kernel/irq.c中定义,声明如下:
asmlinkage void __exception asm_do_IRQ(unsigned int irq, struct pt_regs *regs)
此函数中调用了generic_handle_irq(irq);而后调用到了desc->handle_irq(irq, desc);函数。
desc->hand_irq函数直接调用desc结构中的hand_irq成员函数,它就是irq_desc[irq].handle.irq
2、irq_desc[irq].handle.irq在中断初始化函数(arch/arm/kernel/irq.c 的init_IRQ( ))中已经通过s3c24xx_init_irq进行了初始化。
以EINT4-EINT23为例:
void __init s3c24xx_init_irq(void)
{
...
for (irqno = IRQ_EINT4; irqno <= IRQ_EINT23; irqno++) {
irqdbf("registering irq %d (extended s3c irq)\n", irqno);
set_irq_chip(irqno, &s3c_irqext_chip);
set_irq_handler(irqno, handle_edge_irq);
set_irq_flags(irqno, IRQF_VALID);
}
...
}
irq_desc[irq].handle.irq函数指向了handle_edge_irq函数
3、handle_edge_irq函数声明如下:
void handle_edge_irq(unsigned int irq, struct irq_desc *desc)
此函数进行了如下的操作:
(1)清除中断
desc->chip->ack(irq);
此函数在init_IRQ( )中被指向static void s3c_irqext_ack(unsigned int irqno)函数
(2)处理中断
action_ret = handle_IRQ_event(irq, action);
取出action链表中的成员执行action->handle
ret = action->handler(irq, action->dev_id);