0

0

c++链接脚本怎么写 c++精准控制内存布局【底层】

穿越時空

穿越時空

发布时间:2026-01-05 04:01:25

|

686人浏览过

|

来源于php中文网

原创

c++kquote>链接脚本核心作用是控制代码段、数据段在可执行文件和内存中的布局,精确指定.text、.data、.bss等段的地址、对齐、合并与加载顺序,支撑裸机驱动、嵌入式OS等底层开发。

c++链接脚本怎么写 c++精准控制内存布局【底层】

c++链接脚本的核心作用

链接脚本(Linker Script)不处理C++语法或类定义,而是由链接器(如GNU ld)在最终链接阶段读取,用于精确指定代码段、数据段在可执行文件和内存中的布局。它直接控制.text、.data、.bss、自定义section的起始地址、对齐方式、合并规则和加载顺序——这是实现裸机驱动、嵌入式OS、安全隔离、内存热补丁等底层场景的关键基础。

最简可用的C++链接脚本结构

一个典型嵌入式或可控环境下的链接脚本(如layout.ld)如下:

MEMORY 定义物理/虚拟地址空间区域
SECTIONS 描述每个段如何映射到这些区域

示例(ARM Cortex-M风格,带C++运行时支持):

立即学习C++免费学习笔记(深入)”;

DeepL
DeepL

DeepL是一款强大的在线AI翻译工具,可以翻译31种不同语言的文本,并可以处理PDF、Word、PowerPoint等文档文件

下载
MEMORY
{
  FLASH (rx) : ORIGIN = 0x08000000, LENGTH = 512K
  RAM  (rwx): ORIGIN = 0x20000000, LENGTH = 128K
}

SECTIONS { .text : ALIGN(4) { (.text.startup) / 启动代码优先 / (.text) / 普通函数 / (.rodata) / 只读数据(含C++字符串字面量、虚表、typeinfo)/ . = ALIGN(4); __exidx_start = .; (.ARM.exidx) / C++异常表入口(若启用-EH) / __exidx_end = .; } > FLASH

.data : ALIGN(4) { data_load_start = LOADADDR(.data); data_start = .; (.data) (.data.) . = ALIGN(4); __data_end = .; } > RAM AT > FLASH / 运行时从FLASH拷贝到RAM */

.bss : ALIGN(4) { __bss_start = .; (.bss) (.bss.) (COMMON) . = ALIGN(4); __bss_end = .; } > RAM

/ C++全局构造函数表(关键!否则ctor不执行) / .init_array : ALIGN(4) { PROVIDE_HIDDEN (__init_array_start = .); KEEP ((SORT(.init_array.))) KEEP (*(.init_array)) PROVIDE_HIDDEN (__init_array_end = .); } > RAM

/ C++全局析构函数表(用于atexit或退出清理) / .fini_array : ALIGN(4) { PROVIDE_HIDDEN (__fini_array_start = .); KEEP ((SORT(.fini_array.))) KEEP (*(.fini_array)) PROVIDE_HIDDEN (__fini_array_end = .); } > RAM

/ 自定义section:比如放特定类实例到固定地址 / .my_section ALIGN(64) : { *(.my_section) } > RAM

/ 符号:栈顶、堆底(供malloc/sbrk使用) / _stack_top = ORIGIN(RAM) + LENGTH(RAM); _heap_start = .; }

C++代码中配合链接脚本的关键写法

仅靠链接脚本无法自动绑定C++对象——必须用__attribute__((section("xxx")))显式指定目标section,并用extern "C"避免符号修饰干扰地址计算:

  • 将某个全局对象强制放入.my_section
    MyDriver driver_instance __attribute__((section(".my_section"), used));
  • 声明启动前必须调用的初始化函数(类似Linux module_init):
    void init_hw() __attribute__((constructor(101))); // 数字越小越早执行
  • 获取链接脚本定义的符号(如__data_start),需声明为外部C符号:
    extern "C" char __data_start[], __data_end[], __data_load_start[];
  • 虚函数表(vtable)、RTTI(typeinfo)默认进.rodata;若需隔离,可重定向:
    *(.rodata.vtable) *(.rodata.typeinfo) 单独归组

验证与调试技巧

写完链接脚本不能只靠“能编过”,必须验证实际布局是否符合预期:

  • arm-none-eabi-objdump -h your.elf 查看各section的VMA/LMA地址和大小
  • arm-none-eabi-nm -n your.elf | grep "__data" 确认符号地址是否落在RAM区间
  • 在C++启动代码(如Reset_Handler)中插入断言:
    static_assert(reinterpret_cast(&driver_instance) == 0x20001000, "driver not at expected addr");
  • 对关键对象加static_assert(sizeof(MyClass) % 64 == 0, "not aligned for cache line") 配合section对齐

不复杂但容易忽略:C++模板实例化、内联函数、异常处理帧信息都会生成隐藏section,务必用objdump -s -j .rodata your.elf抽样检查内容分布。

相关专题

更多
javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

97

2025.11.27

磁盘配额是什么
磁盘配额是什么

磁盘配额是计算机中指定磁盘的储存限制,就是管理员可以为用户所能使用的磁盘空间进行配额限制,每一用户只能使用最大配额范围内的磁盘空间。php中文网为大家提供各种磁盘配额相关的内容,教程,供大家免费下载安装。

1348

2023.06.21

如何安装LINUX
如何安装LINUX

本站专题提供如何安装LINUX的相关教程文章,还有相关的下载、课程,大家可以免费体验。

701

2023.06.29

linux find
linux find

find是linux命令,它将档案系统内符合 expression 的档案列出来。可以指要档案的名称、类别、时间、大小、权限等不同资讯的组合,只有完全相符的才会被列出来。find根据下列规则判断 path 和 expression,在命令列上第一个 - ( ) , ! 之前的部分为 path,之后的是 expression。还有指DOS 命令 find,Excel 函数 find等。本站专题提供linux find相关教程文章,还有相关

294

2023.06.30

linux修改文件名
linux修改文件名

本专题为大家提供linux修改文件名相关的文章,这些文章可以帮助用户快速轻松地完成文件名的修改工作,大家可以免费体验。

776

2023.07.05

linux系统安装教程
linux系统安装教程

linux系统是一种可以免费使用,自由传播,多用户、多任务、多线程、多CPU的操作系统。本专题提供linux系统安装教程相关的文章,大家可以免费体验。

572

2023.07.06

linux查看文件夹大小
linux查看文件夹大小

Linux是一种自由和开放源码的类Unix操作系统,存在着许多不同的Linux版本,但它们都使用了Linux内核。Linux可安装在各种计算机硬件设备中,比如手机、平板电脑、路由器、视频游戏控制台、台式计算机、大型机和超级计算机。linux怎么查看文件夹大小呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

545

2023.07.20

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.2万人学习

Git 教程
Git 教程

共21课时 | 2.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号