init是Linux内核启动的第一个用户级进程,系统的很多初始化工作都是由它开始的,当然也包括中断的初始化工作。其主要函数在init/main.c中,汇编程序会跳到该文件的start_kernel函数执行,我们就由这个函数开始分析Linux中断初始化相关的主要流程。
asmlinkage void __init start_kernel(void)
{ … …
setup_arch(&command_line);
… …
init_IRQ();
… …
}
setup_arch(&command_line)很明显是用来设置体系结构的,我们这里以arm为例来分析整个流程。先来看看它做了哪些与中断有关的工作arch\arm\kernel\setup.c:
void __init setup_arch(char **cmdline_p)
{ … …
struct machine_desc *mdesc;
mdesc = setup_machine(machine_arch_type);
init_arch_irq = mdesc->init_irq;
… …
}
其中init_arch_irq是在arch\arm\kernel\irq.c中定义的:
void (*init_arch_irq)(void) __initdata = NULL;
很明显init_irq应该是一个函数指针并且是和具体的版型相关的,我们可以找一个版型的相关代码来查一下。比如mini2440的板子mach-mini2440.c:
MACHINE_START(MINI2440, "MINI2440")
/* Maintainer: Michel Pollet */
.phys_io = S3C2410_PA_UART,
.io_pg_offst = (((u32)S3C24XX_VA_UART) >> 18) & 0xfffc,
.boot_params = S3C2410_SDRAM_PA + 0x100,
.map_io = mini2440_map_io,
.init_machine = mini2440_init,
.init_irq = s3c24xx_init_irq,
.timer = &s3c24xx_timer,
MACHINE_END
由上面的代码可知init_irq是指向s3c24xx_init_irq的。s3c24xx_init_irq的源代码如下:
void __init s3c24xx_init_irq(void)
{ … …
set_irq_chip(irqno, &s3c_irq_chip);
set_irq_handler(irqno, handle_edge_irq);
… …
}
这个函数主要是初始化了irp_chip和irq_handler,在后面的分析中会用到,沿着setup_arch这条线的分析就到此为止,有兴趣的话可以继续跟下去。
下面我们来分析init_IRQ(),它的源代码在arch\arm\kernel\irq.c中:
void __init init_IRQ(void)
{
int irq;
for (irq = 0; irq < NR_IRQS; irq++)
irq_desc[irq].status |= IRQ_NOREQUEST | IRQ_NOPROBE;
init_arch_irq();
}
它的主要工作是初始化全局变量irq_desc,并调用init_arch_irq()。init_arch_irq最终会调用到s3c24xx_init_irq。而irq_desc的定义在kernel\irq\handle.c中:
struct irq_desc irq_desc[NR_IRQS] __cacheline_aligned_in_smp = {
[0 ... NR_IRQS-1] = {
.status = IRQ_DISABLED,
.chip = &no_irq_chip,
.handle_irq = handle_bad_irq,
.depth = 1,
.lock = __SPIN_LOCK_UNLOCKED(irq_desc->lock),
}
};
在s3c24xx_init_irq中也会对irq_desc的一些成员进行赋值,如将.handle_irq有默认的handle_bad_irq赋值为handle_edge_irq。