start_gcc.S 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487
  1. /*
  2. * Copyright (c) 2006-2018, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. */
  9. @-------------------------------------------------------------------------------
  10. @ sys_core.asm
  11. @
  12. @ (c) Texas Instruments 2009-2013, All rights reserved.
  13. @
  14. #include <rtconfig.h>
  15. .equ Mode_USR, 0x10
  16. .equ Mode_FIQ, 0x11
  17. .equ Mode_IRQ, 0x12
  18. .equ Mode_SVC, 0x13
  19. .equ Mode_ABT, 0x17
  20. .equ Mode_UND, 0x1B
  21. .equ Mode_SYS, 0x1F
  22. .equ I_Bit, 0x80 @ when I bit is set, IRQ is disabled
  23. .equ F_Bit, 0x40 @ when F bit is set, FIQ is disabled
  24. .equ UND_Stack_Size, 0x00000000
  25. .equ SVC_Stack_Size, 0x00000000
  26. .equ ABT_Stack_Size, 0x00000000
  27. .equ FIQ_Stack_Size, 0x00001000
  28. .equ IRQ_Stack_Size, 0x00001000
  29. .section .bss.noinit
  30. /* stack */
  31. .globl stack_start
  32. .globl stack_top
  33. .align 3
  34. stack_start:
  35. .rept (UND_Stack_Size + SVC_Stack_Size + ABT_Stack_Size + FIQ_Stack_Size + IRQ_Stack_Size)
  36. .byte 0
  37. .endr
  38. stack_top:
  39. .section .text, "ax"
  40. .text
  41. .arm
  42. .globl _c_int00
  43. .globl _reset
  44. _reset:
  45. @-------------------------------------------------------------------------------
  46. @ Initialize CPU Registers
  47. @ After reset, the CPU is in the Supervisor mode (M = 10011)
  48. cpsid if, #19
  49. #if defined (__VFP_FP__) && !defined(__SOFTFP__) && defined(RT_VFP_LAZY_STACKING)
  50. @ Turn on FPV coprocessor
  51. mrc p15, #0x00, r2, c1, c0, #0x02
  52. orr r2, r2, #0xF00000
  53. mcr p15, #0x00, r2, c1, c0, #0x02
  54. fmrx r2, fpexc
  55. orr r2, r2, #0x40000000
  56. fmxr fpexc, r2
  57. #endif
  58. @-------------------------------------------------------------------------------
  59. @ Initialize Stack Pointers
  60. ldr r0, =stack_top
  61. @ Set the startup stack for svc
  62. mov sp, r0
  63. @ Enter Undefined Instruction Mode and set its Stack Pointer
  64. msr cpsr_c, #Mode_UND|I_Bit|F_Bit
  65. mov sp, r0
  66. sub r0, r0, #UND_Stack_Size
  67. @ Enter Abort Mode and set its Stack Pointer
  68. msr cpsr_c, #Mode_ABT|I_Bit|F_Bit
  69. mov sp, r0
  70. sub r0, r0, #ABT_Stack_Size
  71. @ Enter FIQ Mode and set its Stack Pointer
  72. msr cpsr_c, #Mode_FIQ|I_Bit|F_Bit
  73. mov sp, r0
  74. sub r0, r0, #FIQ_Stack_Size
  75. @ Enter IRQ Mode and set its Stack Pointer
  76. msr cpsr_c, #Mode_IRQ|I_Bit|F_Bit
  77. mov sp, r0
  78. sub r0, r0, #IRQ_Stack_Size
  79. @ Switch back to SVC
  80. msr cpsr_c, #Mode_SVC|I_Bit|F_Bit
  81. bl next1
  82. next1:
  83. bl next2
  84. next2:
  85. bl next3
  86. next3:
  87. bl next4
  88. next4:
  89. ldr lr, =_c_int00
  90. bx lr
  91. .globl data_init
  92. data_init:
  93. /* copy .data to SRAM */
  94. ldr r1, =_sidata /* .data start in image */
  95. ldr r2, =_edata /* .data end in image */
  96. ldr r3, =_sdata /* sram data start */
  97. data_loop:
  98. ldr r0, [r1, #0]
  99. str r0, [r3]
  100. add r1, r1, #4
  101. add r3, r3, #4
  102. cmp r3, r2 /* check if data to clear */
  103. blo data_loop /* loop until done */
  104. /* clear .bss */
  105. mov r0,#0 /* get a zero */
  106. ldr r1,=__bss_start /* bss start */
  107. ldr r2,=__bss_end /* bss end */
  108. bss_loop:
  109. cmp r1,r2 /* check if data to clear */
  110. strlo r0,[r1],#4 /* clear 4 bytes */
  111. blo bss_loop /* loop until done */
  112. /* call C++ constructors of global objects */
  113. ldr r0, =__ctors_start__
  114. ldr r1, =__ctors_end__
  115. ctor_loop:
  116. cmp r0, r1
  117. beq ctor_end
  118. ldr r2, [r0], #4
  119. stmfd sp!, {r0-r3, ip, lr}
  120. mov lr, pc
  121. bx r2
  122. ldmfd sp!, {r0-r3, ip, lr}
  123. b ctor_loop
  124. ctor_end:
  125. bx lr
  126. @-------------------------------------------------------------------------------
  127. @ Enable RAM ECC Support
  128. .globl _coreEnableRamEcc_
  129. _coreEnableRamEcc_:
  130. stmfd sp!, {r0}
  131. mrc p15, #0x00, r0, c1, c0, #0x01
  132. orr r0, r0, #0x0C000000
  133. mcr p15, #0x00, r0, c1, c0, #0x01
  134. ldmfd sp!, {r0}
  135. bx lr
  136. @-------------------------------------------------------------------------------
  137. @ Disable RAM ECC Support
  138. .globl _coreDisableRamEcc_
  139. _coreDisableRamEcc_:
  140. stmfd sp!, {r0}
  141. mrc p15, #0x00, r0, c1, c0, #0x01
  142. bic r0, r0, #0x0C000000
  143. mcr p15, #0x00, r0, c1, c0, #0x01
  144. ldmfd sp!, {r0}
  145. bx lr
  146. @-------------------------------------------------------------------------------
  147. @ Enable Flash ECC Support
  148. .globl _coreEnableFlashEcc_
  149. _coreEnableFlashEcc_:
  150. stmfd sp!, {r0}
  151. mrc p15, #0x00, r0, c1, c0, #0x01
  152. orr r0, r0, #0x02000000
  153. dmb
  154. mcr p15, #0x00, r0, c1, c0, #0x01
  155. ldmfd sp!, {r0}
  156. bx lr
  157. @-------------------------------------------------------------------------------
  158. @ Disable Flash ECC Support
  159. .globl _coreDisableFlashEcc_
  160. _coreDisableFlashEcc_:
  161. stmfd sp!, {r0}
  162. mrc p15, #0x00, r0, c1, c0, #0x01
  163. bic r0, r0, #0x02000000
  164. mcr p15, #0x00, r0, c1, c0, #0x01
  165. ldmfd sp!, {r0}
  166. bx lr
  167. @-------------------------------------------------------------------------------
  168. @ Get data fault status register
  169. .globl _coreGetDataFault_
  170. _coreGetDataFault_:
  171. mrc p15, #0, r0, c5, c0, #0
  172. bx lr
  173. @-------------------------------------------------------------------------------
  174. @ Clear data fault status register
  175. .globl _coreClearDataFault_
  176. _coreClearDataFault_:
  177. stmfd sp!, {r0}
  178. mov r0, #0
  179. mcr p15, #0, r0, c5, c0, #0
  180. ldmfd sp!, {r0}
  181. bx lr
  182. @-------------------------------------------------------------------------------
  183. @ Get instruction fault status register
  184. .globl _coreGetInstructionFault_
  185. _coreGetInstructionFault_:
  186. mrc p15, #0, r0, c5, c0, #1
  187. bx lr
  188. @-------------------------------------------------------------------------------
  189. @ Clear instruction fault status register
  190. .globl _coreClearInstructionFault_
  191. _coreClearInstructionFault_:
  192. stmfd sp!, {r0}
  193. mov r0, #0
  194. mcr p15, #0, r0, c5, c0, #1
  195. ldmfd sp!, {r0}
  196. bx lr
  197. @-------------------------------------------------------------------------------
  198. @ Get data fault address register
  199. .globl _coreGetDataFaultAddress_
  200. _coreGetDataFaultAddress_:
  201. mrc p15, #0, r0, c6, c0, #0
  202. bx lr
  203. @-------------------------------------------------------------------------------
  204. @ Clear data fault address register
  205. .globl _coreClearDataFaultAddress_
  206. _coreClearDataFaultAddress_:
  207. stmfd sp!, {r0}
  208. mov r0, #0
  209. mcr p15, #0, r0, c6, c0, #0
  210. ldmfd sp!, {r0}
  211. bx lr
  212. @-------------------------------------------------------------------------------
  213. @ Get instruction fault address register
  214. .globl _coreGetInstructionFaultAddress_
  215. _coreGetInstructionFaultAddress_:
  216. mrc p15, #0, r0, c6, c0, #2
  217. bx lr
  218. @-------------------------------------------------------------------------------
  219. @ Clear instruction fault address register
  220. .globl _coreClearInstructionFaultAddress_
  221. _coreClearInstructionFaultAddress_:
  222. stmfd sp!, {r0}
  223. mov r0, #0
  224. mcr p15, #0, r0, c6, c0, #2
  225. ldmfd sp!, {r0}
  226. bx lr
  227. @-------------------------------------------------------------------------------
  228. @ Get auxiliary data fault status register
  229. .globl _coreGetAuxiliaryDataFault_
  230. _coreGetAuxiliaryDataFault_:
  231. mrc p15, #0, r0, c5, c1, #0
  232. bx lr
  233. @-------------------------------------------------------------------------------
  234. @ Clear auxiliary data fault status register
  235. .globl _coreClearAuxiliaryDataFault_
  236. _coreClearAuxiliaryDataFault_:
  237. stmfd sp!, {r0}
  238. mov r0, #0
  239. mcr p15, #0, r0, c5, c1, #0
  240. ldmfd sp!, {r0}
  241. bx lr
  242. @-------------------------------------------------------------------------------
  243. @ Get auxiliary instruction fault status register
  244. .globl _coreGetAuxiliaryInstructionFault_
  245. _coreGetAuxiliaryInstructionFault_:
  246. mrc p15, #0, r0, c5, c1, #1
  247. bx lr
  248. @-------------------------------------------------------------------------------
  249. @ Clear auxiliary instruction fault status register
  250. .globl _coreClearAuxiliaryInstructionFault_
  251. _coreClearAuxiliaryInstructionFault_:
  252. stmfd sp!, {r0}
  253. mov r0, #0
  254. mrc p15, #0, r0, c5, c1, #1
  255. ldmfd sp!, {r0}
  256. bx lr
  257. @-------------------------------------------------------------------------------
  258. @ Clear ESM CCM errorss
  259. .globl _esmCcmErrorsClear_
  260. _esmCcmErrorsClear_:
  261. stmfd sp!, {r0-r2}
  262. ldr r0, ESMSR1_REG @ load the ESMSR1 status register address
  263. ldr r2, ESMSR1_ERR_CLR
  264. str r2, [r0] @ clear the ESMSR1 register
  265. ldr r0, ESMSR2_REG @ load the ESMSR2 status register address
  266. ldr r2, ESMSR2_ERR_CLR
  267. str r2, [r0] @ clear the ESMSR2 register
  268. ldr r0, ESMSSR2_REG @ load the ESMSSR2 status register address
  269. ldr r2, ESMSSR2_ERR_CLR
  270. str r2, [r0] @ clear the ESMSSR2 register
  271. ldr r0, ESMKEY_REG @ load the ESMKEY register address
  272. mov r2, #0x5 @ load R2 with 0x5
  273. str r2, [r0] @ clear the ESMKEY register
  274. ldr r0, VIM_INTREQ @ load the INTREQ register address
  275. ldr r2, VIM_INT_CLR
  276. str r2, [r0] @ clear the INTREQ register
  277. ldr r0, CCMR4_STAT_REG @ load the CCMR4 status register address
  278. ldr r2, CCMR4_ERR_CLR
  279. str r2, [r0] @ clear the CCMR4 status register
  280. ldmfd sp!, {r0-r2}
  281. bx lr
  282. ESMSR1_REG: .word 0xFFFFF518
  283. ESMSR2_REG: .word 0xFFFFF51C
  284. ESMSR3_REG: .word 0xFFFFF520
  285. ESMKEY_REG: .word 0xFFFFF538
  286. ESMSSR2_REG: .word 0xFFFFF53C
  287. CCMR4_STAT_REG: .word 0xFFFFF600
  288. ERR_CLR_WRD: .word 0xFFFFFFFF
  289. CCMR4_ERR_CLR: .word 0x00010000
  290. ESMSR1_ERR_CLR: .word 0x80000000
  291. ESMSR2_ERR_CLR: .word 0x00000004
  292. ESMSSR2_ERR_CLR: .word 0x00000004
  293. VIM_INT_CLR: .word 0x00000001
  294. VIM_INTREQ: .word 0xFFFFFE20
  295. @-------------------------------------------------------------------------------
  296. @ Work Around for Errata CORTEX-R4#57:
  297. @
  298. @ Errata Description:
  299. @ Conditional VMRS APSR_Nzcv, FPSCR May Evaluate With Incorrect Flags
  300. @ Workaround:
  301. @ Disable out-of-order single-precision floating point
  302. @ multiply-accumulate instruction completion
  303. .globl _errata_CORTEXR4_57_
  304. _errata_CORTEXR4_57_:
  305. push {r0}
  306. mrc p15, #0, r0, c15, c0, #0 @ Read Secondary Auxiliary Control Register
  307. orr r0, r0, #0x10000 @ Set BIT 16 (Set DOOFMACS)
  308. mcr p15, #0, r0, c15, c0, #0 @ Write Secondary Auxiliary Control Register
  309. pop {r0}
  310. bx lr
  311. @-------------------------------------------------------------------------------
  312. @ Work Around for Errata CORTEX-R4#66:
  313. @
  314. @ Errata Description:
  315. @ Register Corruption During A Load-Multiple Instruction At
  316. @ an Exception Vector
  317. @ Workaround:
  318. @ Disable out-of-order completion for divide instructions in
  319. @ Auxiliary Control register
  320. .globl _errata_CORTEXR4_66_
  321. _errata_CORTEXR4_66_:
  322. push {r0}
  323. mrc p15, #0, r0, c1, c0, #1 @ Read Auxiliary Control register
  324. orr r0, r0, #0x80 @ Set BIT 7 (Disable out-of-order completion
  325. @ for divide instructions.)
  326. mcr p15, #0, r0, c1, c0, #1 @ Write Auxiliary Control register
  327. pop {r0}
  328. bx lr
  329. .globl turnon_VFP
  330. turnon_VFP:
  331. @ Enable FPV
  332. STMDB sp!, {r0}
  333. fmrx r0, fpexc
  334. orr r0, r0, #0x40000000
  335. fmxr fpexc, r0
  336. LDMIA sp!, {r0}
  337. subs pc, lr, #4
  338. .macro push_svc_reg
  339. sub sp, sp, #17 * 4 @/* Sizeof(struct rt_hw_exp_stack) */
  340. stmia sp, {r0 - r12} @/* Calling r0-r12 */
  341. mov r0, sp
  342. mrs r6, spsr @/* Save CPSR */
  343. str lr, [r0, #15*4] @/* Push PC */
  344. str r6, [r0, #16*4] @/* Push CPSR */
  345. cps #Mode_SVC
  346. str sp, [r0, #13*4] @/* Save calling SP */
  347. str lr, [r0, #14*4] @/* Save calling PC */
  348. .endm
  349. .globl vector_svc
  350. vector_svc:
  351. push_svc_reg
  352. bl rt_hw_trap_svc
  353. b .
  354. .globl vector_pabort
  355. vector_pabort:
  356. push_svc_reg
  357. bl rt_hw_trap_pabt
  358. b .
  359. .globl vector_dabort
  360. vector_dabort:
  361. push_svc_reg
  362. bl rt_hw_trap_dabt
  363. b .
  364. .globl vector_resv
  365. vector_resv:
  366. push_svc_reg
  367. bl rt_hw_trap_resv
  368. b .