【STM32-MAP文件分析】

STM32-MAP文件分析

  • ■ MDK编译生成文件简介
    • ■ .o
    • ■ .axf
    • ■ .hex
    • ■ .crf
    • ■ .d
    • ■ .dep
    • ■ .lnp
    • ■ .lst
    • ■ .map
    • ■ .build_log.htm
    • ■ .htm 文件
    • ■ .map 文件
  • ■ map文件分析
    • ■ map 文件的 MDK 设置
      • ■ 1. 要生成 map 文件 在 Listing 选项卡里面
      • ■ 2. Keil5 中打开.map 文件
    • ■ map 文件的基础概念
      • ■ 1. Section:程序段
      • ■ 2. RO: Read Only 的缩写,
      • ■ 3. RW: Read Write 的缩写,
      • ■ 4. ZI: Zero initialized 的缩写,
      • ■ 5. .text:相当于 RO code
      • ■ 6. .constdata:相当于 RO data
      • ■ 7. .bss:相当于 ZI data
      • ■ 8. .data:相当于 RW data
    • ■ map 文件的组成部分说明
      • ■ 1. 程序段交叉引用关系(Section Cross References)
      • ■ 2. 删除映像未使用的程序段(Removing Unused input sectionsfrom the image)
      • ■ 3. 映像符号表(Image Symbol Table)
        • ■ 3.1 本地符号(Local Symbols) 地址和大小
        • ■ 3.2 全局符号( Global Symbols)地址和大小
      • ■ 4. 映像内存分布图( Memory Map of the image)
      • ■ 5. 映像组件大小(Image component sizes)

■ MDK编译生成文件简介

MDK 编译工程,会生成一些中间文件(如 .axf、 .crf、 .d、 .dep、.hex、 .lnp、 .lst、 .o、 .htm、 bulild_log.htm 和.map等),最终生成 hex 文件,编译过程产生的所有文件,都存放在 Output 文件夹下
在这里插入图片描述

■ .o

可重定向 1对象文件,每个源文件(.c/.s 等)编译都会生成一个.o 文件
它是由编译器编译.c/.s 文件时所产生的可重定向对象文件,其文件名同.c/.s 文件一模一样,只是后缀为.o。所以,我们看到.o 就应该想到与之对应的.c 或者.s 文件,我们在分散加载里面经常会用到.o 文件。

■ .axf

由 ARMCC 编译生产的可执行对象文件,不可重定向 2(绝对地址)
多个.o 文件链接生成.axf 文件,我们在仿真的时候,需要用到该文件,
由 armlink 链接器,将整个工程参与编译的.o 文件链接成一个可执行对象文件,它是不可重定向的。有了该文件,我们就可以用仿真器来下载到 MCU 进行仿真调试了。
注意:各类仿真器,在进行下载调试的时候,都是使用的.axf 文件。

■ .hex

Intel Hex 格式文件,可用于下载到 MCU, .hex 文件和.bin 文件由.axf 文件转换而来可执行的对象文件
.hex 文件和.bin 文件的区别是: .bin 文件不含地址信息,全部都是可执行代码;而.hex 文件则是包含地址信息的可自行代码。
我们在使用 ISP 软件进行程序下载的时候,
一般使用的是.hex 文件,由 ISP 软件解析.hex 文件包含的地址信息来实现程序下载。而我们在进行 BootLoader 升级的时候,一般使用.bin 文件,地址由 Bootloader 程序指定。

■ .crf

交叉引用文件,包含浏览信息(定义、标识符、引用)

■ .d

由 ARMCC/GCC 编译生产的依赖文件(.o 文件所对应的依赖文件)
每个.o 文件,都有一个对应的.d 文件

■ .dep

整个工程的依赖文件

■ .lnp

MDK 生成的链接输入文件,用于命令输入

■ .lst

C 语言或汇编编译器生成的列表文件

■ .map

连接器生成的列表文件/MAP 文件, 该文件对我们非常有用!

■ .build_log.htm

最近一次编译工程时的日志记录文件

■ .htm 文件

链接生成的列表文件
它是编译器在编译代码的时候生成的一个列表文件, 包含了整个工程的静态调用图, 最大的用处就是可以查看栈深度(最小深度),方便设置栈大小。
它是编译器在编译代码的时候生成的一个列表文件, 包含了整个工程的静态调用图, 最大的用处就是可以查看栈深度(最小深度),方便设置栈大小。 .htm 文件可以直接由浏览器打开(双击打开)。 .htm 文件包含两部分内容:
1, 整个工程最大栈(Stack) 深度及其调用关系
我们打开:新建工程例程→Output→atk_h750.htm 文件(双击,注意:必须整个工程编译一遍,才会生成 atk_h750.htm 文件,否则是找不到这个文件的!),可以看到如图 1.2 所示:
在这里插入图片描述
可以看到,例程的最大栈深度是 416 字节,最大栈深时的调用关系为: __rt_entry_main⇒ main ⇒ sys_stm32_clock_init ⇒AL_RCC_ClockConfig ⇒ HAL_InitTick ⇒HAL_NVIC_SetPriority ⇒ __NVIC_SetPriority。

不过需要注意的是,这里的最大栈深度仅仅是最低要求(静态栈),因为它并没有统计无栈深的函数(用内存管理)、递归函数、以及无法追踪的函数(函数指针)等所包含的栈(Stack)。
不过它给我们指明了最低需求,我们在分配栈深度的时候,就可以参考这个值来做设置,一般不低于静态栈的2倍。我们例程默认设置的栈深(Stack_Size)为0X800(通过.s文件设置),

2, 各个函数的栈深及其调用关系
.htm 文件还给出了每个函数所使用的栈深度以及其调用关系,如图 1.3 所示:
在这里插入图片描述
上图给出了 sys_stm32_clock_init 函数的最大栈深度及其调用关系,并且列出了其所调用的函数及其所被调用的函数。

■ .map 文件

对我们编程非常有帮助,是本文档重点要给大家介绍的,因此另起一节,进行重点说明。
其他文件类型及说明,请大家参考: MDK→help→uVision Help→B. File Types,如图 1.3所示:
在这里插入图片描述

■ map文件分析

map 文件是编译器链接时生成的一个文件,它主要包含了交叉链接信息。通过.map 文件,我们可以知道整个工程的函数调用关系、 FLASH 和 RAM 占用情况及其详细汇总信息,能具体到单个源文件(.c/.s)的占用情况,根据这些信息,我们可以对代码进行优化。

.map 文件可以分为以下 5 个组成部分:
1, 程序段交叉引用关系(Section Cross References)
2, 删除映像未使用的程序段(Removing Unused input sections from the image)
3, 映像符号表(Image Symbol Table)
4, 映像内存分布图(Memory Map of the image)
5, 映像组件大小(Image component sizes)

接下来,我们将分三个部分对 map 文件进行详细介绍:
1, map 文件的 MDK 设置
2, map 文件的基础概念
3, map 文件的组成部分说明

■ map 文件的 MDK 设置

■ 1. 要生成 map 文件 在 Listing 选项卡里面

要生成 map 文件,我们需要在 MDK 的魔术棒→Listing 选项卡里面,进行相关设置,如图 2.1.1 所示:
在这里插入图片描述
中红框框出的部分就是我们需要设置的,默认情况下, MDK 这部分设置就是全勾选的,如果我们想取消掉一些信息的输出,则取消相关勾选即可(一般不建议)。

■ 2. Keil5 中打开.map 文件

双击工程目标 LED(不同工程名字可能不一样),打开.map 文件。
在这里插入图片描述

■ map 文件的基础概念

为了更好的分析 map 文件,我们先对需要用到的一些基础概念进行一个简单介绍,相关概念如下:

■ 1. Section:程序段

描述映像文件的代码或数据块,我们简称程序段

■ 2. RO: Read Only 的缩写,

包括只读数据(RO data)和代码(RO code)两部分内容,占用 FLASH 空间

■ 3. RW: Read Write 的缩写,

包含可读写数据(RW data,有初值,且不为 0),占用 FLASH(存储初值)和 RAM(读写操作)

■ 4. ZI: Zero initialized 的缩写,

包含初始化为 0 的数据(ZI data),占用 RAM 空间。

■ 5. .text:相当于 RO code

■ 6. .constdata:相当于 RO data

■ 7. .bss:相当于 ZI data

■ 8. .data:相当于 RW data

■ map 文件的组成部分说明

■ 1. 程序段交叉引用关系(Section Cross References)

这部分内容描述了各个文件(.c/.s 等)之间函数(程序段)的调用关系,在这里插入图片描述
main.o(i.main) refers to sys.o(i.sys_stm32_clock_init) for sys_stm32_clock_init, 表示:
main.c文件中的main函数,调用了sys.c中的sys_stm32_clock_init函数。
其中:
i.main 表示 main 函数的入口地址
i.sys_stm32_clock_init 表示sys_stm32_clock_init 的入口地址。

■ 2. 删除映像未使用的程序段(Removing Unused input sectionsfrom the image)

这部分内容描述了工程中由于未被调用而被删除的冗余程序段(函数/数据),
在这里插入图片描述
例如 main.c 中没有调用到的rev16_text 等文件将被移除。

另外,在最后还有一个统计信息: 361 unused section(s) (total 43234 bytes) removed fromthe image. 表示总共移除了 361 个程序段(函数/数据),大小为 43234 字节。即给我们的MCU 节省了 43234 字节的程序空间。

为了更好的节省空间,我们一般在 MDK→魔术棒→C/C++选项卡里面勾选: One ELFSection per Function,如图 2.1.2.2 所示:
在这里插入图片描述

■ 3. 映像符号表(Image Symbol Table)

映像符号表(Image Symbol Table)描述了被引用的各个符号(程序段/数据)在存储器中的存储地址、类型、 大小等信息。
映像符号表分为两类:本地符号(Local Symbols)和全局符号(Global Symbols)。

■ 3.1 本地符号(Local Symbols) 地址和大小

记录了用 static 声明的全局变量地址和大小, c 文件中函数的地址和 用 static 声明的函数代码大小,汇编文件中的标号地址(作用域: 限本文件)
在这里插入图片描述
图中红框框处部分,表示 sys.c 文件中的 sys_stm32_clock_init 函数的入口地址为:0x08002bc8,类型为: Section(程序段),大小为 0。因为: i. sys_stm32_clock_init 仅仅表示 sys_stm32_clock_init 函数入口地址,并不是指令,所以没有大小。
在全局符号段,会列出sys_stm32_clock_init 函数的大小。
在这里插入图片描述

■ 3.2 全局符号( Global Symbols)地址和大小

全局符号(Global Symbols) 记录了全局变量的地址和大小, C 文件中函数的地址及其代码大小,汇编文件中的标号地址(作用域: 全工程), 全局符号如图 2.1.3.2.1 所示:
在这里插入图片描述
图中红框框处部分
表示 sys.c 文件中的 sys_stm32_clock_init 函数的入口地址为:0x08002bc9,类型为: Thumb Code(程序段),大小为 344 字节。
在这里插入图片描述

注意,此处Local Symbols: 的地址用的 0x080003e8,和Global Symbols: 0x080003e9地址不符,这是因为ARM 规定 Thumb 指令集的所有指令,其最低位必须为 1, 0x080003e9= 0x080003e8+ 1,所以才会有 2 个不同的地址,且总是差 1,实际上就是同一个函数。

Local Symbols:
在这里插入图片描述
Global Symbols:
在这里插入图片描述

■ 4. 映像内存分布图( Memory Map of the image)

映像文件分为加载域( Load Region)和运行域( Execution Region)
一个程序可以有多个加载域, 一个加载域必须有至少一个运行域(可以有多个运行域)。
加载域为映像程序的实际存储区域,而运行域则是 MCU 上电后的运行状态。加载域和运行域的简化关系(这里仅表示一个加载域的情况)
在这里插入图片描述
由图可知, RW 区也是存放在 ROM(FLASH)里面的,在执行 main 函数之前, RW(有初值且不为 0 的变量)数据会被拷贝到 RAM 区,同时还会在 RAM 里面创建 ZI 区(初始化为 0 的变量)。
在这里插入图片描述

序号描述
① 处表示映像的入口地址,也就是整个程序运行的起始地址,为: 0X0800 0299。 实际地址为: 0X0800 0298(Thumb 指令最低位是 1)。
② 处表示 LR_m_stmflash 加载域, 其起始地址为: 0X0800 0000;占用大小为: 0X0000 2D8C; 最大地址范围为: 0X0002 0000。
其内部包含两个运行域: ER_m_stmflash和 RW_m_stmsram。
③ 处表示 ER_m_stmflash 运行域,其起始地址为: 0X0800 0000;占用大小为: 0X0000 2D6C;最大地址范围为: 0X0002 0000;即 内部 FLASH 运行域,所有需要放内部 FLASH 的代码,都应该放到这个运行域里面。
对于 STM32F1/F4/F767 等开发板, 我们例程所有的代码,都是放在这个运行域的(名字可能不一样)。
④ 处表示 RW_m_stmsram 运行域,其起始地址为:0X2400 0000;占用大小为: 0X0000 2D6C;最大地址范围为: 0X0008 0000;即内部 SRAM 运行域,所有 RAM(包括 RW 和 ZI)都是放在这个运行域里面。
⑤ 处表示 LR_m_qspiflash 加载域,其起始地址为: 0X9000 0000;占用大小为: 0X0000 0720;最大地址范围为: 0X0080 0000。
其内部包含一个运行域: ER_m_qspiflash。
⑥ 处表示 ER_m_qspiflash 运行域,其起始地址为: 0X9000 0000;占用大小为: 0X0000 0720;最大地址范围为: 0X0080 0000;即外部 QSPI FLASH 运行域,所有需要放 外部 QSPI FLASH 的代码,都应该放到这个运行域里面。

如 sys_stm32_clock_init 函数:该函数在 ER_m_stmflash 运行域;入口地址为: 0X0800 2BC8;大小为: 0X168 字节;是 sys.c里面的函数。了解这些信息,对我们分析及优化程序非常有用。

■ 5. 映像组件大小(Image component sizes)

映像组件大小(Image component sizes)给出了整个映像所有代码(.o)占用空间的汇总信息,对我们比较有用,

在这里插入图片描述

在这里插入图片描述

① 处,表示.c/.s 文件生成对象所占空间大小(单位:字节,下同),即.c/.s 文件编译后所占代码空间的大小。
每个项所代表的意义如下:

类型作用
Code(inc.data):表示包含内联数据(inc.data)后的代码大小。
如 delay.o(即delay.c)所占的 Code 大小为 142 字节,其中 8 字节是内联数据。
RO Data:表示只读数据所占的空间大小,一般是指 const 修饰的数据大小。
RW Data:表示有初值(且非 0)的可读写数据所占的空间大小,它同时占用 FLASH和 RAM 空间。
ZI Data:表示初始化为 0 的可读写数据所占空间大小,它只占用 RAM 空间。
Debug:表示调试数据所占的空间大小,如调试输入节及符号和字符串。
Object Totals:表示以上部分链接到一起后,所占映像空间的大小。

(incl.Generated):表示链接器生产的映像内容大小,它包含在 Object Totals 里面了,这里仅仅是单独列出,我们一般不需要关心。
(incl.Padding) :表示链接器根据需要插入填充以保证字节对齐的数据所占空间的大小,它也包含在 Object Totals 里面了,这里单独列出,一般无需关心。

② 处,表示被提取的库成员( .lib)添加到映像中的部分所占空间大小。
各项意义同①中的说明。我们一般只用看 Library Totals 来分析库所占空间的大小即可。

③ 处,表示本工程全部程序汇总后的占用情况。其中:

类型作用
Grand Totals:表示整个映像所占空间大小。
ELF Image Totals:表示 ELF 可执行链接格式映像文件的大小,一般和 Grand Totals一样大小。
ROM Totals:表示整个映像所需要的 ROM 空间大小,不含 ZI 和 Debug 数据。
Total RO Size:表示 Code 和 RO 数据所占空间大小,本例程为: 13452 字节。
Total RW Size:表示 RW 和 ZI 数据所占空间大小,即本映像所需 SRAM 空间的大小,本例程为: 3032 字节。
Total ROM Size:表示 Code、 RO 和 RW 数据所占空间大小,即本映像所需 FLASH空间的大小,本例程为: 13484 字节。

打开debug后的Image component sizes
在这里插入图片描述

在这里插入图片描述
关闭debug后的Image component sizes
在这里插入图片描述

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

来自 Kubecon Paris 的快讯

一年一度的 KubeconEU 审查时间到了——它未经过滤,偶尔不受 CNCF 的欢迎——但剧透警告,巴黎取得了巨大的成功。我们总是爱这里的人,我们并不总是爱场地或演出管理,但巴黎是一场胜利,更重要的是,Kubernete…

无线麦克风一拖二哪个牌子好,揭秘目前音质最好的麦克风

在自媒体的世界里,无线领夹麦克风已经成了一种标准配置,它受到广大视频创作者的青睐。无论是刚刚踏入短视频领域的新手,还是已经拥有丰富经验的资深博主,都深知一款好的领夹麦克风对于提升视频音频质量的重要性。它不仅能帮助创作…

从数据资源到数据资产之数据要素及其市场运作【AMT企源】

题记 本文旨在探讨企业数据资产化的过程中,数据要素及其资产化市场的构成,形成可交易的数据产品的过程,以及如何在大数据交易市场中发挥价值。最后通过一个典型案例的分析,浅谈数据要素的社会化配置与数据资产的入表,…

深度学习Day-21:ResNet与DenseNet结合

🍨 本文为:[🔗365天深度学习训练营] 中的学习记录博客 🍖 原作者:[K同学啊 | 接辅导、项目定制] 要求: 探索ResNet与DenseNet结合的可能性根据模型特性构建新的模型框架验证改进后模型的效果 一、 基础配…

【教程】PVE下uhd630核显直通HDMI输出 以NUC9为例村雨Murasame

大家好,村雨本雨又来发教程了 最近在搞小主机,之前hp400g3仅仅200多元成功核显直通HDMI,作为简单NAS、解码机、伺服机、中控都非常棒,待机仅9w 村雨Murasame:【教程】7代核显直通HDMI成功输出画面 PVE下7代intel核显…

Android studio安卓期末大作业,智学英语App

目录 1 系统主要功能 2 登陆 3 主菜单界面 4 单词查询 5 美文阅读 6 客服聊天 7 关于我们 1 系统主要功能 应用启动 打开应用,将看到一个欢迎界面,这通常是一个简短的动画或静态图像,用于引导进入主界面。 登录功能 (1…

R语言数据分析案例32-针对芬兰污染指数的分析与考察

一、 研究背景及意义 近年来,随着我国科技和经济高速发展,人们生活质量也随之显著提高。但是, 环境污染问题也日趋严重,给人们的生活质量和社会生产的各个方面都造成了许多不 利的影响。空气污染作为环境污染主要方面&#xff0c…

keepalive+nginx高可用架构

keepalivenginx架构 一.配置真实服务器web1和web2 1.关闭防火墙,并在真实服务器下载http服务 [rootlocalhost ~]# systemctl stop firewalld.service [rootlocalhost ~]# setenforce 0 [rootlocalhost ~]# yum install httpd -y 2.分别在web1和web2上制作网页…

慎投!新增7本期刊被“On Hold“,14本影响因子下降!

本周投稿推荐 SSCI • 中科院2区,6.0-7.0(录用友好) EI • 各领域沾边均可(2天录用) CNKI • 7天录用-检索(急录友好) SCI&EI • 4区生物医学类,0.5-1.0(录用…

如何使用Excel与Outlook实现邮件群发:详细教程

引言 在工作中,我们经常需要发送大量邮件。手动发送既费时又容易出错。本教程将教你如何使用Excel和Outlook,通过简单的VBA代码实现邮件的自动群发,提高工作效率。 准备工作 在开始之前,你需要确保以下工具已经安装在电脑上&am…

骨传导耳机值不值得入手?五款运动好物骨传导耳机推荐!

开放式耳机在如今社会中已经迅速成为大家购买耳机的新趋势,开放式蓝牙耳机作为骨传导耳机,深受喜欢听歌和热爱运动的人群欢迎。当大家谈到佩戴的稳固性时,后挂式骨传导耳机都会收到一致好评。对于热爱运动的人士而言,高品质的骨传…

WordPress插件:子比zibll主题插件 炙焰美化全开源插件V3.2

在网络世界中,开源内容管理系统(CMS)已经成为了网站构建的关键工具之一。WordPress,作为最受欢迎的开源CMS之一,其广泛的应用及其灵活性使得它成为了创建和管理各种类型网站的理想选择。而Zibll主题插件,作…

vue3中h函数的使用

h函数是用于创建一个 vnodes ,它既可以用于创建原生元素,也可以创建组件,其渲染后的效果等同于使用模版语言来进行创建。 h函数的传参如下: // 完整参数签名 function h(type: string | Component,props?: object | null,child…

【GD32F303红枫派使用手册】第二十二节 IIC-IIC OLED显示实验

22.1 实验内容 通过本实验主要学习以下内容: OLED驱动原理 IIC驱动OLED显示操作 22.2 实验原理 OLED模块的驱动芯片为SSD1306,其显存大小总共为 128*64bit 大小,SSD1306 将这些显存分为了 8 页,其对应关系如下所示&#xff1…

如何将办公文档压缩成rar格式文件?

压缩包格式是我们生活工作中常用到的文件格式,那么如何得到一个rar格式的压缩文件?或者说如何将文件压缩成rar格式而不是zip格式呢?今天我们来了解一下如何压缩为rar格式文件。 首先,下载并安装WinRAR,然后用鼠标选择需…

FlinkCDC介绍及使用

CDC简介 什么是CDC? cdc是Change Data Capture(变更数据获取)的简称。核心思想是,监测并捕获数据库的 变动(包括数据或数据表的插入,更新以及删除等),将这些变更按发生的顺序完整记录下来,写入到消息中间件以供其它服…

修复 Android 手机卡在启动屏幕上的 7 种方法

Android 手机卡在启动屏幕上的情况并不常见。通常,问题出现在应用新更新或安装未知来源的应用程序后。幸运的是,您可以让您的 Android 手机跳过启动屏幕,而无需前往最近的服务中心。 当您的 Android 手机在启动屏幕上陷入无限循环时&#xf…

计算机组成原理 —— 存储系统(概述)

计算机组成原理 —— 存储系统(概述) 存储系统按层次划分按照存储介质分类按照存储方式分类按照信息可更改性分类根据信息的可保存性分类存储器性能指标 我们今天来学习计算机组成原理中的存储系统: 存储系统 存储系统是计算机系统中用于存…

css文字镂空加描边

css文字镂空加描边 <!DOCTYPE html> <html><head><meta charset"utf-8"><title>文字镂空</title><style>/* 公用样式 */html,body{width: 100%;height: 100%;position: relative;}/* html{overflow-y: scroll;} */*{margi…

【机器学习 复习】 第1章 概述

一、概念 1.机器学习是一种通过先验信息来提升模型能力的方式。 即从数据中产生“模型”( model )的算法&#xff0c;然后对新的数据集进行预测。 2.数据集&#xff08;Dataset&#xff09;&#xff1a;所有数据的集合称为数据集。 训练集&#xff1a;用来训练出一个适合模…