Linux学习笔记——零基础详解:什么是Bootloader?U-Boot启动流程全解析!

零基础详解:什么是Bootloader?U-Boot启动流程全解析!

    • 一、什么是Bootloader?
      • 📌 举个例子:
    • 二、U-Boot 是什么?
    • 三、U-Boot启动过程:分为两个阶段
      • 🔹 第一阶段(汇编阶段)
      • 🔹 第二阶段(C语言阶段)
    • 四、U-Boot是怎么启动Linux内核的?
    • 五、U-Boot与Linux之间的参数传递机制
      • 💡 什么是 tag(启动参数)?
    • 六、为什么U-Boot要关闭 Cache 和 MMU?
    • 七.为什么要给内核传递参数?
      • 为什么要传递这些参数?
    • 八、系统启动完整流程回顾
    • 九、总结一句话
    • 📌 小贴士:常见面试题汇总

一、什么是Bootloader?

Bootloader(引导加载程序),是一段特殊的“小程序”,它运行在 系统刚上电时,主要目的是:

  • 初始化硬件环境(如:内存、时钟、中断、串口等)
  • 然后将 Linux内核从 flash(NAND, NOR FLASH,SD,MMC等)拷贝到SDRAM中,最后启动Linux内核

通俗一点讲,Bootloader 就像电脑中的 BIOS —— 当你按下电源开关,它最先启动,完成最基础的硬件检测后,才将控制权交给操作系统(比如 Windows 或 Linux)。

📌 举个例子:

项目PC系统嵌入式Linux系统
初始程序BIOSBootloader
操作系统WindowsLinux
存储介质硬盘Flash / SD卡等

二、U-Boot 是什么?

U-Boot(全称为 Universal Bootloader)是一种 开源、通用的Bootloader,支持多种架构(ARM、MIPS、x86等),被广泛应用于嵌入式Linux开发中。

它的主要任务:

  • 初始化硬件资源(如内存、串口、存储器等)
  • 加载Linux内核到内存中
  • 传递启动参数给内核
  • 启动Linux内核

三、U-Boot启动过程:分为两个阶段

🔹 第一阶段(汇编阶段)

此阶段主要用 汇编代码编写,功能包括:

  1. 初始化时钟系统
  2. 关闭看门狗(防止系统误复位)
  3. 关闭中断
  4. 启动指令Cache(ICache)
  5. 关闭数据Cache与TLB
  6. 关闭MMU
  7. 初始化SDRAM(用于加载后续程序)
  8. 初始化NAND Flash
  9. 代码重定位(将U-Boot代码搬运至SDRAM中运行)

💡 注意:这部分代码位置受限(通常在片内RAM或FLASH运行),所以通常要写得很精炼。


🔹 第二阶段(C语言阶段)

这部分用 C语言编写,主要功能是:

  1. 初始化串口(便于调试)
  2. 打印启动信息
  3. 检测内存映射情况
  4. 从存储介质读取Linux内核镜像和根文件系统(initrd)将内核映象和根文件系统映象从 Flash上读到SDRAM空间中
  5. 设置内核启动参数
  6. 跳转到Linux内核执行入口

四、U-Boot是怎么启动Linux内核的?

当U-Boot完成自身工作后,就会把控制权交给Linux内核。那么,怎么跳转呢?

👉 本质就是:修改PC寄存器的值为内核所在地址

这样CPU下一条执行指令就来自Linux内核,开始正式启动!


五、U-Boot与Linux之间的参数传递机制

这部分是面试常考,请务必掌握!

在跳转到内核之前,U-Boot 会设置好三个寄存器:

寄存器含义示例说明
R0保持为0固定写0即可
R1机器ID(开发板型号)如友善之臂开发板是199,见mach-types.h
R2参数的地址(tag结构体)tag链表,里面保存了内存大小、命令行、根文件系统等

📌 这些参数是内核“启动时所需的环境信息”,如果不给内核传递参数,它无法知道设备的具体配置!

  1. R0寄存器:设置为0,通常用于标识内核启动时的参数。
  2. R1寄存器:传递给内核一个“机器ID”。这是一个唯一标识当前硬件平台的ID,每个开发板或CPU都会有一个唯一的ID。内核会根据该ID来判断是否支持当前硬件平台。
  3. R2寄存器:存储一个指向参数列表的指针,该列表包含了内核启动时需要的所有环境信息(例如内存大小、内存地址、文件系统等)。

💡 什么是 tag(启动参数)?

tag 是一组结构体,存放了启动参数,形式如下:

struct tag {struct tag_header {uint32_t size;   // tag大小(单位:4字节)uint32_t tag;    // tag类型(如ATAG_MEM、ATAG_CMDLINE等)} hdr;union {struct tag_mem32 mem;struct tag_cmdline cmdline;// 其他类型参数...} u;
};

📌 常见的tag有:

  • ATAG_CORE:表示起始
  • ATAG_MEM:内存起始地址与大小
  • ATAG_CMDLINE:命令行参数(如根文件系统路径)
  • ATAG_NONE:表示结束

六、为什么U-Boot要关闭 Cache 和 MMU?

  • 上电后RAM未初始化前,Cache 不能正常工作。
  • 如果数据 Cache 开启了,会导致从未初始化的Cache中读取错误数据,程序可能崩溃!
  • 所以:必须关闭MMU和Data Cache,指令Cache可以开也可以不开。

七.为什么要给内核传递参数?

在启动Linux内核之前,U-Boot完成了对硬件的基本初始化。此时,U-Boot已经“适应”了当前的开发板,但Linux内核并不一定能直接支持所有开发板。因为每个开发板的硬件配置都可能有所不同,例如不同的CPU型号、内存配置、外设接口等。而内核本身并不是为每个具体的开发板都做了硬件适配的,因此,内核在启动时需要了解当前开发板的硬件环境。为此,U-Boot需要将这些硬件信息传递给内核,以帮助内核做相应的配置和适配。

为什么要传递这些参数?

  1. 硬件适配:Linux内核需要了解当前硬件平台的配置(例如内存大小、CPU架构、设备树等)才能正确配置和初始化硬件资源。如果没有这些信息,内核就无法正确识别或初始化硬件设备。
  2. 启动过程的灵活性:开发板通常会有不同的硬件平台,U-Boot的作用就是提供一个通用的引导机制,而通过参数传递,U-Boot使得内核在加载后可以根据传递的参数调整自身的行为和配置。
  3. 内核启动时的配置:内核需要根据不同的硬件环境来调整内存、CPU等硬件的使用模式。例如,在某些开发板上可能需要配置特定的硬件接口,或者根据内存的大小调整内核的行为。通过参数传递,内核能够灵活地进行这些调整。

八、系统启动完整流程回顾

下面是系统上电到进入Linux系统的完整流程图:

[ 上电/复位 ]↓
[ ROM代码执行 ]↓
[ 初始化外设 & SDRAM ]↓
[ 加载并执行U-Boot ]↓
[ U-Boot初始化硬件 ]↓
[ 加载Linux内核 + initrd ]↓
[ 设置启动参数(R0, R1, R2)]↓
[ 跳转到内核入口 ]↓
[ Linux内核初始化 ]↓
[ 挂载根文件系统 ]↓
[ 启动用户空间进程 ]

九、总结一句话

Bootloader 就是操作系统的“搬运工 + 启动器”,而 U-Boot 就是最流行的 Bootloader,它让 Linux 成为可能!


📌 小贴士:常见面试题汇总

面试问题回答要点
什么是Bootloader?启动时执行的第一段程序,初始化硬件,加载并启动内核
U-Boot做了哪些事?两阶段:汇编(初始化)、C语言(加载内核+参数设置+跳转)
U-Boot如何跳转到Linux内核?设置PC寄存器为内核地址
参数是如何传递给内核的?用R0=0, R1=机器ID, R2=tag结构体地址传递
为什么要关闭Data Cache?避免RAM未初始化时Cache取错误数据导致程序异常
什么是tag结构?如何组织?struct tag,含header+联合体,ATAG_CORE起始、ATAG_NONE结束

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/74886.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Word 页眉设置(不同章节不同页眉)

需求分析 要给文档设置页眉,但是要不同的页眉不同的页眉 问题点:一旦设置页眉 每个页眉都是一样的 现在要设置不一样的 设置了页眉但是整个文章的页眉都一样 问题解决 取消链接 前一节(不和前面的页眉同步更新) 小结 不同的…

Debezium日常分享系列之:Debezium3.1版本之增量快照

Debezium日常分享系列之:Debezium3.1版本之增量快照 按需快照触发一次临时增量快照触发临时阻塞快照增量快照增量快照过程如何 Debezium 解决具有相同主键的记录之间的冲突快照窗口触发增量快照使用附加条件运行临时增量快照使用 Kafka 信号通道触发增量快照临时增量…

音视频开发从入门到精通:编解码、流媒体协议与FFmpeg实战指南

音视频开发从入门到精通:编解码、流媒体协议与FFmpeg实战指南 音视频技术作为数字媒体领域的核心,正在成为互联网和移动应用的重要组成部分。本文将全面介绍音视频开发的学习路径,从基础概念到高级应用,从编解码原理到实战案例&a…

bookkeeper基本概念

Apache BookKeeper 架构与基本概念 Apache BookKeeper 的架构 Apache BookKeeper 是一个高性能的分布式日志存储系统,主要用于存储和管理顺序写入的数据。它被设计用来提供低延迟、高吞吐量和强一致性的服务,常用于分布式系统中的日志存储需求&#xf…

Scala相关知识学习总结3

包 - 包声明:和Java类似,作用是区分同名类、管理类命名空间。Scala包名只能含数字、字母等,不能数字开头、不能用关键字。 - 包说明:有类似Java的包管理风格,也有独特嵌套风格。嵌套风格有两个特点,一是&…

在Spring Boot中实现图片上传和修改

1. 图片上传实现步骤 1.1 添加依赖 确保 spring-boot-starter-web 和 spring-boot-starter-validation 已存在&#xff1a; <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId> <…

网络原理 - HTTP/HTTPS

1. HTTP 1.1 HTTP是什么&#xff1f; HTTP (全称为 “超文本传输协议”) 是⼀种应用非常广泛的应用层协议. HTTP发展史&#xff1a; HTTP 诞生于1991年. 目前已经发展为最主流使用的⼀种应用层协议 最新的 HTTP 3 版本也正在完善中, 目前 Google / Facebook 等公司的产品已经…

第十届MathorCup高校数学建模挑战赛-A题:无车承运人平台线路定价问题

目录 摘 要 一、问题提出 1.1 背景 1.2 问题重述 二、基本假设 三、符号说明 四、问题分析 4.1 问题一的分析 4.2 问题二的分析 4.3 问题三的分析 4.4 问题四的分析 五、模型的建立与求解 5.1 问题一模型的建立与求解 5.1.1 数据预处理 5.1.2 问题一结果检验:因子分析模型 5.2…

C++假期练习

思维导图 牛客练习

Go语言-初学者日记(四):包管理

众所周知——“包”治百病。 理解包与模块&#xff0c;是 Go 迈向工程化开发的关键一环&#xff01; &#x1f4c2; 一、包&#xff08;Package&#xff09;是 Go 的基本组织单位 在 Go 中&#xff0c;每个 .go 文件都属于某个包&#xff08;package&#xff09;&#xff1a; …

Scala面向对象2

1. 抽象属性和方法&#xff1a;用 abstract 关键字定义抽象类&#xff0c;其中抽象属性无初始值&#xff0c;抽象方法无实现 。重写抽象方法需用 override &#xff0c;重写抽象属性时&#xff0c;可变属性用 var &#xff0c;不可变属性用 val 。 匿名子类&#xff1a;和 Jav…

DiffAD:自动驾驶的统一扩散建模方法

25年3月来自新加坡公司 Carion 和北航的论文“DiffAD: A Unified Diffusion Modeling Approach for Autonomous Driving”。 端到端自动驾驶 (E2E-AD) 已迅速成为实现完全自动驾驶的一种有前途的方法。然而&#xff0c;现有的 E2E-AD 系统通常采用传统的多任务框架&#xff0c…

Python四大核心数据结构深度解析:列表、元组、字典与集合

在Python编程语言中&#xff0c;数据结构是组织和存储数据的基本方式。Python提供了四种内置的核心数据结构&#xff1a;列表&#xff08;List&#xff09;、元组&#xff08;Tuple&#xff09;、字典&#xff08;Dictionary&#xff09;和集合&#xff08;Set&#xff09;。这…

网络编程—Socket套接字(TCP)

上篇文章&#xff1a; 网络编程—Socket套接字&#xff08;UDP&#xff09;https://blog.csdn.net/sniper_fandc/article/details/146923670?fromshareblogdetail&sharetypeblogdetail&sharerId146923670&sharereferPC&sharesourcesniper_fandc&sharefro…

SkyWalking+Springboot实战(最详细)

本篇文章记录了作者在0到1学习SkyWalking的过程&#xff0c;记录了对SkyWalking的部署&#xff0c;学习&#xff0c;使用Bug解决等等过程 一、什么是SkyWalking 官方文档&#xff1a; Apache SkyWalkinghttps://skywalking.apache.org/ SkyWalking 是一个开源的分布式追踪、性…

Arduino示例代码讲解:Row-Column Scanning an 8x8 LED matrix with X-Y input LED矩阵

Arduino示例代码讲解:Row-Column Scanning an 8x8 LED matrix with X-Y input LED矩阵 Row-Column Scanning an 8x8 LED matrix with X-Y input LED矩阵功能概述硬件部分:软件部分:代码逐行解释定义常量定义变量`setup()` 函数`loop()` 函数`readSensors()` 函数`refreshScr…

多线程编程中的锁策略

目录 1.悲观锁vs乐观锁 关键总结 悲观锁&#xff1a; 乐观锁&#xff1a; 选择建议 用 悲观锁 当&#xff1a; 用 乐观锁 当&#xff1a; 2.重量级锁vs轻量级锁 选择建议 用 轻量级锁&#xff1a; 用 重量级锁&#xff1a; 3.挂起等待锁vs自旋锁 关键细节说明 选择…

负载均衡是什么,Kubernetes如何自动实现负载均衡

负载均衡是什么&#xff1f; 负载均衡&#xff08;Load Balancing&#xff09; 是一种网络技术&#xff0c;用于将网络流量&#xff08;如 HTTP 请求、TCP 连接等&#xff09;分发到多个服务器或服务实例上&#xff0c;以避免单个服务器过载&#xff0c;提高系统的可用性、可扩…

React-01React创建第一个项目(npm install -g create-react-app)

1. React特点 JSX是javaScript语法的扩展&#xff0c;React开发不一定使用JSX。单向响应的数据流&#xff0c;React实现单向数据流&#xff0c;减少重复代码&#xff0c;比传统数据绑定更简单。等等 JSX是js的语法扩展&#xff0c;允许在js中编写类似HTML的代码 const …

小程序中的网络请求

在小程序中&#xff0c;使用 wx.request( ) 这个方法来发送网路请求&#xff0c;整个请求的方式和 jQuery 里面的 $.ajax 方法是非常相似的。 在 wx.request( ) 这个方法中&#xff0c;接收一个配置对象&#xff0c;该配置对象中能够配置的项目如下表&#xff1a; 关于服务器…