ARM CM3核 压栈流程

STM32F103 使用 ARM Cortex-M3 内核,与 STM32F013(Cortex-M0)相比,其压栈行为有所不同,主要体现在异常自动压栈和**手动压栈(函数调用)**两方面。


1. 进入异常/中断时的自动压栈

当 STM32F103 发生 中断或异常 时,Cortex-M3 内核会 自动 将以下 8 个寄存器 按固定顺序 压入栈中:

寄存器 说明
R0 通用寄存器 R0 (函数参数)
R1 通用寄存器 R1
R2 通用寄存器 R2
R3 通用寄存器 R3
R12 通用寄存器 R12
LR (R14) 链接寄存器(异常返回地址)
PC 发生异常时的程序计数器
xPSR 程序状态寄存器

压栈顺序(从低地址到高地址)

plaintext 复制代码
低地址
  ↓
[ xPSR  ]  --> 最高位 (最先压入栈)
[   PC   ]
[   LR   ]
[  R12   ]
[  R3    ]
[  R2    ]
[  R1    ]
[  R0    ]  --> 最低位 (最后压入栈)
  ↑
高地址

说明

  • 进入中断/异常时,处理器自动保存这些寄存器,以便中断返回时恢复现场。
  • PC 记录异常发生时的地址,返回时用于恢复执行。
  • xPSR 记录状态信息(如中断号、Thumb 状态等)。
  • 退出中断时,处理器会自动弹出这些寄存器并恢复 CPU 状态。

比 Cortex-M0 额外支持

Cortex-M3 支持硬件堆栈对齐 (8 字节对齐),同时具备 可选的浮点扩展(如果有 FPU) ,但 STM32F103 没有 FPU,因此不会涉及浮点寄存器的压栈。


2. 函数调用时的手动压栈

当 C 语言函数调用时,ARM 编译器根据 ARM AAPCS(ARM 过程调用标准),会自动进行手动压栈:

  1. 函数参数传递

    • 前 4 个参数 传递给 R0-R3,超过 4 个的参数需要压入栈

    • 例如:

      c 复制代码
      void test(int a, int b, int c, int d, int e);  // e 需要压栈
  2. 局部变量存放

    • 若寄存器不够用,局部变量会被压入栈中。
  3. 保护非易失性寄存器

    • 非易失性寄存器 (R4-R11LR) 必须在函数调用前保存,否则返回时数据会丢失。
    • 编译器可能会在函数入口 push 这些寄存器,函数结束时 pop 恢复。

示例

assembly 复制代码
push {r4-r7, lr}   ; 进入函数时,保存 r4-r7 和 lr
...
pop {r4-r7, lr}    ; 离开函数时,恢复寄存器
bx lr              ; 返回调用方

说明

  • push {r4-r7, lr}:手动压栈,保护 R4-R7LR(返回地址)。
  • pop {r4-r7, lr}:弹出栈,恢复寄存器。
  • bx lr:返回调用方。

3. STM32F103(Cortex-M3) vs. STM32F013(Cortex-M0)

对比项 STM32F013(Cortex-M0) STM32F103(Cortex-M3)
自动压栈寄存器 R0-R3, R12, LR, PC, xPSR R0-R3, R12, LR, PC, xPSR
异常返回自动弹栈
支持 8 字节堆栈对齐
支持可选浮点寄存器压栈 ❌(无 FPU)
函数调用手动压栈 R4-R7, LR R4-R11, LR(更多可用寄存器)

总结

  1. 异常/中断发生时,Cortex-M3 会自动压栈

    • R0-R3, R12, LR, PC, xPSR 这 8 个寄存器。
    • 返回时自动弹栈恢复。
  2. 普通函数调用时,手动压栈

    • push {r4-r7, lr} 保护调用者的寄存器。
    • 超过 4 个参数的函数参数和局部变量会存入栈中。
  3. 与 Cortex-M0(STM32F013)相比

    • Cortex-M3 支持更多的寄存器 (R4-R11),函数调用时可能会压栈更多寄存器。
    • Cortex-M3 支持 8 字节对齐,优化堆栈访问效率。

这样,STM32F103 在不同场景下的压栈行为就清楚了!如果你有更具体的应用场景,可以进一步探讨 😃

相关推荐
KeLin&1 小时前
讯为iTOP4412-Qt5.7环境搭建
开发语言·arm开发·qt·arm
切糕师学AI1 天前
ARM 应用程序状态寄存器APSR的整数状态(NZCV)位
arm开发·寄存器·apsr
早日退休!!!2 天前
ARM Cortex-M核 【保存上下文&恢复上下文】
arm开发·单片机·嵌入式硬件
早日退休!!!2 天前
ARM A核、ARM M核、X86与RISC-V架构:寄存器作用及上下文处理差异报告
arm开发·架构·risc-v
A-花开堪折2 天前
Qemu-NUC980(十一):SPI Controller
linux·arm开发·驱动开发·嵌入式硬件
Q_21932764553 天前
基于ARM的嵌入式温度控制系统的设计
arm开发
想你依然心痛3 天前
从x86到ARM的HPC之旅:鲲鹏开发工具链(编译器+数学库+MPI)上手与实战
java·开发语言·arm开发·鲲鹏·昇腾
MUTA️3 天前
x86 架构下运行 ARM-ROS2 Docker 镜像操作指南
arm开发·docker·架构
比奇堡派星星4 天前
Linux 杂项设备驱动框架详解
linux·arm开发·驱动开发
xingzhemengyou14 天前
LINUX 通过/proc/interrupts了解系统硬件中断的使用情况
linux·arm开发