init_machine 在Kernel中被调用的过程

发布时间:
来源: 电子工程世界

以tiny4412为例:


arch/arm/mach-exynos/mach-tiny4412.c


MACHINE_START(TINY4412, "TINY4412")

    /* Maintainer: FriendlyARM (www.arm9.net) */

    .boot_params    = S5P_PA_SDRAM + 0x100,

    .init_irq    = exynos4_init_irq,

    .map_io        = smdk4x12_map_io,

    .init_machine    = smdk4x12_machine_init,

    .timer        = &exynos4_timer,

    .reserve    = &exynos4_reserve,

MACHINE_END


其中:


#define MACHINE_START(_type,_name)           

static const struct machine_desc __mach_desc_##_type   

 __used                           

 __attribute__((__section__(".arch.info.init"))) = {   

    .nr        = MACH_TYPE_##_type,       

    .name        = _name,


#define MACHINE_END               

};


启动时:


start_kernel  ----- init/main.c


      ---->  setup_arch ---- arch/arm/kernel/setup.c


                      ---->  mdesc = setup_machine_tags(machine_arch_type);   到这里,根据machine_arch_type就找到上面这个结构体了。


                      ---->  machine_desc = mdesc;


                      ---->  paging_init(mdesc)   (arch/arm/mm/mmu.c)


                                      ----> devicemaps_init(mdesc)


                                                        ----> mdesc->map_io()   调用了函数 smdk4x12_map_io


     ---->  init_IRQ()   (arch/arm/kernel/irq.c)


                 ----> machine_desc->init_irq()    调用 exynos4_init_irq



     ----> time_init()


                  ---->      system_timer = machine_desc->timer;   其中, system_timer 就是 exynos4_timer

                  ---->      system_timer->init();          其中, init 是 exynos4_timer_init


     ----> rest_init()


                  ---->  kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND)                                   


                                      ---->  kernel_init


                                                    ---->  do_basic_setup()


                                                                    ---->  driver_init()


                                                                                     ---->  platform_bus_init();


                                                                    ---->  do_initcalls()


                                                                                     ----> 


static void __init do_initcalls(void)

{

    initcall_t *fn;


    for (fn = __early_initcall_end; fn < __initcall_end; fn++)

        do_one_initcall(*fn);

}


在arch/arm/kernel/vmlinux.lds中:


  __initcall_start = .; *(.initcallearly.init) __early_initcall_end = .; *(.initcall0.init) *(.initcall0s.init) *(.initcall1.init) *(.initcall1s.init) *(.initcall2.init) *(.initcall2s.init) *(.initcall3.init) *(.initcall3s.init) *(.initcallbresume.init) *(.initcallresume.init) *(.initcall4.init) *(.initcall4s.init) *(.initcall5.init) *(.initcall5s.init) *(.initcallrootfs.init) *(.initcall6.init) *(.initcall6s.init) *(.initcall7.init) *(.initcall7s.init) __initcall_end = .;

即: do_initcalls 会一次执行上面的链接脚本指定的段中的函数,其中在arch/arm/kernel/setup.c中:


static int __init customize_machine(void)

{

    /* customizes platform devices, or adds new ones */

    if (machine_desc->init_machine)

        machine_desc->init_machine();   // 执行了smdk4x12_machine_init

    return 0;

}

arch_initcall(customize_machine); 


其中在include/linux/init.h中:


#define __define_initcall(level,fn,id)

    static initcall_t __initcall_##fn##id __used

    __attribute__((__section__(".initcall" level ".init"))) = fn



#define arch_initcall(fn)     __define_initcall("3",fn,3)


所以, customize_machine 被链接到了 ".initcall3.init" 段, 会被 do_initcalls执行。


文章来源于: 电子工程世界 原文链接

本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。