【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开启后,立即跳转到虚拟地址

相关推荐
思尔芯S2C11 小时前
FPGA原型验证实战:如何应对外设连接问题
fpga开发·risc-v·soc设计·prototyping·原型验证
加强洁西卡11 小时前
【RISC-V】从C到可执行文件分析链接重定位的过程
c语言·开发语言·risc-v
硬汉嵌入式11 小时前
基于Rust构建的单片机Ariel RTOS,支持Cortex-M、RISC-V 和 Xtensa
单片机·rust·risc-v
MounRiver_Studio11 小时前
RISC-V IDE MRS2进阶分享(三):MRS语言服务器
ide·mcu·risc-v·嵌入式开发
加强洁西卡11 小时前
【RISC-V】解决链接器加入全局变量优化后操作系统无法启动的问题
risc-v
MounRiver_Studio11 小时前
RISC-V IDE MRS2进阶分享(四):CH32H417双核芯片项目开发
ide·mcu·risc-v·嵌入式开发
飞凌嵌入式2 天前
1块集成了4核Cortex-A7高性能CPU、1颗RISC-V MCU、多种高速总线、还兼容树莓派的T153低成本开发板
linux·arm开发·嵌入式硬件·risc-v
加强洁西卡2 天前
【RISC-V】riscv64-linux-gnu工具链都有哪些工具
linux·gnu·risc-v
jerwey2 天前
RISC-V VP 中 TLM 精度
risc-v