历史上的今天
返回首页

历史上的今天

今天是:2025年03月24日(星期一)

正在发生

2019年03月24日 | STM32与ARM启动代码比较分析

2019-03-24 来源:eefocus

从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()函数地址填入。


推荐阅读

史海拾趣

E-Switch公司的发展小趣事

自1979年成立以来,E-Switch公司始终将创新作为发展的核心动力。在公司成立初期,面对激烈的市场竞争,E-Switch决定走差异化路线,专注于开发和生产高品质的开关产品。通过不断的技术创新和产品优化,E-Switch逐渐在电子行业中树立了自己的品牌形象。

CMOS Sensor Inc公司的发展小趣事

随着技术的不断发展,CMOS Sensor Inc公司不断推出创新产品。其中,线型CMOS图像传感器的研发成功,为公司带来了重要的突破。这一产品具有更高的分辨率和更低的噪声,为工业质量保证、商业成像等领域提供了更优质的解决方案。此后,公司又不断对产品进行升级和优化,以满足不同客户的需求。

AINFO Inc公司的发展小趣事

随着技术的不断进步,AINFO Inc公司不断推出创新产品,满足市场的多样化需求。公司注重产品的质量和性能,通过优化设计和生产工艺,提高了产品的竞争力。同时,公司积极开拓国内外市场,与多家知名企业建立了合作关系,实现了产品的广泛应用。

台湾美丽微(FMS)公司的发展小趣事

FTDI的历史可以追溯到1992年3月13日,由弗雷德·达特(Fred Dart)创立。公司初期专注于个人电脑市场,推出了针对主板芯片组的解决方案。其中,早期的主要客户之一是IBM,这些芯片被应用于AMBRA和PS/2系列的个人电脑中,为当时的计算机技术发展做出了重要贡献。这一时期的成功不仅奠定了FTDI的技术基础,也为其后续的市场拓展奠定了坚实的客户基础。

Chen Yang Technologies GmbH & Co KG公司的发展小趣事

随着业务的发展,Chen Yang Technologies意识到单靠自身力量难以实现快速扩张。因此,公司开始积极寻求与其他电子行业巨头的战略合作。通过与这些公司的合作,Chen Yang Technologies不仅获得了更多的资金支持和技术资源,还成功进入了新的市场领域。这些合作使得公司的业务范围得到了极大的拓展,同时也提升了公司的品牌影响力。

复旦微电子(FM)公司的发展小趣事

在快速发展的过程中,Chen Yang Technologies始终重视内部管理和人才培养。公司推行了一系列创新的管理理念和措施,如扁平化管理、项目制运作等,这些措施有效提高了工作效率和团队协作能力。同时,公司还注重人才培养和引进,通过设立激励机制、提供培训和发展机会等方式,吸引和留住了一批优秀的技术人才和管理人才。

问答坊 | AI 解惑

JZC-23F继电器

还有一个问题:这个继电器的用什么样的三极管来带动?上面有“10A/28VDC,5A/220VAC,DC5V“字样,怎样解释?它对吸合电流又没有要求?…

查看全部问答>

RS232 起始位 停止位 校验位 需要自己实现吗?

ModBus帧格式为:从机地址(1字节),功能码(1字节),起始地址(2字节),数据长度(2字节),CRC码(2字节) 然后数据传输方式规定为串行异步:1位起始位,8位数据位,1偶校验位,1位停止位。 我的疑问是: 1)CRC只对数据位进行校验,那就是 ...…

查看全部问答>

S3C44B0X板配什么LCD好?学习用的,价格一般

麻烦大家推荐一下,要是有卖的话直接卖给我吧,免得我东找西找,好辛苦~…

查看全部问答>

【求助】eVB读写文件时“语句未结束”?

源代码如下: Private Sub Form_Load()    Dim word, meaning As String        Open \"words.dat\" For Input Access Read As #1    Do While Not EOF(1)      Input #1, word ...…

查看全部问答>

【求助】有关无线通讯(实在没办法了)

我已经从21号弄到现在了,本来年前想弄一点效果出来,但这几天一直试却实在调不通,所以只好向大家求助了。 我计划做两块电路板之间的无线通讯,用的是149,通讯模块是prt2000。 电路版已做好,板内控制灯等元件都是灵的,但是两块板之间的控一直 ...…

查看全部问答>

【征文】51单片机从入门到……到刚入门

       嘿嘿,其实《51单片机从入门到精通》算是我的单片机启蒙书了,其实这个书名毫无新意,类似的简直是一抓一大把。但是我的经历告诉我,即便是再平凡的书,善加利用也会有很大帮助。就是这本在普通不过,甚至还有几处错误 ...…

查看全部问答>

综述嵌入式开发

 嵌入式系统无疑是当前最热门最有发展前途的IT应用领域之一,特别是随着消费家电的智能化,嵌入式更显重要。下面就由福州卓跃教育为大家做一个具体的介绍。        嵌入式系统是软硬结合的东西,福州嵌入式开 ...…

查看全部问答>

学模拟+PSRR和电源的关系

本帖最后由 dontium 于 2015-1-23 11:39 编辑 放大器具有高 PSRR,就不需担心电源变化? 链接deyisupport./blog/b/signalchain/archive/2013/12/16/psrr.aspx 本人模拟新手,看了这篇文章了解了PSRR和电源的关系,重点了解了规格书中写的DC PS是R ...…

查看全部问答>

52单片机T2的用法

众所周知,52单片机相比较51多了一个T2定时器,楼主最近搞的一个小车需要三个定时器,所以T2也派上了用场。貌似这个T2的资料比较少,我看了郭天祥老师的书里关于T2也讲了一些,可是不咋详细。关于T2定时器的程序我在网上找到了,经实验测试没有问题 ...…

查看全部问答>