之前虽然已经为sep4020移植内核了linux2.6的内核,但当初做的时候还是很多地方不是很明白为什么要这样做,仅仅是把项目完成了,没有真正理解透彻,最近有点时间再把以前的工作好好理清楚。
1.之前谈到我们要为我们的处理器设置一个它独有的机器号,在arch/arm/tools/mach-types中,我写了这么一句话:
# machine_is_xxx CONFIG_xxxx MACH_TYPE_xxx number
sep4020 ARCH_4020 GFD4020 194
我们说bootloader一定要把机器号传给内核,这样内核才能找到相应的处理器,这里uboot传的机器号就是(number)194,但是内核又是如何根据这个194找到我们的处理器,找到我们的板级设备。
2.上面那段定义看似一定意义都没有,但是实际是有的,通过我们的gen-mach-types会在include/asm-arm/mach-types.h中自动生成这么一段话:
#define MACH_TYPE_GFD4020 194
#ifdef CONFIG_ARCH_4020
# ifdef machine_arch_type
# undef machine_arch_type
# define machine_arch_type __machine_arch_type
# else
# define machine_arch_type MACH_TYPE_GFD4020
# endif
# define machine_is_sep4020() (machine_arch_type == MACH_TYPE_GFD4020)
#else
# define machine_is_sep4020() (0)
#endif
3.在我们的arch/arm/4020.c中有一个函数,这个函数包括了我们板级初始化的所有初始化步骤。
MACHINE_START(GFD4020, "4020 board")
.phys_io = 0x10000000,
.io_pg_offst = ((0xe0000000) >> 18) & 0xfffc,
.boot_params = 0x30000100,
.fixup = fixup_gfd4020,
.map_io = sep4020_map_io,
.init_irq = sep4020_init_irq,
.init_machine = sep4020_init,
.timer = &sep4020_timer,
MACHINE_END
跟进去发现其实这一步其实是定义了一个结构体,
struct machine_desc __mach_desc_##_type
{
.nr = MACH_TYPE_GFD4020, //可见就是194
.name = "4020 board",
//重要的板级初始化函数
.map_io = sep4020_map_io,
.init_irq = sep4020_init_irq,
.init_machine = sep4020_init,
.timer = &sep4020_timer,
}
4.linux起来之后会进入start_kernel大函数,在这个启动的大函数中的最前面几步会到一个setup_arch的子函数,我们的板级初始化过程就是通过这个函数实现的。
setup_arch->setup_machine->lookup_machine_type(nr);
mdesc = setup_machine(machine_arch_type);这个参数就是通过uboot传过来的机器号194;然后通过lookup_machine_type函数得到之前提到的machine_desc结构体;
得到了这个板级结构体之后,就把这个板级结构体中的我们cpu的信息赋给系统通用的全局变量,如下:
machine_name = mdesc->name;
init_arch_irq = mdesc->init_irq;
system_timer = mdesc->timer;
init_machine= mdesc->init_machine;
到此板级的钩子算是勾上了,但是还没有真正的实现注册,具体的timer,irq注册就是通过红字标出的这几个全局变量结构体在后面的init_IRQ,init_timers,中实现的。
阅读(431) | 评论(0) | 转发(0) |