【RISC-V】区分加载地址、链接地址、运行地址

原本是没有这个概念区分的,现代通用计算机(PC、手机、服务器)都有MMU,你的操作系统内核会自动处理了这一切,你不需要区分这些,但是当你的机器上电时,还没有操作系统内核接管时,你开发的裸机/嵌入式/引导程序需要明确区分这些地址:

┌─上电过程─────────────────────────────────────────┐

│ 上电复位 → BIOS/UEFI固件 → Bootloader → 操作系统内核 → 用户空间 │

└────────────────────────────────────────────────┘

  • 加载地址:程序真正复制或存储(有些固件程序就存储在ROM中)到内存中的起始物理。
  • 链接地址:链接地址是链接器(linker)在生成可执行文件或目标文件时,为每个符号和段分配的虚拟地址。
  • 运行地址:程序真正运行时的内存地址,也是PC在执行指令或访问数据时,实际使用的内存地址

所以

什么情况下一段程序三者地址相同:

比较少见了,现在程序基本都在虚拟内存下运行,只有标准静态链接的非 PIE(Position-Independent Executable)用户态可执行程序三者地址相同(在编译时要明确指明-no-pie参数)

什么情况下加载地址与链接地址不同,但是链接地址与运行地址相同:

一般你编写的程序(大多数用户态程序)都是这个模式,上面也说了,当你的程序运行在 带 MMU 的操作系统 (如 Linux、Windows)时,你的操作系统加载器会将程序加载到内存的虚拟地址。CPU 运行时使用该虚拟地址访问代码和数据。此时链接地址与运行地址与加载地址都相同。

什么情况下链接地址与运行地址不同:

内核或裸机程序在 MMU 开启之前的程序,比如Bootloader 的第二阶段,还有RISCV架构中MMU 开启前会调用固件接口程序和启动操作系统内核的汇编代码(因为MMU启动后归MMU接管了)

注意!!!:这些任务不能依赖虚拟内存 ,代码必须是位置无关(Position-Independent) 或已知加载地址的。

举一个真实的例子,Linux 内核在编译链接时,会被链接脚本链接到一个高的虚拟地址 0xffffffff80000000(. = 0xffffffff80000000,但Bootloader中(如 OpenSBI 或 U-Boot)将内核镜像加载地址编写在了内存的0x80200000处

如何解决:内核代码里的所有MMU启动前的汇编代码都不基于基址编写,而是基于PC编写,用auipc和lla这些,当MMU开启后,立即跳转到虚拟地址

相关推荐
木由里予1 天前
RISC-V 32 位基础指令集(RV32I)完整参考手册
risc-v
一枝小雨3 天前
RISC-V架构sp寄存器 & RISC-V架构下FreeRTOS任务上下文保存与恢复
单片机·架构·嵌入式·risc-v·rtos·内核原理
一枝小雨3 天前
RISC-V架构的中断与异常处理机制学习笔记
单片机·架构·嵌入式·risc-v·内核原理·中断与异常
一枝小雨3 天前
什么是标准C函数:以RISC-V架构下的C函数为例
c语言·risc-v·内核原理
咕咚.萌西4 天前
RISCV AS汇编器
risc-v
嵌入式小企鹅4 天前
UiPath推出AI编程“总指挥台”,SiFive发布RISC-V第三代猛兽
人工智能·学习·google·程序员·ai编程·risc-v·开源工具
阿祖_in_coding7 天前
RISC-V ACT测试
risc-v
yusur7 天前
开芯院院长唐丹一行来访中科驭数 共探RISC-V与DPU算力协同创新之路
risc-v
加强洁西卡7 天前
【RISC-V】RVV选摘
risc-v
加强洁西卡8 天前
【RISC-V】fclass数值类型对照表
risc-v