0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

kernel的启动分为哪两个阶段

麦辣鸡腿堡 来源:TrustZone 作者:Hcoco 2023-12-04 16:45 次阅读

kernel的启动主要分为两个阶段。

1、阶段一

从入口跳转到start_kernel之前的阶段。

对应代码arch/arm/kernel/head.S中stext的实现:

ENTRY(stext)
  • 这个阶段主要由汇编语言实现。
  • 这个阶段主要负责MMU打开之前的一些操作,以及打开MMU的操作。
  • 由于这个阶段MMU还没有打开,并且kernel加载地址和连接地址并一致,所以需要使用位置无关设计。在运行过程中运行地址和加载地址一致(如果不明白的话建议先参考一下《[kernel 启动流程] 前篇——vmlinux.lds分析》)。

(上一篇从uboot到kernel的地方,讲了kernel启动后的几个阶段,停在start_kernel部分)

2、阶段二

start_kernel开始的阶段。

正题-kernel-uboot

Android生在linux内核基础上,linux内核启动的最后一步,一定是启动的android的进程。

然后我们也知道了内核启动分为三个阶段,

  • 第一二是运行head.S文件和head-common.S,
  • 第三个阶段是允许第二是运行main.c文件。

对于ARM的处理器,内核第一个启动的文件是arc/arm/kernel下面的head.S文件。、

当然arc/arm/boot/compress下面 也有这个文件,这个文件和上面的文件略有不同,当要生成压缩的内核时zImage时, 启动的是后者后者与前者不同的是:它前面的代码是做自解压的,后面的代码都相同。

我们这里这分析arc/arm/kernel下面的head.S文件。当head.S所作的工作完成后它会跳到init/目录下跌的 main.c的start_kernel函数开始执行。

因为我们要研究的是过渡阶段,而不是整个启动流程。(后面会研究的。)这里直接看第三个--start_kernel阶段。

asmlinkage void __init start_kernel(void)  
{  
       …………………….  
       ……………………..  
       printk(KERN_NOTICE);  
       printk(linux_banner);  
       setup_arch(&command_line);  
       setup_command_line(command_line);  


       parse_early_param();  
       parse_args("Booting kernel",static_command_line, __start___param,  
                __stop___param - __start___param,  
                &unknown_bootoption);  
……………………  
…………………………        
       init_IRQ();  
       pidhash_init();  
       init_timers();  
       hrtimers_init();  
       softirq_init();  
       timekeeping_init();  
       time_init();  
       profile_init();  
…………………………  
……………………………  
       console_init();  
………………………………  
………………………………  
       rest_init();  
}

从上面可以看出start_kernel首先是打印内核信息,然后对bootloader传进来的一些参数进行处理,再接着执行各种各样的初始化,在这其中会初始化控制台。最后会调用rest_init();

我们再来看 rest_init ()函数

static void noinline __init_refok rest_init(void)  
    __releases(kernel_lock)  
{  
    int pid;  

    kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND);  
    ............      
}

他启动了kernel_init这个函数,再来看kerne_init函数

static int __init kernel_init(void * unused)  
{  
    ..............................  

    if (!ramdisk_execute_command)  
        ramdisk_execute_command = "/init";  

    if (sys_access((const char __user *) ramdisk_execute_command, 0) != 0) {  
        ramdisk_execute_command = NULL;  
        prepare_namespace();  
    }  

    /*  
     * Ok, we have completed the initial bootup, and  
     * we're essentially up and running. Get rid of the  
     * initmem segments and start the user-mode stuff..  
     */  
    init_post();  
    return 0;  
}

kernel_init先调用了 prepare_namespace() ;然后调用了init_post函数

void __init prepare_namespace(void)  
{  
    ..........................  
    mount_root();  
    .....................  
}

可以看出prepare_namespace调用了mount_root挂接根文件系统。接着kernel_init再执行init_post

static int noinline init_post(void)  
{  
    .......................................  
    /*打开dev/console控制台,并设置为标准输入、输出*/  

    if (sys_open((const char __user *) "/dev/console", O_RDWR, 0) < 0)  
        printk(KERN_WARNING "Warning: unable to open an initial console.n");  

    (void) sys_dup(0);  
    (void) sys_dup(0);  

    if (ramdisk_execute_command) {  
        run_init_process(ramdisk_execute_command);  
        printk(KERN_WARNING "Failed to execute %sn",  
                ramdisk_execute_command);  
    }  

    /*  
     * We try each of these until one succeeds.  
     *  
     * The Bourne shell can be used instead of init if we are  
     * trying to recover a really broken machine.  
     */  

    //如果bootloader指定了init参数,则启动init参数指定的进程  
    if (execute_command) {  
        run_init_process(execute_command);  
        printk(KERN_WARNING "Failed to execute %s.  Attempting "  
                    "defaults...n", execute_command);  
    }  

    //如果没有指定init参数,则分别带sbin、etc、bin目录下启动init进程  
    run_init_process("/sbin/init");  
    run_init_process("/etc/init");  
    run_init_process("/bin/init");  
    run_init_process("/bin/sh");  

    panic("No init found.  Try passing init= option to kernel.");  
}

注意上面的run_init_process的会等待init进程返回才往后面执行,所有它一旦找到一个init可执行的文件它将一去不复返。

综上,内核启动的过程大致为以下几步:

  • 1.检查CPU和机器类型
  • 2.进行堆栈、MMU等其他程序运行关键的东西进行初始化
  • 3.打印内核信息
  • 4.执行各种模块的初始化
  • 5.挂接根文件系统
  • 6.启动第一个init进程
  • 7.android启动
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Android
    +关注

    关注

    12

    文章

    3939

    浏览量

    127589
  • Linux
    +关注

    关注

    87

    文章

    11322

    浏览量

    209866
  • MMU
    MMU
    +关注

    关注

    0

    文章

    91

    浏览量

    18320
  • Kernel
    +关注

    关注

    0

    文章

    48

    浏览量

    11206
收藏 人收藏

    评论

    相关推荐

    底部有4点的无源晶振,两个点是有用的

    底部有4点的无源晶振,请问两个点是有用的?
    发表于 09-01 10:19

    双绞线分为

    `  谁知道双绞线分为种?`
    发表于 12-31 15:53

    双绞线分为

    `  谁来阐述一下双绞线分为种?`
    发表于 03-16 17:03

    BootLoader启动过程分为哪几个阶段

    BootLoarder是什么?BootLoader的操作模式有哪几种呢?BootLoader启动过程分为哪几个阶段呢?
    发表于 11-30 07:02

    技术分享|TQ3568如何更换启动logo(包括uboot和kernel)

    TQ3568平台的开机logo分为两个阶段,分别为u-boot 阶段的logo 和内核阶段的logo如果
    发表于 04-17 09:39

    伦敦区块链银行金融初创公司BABB启动ICO 欲分两个阶段稳扎稳打

    服务。 据悉,BABB将会在1月15日开始预售,目前注册工作已经开始,正式公开发售将会在下个月启动。和绝大多数ICO不同,BABB计划将本次ICO项目分为两个阶段,第二
    发表于 02-18 16:45 1193次阅读

    两个接触器如何实现顺序启动

    今天小编给大家分享一电路,两个电机顺序启动电路,这张图是实物接线,我们可以看到两个接触器,一时间继电器,
    的头像 发表于 05-14 17:10 7783次阅读

    两个阶段来看新能源电动汽车的发展前景

    新能源电动车的发展前景可以分为两个阶段来看。第1就是20年之内。第2阶段是20年之后。在第
    发表于 10-23 08:58 2114次阅读

    两个变频器不能启动的故障处理案例分享

    本文通过两个变频器不能启动的故障处理案例,和大家分享控制电路设计存在缺陷和变频器参数设置不符合应用要求导致的变频器不能启动的原因和处理办法以及从这些故障处理中总结的经验。
    的头像 发表于 11-01 11:04 5313次阅读
    <b class='flag-5'>两个</b>变频器不能<b class='flag-5'>启动</b>的故障处理案例分享

    光电编码器分为

    光电编码器分为种?
    的头像 发表于 07-29 15:52 9775次阅读

    两个LED和两个按钮的使用

    电子发烧友网站提供《两个LED和两个按钮的使用.zip》资料免费下载
    发表于 01-30 16:04 1次下载
    <b class='flag-5'>两个</b>LED和<b class='flag-5'>两个</b>按钮的使用

    kernel到android核心启动过程

    总结一图:kernel 到android核心启动过程 kernel镜像执行跳转到start_kernel开始执行,在rest_init会创
    的头像 发表于 12-04 16:59 1048次阅读
    <b class='flag-5'>kernel</b>到android核心<b class='flag-5'>启动</b>过程

    uboot的启动BL1和BL2两个阶段介绍

    之前对这个uboot的源码了解有些许遗忘。最近做AVB校验,需要uboot到kernel的这个过程。这里再复习一下。 与大多数BootLoader一样,uboot的启动过程分为BL1和BL2
    的头像 发表于 12-04 17:53 3125次阅读

    电磁继电器分为两个电路

    的控制。根据其结构和工作原理,电磁继电器可以分为两个电路:控制电路和工作电路。 一、控制电路 控制电路是电磁继电器的重要组成部分,它的作用是提供电磁铁所需的电流,使其产生磁场。控制电路主要由电源、控制开关和
    的头像 发表于 06-21 09:28 728次阅读

    二极管的伏安特性分为两个部分?

    二极管是一种半导体器件,具有单向导电性。其伏安特性是描述二极管在不同电压下电流变化的曲线。二极管的伏安特性可以分为两个部分:正向特性和反向特性。 正向特性 正向特性是指二极管在正向偏置下,即阳极接
    的头像 发表于 08-16 11:16 930次阅读