【ARM 嵌入式 编译系列 7.1 -- GCC 链接脚本中节区及各个段的详细介绍】
文章目录
- 什么是Section(节区)
- 输入文件常见节区有哪些?
- 什么是 glue code?.glue_7和.glue_7的作用是什么?
- 链接脚本中的 KEEP 关键字是什么呢作用?
- 链接脚本中的 PROVIDE 关键字是什么呢作用?
上篇文章:ARM 嵌入式 编译系列 7 – ARM GCC 链接脚本详细讲解
下篇文章:ARM 嵌入式 番外篇 编译系列 8 – RT-Thread 编译命令 Scons 详细讲解
什么是Section(节区)
在编程语言中,节区(Section)是对程序的一个逻辑分割,它包含了程序中特定类型的信息。一个完整的程序通常会被分割成几个节区,比如代码节区(.text)、数据节区(.data)和 BSS 节区(.bss)等。
在 GCC 链接脚本中,你可以使用 SECTIONS 命令来定义输出文件中各个节区的位置和顺序。其中,每个节区都用花括号 {} 括起来,并且可以包含多个输入节区和符号。
以下是一个简单的例子:
SECTIONS
{ .text : { *(.text) } > FLASH .data : { *(.data) } > RAM .bss : { *(.bss) } > RAM
}
在这个示例中,.text、.data 和 .bss 是输出文件中的节区,而 *(.text)、*(.data) 和 *(.bss) 是输入文件中的节区。> FLASH 和 > RAM 指定了节区应该被放置到哪个内存区域。
需要注意的是,链接器会按照 SECTIONS 命令中节区的顺序来布局输出文件中的节区。所以,如果你需要特定节区在内存中的位置或顺序,请在 SECTIONS 命令中进行指定。
输入文件常见节区有哪些?
在 ARM GCC 链接脚本中,常见的输入文件节区主要包括以下几种:
-
.text:存放代码的节区,通常对应 FLASH 等只读存储区; -
.rodata:存放只读数据的节区,比如常量字符串、const 声明的变量等,也通常对应 FLASH 等只读存储区; -
.data:存放已初始化的全局变量和静态变量的节区。此节区的初始化值存储在 FLASH 中,但在程序启动时会被复制到 RAM 中; -
.bss:存放未初始化的全局变量和静态变量的节区。此节区不占用 FLASH 空间,仅在 RAM 中预留空间; -
.heap:堆区,用于动态内存分配,如malloc、free等; -
.stack:栈区,用于存放局部变量和函数调用的上下文; -
.isr_vector:这个节区通常用于存放中断向量表。对于基于 ARM Cortex-M 的微控制器,中断向量表包括了复位向量、异常处理函数和中断处理函数的入口地址等。在程序启动时,CPU 会加载中断向量表到特定的地址,并在中断发生时通过中断向量表跳转到相应的处理函数; -
.init:这个节区包含了程序的初始化代码。在 C 程序中,全局变量和静态变量的初始化代码通常就在这个节区。此外,如果程序使用了 C++,那么全局对象的构造函数也会被放到这个节区; -
.fini:这个节区包含了程序的终止代码。在 C 程序中,这个节区通常为空。但如果程序使用了 C++,那么全局对象的析构函数会被放到这个节区。
这些节区在链接脚本中的定义和布局将决定了它们在内存中的位置和大小。并且这些节区的布局需要与 MCU 的内存布局匹配,以确保代码和数据能够被正确地加载和执行。
实际使用的链接脚本中内容设置比较多,如下:
SECTIONS
{.text :{. = ALIGN(4);_stext = .;KEEP(*(.isr_vector)) /* Startup code */. = ALIGN(4);*(.text) /* remaining code */*(.text.*) /* remaining code */*(.rodata) /* read-only data (constants) */*(.rodata*)*(.glue_7)*(.glue_7t)*(.gnu.linkonce.t*)/* section information for finsh shell */. = ALIGN(4);__fsymtab_start = .;KEEP(*(FSymTab))__fsymtab_end = .;. = ALIGN(4);__vsymtab_start = .;KEEP(*(VSymTab))__vsymtab_end = .;/* section information for initial. */. = ALIGN(4);__rt_init_start = .;KEEP(*(SORT(.rti_fn*)))__rt_init_end = .;. = ALIGN(4);PROVIDE(__ctors_start__ = .);KEEP (*(SORT(.init_array.*)))KEEP (*(.init_array))...
什么是 glue code?.glue_7和.glue_7的作用是什么?
GCC 编译器会自动生成一些代码来处理 ARM 和 Thumb 指令集之间的交互,这些代码通常被称为 “glue code”。在链接脚本中,这些 glue code 通常被放在 .glue_7 和 .glue_7节区中。
.glue_7节区包含了从 ARM 模式到 Thumb 模式的跳转代码;.glue_7t节区包含了从 Thumb 模式到 ARM 模式的跳转代码。
这些节区中的代码通常会被自动插入到需要进行模式转换的地方。
在链接脚本中,你通常需要将这两个节区放到 .text 节区中,例如:
SECTIONS
{ .text : { *(.text) *(.glue_7) *(.glue_7t) } > FLASH
}
在这个示例中,.glue_7 和 .glue_7t 节区被放置在 .text 节区中,这样它们就会被加载到 FLASH 中,并能在程序运行时被执行。
注意 只有在混合使用 ARM 和 Thumb 指令集的情况下,编译器才会生成 glue code。如果你的程序只使用一种指令集,那么就不需要关心这两个节区。
链接脚本中的 KEEP 关键字是什么呢作用?
在 GCC 链接脚本中,KEEP 关键字用于防止某些节区在链接过程中被丢弃。
在链接过程中,如果链接器发现某些节区没有被引用,链接器可能会选择丢弃这些节区以节省空间。但是有些时候,你可能需要保留这些节区,即使它们没有被直接引用。这时,你就可以使用 KEEP 关键字。
KEEP 关键字的语法如下:
KEEP(file section)
file 是输入的文件列表,可以使用通配符。
section 是文件中需要保留的节区列表,也可以使用通配符。
比如说,以下的链接脚本会保留所有输入文件中的 .init 和 .fini 节区:
SECTIONS
{ .init : { KEEP(*(.init)) } > FLASH .fini : { KEEP(*(.fini)) } > FLASH
}
在这个示例中,KEEP(*(.init)) 和 KEEP(*(.fini)) 分别会保留所有输入文件中的 .init 和 .fini 节区,这些节区在链接过程中不会被丢弃。
链接脚本中的 PROVIDE 关键字是什么呢作用?
见文章:ARM 嵌入式 编译系列 7 – ARM GCC 链接脚本详细讲解
链接脚本中定义了 FSymTab ,VSymTab及 .rti_fn 这几个节区,以 .rti_fn 为例,在代码中有很多初始化函数需要按照先后顺序进行执行,编译的时候会统一将这些初始化函数放到某一个节区中,比如 .rti_fn 节区。如下代码是将函数 fn 放到 .rti_fn 节区中,然后根据 level 的值进行排序。
#if RT_DEBUG_INITstruct rt_init_desc{const char* fn_name;const init_fn_t fn;};#define INIT_EXPORT(fn, level) \const char __rti_##fn##_name[] = #fn; \RT_USED const struct rt_init_desc __rt_init_desc_##fn RT_SECTION(".rti_fn." level) = \{ __rti_##fn##_name, fn};#else#define INIT_EXPORT(fn, level) \RT_USED const init_fn_t __rt_init_##fn RT_SECTION(".rti_fn." level) = fn#endif
在 rt-thread/include/rtdef.h 中实现了如下宏定义来定义初始化函数的执行先后顺序。
/* board init routines will be called in board_init() function */
#define INIT_BOARD_EXPORT(fn) INIT_EXPORT(fn, "1")/* pre/device/component/env/app init routines will be called in init_thread */
/* components pre-initialization (pure software initilization) */
#define INIT_PREV_EXPORT(fn) INIT_EXPORT(fn, "2")
/* device initialization */
#define INIT_DEVICE_EXPORT(fn) INIT_EXPORT(fn, "3")
/* components initialization (dfs, lwip, ...) */
#define INIT_COMPONENT_EXPORT(fn) INIT_EXPORT(fn, "4")
/* environment initialization (mount disk, ...) */
#define INIT_ENV_EXPORT(fn) INIT_EXPORT(fn, "5")
/* appliation initialization (rtgui application etc ...) */
#define INIT_APP_EXPORT(fn)
上篇文章:ARM 嵌入式 编译系列 7 – ARM GCC 链接脚本详细讲解
下篇文章:ARM 嵌入式 番外篇 编译系列 8 – RT-Thread 编译命令 Scons 详细讲解
相关文章:
【ARM 嵌入式 编译系列 7.1 -- GCC 链接脚本中节区及各个段的详细介绍】
文章目录 什么是Section(节区)输入文件常见节区有哪些?什么是 glue code?.glue_7和.glue_7的作用是什么?链接脚本中的 KEEP 关键字是什么呢作用?链接脚本中的 PROVIDE 关键字是什么呢作用? 上篇文章:ARM 嵌…...
一文读懂HTML
文章目录 HTML的历史HTML的作用HTML的基本语言 HTML的历史 HTML(HyperText Markup Language)的历史可以追溯到20世纪90年代早期,它是互联网发展的重要里程碑之一。以下是HTML的历史概述: 早期阶段(1980年代末 - 1990年…...
MOCK测试
介绍 mock:就是对于一些难以构造的对象,使用虚拟的技术来实现测试的过程。 mock测试:在测试过程中,对于某些不容易构造或者不容易获取的对象,可以用一个虚拟的对象来代替的测试方 法。 接口Mock测试:在接口…...
Flutter源码分析笔记:Widget类源码分析
Flutter源码分析笔记 Widget类源码分析 - 文章信息 - Author: 李俊才 (jcLee95) Visit me at: https://jclee95.blog.csdn.netEmail: 291148484163.com. Shenzhen ChinaAddress of this article:https://blog.csdn.net/qq_28550263/article/details/132259681 【介绍】&#x…...
PyTorch 微调终极指南:第 2 部分 — 提高模型准确性
一、说明 如今,在训练深度学习模型时,通过在自己的数据上微调预训练模型来迁移学习已成为首选方法。通过微调这些模型,我们可以利用他们的专业知识并使其适应我们的特定任务,从而节省宝贵的时间和计算资源。本文分为四个部分&…...
MySQL数据库----------安装anaconda---------python与数据库的链接
作者前言 🎂 ✨✨✨✨✨✨🍧🍧🍧🍧🍧🍧🍧🎂 🎂 作者介绍: 🎂🎂 🎂 🎉🎉🎉…...
nuxt页面布局
nuxt页面默认布局文件在layouts目录下default.vue,可将页面的头部和脚部提取出来,形成布局页,将主内容区域的内容替换成<nuxt />。附default.vue代码: <template><div class"app-container"><div…...
mac编译ffmpeg
- code: git clone https://git.ffmpeg.org/gitweb/ffmpeg.git - 编译安装 https://trac.ffmpeg.org/wiki/CompilationGuide - 使用homebrew安装dependency brew install automake fdk-aac git lame libass libtool libvorbis libvpx \ opus sdl shtool texi2ht…...
如何让你的图片服务也有类似OSS的图片处理功能
原文链接 前言 有自己机房的公司一般都有一套存储系统用于存储公司的图片、视频、音频、文件等数据,常见的存储系统有以NAS、FASTDFS为代表的传统文件存储,和以Minio为代表的对象存储系统,随着云服务的兴起很多公司逐渐将数据迁移到以阿里云…...
Oracle PL/SQL 类型(Type):索引表、嵌套表、变长数组、pipelined 管道
1、Oracle 新建员工表和部门表.sql。 集合类型 1、Oracle 集合是相同类型元素的组合,在集合中,使用唯一的下标来标识其中的每个元素,与 Java 的 List 很像。 2、常用集合方式: 类型语法下标元素个数初始值.extend能否存在DB中…...
Web 服务器 -【Tomcat】的简单学习
Tomcat1 简介1.1 什么是Web服务器 2 基本使用2.1 下载2.2 安装2.3 卸载2.4 启动2.5 关闭2.6 配置2.7 部署 3 Maven创建Web项目3.1 Web项目结构3.2 创建Maven Web项目 4 IDEA使用Tomcat4.1 集成本地Tomcat4.2 Tomcat Maven插件 Tomcat 1 简介 1.1 什么是Web服务器 Web服务器是…...
armbian使用1panel快速部署部署springBoot项目后端
文章目录 前言环境准备实现步骤第一步:Armbian安装1panel第二步:安装数据库第三步:查看数据库容器重要信息【重要】查看容器所在的网络查看容器连接地址 第四步:项目配置和打包第五步:构建项目镜像 前言 这里只是简单记录部署spr…...
Streamlit 讲解专栏(八):图像、音频与视频魔法
文章目录 1 前言2 st.image:嵌入图像内容2.1 图像展示与描述2.2 调整图像尺寸2.3 使用本地文件或URL 3 st.audio:嵌入音频内容3.1 播放音频文件3.2 生成音频数据播放 4 st.video:嵌入视频内容4.1 播放视频文件4.2 嵌入在线视频 5 结语&#x…...
python使用装饰器记录方法耗时
思路 python使用修饰器记录方法耗时,目的是每当方法执行完后,可以记录该方法耗时,而不需要在每个方法的执行前后,去创建一个临时变量,来记录耗时。 方式一(不推荐): 在每个方法的…...
JavaWeb课程学习--Day01
HTML 建立css文件: css使用方式: <span>...</span>无语意包裹标签 css中的三种选择器: 注意:播放视音频时要留出播放空间 盒子模型: 表格标签: 以上表格: 表单标签: 表…...
Spring Boot单元测试使用MockBean注解向Service注入Mock对象
1. 背景介绍 我们在测试时有一个Service,我们需要测试Service,但Service内部依赖ServiceA、ServiceB,此时我们希望Mock ServiceA,ServiceB 注入真实对象。 class Service {private ServiceA A;private ServiceB B;public int me…...
Java中使用instanceof判断对象类型
记录:470 场景:Java中使用instanceof判断对象类型。例如在解析JSON字符串转换为指定类型时,先判断类型,再定向转换。在List<Object>中遍历Object时,先判断类型,再定向转换。 版本:JDK 1…...
postman测试后端增删改查
目录 一、本文介绍 二、准备工作 (一)新建测试 (二)默认url路径查看方法 三、增删改查 (一)查询全部 (二)增加数据 (三)删除数据 (四&…...
根据源码,模拟实现 RabbitMQ - 通过 SQLite + MyBatis 设计数据库(2)
目录 一、数据库设计 1.1、数据库选择 1.2、环境配置 1.3、建库建表接口实现 1.4、封装数据库操作 1.5、针对 DataBaseManager 进行单元测试 一、数据库设计 1.1、数据库选择 MySQL 是我们最熟悉的数据库,但是这里我们选择使用 SQLite,原因如下&am…...
1、基于 CentOS 7 构建 LVS-DR 群集。 2、配置nginx负载均衡
一、基于CentOS7和、构建LVS-DR群集 准备四台虚拟机 ip作用192.168.27.150客户端192.168.27.151LVS192.168.27.152RS192.168.27.152RS 关闭防火墙 [rootlocalhost ~]# systemctl stop firewalld安装ifconfig yum install net-tools.x86_64 -y1、DS上 1.1 配置LVS虚拟IP …...
从零实现富文本编辑器#5-编辑器选区模型的状态结构表达
先前我们总结了浏览器选区模型的交互策略,并且实现了基本的选区操作,还调研了自绘选区的实现。那么相对的,我们还需要设计编辑器的选区表达,也可以称为模型选区。编辑器中应用变更时的操作范围,就是以模型选区为基准来…...
Qt Http Server模块功能及架构
Qt Http Server 是 Qt 6.0 中引入的一个新模块,它提供了一个轻量级的 HTTP 服务器实现,主要用于构建基于 HTTP 的应用程序和服务。 功能介绍: 主要功能 HTTP服务器功能: 支持 HTTP/1.1 协议 简单的请求/响应处理模型 支持 GET…...
Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...
DBAPI如何优雅的获取单条数据
API如何优雅的获取单条数据 案例一 对于查询类API,查询的是单条数据,比如根据主键ID查询用户信息,sql如下: select id, name, age from user where id #{id}API默认返回的数据格式是多条的,如下: {&qu…...
GitHub 趋势日报 (2025年06月08日)
📊 由 TrendForge 系统生成 | 🌐 https://trendforge.devlive.org/ 🌐 本日报中的项目描述已自动翻译为中文 📈 今日获星趋势图 今日获星趋势图 884 cognee 566 dify 414 HumanSystemOptimization 414 omni-tools 321 note-gen …...
Rust 异步编程
Rust 异步编程 引言 Rust 是一种系统编程语言,以其高性能、安全性以及零成本抽象而著称。在多核处理器成为主流的今天,异步编程成为了一种提高应用性能、优化资源利用的有效手段。本文将深入探讨 Rust 异步编程的核心概念、常用库以及最佳实践。 异步编程基础 什么是异步…...
大数据学习(132)-HIve数据分析
🍋🍋大数据学习🍋🍋 🔥系列专栏: 👑哲学语录: 用力所能及,改变世界。 💖如果觉得博主的文章还不错的话,请点赞👍收藏⭐️留言Ǵ…...
GC1808高性能24位立体声音频ADC芯片解析
1. 芯片概述 GC1808是一款24位立体声音频模数转换器(ADC),支持8kHz~96kHz采样率,集成Δ-Σ调制器、数字抗混叠滤波器和高通滤波器,适用于高保真音频采集场景。 2. 核心特性 高精度:24位分辨率,…...
均衡后的SNRSINR
本文主要摘自参考文献中的前两篇,相关文献中经常会出现MIMO检测后的SINR不过一直没有找到相关数学推到过程,其中文献[1]中给出了相关原理在此仅做记录。 1. 系统模型 复信道模型 n t n_t nt 根发送天线, n r n_r nr 根接收天线的 MIMO 系…...
MySQL账号权限管理指南:安全创建账户与精细授权技巧
在MySQL数据库管理中,合理创建用户账号并分配精确权限是保障数据安全的核心环节。直接使用root账号进行所有操作不仅危险且难以审计操作行为。今天我们来全面解析MySQL账号创建与权限分配的专业方法。 一、为何需要创建独立账号? 最小权限原则…...
