在了解STM32内存之前需要了解 MCU 的型号和MDK 中的.map 文件,很多刚学习 stm32 时都不会过多的去了解 MCU 的选型,是在太枯燥了。这里在从新了解一下,久了就熟悉了。

一、STM32命令规则

二、MDK下生成.map文件

  1. 在MDK中勾选.map文件的生成,确认后编译一下工程即可生成,map文件。

  2. 打开.map文件

三、MDK下文件基本概念

在.map文件的最后可以看到文件信息的统计,如下图所示:

当然每次编译完成后也可以看到统计信息,如下图所示:

了解MDK下的一些常用变量名:

变量 作用
code 代码存储区,存放函数体的二进制代码
Ro-data 只读数据存储区,存放字常量数据类型(如const类型)程序结束后有系统自动释放
RW-data 初始化可读写变量的大小,程序结束后由系统自动释放。
ZI-data 没有初始化的可读写变量大小,程序结束后由系统自动释放。
heap 堆区,一般由程序员分配释放,若程序员不释放,程序结束时可能由OS释放。
stack 栈区,由编译器自动分配释放,存放函数的参数值,局部变量的值等。
.text 与RO-code同义
.constdata 与RO-data同义
.bss 未初始化的全局和静态变量,编译器自动初始化为0
.data 初始化的全局和静态变量(与RW-data同义)
PAD 地址空间对齐
RO Size 包含Code及RO Data,表示只读数据占用Flash空间的大小。
RW Size 包含RW Data及ZI Data,表示运行时占用的RAM的大小。
ROM Size 包含Code,RO Data及RW Data,表示烧写程序所占用的Flash的大小。

注意:栈向下生长,内存地址由高至低;堆向上,内存地址由低至高

通过上面表格可知STM32在编程时所用RAM和ROM的大小:

Flash(ROM)=Code+Ro-data
Sram(RAM)=Rw-data+ZI-data

四、STM32内存

这里我找了一位大佬总结的博客“STM32内存知识你真的了解吗?”,感觉挺好的,所以我直接引用一下启动的图片,如下图所示:



  • STM32程序运行的流程

    程序在运行之前,需要可执行将镜像文件(一般是bin或hex文件),通过烧写工具写入STM32的Flash中。STM32上电启动(从Flash启动时)后会将RW段中的RW-data(初始化的全局变量)拷贝到RAM中,然后根据编译器给出的ZI地址和大小分配出ZI段,并将这块RAM区域清零。如下图所示:左边是每上电flash+ram的状态,右边是上电后运行时flash+ram的状态。

注意:

  • 可执行映像文件烧录到 STM32 后的内存分布包含 RO 段和 RW 段两个部分,其中其中 RO 段中保存了 Code、RO-data 的数据,RW 段保存了 RW-data 的数据,由于 ZI-data 都是 0,所以未包含在映像文件中。
  • STM32运行时不会拷贝RO段,因为CPU的可执行代码是直接从Flash中读取的。

STM32编程时需要注意的事项

  • 堆栈的大小在编译器编译之后是不知道的,只有在运行时才知道,所以容易造成堆栈溢出(发生hardfault错误),那怎么知道自己的内存大小了,通过选型手册就知道了,比如我使用的是STM32F103C8T6,ROM是64k,RAM是20k,如下图所示:

  • 程序中的常量,如果没加const也会编译到SRAM里,加了const会被编译到flash中。

  • 栈向下生长,内存地址由高至低;堆向上,内存地址由低至高,堆栈之间没有固定的界限,堆栈冲突时会导致系统崩溃,如下图所示:

五、.map文件

  1. 不同文件中函数调用的关系

    ==============================================================================
    
    Section Cross References
    
        startup_stm32f10x_hd.o(RESET) refers to startup_stm32f10x_hd.o(.text) for Reset_Handler
    startup_stm32f10x_hd.o(.text) refers to system_stm32f10x.o(.text) for SystemInit
    startup_stm32f10x_hd.o(.text) refers to entry.o(.ARM.Collect$$$$00000000) for __main
    stm32f10x_rcc.o(.text) refers to stm32f10x_rcc.o(.data) for APBAHBPrescTable
    stm32f10x_gpio.o(.text) refers to stm32f10x_rcc.o(.text) for RCC_APB2PeriphResetCmd
    stm32f10x_usart.o(.text) refers to stm32f10x_rcc.o(.text) for RCC_APB2PeriphResetCmd
    led.o(.text) refers to stm32f10x_rcc.o(.text) for RCC_APB2PeriphClockCmd
    led.o(.text) refers to stm32f10x_gpio.o(.text) for GPIO_Init
    main.o(.text) refers to led.o(.text) for LED_GPIO_Config
    main.o(.text) refers to stm32f10x_gpio.o(.text) for GPIO_ResetBits
    entry.o(.ARM.Collect$$$$00000000) refers (Special) to entry10a.o(.ARM.Collect$$$$0000000D) for __rt_final_cpp
    entry.o(.ARM.Collect$$$$00000000) refers (Special) to entry11a.o(.ARM.Collect$$$$0000000F) for __rt_final_exit
    entry.o(.ARM.Collect$$$$00000000) refers (Special) to entry7b.o(.ARM.Collect$$$$00000008) for _main_clock
    entry.o(.ARM.Collect$$$$00000000) refers (Special) to entry8b.o(.ARM.Collect$$$$0000000A) for _main_cpp_init
    entry.o(.ARM.Collect$$$$00000000) refers (Special) to entry9a.o(.ARM.Collect$$$$0000000B) for _main_init
    entry.o(.ARM.Collect$$$$00000000) refers (Special) to entry5.o(.ARM.Collect$$$$00000004) for _main_scatterload
    entry.o(.ARM.Collect$$$$00000000) refers (Special) to entry2.o(.ARM.Collect$$$$00000001) for _main_stk
    entry2.o(.ARM.Collect$$$$00000001) refers to entry2.o(.ARM.Collect$$$$00002712) for __lit__00000000
    entry2.o(.ARM.Collect$$$$00002712) refers to startup_stm32f10x_hd.o(STACK) for __initial_sp
    entry2.o(__vectab_stack_and_reset_area) refers to startup_stm32f10x_hd.o(STACK) for __initial_sp
    entry2.o(__vectab_stack_and_reset_area) refers to entry.o(.ARM.Collect$$$$00000000) for __main
    entry5.o(.ARM.Collect$$$$00000004) refers to init.o(.text) for __scatterload
    entry9a.o(.ARM.Collect$$$$0000000B) refers to main.o(.text) for main
    entry9b.o(.ARM.Collect$$$$0000000C) refers to main.o(.text) for main
    init.o(.text) refers to entry5.o(.ARM.Collect$$$$00000004) for __main_after_scatterload ==============================================================================

    main.o(.text) refers to led.o(.text) for LED_GPIO_Config,是main.c文件中调用了led.c文件中的LED_GPIO_Config函数

  2. 被删除的冗余函数

    ==============================================================================
    
    Removing Unused input sections from the image.
    
        Removing startup_stm32f10x_hd.o(HEAP), (0 bytes).
    Removing core_cm3.o(.emb_text), (32 bytes).
    Removing system_stm32f10x.o(.constdata), (20 bytes).
    Removing misc.o(.text), (220 bytes).
    Removing stm32f10x_usart.o(.text), (880 bytes). 5 unused section(s) (total 1152 bytes) removed from the image. ==============================================================================

    删除冗余的函数,有效降低程序的代码量,MDK自动优化,可以通过“One ELF Section per Function”选项开启,开启后可以大大优化程序代码量,打开方式是如下图所示:

    打开后再次编译,看看.map文件中删除了81个函数,优化了2762字节,如下图所以:

  3. 局部标号和全局标号

    • 局部标号

      主要是在文件中用static声明的全局变量和函数。汇编文件中的标号地址(作用域限本文件)

    • 全局标号

      非static声明的变量和函数,汇编文件中的标号地址(作用域全局工程)

    注意:

    • Number 不站地址空间,大小为0。
    • DATA 只读数据
    • 文件中的标号再次用i.声明,说明在c文件中用static声明了的,如下图所示:



  4. 映像文件

    映像文件可以分为加载域和运行域

    加载域反应了ARM可执行映像文件各个段存放在寄存器中的位置关系。

  5. 组件大小

  6. 映像的真实大小

六、.htm文件

文件中做大的作用就是基本统计了所有被调用函数的栈stack使用的情况(不考虑中断嵌套)

  1. 栈的最大深度,调用路劲是main ⇒ LED_GPIO_Config ⇒ GPIO_Init

  2. 递归调用函数

  3. 函数指针

  4. 全局标号

    比如复位中断函数,使用的是Thumb指令,占用0字节栈空间,函数代码大小2字节

    比如系统时钟初始化函数SystemInit ,使用的是Thumb指令,函数代码大小68字节,占用栈空间8字节,代码深度28字节,函数调用路径是Call Chain = SetSysClock ⇒ SetSysClockTo72

参考文献

stm32的内存分布:https://blog.csdn.net/BooleanWater/article/details/119278723

STM32单片机的内存分布详解(1):https://www.bilibili.com/read/cv13912565

STM32内存知识你真的了解吗?:https://blog.csdn.net/qq_49864684/article/details/119887704

MDK生成的map和htm文件分析:https://www.bilibili.com/video/BV1t3411C7Pu/?spm_id_from=autoNext

STM32内存知识的更多相关文章

  1. STM32内存结构介绍和FreeRTOS内存分配技巧

    这是我第一次使用FreeRTOS构建STM32的项目,踩了好些坑,又发现了我缺乏对于操作系统的内存及其空间的分配的知识,故写下文档记录学习成果. 文章最后要解决的问题是,如何恰当地分配FreeRTOS ...

  2. STM32内存映射

    一.概述 STM32内存映射是STM32的架构的重要组成部分,不可或缺. 二.STM32内存映射图 1.内存映射图--摘自<CM3权威指南> 2.内存映射图--摘自<STM32F10 ...

  3. 舵机&数据处理&stm32内存之堆栈溢出(遇到的问题)

    产品名称:TOWER PRO(辉盛)大扭力舵机MG996R (MG995升级产品)6v/11Kg厂家编号:MG996R产品净重: 55g产品尺寸: 40.7*19.7*42.9mm产品拉力: 9.4k ...

  4. C程序员必须知道的内存知识【英】

    C程序员必须知道的内存知识[英] 时间 2015-03-08 14:16:11  极客头条原文  http://marek.vavrusa.com/c/memory/2015/02/20/memory ...

  5. Java-100天知识进阶-Java内存-知识铺(四)

    知识铺: 致力于打造轻知识点,持续更新每次的知识点较少,阅读不累.不占太多时间,不停的来唤醒你记忆深处的知识点. 1.Java内存模型是每个java程序员必须掌握理解的 2.Java内存模型的主要目标 ...

  6. JVM内存知识备忘

    又是一篇备忘... 主要记录一些知识,进行一些资源的汇总. 先来群里liufor大大提供的两张图,清晰易懂: Dockerized Java https://www.youtube.com/watch ...

  7. [转帖] 学习 Linux 大页的内存知识

    一.在解释什么情况下需要开启大页和为啥需要开启大页前先了解下Linux下页的相关的知识:以下的内容是基于32位的系统,4K的内存页大小做出的计算1)目录表,用来存放页表的位置,共包含1024个目录en ...

  8. STM32 内存管理实验

    参考原文<STM32F1开发指南> 内存管理简介 内存管理,是指软件运行时对计算机内存资源的分配和使用的技术.最主要的目的是如何高效.快速的分配,并且在适当的时候释放和回收内存资源.内存管 ...

  9. Java-100天知识进阶-JVM内存-知识铺(三)

    知识铺: 致力于打造轻知识点,持续更新每次的知识点较少,阅读不累.不占太多时间,不停的来唤醒你记忆深处的知识点. Java内存模型(JMM) JVM内存模式是JVM的内存分区 Java内存模式是一种虚 ...

随机推荐

  1. 微信小程序列表拖动排序Demo

    wxml页面编写 <view class="container"> <view bindtap="box" class="box&q ...

  2. 安卓电池健康查看软件AccuBattery 分享

    一.天下苦秦久矣 说实话,我是小米的忠实粉丝(雷总打钱),手里目前是红米k30pro标准版, 室友中有用华为也有苹果的,据我所知苹果系统是可以看到电池健康的,但是安卓却不行, 所以推荐大家一个安卓软件 ...

  3. 修复tunl0-二进制安装calico

    这篇博文很重要,出现这个问题导致pod之间无法通讯,pod无法连接外网. 出现的问题是二进制方式安装了节点之后, tunl0没有显示,通过ifconfig tunl0 up 启动tunl0 没有意义, ...

  4. 《java基础——对象的拷贝》

    java基础--对象的拷贝 一.浅拷贝: 规则: 1. 浅拷贝只是拷贝了源对象的地址,所以源对象的值发生变化时,拷贝对象的值也会发生变化. 2. 浅拷贝相当于两个对象共用一套实例. 格式: 类名 对象 ...

  5. NodeJs学习日报day9——操作数据库

    const mysql = require('mysql') const db = mysql.createPool({ // 数据库的ip地址 host: 'localhost', user: 'r ...

  6. Java学习day14

    可变参数用作方法的形参,方法参数的个数就可变 格式:修饰符 返回值类型 方法名(数据类型...变量名){ } 方法内的形参只能有一个,这里的变量是一个数组 public static <T> ...

  7. Codeforces Round #706 (Div. 2)B. Max and Mex __ 思维, 模拟

    传送门 https://codeforces.com/contest/1496/problem/B 题目 Example input 5 4 1 0 1 3 4 3 1 0 1 4 3 0 0 1 4 ...

  8. 前端vue之属性指令、style和class、条件渲染、列表渲染、事件处理、数据双向绑定、表单控制、v-model进阶

    今日内容概要 属性指令 style和class 条件渲染 列表渲染 事件处理 数据的双向绑定 v-model进阶 购物车案例 内容详细 1.属性指令 <!DOCTYPE html> < ...

  9. 2021.05.14 tarjan

    2021.05.14 tarjan 标准版tarjan 这里使用数组来模拟栈 void tarjan(int x){ ++ind; dfn[x]=low[x]=ind; stacki[++top]=x ...

  10. Java 获取Word中的所有插入和删除修订

    在 Word 文档中启用跟踪更改功能后,会记录文档中的所有编辑行为,例如插入.删除.替换和格式更改.对插入或删除的内容,可通过本文中介绍的方法来获取. 引入Jar 方法1 手动引入:将 Free Sp ...