Linux设备驱动程序学习笔记14:中断的初始化流程

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。









你可能感兴趣的:(Linux)