当前位置: 首页 > ds >正文

STM32 __main汇编分析

在STM32的启动流程中,__main是一个由编译器自动生成的C标准库函数,其汇编级调用逻辑可通过启动文件(如startup_stm32fxxx.s)观察到,但具体实现细节被封装在编译器的运行时库中。以下是其核心逻辑解析:


一、__main的汇编级调用方式

在STM32的启动文件中,__main的调用流程如下(以Cortex-M系列为例):

Reset_Handler PROC; 1. 调用系统初始化函数LDR R0, =SystemInit    ; 加载SystemInit函数地址到R0BLX R0                ; 跳转到SystemInit执行; 2. 准备进入__main函数LDR R0, =__main       ; 加载__main函数地址到R0BX R0                 ; 跳转到__main执行
ENDP

这段代码表明:

  1. 复位处理程序​(Reset_Handler)首先调用SystemInit函数完成时钟配置等硬件初始化;
  2. 通过LDR指令将__main的地址加载到寄存器R0;
  3. BX R0指令实现跳转,进入__main的执行流程。

二、__main的内部行为(编译器实现)

虽然无法直接查看__main的源码,但其核心功能可通过反汇编和调试观察:

  1. 初始化数据段(.data)​
    将Flash中的已初始化全局变量拷贝到RAM中:

    LDR R0, =sdata         ; Flash中.data段的起始地址
    LDR R1, =_sidata       ; RAM中.data段的起始地址
    LDR R2, =_edata        ; RAM中.data段的结束地址
    copy_loop:CMP R1, R2         ; 检查是否完成拷贝BGE copy_doneLDR R3, [R0], #4   ; 从Flash加载4字节到R3STR R3, [R1], #4   ; 将R3内容存入RAMB copy_loop
    copy_done:
  2. 清零未初始化数据段(.bss)​
    将未初始化的全局变量内存区域置零:

    LDR R0, =_sbss         ; .bss段起始地址
    LDR R1, =_ebss         ; .bss段结束地址
    MOV R2, #0             ; 清零寄存器
    zero_loop:CMP R0, R1BGE zero_doneSTR R2, [R0], #4   ; 写入4字节0B zero_loop
    zero_done:
  3. 初始化堆栈指针
    根据启动文件中定义的Stack_SizeHeap_Size配置堆栈指针(MSP/PSP)。

  4. 跳转至用户main()函数
    通过BL main指令进入用户编写的C语言主函数。


三、调试观察__main的执行流程

在调试器中(如STM32CubeIDE):

  1. 反汇编窗口
    单步调试时,可观察到程序从Reset_HandlerSystemInit__mainmain()的跳转过程。

  2. 内存窗口验证

    • 查看0x20000000(RAM起始地址)附近的数据变化,确认.data段已正确初始化;
    • 检查.bss段内存是否被清零。

四、注意事项

  1. ​不可直接修改__main
    用户无法修改__main的实现,否则会导致C运行时环境初始化失败。

  2. 优化等级影响
    若编译器优化等级过高(如-O2),可能导致部分初始化逻辑被优化,需设置为-O0调试。


总结

__main的汇编级调用在启动文件中表现为简单的地址跳转(LDR+BX),但其内部逻辑由编译器自动生成,负责初始化C程序的运行时环境。通过调试器反汇编和内存观察,可间接验证其行为逻辑。

http://www.xdnf.cn/news/5986.html

相关文章:

  • 基于 mathematical-expression 的动态数学方程求解器实现
  • 如何使用C51的Timer0实现定时功能
  • DeepBook 与 CEX 的不同
  • HDFS的客户端操作(2)文件上传
  • 重庆医科大学附属第二医院外科楼外挡墙自动化监测
  • 鸿蒙5.0项目开发——鸿蒙天气项目的实现(介绍)
  • OJ判题系统第6期之判题逻辑开发——设计思路、实现步骤、代码实现(策略模式)
  • 目标检测任务 - 数据增强
  • Linux文件编程——标准库函数(fopen等)和系统调用函数(open等)的区别
  • 无锡哲讯科技:引领芯片封装SAP系统的智能化革命
  • Flannel Host-gw模式的优缺点
  • Leetcode 3544. Subtree Inversion Sum
  • 深入学习 Java 泛型实现方式:擦除法!
  • 43、Server.UrlEncode、HttpUtility.UrlDecode的区别?
  • 物理:篮球为什么能被拍起来?
  • .Net HttpClient 使用Json数据
  • Centos7安装部署wordpress个人博客保姆级教程
  • iVX 研发基座:大型系统开发的协作与安全架构实践
  • 基于MATLAB的生物量数据拟合模型研究
  • 云蝠智能大模型呼叫优势:技术驱动全链路升级,重塑智能交互服务新体验
  • 前端性能优化3:深入分析 Web Worker 和 Service Worker
  • 【源码+文档+调试讲解】驾校报名小程序2
  • python打卡day24
  • ppy/osu构建
  • window 显示驱动开发-创建分配时指定段
  • 块设备代码分析
  • 测试集群的功能-执行wordcount程序
  • uniapp|实现获取手机摄像头权限,调用相机拍照实现人脸识别相似度对比,拍照保存至相册,多端兼容(APP/微信小程序)
  • 什么情况会导致JVM退出?
  • 【机器学习赋能的智能光子学器件系统研究与应用】