当前位置:   article > 正文

STM32启动过程详解_stm32启动.s引脚

stm32启动.s引脚

一、STM32启动文件详细解析

STM32启动文件详细解析(V3.5.0) 以:startup_stm32f10x_hd.s为例

  1. ;******************** (C) COPYRIGHT 2011 STMicroelectronics ********************
  2. ;* File Name : startup_stm32f10x_hd.s
  3. ;* Author : MCD Application Team
  4. ;* Version : V3.5.0
  5. ;* Date : 11-March-2011
  6. ;* Description : STM32F10x High Density Devices vector table for MDK-ARM
  7. ;* toolchain.
  8. ;* This module performs:
  9. ;* - Set the initial SP
  10. ;* - Set the initial PC == Reset_Handler
  11. ;* - Set the vector table entries with the exceptions ISR address
  12. ;* - Configure the clock system and also configure the external
  13. ;* SRAM mounted on STM3210E-EVAL board to be used as data
  14. ;* memory (optional, to be enabled by user)
  15. ;* - Branches to __main in the C library (which eventually
  16. ;* calls main()).
  17. ;* After Reset the CortexM3 processor is in Thread mode,
  18. ;* priority is Privileged, and the Stack is set to Main.
  19. ;* <<< Use Configuration Wizard in Context Menu >>>
  20. ;*******************************************************************************
  21. ; THE PRESENT FIRMWARE WHICH IS FOR GUIDANCE ONLY AIMS AT PROVIDING CUSTOMERS
  22. ; WITH CODING INFORMATION REGARDING THEIR PRODUCTS IN ORDER FOR THEM TO SAVE TIME.
  23. ; AS A RESULT, STMICROELECTRONICS SHALL NOT BE HELD LIABLE FOR ANY DIRECT,
  24. ; INDIRECT OR CONSEQUENTIAL DAMAGES WITH RESPECT TO ANY CLAIMS ARISING FROM THE
  25. ; CONTENT OF SUCH FIRMWARE AND/OR THE USE MADE BY CUSTOMERS OF THE CODING
  26. ; INFORMATION CONTAINED HEREIN IN CONNECTION WITH THEIR PRODUCTS.
  27. ;*******************************************************************************
  28. ; Amount of memory (in bytes) allocated for Stack
  29. ; Tailor this value to your application needs
  30. ; <h> Stack Configuration ;栈定义
  31. ; <o> Stack Size (in Bytes) <0x0-0xFFFFFFFF:8>
  32. ; </h>
  33. Stack_Size EQU 0x00000400 ;EQU伪指令,作用是左边的符号名代表右边的表达式</span>
  34. AREA STACK, NOINIT, READWRITE, ALIGN=3 ;定义栈段:名称为STACK,未初始化,可读写,ELF 的栈段按2^3=8对齐
  35. Stack_Mem SPACE Stack_Size ;分配一片连续的存储区域并初始化为 0,栈空间:0x400个字节
  36. __initial_sp ;栈空间顶地址
  37. ; <h> Heap Configuration ;堆定义
  38. ; <o> Heap Size (in Bytes) <0x0-0xFFFFFFFF:8>
  39. ; </h>
  40. Heap_Size EQU 0x00000200
  41. AREA HEAP, NOINIT, READWRITE, ALIGN=3
  42. __heap_base ;堆空间起始地址
  43. Heap_Mem SPACE Heap_Size ;堆空间:0x200个字节
  44. __heap_limit ;堆空间结束地址
  45. PRESERVE8 ;PRESERVE8 指令指定当前文件保持堆栈八字节对齐
  46. THUMB ;告诉汇编器下面是32位的Thumb指令,如果需要汇编器将插入位以保证对齐
  47. ; Vector Table Mapped to Address 0 at Reset ;中断向量表定义
  48. ;实际上是在CODE区(假设STM32从FLASH启动,则此中断向量表起始地址即为0x8000000
  49. AREA RESET, DATA, READONLY ;定义一块数据段<DATA>,只可读<READONLY,默认READWRITE>,段名字是RESET
  50. EXPORT __Vectors ;EXPORT:在程序中声明一个全局的标号__Vectors,该标号可在其他的文件中引用
  51. EXPORT __Vectors_End ;在程序中声明一个全局的标号__Vectors_End
  52. EXPORT __Vectors_Size ;在程序中声明一个全局的标号__Vectors_Size
  53. ;DCD(DCDU)用于分配一片连续的字存储单元并用指定的数据初始化。
  54. __Vectors DCD __initial_sp ; Top of Stack ;该处物理地址值存储__initial_sp所表示的地址值,即为 __Vetors 标号所表示的值
  55. DCD Reset_Handler ; Reset Handler
  56. DCD NMI_Handler ; NMI Handler
  57. DCD HardFault_Handler ; Hard Fault Handler
  58. DCD MemManage_Handler ; MPU Fault Handler
  59. DCD BusFault_Handler ; Bus Fault Handler
  60. DCD UsageFault_Handler ; Usage Fault Handler
  61. DCD 0 ; Reserved
  62. DCD 0 ; Reserved
  63. DCD 0 ; Reserved
  64. DCD 0 ; Reserved
  65. DCD SVC_Handler ; SVCall Handler
  66. DCD DebugMon_Handler ; Debug Monitor Handler
  67. DCD 0 ; Reserved
  68. DCD PendSV_Handler ; PendSV Handler
  69. DCD SysTick_Handler ; SysTick Handler
  70. ; External Interrupts ;以下为外部中断向量表
  71. DCD WWDG_IRQHandler ; Window Watchdog
  72. DCD PVD_IRQHandler ; PVD through EXTI Line detect
  73. DCD TAMPER_IRQHandler ; Tamper
  74. DCD RTC_IRQHandler ; RTC
  75. DCD FLASH_IRQHandler ; Flash
  76. DCD RCC_IRQHandler ; RCC
  77. DCD EXTI0_IRQHandler ; EXTI Line 0
  78. DCD EXTI1_IRQHandler ; EXTI Line 1
  79. DCD EXTI2_IRQHandler ; EXTI Line 2
  80. DCD EXTI3_IRQHandler ; EXTI Line 3
  81. DCD EXTI4_IRQHandler ; EXTI Line 4
  82. DCD DMA1_Channel1_IRQHandler ; DMA1 Channel 1
  83. DCD DMA1_Channel2_IRQHandler ; DMA1 Channel 2
  84. DCD DMA1_Channel3_IRQHandler ; DMA1 Channel 3
  85. DCD DMA1_Channel4_IRQHandler ; DMA1 Channel 4
  86. DCD DMA1_Channel5_IRQHandler ; DMA1 Channel 5
  87. DCD DMA1_Channel6_IRQHandler ; DMA1 Channel 6
  88. DCD DMA1_Channel7_IRQHandler ; DMA1 Channel 7
  89. DCD ADC1_2_IRQHandler ; ADC1 & ADC2
  90. DCD USB_HP_CAN1_TX_IRQHandler ; USB High Priority or CAN1 TX
  91. DCD USB_LP_CAN1_RX0_IRQHandler ; USB Low Priority or CAN1 RX0
  92. DCD CAN1_RX1_IRQHandler ; CAN1 RX1
  93. DCD CAN1_SCE_IRQHandler ; CAN1 SCE
  94. DCD EXTI9_5_IRQHandler ; EXTI Line 9..5
  95. DCD TIM1_BRK_IRQHandler ; TIM1 Break
  96. DCD TIM1_UP_IRQHandler ; TIM1 Update
  97. DCD TIM1_TRG_COM_IRQHandler ; TIM1 Trigger and Commutation
  98. DCD TIM1_CC_IRQHandler ; TIM1 Capture Compare
  99. DCD TIM2_IRQHandler ; TIM2
  100. DCD TIM3_IRQHandler ; TIM3
  101. DCD TIM4_IRQHandler ; TIM4
  102. DCD I2C1_EV_IRQHandler ; I2C1 Event
  103. DCD I2C1_ER_IRQHandler ; I2C1 Error
  104. DCD I2C2_EV_IRQHandler ; I2C2 Event
  105. DCD I2C2_ER_IRQHandler ; I2C2 Error
  106. DCD SPI1_IRQHandler ; SPI1
  107. DCD SPI2_IRQHandler ; SPI2
  108. DCD USART1_IRQHandler ; USART1
  109. DCD USART2_IRQHandler ; USART2
  110. DCD USART3_IRQHandler ; USART3
  111. DCD EXTI15_10_IRQHandler ; EXTI Line 15..10
  112. DCD RTCAlarm_IRQHandler ; RTC Alarm through EXTI Line
  113. DCD USBWakeUp_IRQHandler ; USB Wakeup from suspend
  114. DCD TIM8_BRK_IRQHandler ; TIM8 Break
  115. DCD TIM8_UP_IRQHandler ; TIM8 Update
  116. DCD TIM8_TRG_COM_IRQHandler ; TIM8 Trigger and Commutation
  117. DCD TIM8_CC_IRQHandler ; TIM8 Capture Compare
  118. DCD ADC3_IRQHandler ; ADC3
  119. DCD FSMC_IRQHandler ; FSMC
  120. DCD SDIO_IRQHandler ; SDIO
  121. DCD TIM5_IRQHandler ; TIM5
  122. DCD SPI3_IRQHandler ; SPI3
  123. DCD UART4_IRQHandler ; UART4
  124. DCD UART5_IRQHandler ; UART5
  125. DCD TIM6_IRQHandler ; TIM6
  126. DCD TIM7_IRQHandler ; TIM7
  127. DCD DMA2_Channel1_IRQHandler ; DMA2 Channel1
  128. DCD DMA2_Channel2_IRQHandler ; DMA2 Channel2
  129. DCD DMA2_Channel3_IRQHandler ; DMA2 Channel3
  130. DCD DMA2_Channel4_5_IRQHandler ; DMA2 Channel4 & Channel5
  131. __Vectors_End ;Vectors结束
  132. __Vectors_Size EQU __Vectors_End - __Vectors ;得到向量表的大小,304个字节也就是0x130个字节
  133. AREA |.text|, CODE, READONLY ;定义一个代码段,可读,段名字是.text
  134. ;|.text| 用于表示由 C 编译程序产生的代码段,或用于以某种方式与 C 库关联的代码段
  135. ; Reset handler ;利用PROC、ENDP这一对伪指令标记程序开始、结束,把程序段分为若干个过程,使程序的结构加清晰
  136. Reset_Handler PROC
  137. EXPORT Reset_Handler [WEAK];WEAK声明其他的同名标号优先于该标号被引用,就是说如果外面声明了的话,调用外面的对应函数
  138. IMPORT __main ;IMPORT:伪指令用于通知编译器要使用的标号在其他的源文件中定义
  139. IMPORT SystemInit
  140. LDR R0, =SystemInit ;系统初始化
  141. BLX R0 ;带链接的跳转,切换指令集,跳到SystemInit
  142. LDR R0, =__main ;__main为运行时库提供的函数;完成堆栈,堆的初始化等工作,会调用下面定义的__user_initial_stackheap
  143. BX R0 ;切换指令集,main函数不返回跳到__main,进入C的世界
  144. ENDP
  145. ; Dummy Exception Handlers (infinite loops which can be modified)
  146. NMI_Handler PROC
  147. EXPORT NMI_Handler [WEAK];不可屏蔽中断处理函数
  148. B .
  149. ENDP
  150. HardFault_Handler\ ;\意为换行
  151. PROC
  152. EXPORT HardFault_Handler [WEAK];硬件错误处理函数
  153. B .
  154. ENDP
  155. MemManage_Handler\
  156. PROC
  157. EXPORT MemManage_Handler [WEAK]
  158. B .
  159. ENDP
  160. BusFault_Handler\
  161. PROC
  162. EXPORT BusFault_Handler [WEAK]
  163. B .
  164. ENDP
  165. UsageFault_Handler\
  166. PROC
  167. EXPORT UsageFault_Handler [WEAK]
  168. B .
  169. ENDP
  170. SVC_Handler PROC
  171. EXPORT SVC_Handler [WEAK]
  172. B .
  173. ENDP
  174. DebugMon_Handler\
  175. PROC
  176. EXPORT DebugMon_Handler [WEAK]
  177. B .
  178. ENDP
  179. PendSV_Handler PROC
  180. EXPORT PendSV_Handler [WEAK]
  181. B .
  182. ENDP
  183. SysTick_Handler PROC
  184. EXPORT SysTick_Handler [WEAK]
  185. B .
  186. ENDP
  187. Default_Handler PROC
  188. ;输出异常向量表标号,方便外部实现异常的具体功能,[WEAK]是弱定义的意思,如果外部定义了,优先执行外部定义,否则下面的函数定义
  189. EXPORT WWDG_IRQHandler [WEAK]
  190. EXPORT PVD_IRQHandler [WEAK]
  191. EXPORT TAMPER_IRQHandler [WEAK]
  192. EXPORT RTC_IRQHandler [WEAK]
  193. EXPORT FLASH_IRQHandler [WEAK]
  194. EXPORT RCC_IRQHandler [WEAK]
  195. EXPORT EXTI0_IRQHandler [WEAK]
  196. EXPORT EXTI1_IRQHandler [WEAK]
  197. EXPORT EXTI2_IRQHandler [WEAK]
  198. EXPORT EXTI3_IRQHandler [WEAK]
  199. EXPORT EXTI4_IRQHandler [WEAK]
  200. EXPORT DMA1_Channel1_IRQHandler [WEAK]
  201. EXPORT DMA1_Channel2_IRQHandler [WEAK]
  202. EXPORT DMA1_Channel3_IRQHandler [WEAK]
  203. EXPORT DMA1_Channel4_IRQHandler [WEAK]
  204. EXPORT DMA1_Channel5_IRQHandler [WEAK]
  205. EXPORT DMA1_Channel6_IRQHandler [WEAK]
  206. EXPORT DMA1_Channel7_IRQHandler [WEAK]
  207. EXPORT ADC1_2_IRQHandler [WEAK]
  208. EXPORT USB_HP_CAN1_TX_IRQHandler [WEAK]
  209. EXPORT USB_LP_CAN1_RX0_IRQHandler [WEAK]
  210. EXPORT CAN1_RX1_IRQHandler [WEAK]
  211. EXPORT CAN1_SCE_IRQHandler [WEAK]
  212. EXPORT EXTI9_5_IRQHandler [WEAK]
  213. EXPORT TIM1_BRK_IRQHandler [WEAK]
  214. EXPORT TIM1_UP_IRQHandler [WEAK]
  215. EXPORT TIM1_TRG_COM_IRQHandler [WEAK]
  216. EXPORT TIM1_CC_IRQHandler [WEAK]
  217. EXPORT TIM2_IRQHandler [WEAK]
  218. EXPORT TIM3_IRQHandler [WEAK]
  219. EXPORT TIM4_IRQHandler [WEAK]
  220. EXPORT I2C1_EV_IRQHandler [WEAK]
  221. EXPORT I2C1_ER_IRQHandler [WEAK]
  222. EXPORT I2C2_EV_IRQHandler [WEAK]
  223. EXPORT I2C2_ER_IRQHandler [WEAK]
  224. EXPORT SPI1_IRQHandler [WEAK]
  225. EXPORT SPI2_IRQHandler [WEAK]
  226. EXPORT USART1_IRQHandler [WEAK]
  227. EXPORT USART2_IRQHandler [WEAK]
  228. EXPORT USART3_IRQHandler [WEAK]
  229. EXPORT EXTI15_10_IRQHandler [WEAK]
  230. EXPORT RTCAlarm_IRQHandler [WEAK]
  231. EXPORT USBWakeUp_IRQHandler [WEAK]
  232. EXPORT TIM8_BRK_IRQHandler [WEAK]
  233. EXPORT TIM8_UP_IRQHandler [WEAK]
  234. EXPORT TIM8_TRG_COM_IRQHandler [WEAK]
  235. EXPORT TIM8_CC_IRQHandler [WEAK]
  236. EXPORT ADC3_IRQHandler [WEAK]
  237. EXPORT FSMC_IRQHandler [WEAK]
  238. EXPORT SDIO_IRQHandler [WEAK]
  239. EXPORT TIM5_IRQHandler [WEAK]
  240. EXPORT SPI3_IRQHandler [WEAK]
  241. EXPORT UART4_IRQHandler [WEAK]
  242. EXPORT UART5_IRQHandler [WEAK]
  243. EXPORT TIM6_IRQHandler [WEAK]
  244. EXPORT TIM7_IRQHandler [WEAK]
  245. EXPORT DMA2_Channel1_IRQHandler [WEAK]
  246. EXPORT DMA2_Channel2_IRQHandler [WEAK]
  247. EXPORT DMA2_Channel3_IRQHandler [WEAK]
  248. EXPORT DMA2_Channel4_5_IRQHandler [WEAK]
  249. ;如下只是定义一些空函数
  250. WWDG_IRQHandler
  251. PVD_IRQHandler
  252. TAMPER_IRQHandler
  253. RTC_IRQHandler
  254. FLASH_IRQHandler
  255. RCC_IRQHandler
  256. EXTI0_IRQHandler
  257. EXTI1_IRQHandler
  258. EXTI2_IRQHandler
  259. EXTI3_IRQHandler
  260. EXTI4_IRQHandler
  261. DMA1_Channel1_IRQHandler
  262. DMA1_Channel2_IRQHandler
  263. DMA1_Channel3_IRQHandler
  264. DMA1_Channel4_IRQHandler
  265. DMA1_Channel5_IRQHandler
  266. DMA1_Channel6_IRQHandler
  267. DMA1_Channel7_IRQHandler
  268. ADC1_2_IRQHandler
  269. USB_HP_CAN1_TX_IRQHandler
  270. USB_LP_CAN1_RX0_IRQHandler
  271. CAN1_RX1_IRQHandler
  272. CAN1_SCE_IRQHandler
  273. EXTI9_5_IRQHandler
  274. TIM1_BRK_IRQHandler
  275. TIM1_UP_IRQHandler
  276. TIM1_TRG_COM_IRQHandler
  277. TIM1_CC_IRQHandler
  278. TIM2_IRQHandler
  279. TIM3_IRQHandler
  280. TIM4_IRQHandler
  281. I2C1_EV_IRQHandler
  282. I2C1_ER_IRQHandler
  283. I2C2_EV_IRQHandler
  284. I2C2_ER_IRQHandler
  285. SPI1_IRQHandler
  286. SPI2_IRQHandler
  287. USART1_IRQHandler
  288. USART2_IRQHandler
  289. USART3_IRQHandler
  290. EXTI15_10_IRQHandler
  291. RTCAlarm_IRQHandler
  292. USBWakeUp_IRQHandler
  293. TIM8_BRK_IRQHandler
  294. TIM8_UP_IRQHandler
  295. TIM8_TRG_COM_IRQHandler
  296. TIM8_CC_IRQHandler
  297. ADC3_IRQHandler
  298. FSMC_IRQHandler
  299. SDIO_IRQHandler
  300. TIM5_IRQHandler
  301. SPI3_IRQHandler
  302. UART4_IRQHandler
  303. UART5_IRQHandler
  304. TIM6_IRQHandler
  305. TIM7_IRQHandler
  306. DMA2_Channel1_IRQHandler
  307. DMA2_Channel2_IRQHandler
  308. DMA2_Channel3_IRQHandler
  309. DMA2_Channel4_5_IRQHandler
  310. B .
  311. ENDP
  312. ALIGN ;默认是字对齐方式,也说明了代码是4字节对齐的
  313. ;*******************************************************************************
  314. ; User Stack and Heap initialization 用户堆栈初始化
  315. ;*******************************************************************************
  316. IF :DEF:__MICROLIB ;判断是否使用DEF:__MICROLIB(micro lib),如果勾选了micro lib
  317. EXPORT __initial_sp ;将栈顶地址、堆起始地址、堆结束地址赋予全局属性,使外部程序可用
  318. EXPORT __heap_base
  319. EXPORT __heap_limit
  320. ELSE ;如果没有勾选micro lib
  321. IMPORT __use_two_region_memory ;两区堆栈空间,堆和栈有各自的空间地址
  322. EXPORT __user_initial_stackheap
  323. __user_initial_stackheap ;标号__user_initial_stackheap,表示用户堆栈初始化程序入口
  324. ;此处是初始化两区的堆栈空间,堆是从由低到高的增长,栈是由高向低生长的,两个是互相独立的数据段,并不能交叉使用。
  325. LDR R0, = Heap_Mem ;保存堆起始地址
  326. LDR R1, =(Stack_Mem + Stack_Size) ;保存栈结束地址
  327. LDR R2, = (Heap_Mem + Heap_Size) ;保存堆结束地址
  328. LDR R3, = Stack_Mem ;保存栈起始地址
  329. BX LR
  330. ALIGN
  331. ENDIF
  332. END ;END命令指示汇编器,已到达一个源文件的末尾
  333. ;******************* (C) COPYRIGHT 2011 STMicroelectronics *****END OF FILE*****


二、STM32 中断向量表的位置 、重定向

知道怎么跳到main函数了,那么,中断发生后,又是怎么跑到中断入口地址的呢?

从stm32f10x.s可以看到,已经定义好了一大堆的中断响应函数,这就是中断向量表,标号__Vectors,表示中断向量表入口地址,例如:

AREA    RESET, DATA, READONLY ; 定义只读数据段,实际上是在CODE区(假设STM32从FLASH启动,则此中断向量表起始地址即为0x8000000)

  1. EXPORT __Vectors
  2. IMPORT OS_CPU_SysTickHandler
  3. IMPORT OS_CPU_PendSVHandler
  4. __Vectors DCD __initial_sp ; Top of Stack
  5. DCD Reset_Handler ; Reset Handler
  6. DCD NMI_Handler ; NMI Handler
  7. DCD HardFault_Handler ; Hard Fault Handler
  8. DCD MemManage_Handler ; MPU Fault Handler
  9. DCD BusFault_Handler ; Bus Fault Handler
  10. DCD UsageFault_Handler ; Usage Fault Handler

这个向量表的编写是有讲究的,跟硬件一一对应不能乱写的,CPU找入口地址就靠它了,bin文件开头就是他们的地址,参考手册RM0008的10.1.2节可以看到排列。

我们再结合CORTEX-M3的特性,他上电后根据boot引脚来决定PC位置,比如boot设置为flash启动,则启动后PC跳到0x08000000。此时CPU会先取2个地址,第一个是栈顶地址,第二个是复位异常地址,故有了上面的写法,这样就跳到reset_handler。

那么这个reset_handler的实际地址是多少.?下面的一堆例如Nmi_handler地址又是多少呢?发生中断是怎么跑到这个地址的呢?下面挨个讲解。

1、我们可以通过反向来得知这些入口地址,查看工程下的map文件就可以看到了,这个地址跟keil里面设置的target->flash起始地址息息相关,实际上我们不太需要关心,让编译器分配,中断向量表放的就是他们的地址。

2、对比ARM7/ARM9内核,Cortex-M3内核则是固定了中断向量表的位置而起始地址是可变化的。

3、进到C语言后会先配置NVIC,NVIC_SetVectorTable()里面可以配置中断向量表的起始地址和偏移,主要是告诉CPU该向量表是位于Flash还是Ram,偏移是多少。例如设置为位于Flash内,偏移就是烧入的程序地址,可在Keil target中设置。这样CPU就知道入口地址了。

4、发生中断后,CPU找到中断向量表地址,然后根据偏移(对号入座)再找到中断地址,这样就跳过去了。

我们截一个图说明一下,map文件:


对应的bin文件,看是不是放的上面地址:


显然,200039c0就是栈顶地址,而08006F21就是reset_handler地址!


如何定位?以放到0x20000000为例

1、keil设置ram起始为0x20000100,我们在0x20000000~0x20000100放中断向量表,其他给程序用

2、设置NVIC_SetVectorTable(NVIC_VectTab_FLASH,0);

3、跳到C时把中断向量表拷贝到0x20000000


声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/56718?site
推荐阅读
相关标签
  

闽ICP备14008679号