datasheet

STM32与ARM启动代码比较分析

2019-03-24来源: eefocus关键字:STM32  ARM  启动代码

从ARM转到STM开发,开发工具也由ADS转到了Keil。借助STM的固件库,使得开发效率更加高效,比如你可以不用关心启动代码的具体实现,只需要专注于具体的应用代码,嵌入式开发也变得越来越“傻瓜”。此事好坏,暂且不论,来看看STM启动代码的特点,或者说相对于ARM的区别。


通常的启动代码结构:


1.  首先是中断向量表的定义.


Ø         ARM


ARM代码在这块的代码为跳转语句,因为指令长度的限制,4个字节也就能放个跳转语就差不多了。通常两种实现方式:


1.       B   Reset_Handler


2.       LDR PC, Reset_Handler


其实都是一个意思,跳转到真正实现Reset_Handler功能的地方去。ARM中断向量在这里总共有8条(复位、未定义、SWI、指令、数据异常、预留、IRQ、FIQ),具体的当前中断类型,在IRQ或FIQ的中断实现里面判断,之后再转到对应的中断处理函数里面。


注意,仔细看,想一想,这里的中断向量处存放的是机器指令码。然而,STM在中断向量处存放的是实现中断功能的入口地址,而不是指令功能码。


Ø         STM


正如上面所说,STM中断向量处存放的是目标地址。但是要注意的是,第一条中断向量存放的堆栈的地址,真正的传统意义上的中断向量从第二条开始。除此之外,STM的中断向量表很长,它不像ARM由IRQ或FIQ进行判断后再处理,而是将所有的中断处理函数入口地址全列在这里:


__Vectors      DCD     __initial_sp          ; Top of Stack


                DCD     Reset_Handler         ; Reset Handler


                DCD     NMI_Handler           ; NMI Handler


                DCD     HardFault_Handler     ; Hard Fault Handler


                DCD     MemManage_Handler     ; MPU Fault Handler


                DCD     BusFault_Handler      ; Bus Fault Handler


                DCD     UsageFault_Handler    ; Usage Fault Handler


                DCD     0                       ; Reserved


                DCD     0                       ; Reserved


                DCD     0                      ; Reserved


                DCD     0                      ; Reserved


                DCD     SVC_Handler           ; SVCall Handler


                DCD     DebugMon_Handler     ; Debug Monitor Handler


                DCD     0                      ; Reserved


                DCD     PendSV_Handler       ; PendSV Handler


                DCD     SysTick_Handler      ; SysTick Handler


 


                ; External Interrupts


                DCD     WWDG_IRQHandler      ; Window Watchdog


                DCD     PVD_IRQHandler       ; PVD through EXTI Line detect


                DCD     TAMPER_IRQHandler    ; Tamper


                …………….


 


 


2.  中断函数的跳转实现


这块功能的实现依赖于编译器、链接器的功能,实现方法各不相同。


Ø         ARM


        CODE32


                AREA    Startup,CODE,READONLY


Vectors


LDR     PC, ResetAddr


              LDR     PC, UndefinedAddr


              LDR     PC, SWI_Addr


              LDR     PC, PrefetchAddr


              LDR     PC, DataAbortAddr


              B            .


              LDR       PC, IRQ_Addr ;跳转至标号IRQ_Addr处


              LDR     PC, FIQ_Addr


 


ResetAddr              DCD     Reset


UndefinedAddr      DCD     Undefined


SWI_Addr               DCD     SoftwareInterrupt


PrefetchAddr         DCD     PrefetchAbort


DataAbortAddr      DCD     DataAbort


Nouse                    DCD     0


IRQ_Addr              DCD     IRQ_Handler ; IRQ_Addr定义为IRQ_Handler地址


FIQ_Addr               DCD     FIQ_Handler


 


; IRQ_Handler在这里定义


IRQ_Handler


           SUB SP, SP, #4


           STMFD SP!, {R8-R9}   


           LDR R9, =INTOFFSET


           LDR R9, [R9]


           LDR       R8, =HandleEINT0


           ADD R8, R8,R9,LSL #2


           LDR R8, [R8]


           STR R8, [SP,#8]


           LDMFD SP!,{R8-R9,PC}


 


注意上面的HandleEINT0标号,它是中断函数的入口首地址,加上当前中断编号的偏移值INTOFFSET。具体对应到哪里呢?看下面:


;这是定义(或者说预留)一个段指定位置开始的内存空间.


         MAP (0x33FFBF00)      


SysRstVector    #     4    


UdfInsVector     #     4    


SwiSvcVector    #     4


InsAbtVector     #     4


DatAbtVector     #     4


ReservedVector       #     4


IrqSvcVector      #     4


FiqSvcVector     #     4


 


HandleEINT0        #   4


HandleEINT1          #   4


HandleEINT2          #   4


HandleEINT3          #   4


HandleEINT4_7     #   4


….


实际上这里也可以理解为定义一个结构体变量,各个标号对应结构体的域,跟C语言不同的是,这里定义的结构体变量可以指定它在内存空间中的地址。


好了,如果当前来了一个IRQ类型的EINT3中断,按照上面的代码应该是跳转至以HandleEINT3这个域存储的值为地址处。那么HandleEINT3这个域里存储的值是什么呢?


下面的代码即可在C语言中定义了。


#define _ISR_STARTADDRESS   0x33FFBF00


#define pISR_EINT3     (*(unsigned *)(_ISR_STARTADDRESS+0x2c))


pISR_EINT3 = (unsigned int)EINT3_Handler; 


static void __irq EINT3_Handler(void)


{


}


 


Ø         STM32


STM32中断处理实现跟ARM不一样。来看代码:


启动代码处的中断向量表(我们以EXTI0为例):


__Vectors      DCD     __initial_sp              ; Top of Stack


                DCD     Reset_Handler           ; Reset Handler


                DCD     NMI_Handler             ; NMI Handler


                DCD     HardFault_Handler       ; Hard Fault Handler


                DCD     MemManage_Handler    ; MPU Fault Handler


                DCD     BusFault_Handler        ; Bus Fault Handler


                DCD     UsageFault_Handler      ; Usage Fault Handler


                DCD     0                         ; Reserved


                DCD     0                         ; Reserved


                DCD     0                         ; Reserved


                DCD     0                         ; Reserved


                DCD     SVC_Handler             ; SVCall Handler


                DCD     DebugMon_Handler       ; Debug Monitor Handler


                DCD     0                         ; Reserved


                DCD     PendSV_Handler          ; PendSV Handler


                DCD     SysTick_Handler          ; SysTick Handler


 


                ; External Interrupts


                DCD     WWDG_IRQHandler        ; Window Watchdog


                DCD     PVD_IRQHandler           ; PVD through EXTI Line detect


                DCD     TAMPER_IRQHandler       ; Tamper


                DCD     RTC_IRQHandler           ; RTC


                DCD     FLASH_IRQHandler         ; Flash


                DCD     RCC_IRQHandler           ; RCC


                DCD     EXTI0_IRQHandler      ; EXTI Line 0 中断发生时跳转至EXT0_IRQHandler地址处。@@@记住这条代码,下面以此处为例@@@


                       ….


 


Default_Handler PROC


                EXPORT  WWDG_IRQHandler           [WEAK]


                EXPORT  PVD_IRQHandler            [WEAK]


                EXPORT  TAMPER_IRQHandler         [WEAK]


                EXPORT  RTC_IRQHandler            [WEAK]


                EXPORT  FLASH_IRQHandler          [WEAK]


                EXPORT  RCC_IRQHandler            [WEAK]


                EXPORT  EXTI0_IRQHandler          [WEAK]


                       …..


WWDG_IRQHandler


PVD_IRQHandler


TAMPER_IRQHandler


RTC_IRQHandler


FLASH_IRQHandler


RCC_IRQHandler


EXTI0_IRQHandler


EXTI1_IRQHandler


….


       B   .


                ENDP


 


这段是啥意思呢?这里是定义各个中断向量的处理函数处,所有列出来的中断向量处理函数地址一致,功能也是一致:原地跳转。


既然所有的中断处理函数功能一致,那它是如何跳转至用户定义在C语言中的中断处理函数的呢?答案是,如果用户没有在用户代码(C语言)中定义对应向量的中断处理函数,则实际起作用的真正的中断处理函数即为上面列出的原地跳转功能处。


它是如何实现的? 注意到在声明导出处理函数后面的[WEAK]了吗?它的功能由链接器实现:如果在别处也定义该标号(函数),在链接时用别处的地址。如果没有其它定方定义,则以此处地址进行链接。


可能不太好理解,实际上是启动代码已经预定义了中断处理函数,它的功能很简单,就是原地跳转。只不过这块预定义的中断处理函数是否真正起作用,要看你是否在别处重定义了相同标号的中断处理函数。如果你已经重定义了,则以你重定义的中断处理函数为准。


以EXTI0中断为列,假设用户在自已的代码中配置好了EXTI0的中断,并且重定义了下面的EXTI0_IRQHandler函数,则链接器会以此函数地址进行链接。


void EXTI0_IRQHandler()


{


}


 


也就是在上面启动代码的@@@标注处(DCD  EXTI0_IRQHandler),会以用户重定义的EXTI0_IRQHandler()函数地址填入。


关键字:STM32  ARM  启动代码

编辑:什么鱼 引用地址:http://www.eeworld.com.cn/mcu/2019/ic-news032443573.html
本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:解决stm32进入HardFault_Handler的定位办法
下一篇:STM32-仿真调试时的SystemInit陷阱

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

STM32堆栈设置

1.堆和栈大小 定义大小在startup_stm32f2xx.sStack_Size      EQU     0x00000400                AREA    STACK, NOINIT, READWRITE, ALIGN=3Stack_Mem      
发表于 2019-04-16
STM32堆栈设置

STM32堆和栈(Heap & Stack)的资料理解

源起:在移植cjson的过程中,解析json包的时候发现动态内存分配不足而导致解析失败,为解决这一问题,而深入了解stm32的堆和栈。stm32的存储器结构。Flash,SRAM寄存器和输入输出端口被组织在同一个4GB的线性地址空间内。可访问的存储器空间被分成8个主要块,每个块为512MB。FLASH存储下载的程序。SRAM是存储运行程序中的数据。而SRAM一般分这几个部分:静态存储区:内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在。它主要存放静态数据、全局数据和常量。栈区:在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率
发表于 2019-04-16
STM32堆和栈(Heap & Stack)的资料理解

STM32定义堆栈地址到ram区顶部

本设置针对stm32f103rbt6的设置,该芯片RAM大小为20kB,故RAM区地址范围为0x20000000—0x20005000,芯片信息如下图所示;第一步:设置.sct文件;;*************************************************************; *** Scatter-Loading Description Filegenerated by uVision ***; *************************************************************LR_IROM1 0x08000000 0x00020000  
发表于 2019-04-16
STM32定义堆栈地址到ram区顶部

STM32之程序如何防止堆栈溢出

近日为某个项目写了个草稿程序,即非正式程序,后来发现老是进入hardfaulthandler,原来是堆栈溢出,后仔细查看发现函数调用纵深太深,最多的时候可保持7个函数在堆栈中调用。因此有心得如下:一、函数调用不要纵深太深,即以下模式:main(){   fun1();}fun1(){  fun2();}fun2(){   fun3();}fun3(){  fun4();}fun4(){  fun5();}fun5(){  fun6();}fun6(){   fun7();}这样子main函数要调用fun1函数完成某个功能,则要一直调到
发表于 2019-04-16

stm32之堆栈

stm32中的堆栈设置keil编译完成时存储情况当编译成功时,会出现: BUILD://Program Size: Code=340 RO-data=252 RW-data=0 ZI-data=1632Code:程序代码部分RO-data: 程序定义的常量const tempRW-data:已初始化的全局变量ZI-data:未初始化的全局变量片中的:flash=Code+RO-data+RW-dataRAM=RW-data+ZI-data通过上面的BUILD可以看出,这个程序已经用了1600多的RAM,为什么会出用到这么多的RAM呢?在startup_stm32f10x_md.s文件中存在:St
发表于 2019-04-16

说说STM32的堆栈与内存

1.概念这里所说的堆栈,是针对单片机所说的“堆”与“栈”,指的是内存中一片特殊用途的区域。而不是数据结构中的堆栈(虽然其实规则一样)。这里所说的内存,是指RAM,RAM包括SRAM,DRAM等。而不是什么手机内存卡之类。这里所说的flash,指的是用作为ROM的存储器,保存代码与常量数据。而不是动画制作。。。栈的生长方向:指的是入栈方向,从高地址向低地址生长叫做向下生长,或逆向生长;反过来就叫向上生长,或正向生长。STM32的栈是向下生长。2.内存中的堆栈安排确切地说,是keil mdk根据STM32的特性,对stm32的RAM甚至flash进行部署。编译工程后,在生成的.map文件里可以看到具体的安排。双击工程界面的工程根目录
发表于 2019-04-16
说说STM32的堆栈与内存

小广播

何立民专栏

单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2019 EEWORLD.com.cn, Inc. All rights reserved