你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32单片机的堆栈

[复制链接]
gaosmile 发布时间:2020-10-31 19:41
    学习STM32单片机的时候,总是能遇到“堆栈”这个概念。分享本文,希望对你理解堆栈有帮助。
    对于了解一点汇编编程的人,就可以知道,堆栈是内存中一段连续的存储区域,用来保存一些临时数据。堆栈操作由PUSH、POP两条指令来完成。而程序内存可以分为几个区:
  • 栈区(stack)
  • 堆区(Heap)
  • 全局区(static)
  • 文字常亮区程序代码区
    % W( }* O: n" }  `% g
    程序编译之后,全局变量,静态变量已经分配好内存空间,在函数运行时,程序需要为局部变量分配栈空间,当中断来时,也需要将函数指针入栈,保护现场,以便于中断处理完之后再回到之前执行的函数。/ x( u, ]4 ^. @) d. B
    栈是从高到低分配,堆是从低到高分配。) W( I% @. P; V; w  p: \
普通单片机与STM32单片机中堆栈的区别
/ ?% J  {) ?% Y1 I. K; X8 }' q    普通单片机启动时,不需要用bootloader将数据 从ROM搬移到RAM。
    但是STM32单片机需要。
    这里我们可以先看看单片机程序执行的过程,单片机执行分三个步骤:
  • 取执行
  • 分析指令
  • 执行指令
    # C% ~( n  F' W0 h: z
    根据PC的值从程序存储器读出指令,送到指令寄存器。然后分析执行执行。这样单片机就从内部程序存储器去代码指令,从RAM存取相关数据。
    RAM取数的速度是远高于ROM的,但是普通单片机因为本身运行频率不高,所以从ROM取指令慢并不影响。
    而STM32的CPU运行的频率高,远大于从ROM读写的速度。所以需要用bootloader将数据 从ROM搬移到RAM。
    使用栈就象我们去饭馆里吃饭,只管点菜(发出申请)、付钱、和吃(使用),吃饱了就走,不必理会切菜、洗菜等准备工作和洗碗、刷锅等扫尾工作,他的好处是快捷,但是自由度小。使用堆就象是自己动手做喜欢吃的菜肴,比较麻烦,但是比较符合自己的口味,而且自由度大。
    其实堆栈就是单片机中的一些存储单元,这些存储单元被指定保存一些特殊信息,比如地址(保护断点)和数据(保护现场)。
    如果非要给他加几个特点的话那就是:
  • 这些存储单元中的内容都是程序执行过程中被中断打断时,事故现场的一些相关参数。如果不保存这些参数,单片机执行完中断函数后就无法回到主程序继续执行了。
  • 这些存储单元的地址被记在了一个叫做堆栈指针(SP)的地方。

    ) [: q9 z% e2 c$ s* o' [" c
结合STM32的开发讲述堆栈
    从上面的描述可以看得出来,在代码中是如何占用堆和栈的。可能很多人还是无法理解,这里再结合STM32的开发过程中与堆栈相关的内容来进行讲述。
    如何设置STM32的堆栈大小?
    在基于MDK的启动文件开始,有一段汇编代码是分配堆栈大小的。
微信图片_20201031193949.png
    这里重点知道堆栈数值大小就行。还有一段AREA(区域),表示分配一段堆栈数据段。数值大小可以自己修改,也可以使用STM32CubeMX数值大小配置,如下图所示。
微信图片_20201031193953.png
    在IAR中,是通过工程配置堆栈大小,如下图所示。
微信图片_20201031193957.png
    STM32F1默认设置值0x400,也就是1K大小。8 G/ s! {7 F# Q2 J' A
  • 6 X. `- |* S$ {  Q/ l3 e& p
Stack_Size EQU 0x400
    函数体内局部变量:

  • $ Q' w# \7 f$ m! S4 M( a) L. A
void Fun(void){ char i; int Tmp[256]; //...}
    局部变量总共占用了256*4 + 1字节的栈空间。所以,在函数内有较多局部变量时,就需要注意是否超过我们配置的堆栈大小。
    函数参数:8 M5 ~6 R4 E" r. ?7 m4 a
  • , o; ^$ C" z4 S$ s$ |
void HAL_GPIO_Init(GPIO_TypeDef *GPIOx, GPIO_InitTypeDef *GPIO_Init)
    这里要强调一点:传递指针只占4字节,如果传递的是结构体,就会占用结构大小空间。提示:在函数嵌套,递归时,系统仍会占用栈空间。
    堆(Heap)的默认设置0x200(512)字节。4 U! B0 ^0 |( w1 P! c; e2 }1 C* A
  • 2 A9 B! J- i3 @3 a
Heap_Size EQU 0x200
    大部分人应该很少使用malloc来分配堆空间。虽然堆上的数据只要程序员不释放空间就可以一直访问,但是,如果忘记了释放堆内存,那么将会造成内存泄漏,甚至致命的潜在错误。
MDK中RAM占用大小分析
    经常在线调试的人,可能会分析一些底层的内容。这里结合MDK-ARM来分析一下RAM占用大小的问题。在MDK编译之后,会有一段RAM大小信息:
微信图片_20201031194001.png
    这里4+6=1640,转换成16进制就是0x668,在进行在调试时,会出现:
2 Z( Y) }  J+ `" j2 k
微信图片_20201031194004.png
    这个MSP就是主堆栈指针,一般我们复位之后指向的位置,复位执向的其实是栈顶:
微信图片_20201031194007.png
    而MSP指向地址0x20000668是0x20000000偏移0x668而得来。具体哪些地方占用了RAM,可以参看map文件中【Image Symbol Table】处的内容:
微信图片_20201031194011.png
6 R7 [( D) \" J5 e" ~* H1 [  \5 S
1 v) F; {# E- b$ k5 H

4 u* _, R, `" N" P  @4 O
3 f6 o3 s* h2 P
收藏 评论0 发布时间:2020-10-31 19:41

举报

0个回答

所属标签

相似分享

关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版