ARM:什么是满减栈?为何选择满减栈?

栈的4种方式(以下以满减为主)

1. 满减栈的定义

  • 满(Full) :栈指针SP始终指向最后一个压入栈的有效数据(即栈顶元素)。

  • 减(Descending) :栈向内存低地址方向增长(栈空间从高地址向低地址扩展)。


2. 压栈操作

当数据被压入栈时:

  1. 先递减SP:SP的值先减去操作所需的内存大小(例如,32位系统每次减4字节)。

  2. 再存储数据:将数据写入SP指向的新地址。


3. 弹栈操作

当数据从栈中弹出时:

  1. 先读取数据:从SP当前指向的地址加载数据。

  2. 再递增SP:SP的值加上相应的内存大小。


4. 硬件指令的支持

ARM通过以下两类指令显式支持满减栈:

a. 多寄存器操作指令
  • 压栈STMDB SP!(Store Multiple Decrement Before)

    复制代码

    STMDB SP!, {R0-R3} ; SP -= 16 → 依次存储R3, R2, R1, R0到新地址

    • STMDB(递减存储)会在存储前递减SP,符合"满减"规则。

    • !表示更新SP的值。

  • 弹栈LDMIA SP!(Load Multiple Increment After)

    复制代码

    LDMIA SP!, {R0-R3} ; 依次加载R0, R1, R2, R3 → SP += 16

    • LDMIA(递增加载)在加载后递增SP,符合"先读后增"逻辑。
b. 专用栈指令(Thumb模式)

在Thumb-2指令集中,PUSHPOP指令进一步简化操作:

复制代码

PUSH {R0, R1} ; SP -= 8 → 存储R1, R0到[SP]和[SP+4] POP {R0, R1} ; 加载[SP]到R0, [SP+4]到R1 → SP += 8


5. 为何选择满减栈?

  • ARM ABI规范 :AAPCS(ARM Architecture Procedure Call Standard)规定使用满减栈作为标准调用约定。

  • 硬件效率:递减存储(STMDB)和递增加载(LDMIA)的组合可以高效处理连续内存操作,且与CPU流水线优化兼容。

  • 自然向下扩展:栈从高地址向低地址增长,与堆(Heap)从低地址向高地址扩展形成互补,避免内存区域重叠。

  • 调试友好:栈溢出时,SP指向低地址的非法区域,便于调试工具检测(如MMU异常)。


6. 实际内存变化示例

假设初始时 SP = 0x2000_1000

  1. 压栈R0(值=0x1234)

    1. SP -= 4SP = 0x2000_0FFC

    2. 存储 0x1234 到地址 0x2000_0FFC

  2. 再次压栈R1(值=0x5678)

    1. SP -= 4SP = 0x2000_0FF8

    2. 存储 0x5678 到地址 0x2000_0FF8

  3. 弹栈到R2

    1. 读取 0x2000_0FF8R2 = 0x5678

    2. SP += 4SP = 0x2000_0FFC


总结

ARM通过硬件指令的寻址模式 (如STMDB/LDMIASTMFD/LDMFD)强制SP在压栈时先减后存 、弹栈时先取后增,从而实现了"满减"栈。这种机制由架构直接支持,确保了栈操作的原子性和高效性,同时符合ARM标准调用约定(AAPCS)。


额外注意:

一般来说栈(stack)从高地址向低地址延申,堆(heap)从低地址向高地址延申,他们从两端向中间生长。

相关推荐
猫猫的小茶馆31 分钟前
【ARM】eclipse 中创建汇编工程
c语言·汇编·arm开发·stm32·单片机·嵌入式硬件·eclipse
猫猫的小茶馆14 小时前
【ARM】ARM的介绍
c语言·开发语言·arm开发·stm32·单片机·嵌入式硬件·物联网
森G17 小时前
五、Linux字符设备驱动
linux·arm开发·c++·ubuntu
贝塔实验室18 小时前
新手如何使用Altium Designer创建第一张原理图(三)
arm开发·单片机·嵌入式硬件·fpga开发·射频工程·基带工程·嵌入式实时数据库
小尧嵌入式19 小时前
QT软件开发知识点流程及图片转换工具的开发
开发语言·arm开发·qt
贝塔实验室20 小时前
Altium Designer全局编辑
arm开发·经验分享·笔记·fpga开发·dsp开发·射频工程·基带工程
dian2008-ic20 小时前
Linux pcie【9】基于GIC-V3 ITS实现pcie msi中断
linux·arm开发·驱动开发·嵌入式硬件
大大菜鸟一枚1 天前
ARM交叉编译环境配置与Qt依赖库部署指南
开发语言·arm开发·qt
CinzWS1 天前
TrustZone-M的设计哲学
arm开发·arm·architecture
macken99991 天前
centos7arm版本镜像下载存档
arm开发·x86_64开发