当前位置:首页 > 嵌入式 > 嵌入式微处理器
[导读][导读] 前面的文章有提到linux启动的第一个进程为init,那么该进程究竟是如何从内核启动入口一步一步运行起来的,而该进程又有些什么作用呢?做嵌入式Linux开发,有必要对这些概念了解清楚。本文基于ARM体系的内核启动做出解析。 跳转内核前基本准备 参考./Do

[导读] 前面的文章有提到linux启动的第一个进程为init,那么该进程究竟是如何从内核启动入口一步一步运行起来的,而该进程又有些什么作用呢?做嵌入式Linux开发,有必要对这些概念了解清楚。本文基于ARM体系的内核启动做出解析。

跳转内核前基本准备

参考./Documentation/arm64/booting.txt

Bootloader至少完成以下基本的初始化准备:

  • 设置并初始化RAM(必须),引导加载程序应找到并初始化内核将用于系统中易失性数据存储的所有RAM。它以机器相关的方式执行此操作。(它可以使用内部算法来自动定位和调整所有RAM的大小,或者可以使用机器中RAM的知识或引导加载程序设计者认为合适的任何其他方法。)

  • 设置设备树dtb(必须) , 设备树blob(dtb)必须8字节对齐,并且大小不能超过2兆字节。由于dtb将使用最大2 MB的块进行映射以可缓存,因此它不能放置在必须使用任何特定属性进行映射的任何2M区域内。注意:v4.2之前的版本还要求将DTB放置在512 MB区域内,从内核映像下方的text_offset字节开始计算。

  • 解压缩内核映像(可选),AArch64内核当前不提供解压缩器,因此如果使用压缩的Image目标(例如Image.gz),则需要由引导加载程序执行解压缩(gzip等)。对于未实现此要求的引导加载程序,可以使用未经压缩内核编译。

  • 调用内核映像(必须)。压缩内核头部如下:

    u32 code0;              /* 可执行code   */
    u32 code1;             /* 可执行code   */
    u64 text_offset;       /* 加载偏移,小端 */
    u64 image_size;        /* 有效映象尺寸,小端 */
    u64 flags;             /* 内核标志, 小端    */
    u64 res2  = 0;         /* 保留 */
    u64 res3  = 0;         /* 保留 */
    u64 res4  = 0;         /* 保留 */
    u32 magic = 0x644d5241/* 幻数,小端, "ARM\x64"  */
    u32 res5;          /* 保留(用于PE COFF偏移量) */

进入内核之前,必须满足以下条件:

  • 禁止所有具有DMA功能的设备,以免内存被虚假错误的网络数据包或磁盘数据损坏。

  • 主CPU通用寄存器设置:

    • x0 =系统RAM中设备树Blob(dtb)的物理地址。

    • x1/x2/x3 = 0(保留供将来使用)

  • CPU模式

    • 所有形式的中断都必须在PSTATE.DAIF中屏蔽(调试,SError,IRQ和FIQ)。

    • CPU必须位于EL2(推荐使用,以便可以访问虚拟化扩展)或非安全EL1中。

  • Caches, MMUs

    • MMU必须关闭。

    • 指令缓存可以打开或关闭。

    • 与加载的内核映像相对应的地址范围必须清除到PoC。如果存在系统缓存或启用了缓存的其他相关主服务器,则通常需要通过VA而不是通过设置/方式操作来维护缓存。

    • 遵循VA对架构化缓存维护的系统缓存。必须配置并启用操作。

    • 不遵循VA对架构化混存维护的系统缓存,必须配置和禁用操作(不推荐)。

  • 架构定时器

    • 必须在所有CPU上以定时器频率设置CNTFRQ,并且必须以一致的值设置CNTVOFF。如果在EL1处进入内核,则CNTHCTL_EL2必须在可用时设置EL1PCTEN(位0)。

  • 连贯性

    • 内核启动时,所有要由内核引导的CPU都必须属于同一一致性域。需要初始化定义的实现,才能在每个CPU上接收维护操作。

  • 系统寄存器

    • 所有将在其中输入内核映像的异常级别的可写体系结构系统寄存器都必须由更高级别的异常级别的软件初始化,以防止在UNKNOWN状态下执行。

    • 对CPU模式,高速缓存,MMU,架构计时器,一致性和系统寄存器的要求适用于所有CPU。所有CPU必须以相同的异常级别进入内核。

  • 主CPU必须直接跳转到内核映像的第一条指令。此CPU传递的设备树Blob必须为每个cpu节点包含一个“启用方法”属性。支持的启用方法如下所述。引导加载程序将生成这些设备树属性,并将其插入内核入口之前的blob中。

  • 具有“旋转表”启用方法的CPU在其cpu节点中必须具有“ cpu-release-addr”属性。此属性标识自然对齐的64位零初始化内存位置。

  • 具有“ psci”启用方法的CPU应该保留在内核之外(即,在内存节点中描述给内核的内存区域之外,或者在内核中通过/ memreserve /描述给内核描述的内存保留区域之外)。设备树)。内核将按照ARM文档编号ARM DEN 0022A(“ ARM处理器上的电源状态协调接口系统软件”)中的说明发出CPU_ON调用,以将CPU带入内核。设备树应包含一个“ psci”节点,参考/bindings/arm/psci.txt.

  • 第二CPU通用寄存器设置的x0/x1/x2/x3都为0,保留。

内核启动init总过程

内核启动有两种方式,压缩格式或不压缩格式,压缩模式所不同的就是其入口位于arch/ /boot/compressed/head.S,为与该路径下的代码主要负责执行执行前期的初始化为解压内核做准备。当完成解压内核后,就跳转到./arm/kernel/head.S开始启动内核。

本文仅分析不压缩方式启动内核,通过分析内核代码,整理出内核启动过程的部分顺序如下:


内核的启动与U-Boot一样,前面一段是汇编代码,然后跳转到C代码。汇编的入口在

./arm/kernel/head.S中,符号名为__HEAD,该文件包含了head-common.S。

所以从启动用户首进程init而言,我将其分成大致分为四大步:

  • head.S ,初始化通用部分环境,与芯片无关

  • start_kernel, head.S完成后,调准到start_kernel,进入C函数执行,该函数为于./init/main.c中

  • rest_init,创建init进程,以及kthredd进程,其中Init进程号为1,kthredd为内核进程。

  • 启动调度器,执行kernel_init,该函数将调用根文件系统中的init执行文件,至此用户空间的init进程就启动起来了。

head.S/head-common.S作用

剖析汇编代码比较枯燥,这里就不进行描述了。仅就其作用进行总结:

  • 检查架构,处理器和机器类型。

  • 配置MMU,创建页表条目并启用虚拟内存。

  • 在init / main.c中调用start_kernel函数。

  • 所有架构的代码相同。这也是为什么采用汇编代码的原因,规避针对不同芯片管理大量重复代码。

start_kernel阶段

该函数主要完成以下以下工作:

  • lockdep 死锁检测模块初始化,

  • RCU机制初始化:RCU(Read-Copy Update),顾名思义就是读-拷贝修改,它是基于其原理命名的。对于被RCU保护的共享数据结构,读者不需要获得任何锁就可以访问它,但写者在访问它时首先拷贝一个副本,然后对副本进行修改,最后使用一个回调(callback)机制在适当的时机把指向原来数据的指针重新指向新的被修改的数据。这个时机就是所有引用该数据的CPU都退出对共享数据的操作。

  • SMP初始化,对称多处理"(Symmetrical Multi-Processing)简称SMP,完成CPU ID的创建。

  • debug_objects_early_init,负责调试对象初始化,以便于内核调试

  • lockdep死锁检测模块初始化,lockdep的工作方式是在内核中的锁定调用包起来。每次采用或释放特定类型的锁时,都会记录该事实以及辅助详细信息,例如处理器当时是否正在处理中断。Lockdep还记录了使用新锁时还持有哪些其他锁;这是lockdep能够执行的许多检查的关键。

  • 调用setup_arch(&command_line),该函数位于arch/

    /kernel/setup.c,用于解析从bootloader传入的引导命令行。
  • 初始化控制台,以打印启动日志。

  • 初始化其他各子系统,如VFS,trace,内存管理子系统,FORK子系统,cgroup,acpi,proc文件系统,内核服务,缓存等等。

  • ……

  • 调用rest_init,以创建init进程以及内核进程,并启动内核调度器。

rest_init阶段

代码如下,其注释如下,主要作用就是先创建init进程使其进程号为1,这是第一个用户空间进程,该进程执行后在衍生出一系列的应用进程。具体取决于启动脚本或者Init的具体实现。然后创建内核进程kthreadd,该进程用于管理内核进程。该进程进程号为2。所有内核进程都是kthreadd的后代, kthreadd枚举其他内核线程;它提供了接口例程,内核服务可以在运行时动态生成其他内核进程。通过kthread_create_list维护其他内核进程。可以使用ps -ef命令从命令行查看内核线程-它们显示在[方括号]中:

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

    rcu_scheduler_starting();
    smpboot_thread_init();

    /*创建init进程,第一个用户空间进程我们
    *需要首先生成init,以便它获得pid 1,但是
    *init任务最终将要创建kthread,如果在创建
    *kthreadd之前对其进行调度,则OOPS。*/

    kernel_thread(kernel_init, NULL, CLONE_FS);
    numa_default_policy();

    /*创建kthreadd用于管理内核线程*/
    pid = kernel_thread(kthreadd, NULL, CLONE_FS | CLONE_FILES);

    /*RCU 锁*/
    rcu_read_lock();
    kthreadd_task = find_task_by_pid_ns(pid, &init_pid_ns);
    rcu_read_unlock();
    /*让内核进程kthreadd处于就绪态TASK_NORMAL*/
    complete(&kthreadd_done);

    /* 启动调度器      */
    init_idle_bootup_task(current);
    schedule_preempt_disabled();
    /* 禁用抢占的情况下调用cpu_idle */
    cpu_startup_entry(CPUHP_ONLINE);
}

kernel_init阶段

当内核调度器运行后,就会执行kernel_init函数:

static int __ref kernel_init(void *unused)
{
    int ret;

    kernel_init_freeable();
    /* 同步完成所有初始化操作 */
    async_synchronize_full();
#ifndef CONFIG_INITCALLS_THREAD
    free_initmem();
#endif
    mark_readonly();
    system_state = SYSTEM_RUNNING;
    numa_default_policy();

    flush_delayed_fput();

    /*如果使能了ramdisk执行命令启动init*/
    if (ramdisk_execute_command) {
        ret = run_init_process(ramdisk_execute_command);
        if (!ret)
            return 0;
        pr_err("Failed to execute %s (error %d)\n",
               ramdisk_execute_command, ret);
    }

    /* 如果execute_command使能,则按命令启动init*/
    if (execute_command) {
        ret = run_init_process(execute_command);
        if (!ret)
            return 0;
        panic("Requested init %s failed (error %d).",
              execute_command, ret);
    }

    /*如果前面两项都没有使能,则依次在根文件系统下寻找并启动Init*/
    if (!try_to_run_init_process("/sbin/init") ||
        !try_to_run_init_process("/etc/init") ||
        !try_to_run_init_process("/bin/init") ||
        !try_to_run_init_process("/bin/sh"))
        return 0;

    panic("No working init found.  Try passing init= option to kernel. "
          "See Linux Documentation/init.txt for guidance.");
}

从而init用户进程就启动起来了,至于最终执行的是哪一个Init可执行文件,取决于系统移植的配置,如前文描述,常见的有busybox init,systemV init,systemD init等等。

本文授权转载自公众号“嵌入式客栈”,作者:逸珺

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

嵌入式ARM

扫描二维码,关注更多精彩内容

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭