STM32
直播中

王涛

7年用户 1221经验值
私信 关注
[问答]

MCU上电后又是如何寻找到入口地址的呢

MCU上电后又是如何寻找到入口地址的呢?
如何去判断STM32芯片的启动文件大小呢?

回帖(1)

李凌平

2021-11-30 15:43:37
   在嵌入式应用程序开发过程里,由于使用C语言编程,基本很少涉及到机器底层寄存器的执行过程,一般都会直接在main函数里开始写代码,似乎main成为了理所当然的起点,尽管从C程序的角度来看程序都是直接从main函数开始执行。然而,MCU上电后,是如何寻找到并执行main函数这一问题却很自然的被忽略了!事实上微控制器是无法从硬件上去定位main函数的入口地址,因为使用c语言作为开发语言后,变量/函数的地址便由编译器在编译时自行分配,因此main函数的入口地址在编译后便不一定是一个绝对地址。MCU上电后又是如何寻找到这个入口地址呢?以前接触无论是PIC、AVR、MSP430或是51过程中都没涉及到启动文件的配置,仅仅只有熔丝位或配置字是需要根据实际使用配置来设置,其实并非没有,而是由于大部分的开发环境往往自动完整地提供了这个启动文件,不需要开发人员再行干预启动过程,只需要从main函数开始进行应用程序的设计即可。然而,但接触到嵌入内核比如Linux系统移植过程“bootloader”却是很重要也是必不可少的一个环节。事实上,每一种微控制器,无论性能高下,结构简繁,价格贵贱都是必须有启动文件才能正常工作的,它的作用同“bootloader”类似。启动文件完成了微控制器从“复位”到“开始执行main函数”中间这段时间的必要启动配置。
       在STM32中,如果是在MDK下创建一个工程,一般都有提示是否加入Star up Code文件,这个就是启动文件,这里有个误区,一般对于初学者来看,很容易误以为STM32F10x.s这个启动文件是STM32所有类型芯片的通用启动文件,因此也自然不会去理会它的作用,事实上,这个启动文件只是针对部分STM32系列,如果仔细看过它的启动代码就会发现里面很多中断函数定义是没有的,甚至有些和STM32F10x_it.c里的函数是有出路的,如果刚好用到了默认的这个中断服务子函数的话,程序一旦运行到了中断是找不到入口地址的,这样就会莫名其妙地不知问题所在。STM32F10x.s是MDK提供的启动代码,从其里面的内容看来,它只定义了3个串口,4个定时器。实际上STM32的系列产品有5个串口的型号,也只有有2个串口的型号,定时器也是,做多的有8个定时器。比如,如果你用的STM32F103ZET6,而启动文件用的是STM32F10x.s的话,你可以正常使用串口1~3的中断,而串口4和5的中断,则无法正常使用。所以STM32F10x.s并不能适用所有的STM32型号,对于不同型号的STM32,正确做法是选择不同的启动文件。ST公司提供了3个启动文件:startup_stm32f10x_ld.s
/startup_stm32f10x_md.s/startup_stm32f10x_hd.s 分别适用于小容量/中容量/大容量的STM32芯片,具体判断方法如下:
                       小容量:FLASH≤32K
                     中容量:64K≤FLASH≤128K
                     大容量:256K≤FLASH
在启动代码中,补充几点:
  启动代码中的两条语句解释:
  一、PROC 为子程序开始,ENDP 为子程序结束
  二、[weak] 的意思是该函数优先级比较弱,如果其它地方定义了一个同名函数,那么此处的这个函数就被取代了。语法格式为 EXPORT 标号 {[WEAK]} 。EXPORT 可用GLOBAL代替。
  对于_main函数的理解:
  事实上,_main 和main是两个完全不同的函数!_main代码是编译器自动创建的,因此无法找到_main代码。MDK文档中有一句说明:it is automatically craated by the linker when it sees a definition of main() .大体意思可以理解为:当编译器发现定义了main函数,那么就会自动创建_main.
  _main 和main的关系
  _main 主要做两件事:其一,C所需的资源;其二,调用main函数。这就不难理解为什么在启动代码调用的是_main ,最后却能转到main函数中去执行的原因了。
   
  AREA指令的理解
  AREA指令是一个伪指令,用于段定义。ARM汇编程序由段组成,段是相对独立的指令或数据单位,每个段由AREA伪指令定义,并定义段的属性。
  AREA参数说明:
  *  STACK——AREA指令的一个参数,定义段名称
  *  NOINIT——AREA指令的一个参数,指定本数据段仅仅保留了内在单元,而将句初始值写入内存单元,此时内存单元值初始化为0
  *  READWRITE——指定本段为可读可写,数据段默认为READWRITE.
  READWRITE(读写)、READONLY(只读)
  *  ALIGN——也是一个伪指令,指定对齐方式。ALIGN n 指令的对齐值有两种选择:n或者2^n
  例子:开辟一个堆栈段,段名为STACK,定义为可读可写,将内存单元初始化为0,对齐方式为8字节对齐。
  AREA STACK,NOINIT,READWRITE,ALIGN=3
举报

更多回帖

发帖
×
20
完善资料,
赚取积分