当前位置: 首页 > backend >正文

Linux系统移植⑨:uboot启动流程详解-bootz启动Linux过程

Linux系统移植⑨:uboot启动流程详解-bootz启动Linux过程

bootz 是 U-Boot 中用于启动 Linux 内核的命令,专为处理 zImage(压缩内核映像) 设计。


启动 Linux 的完整过程:

1. 加载内核与相关文件

U-Boot 先将以下文件加载到内存指定地址:

  • zImage:压缩的 Linux 内核映像(位于内存地址 KERNEL_ADDR)。
  • 设备树二进制文件(DTB):硬件描述文件(位于 FDT_ADDR)。
  • initramfs(可选):初始内存文件系统(位于 INITRD_ADDR)。

2. 执行 bootz 命令

bootz [内核地址] [initramfs地址]:[大小] [设备树地址]

  • 启动命令

    bootz ${KERNEL_ADDR} ${INITRD_ADDR}:${INITRD_SIZE} ${FDT_ADDR}

    若无需 initramfs:
    bootz ${KERNEL_ADDR} - ${FDT_ADDR}

3. zImage 自解压

  • U-Boot 跳转到 KERNEL_ADDR,执行 zImage 头部的小型解压程序。
  • 解压程序将内核解压到预设的 物理内存地址(如 ARM 架构通常为 0x8000)。

4. 传递启动参数

U-Boot 通过寄存器/设备树传递关键参数:

架构寄存器参数说明
ARMr00(保留)
r1机器 ID(旧)或 0xFFFFFFFF(新)
r2设备树地址(FDT_ADDR
RISC-Va0设备树地址
a10(保留)

设备树(DTB) 包含 CPU、内存、外设等硬件信息,内核据此初始化硬件。


5. 内核初始化

Linux 内核接管后执行:

  1. 硬件初始化:基于 DTB 设置 CPU、内存控制器、时钟等。
  2. 驱动加载:识别并初始化存储、网络等设备驱动。
  3. 挂载根文件系统
    • 若有 initramfs,将其作为临时根文件系统。
    • 否则直接挂载 root= 参数指定的文件系统(如 /dev/mmcblk0p2)。

6. 用户空间启动

  1. 内核启动第一个用户进程 /init(位于 initramfs 或根文件系统)。
  2. init 进程加载系统服务(如 systemd 或 SysVinit)。
  3. 最终进入登录界面或指定应用。

关键依赖

  • zImage 格式:必须为 gzip 压缩的 Image 文件(非 uImage)。
  • 设备树支持:现代内核强制要求 DTB(无 DTB 的内核无法启动)。
  • 地址对齐:内核/DTB 的加载地址需符合 CPU 架构要求(如 ARM 需 4KB 对齐)。

do_bootz源码如下:

在这里插入图片描述

其内部主要函数调用关系如下:

do_bootz
-> bootz_start
-> do_bootm_states 阶段为BOOTM_STATE_START
-> bootm_start 对images全局变量清零,
-> images->ep = 0X80800000
->bootz_setup 判断zImage是否正确
-> bootm_find_images
-> boot_get_fdt 找到设备树,然后将设备树起始地址和长度,写入到images的ft_addr和ft_len成员变量中。
-> bootm_disable_interrupts 关闭中断相关
-> images.os.os = IH_OS_LINUX; 表示要启动Linux系统
-> do_bootm_states 状态BOOTM_STATE_OS_PREP 、BOOTM_STATE_OS_FAKE_GO 、BOOTM_STATE_OS_GO,
-> bootm_os_get_boot_func 查找Linux内核启动函数。找到Linux内核启动函数do_bootm_linux,赋值给boot_fn。
-> boot_fn(BOOTM_STATE_OS_PREP, argc, argv, images); 就是do_bootm_linux。
-> boot_prep_linux 启动之前的一些工作,对于使用设备树来说,他会将Bootargs传递给Linux内核,通过设备树完成。也就是向Linux内核传参。
-> boot_selected_os BOOTM_STATE_OS_GO, do_bootm_linux
-> do_bootm_linux,BOOTM_STATE_OS_GO
-> boot_jump_linux
-> machid= gd->bd->bi_arch_number;
-> void (kernel_entry)(int zero, int arch, uint params);
-> kernel_entry = (void (
)(int, int, uint))images->ep; 0X80800000。
-> announce_and_cleanup 输出Starting kernel……
-> kernel_entry(0, machid, r2); 启动Linux内核。 Uboot的最终使命,启动Linux内核。

在这里插入图片描述

http://www.xdnf.cn/news/14572.html

相关文章:

  • 如何在 MX Linux 上安装 Mint 的 Cinnamon 桌面 UI
  • C语言——枚举
  • STM32 GPIO 寄存器开发
  • Vue 二维码组件
  • 力扣-416.分割等和子集
  • shelve模块的使用
  • 数据结构--栈和队列
  • Cobbler批量安装流程及具体配置步骤
  • 鸿蒙开发:基于最新API,如何实现组件化运行
  • 如何用K8s+Istio进行云原生开发?
  • mysql8 sql脚本转mysql5.7
  • 【网工】华为配置专题进阶篇①
  • C/C++中的位域(Bit-field)是什么?
  • 【6G技术探索】MCP协议整理分享
  • 黑马python(七)
  • Java 常用类 Time API:现代时间处理的艺术
  • MIT 6.S081 2020 Lab9 File Systems 个人全流程
  • 部署Maven Java Web项目
  • 什么是状态机?状态机入门
  • 【超详细】讯飞智能车PC电脑烧录指南(高级系统部署与恢复)
  • 《深度学习基础与概念》task2/3
  • 编译器、调试器、仿真器:嵌入式开发的“三把刀”深度解析
  • 目标检测相关【清晰易懂】
  • BloodyAD 命令使用详解
  • WinRAR隐藏技能:给压缩包添加注释
  • Mac Mini M4 安装 jdk8 以及 隐藏 设置内的Java菜单
  • Prompt:更好的提示与迭代
  • c++面试题(14)------顺时针打印矩阵
  • VSCode -配置为中文界面
  • Javaweb学习day4——(MVC架构模式)