Keyboard firmwares for Atmel AVR and Cortex-M
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

HAL_CA9.s 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474
  1. /*----------------------------------------------------------------------------
  2. * RL-ARM - RTX
  3. *----------------------------------------------------------------------------
  4. * Name: HAL_CA9.c
  5. * Purpose: Hardware Abstraction Layer for Cortex-A9
  6. * Rev.: 3 Sept 2013
  7. *----------------------------------------------------------------------------
  8. *
  9. * Copyright (c) 2012 - 2013 ARM Limited
  10. * All rights reserved.
  11. * Redistribution and use in source and binary forms, with or without
  12. * modification, are permitted provided that the following conditions are met:
  13. * - Redistributions of source code must retain the above copyright
  14. * notice, this list of conditions and the following disclaimer.
  15. * - Redistributions in binary form must reproduce the above copyright
  16. * notice, this list of conditions and the following disclaimer in the
  17. * documentation and/or other materials provided with the distribution.
  18. * - Neither the name of ARM nor the names of its contributors may be used
  19. * to endorse or promote products derived from this software without
  20. * specific prior written permission.
  21. *
  22. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  23. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  24. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  25. * ARE DISCLAIMED. IN NO EVENT SHALL COPYRIGHT HOLDERS AND CONTRIBUTORS BE
  26. * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
  27. * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
  28. * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
  29. * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  30. * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  31. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  32. * POSSIBILITY OF SUCH DAMAGE.
  33. *---------------------------------------------------------------------------*/
  34. .global rt_set_PSP
  35. .global rt_get_PSP
  36. .global _alloc_box
  37. .global _free_box
  38. .global PendSV_Handler
  39. .global OS_Tick_Handler
  40. .EQU CPSR_T_BIT, 0x20
  41. .EQU CPSR_I_BIT, 0x80
  42. .EQU CPSR_F_BIT, 0x40
  43. .EQU MODE_USR, 0x10
  44. .EQU MODE_FIQ, 0x11
  45. .EQU MODE_IRQ, 0x12
  46. .EQU MODE_SVC, 0x13
  47. .EQU MODE_ABT, 0x17
  48. .EQU MODE_UND, 0x1B
  49. .EQU MODE_SYS, 0x1F
  50. .EQU TCB_TID, 3 /* 'task id' offset */
  51. .EQU TCB_STACKF, 32 /* 'stack_frame' offset */
  52. .EQU TCB_TSTACK, 36 /* 'tsk_stack' offset */
  53. .extern rt_alloc_box
  54. .extern os_tsk
  55. .extern GICInterface_BASE
  56. .extern rt_pop_req
  57. .extern os_tick_irqack
  58. .extern rt_systick
  59. /*----------------------------------------------------------------------------
  60. * Functions
  61. *---------------------------------------------------------------------------*/
  62. .text
  63. @ For A-class, set USR/SYS stack
  64. @ __asm void rt_set_PSP (U32 stack) {
  65. rt_set_PSP:
  66. .arm
  67. MRS R1, CPSR
  68. CPS #MODE_SYS @no effect in USR mode
  69. ISB
  70. MOV SP, R0
  71. MSR CPSR_c, R1 @no effect in USR mode
  72. ISB
  73. BX LR
  74. @ }
  75. @ For A-class, get USR/SYS stack
  76. @ __asm U32 rt_get_PSP (void) {
  77. rt_get_PSP:
  78. .arm
  79. MRS R1, CPSR
  80. CPS #MODE_SYS @no effect in USR mode
  81. ISB
  82. MOV R0, SP
  83. MSR CPSR_c, R1 @no effect in USR mode
  84. ISB
  85. BX LR
  86. @ }
  87. /*--------------------------- _alloc_box ------------------------------------*/
  88. @ __asm void *_alloc_box (void *box_mem) {
  89. _alloc_box:
  90. /* Function wrapper for Unprivileged/Privileged mode. */
  91. .arm
  92. LDR R12,=rt_alloc_box @ __cpp(rt_alloc_box)
  93. MRS R2, CPSR
  94. LSLS R2, #28
  95. BXNE R12
  96. SVC 0
  97. BX LR
  98. @ }
  99. /*--------------------------- _free_box -------------------------------------*/
  100. @ __asm int _free_box (void *box_mem, void *box) {
  101. _free_box:
  102. /* Function wrapper for Unprivileged/Privileged mode. */
  103. .arm
  104. LDR R12,=rt_free_box @ __cpp(rt_free_box)
  105. MRS R2, CPSR
  106. LSLS R2, #28
  107. BXNE R12
  108. SVC 0
  109. BX LR
  110. @ }
  111. /*-------------------------- SVC_Handler -----------------------------------*/
  112. @ #pragma push
  113. @ #pragma arm
  114. @ __asm void SVC_Handler (void) {
  115. .type SVC_Handler, %function
  116. .global SVC_Handler
  117. SVC_Handler:
  118. @ PRESERVE8
  119. .arm
  120. .extern rt_tsk_lock
  121. .extern rt_tsk_unlock
  122. .extern SVC_Count
  123. .extern SVC_Table
  124. .extern rt_stk_check
  125. .extern FPUEnable
  126. .EQU Mode_SVC, 0x13
  127. SRSDB SP!, #Mode_SVC @ Push LR_SVC and SPRS_SVC onto SVC mode stack
  128. PUSH {R4} @ Push R4 so we can use it as a temp
  129. MRS R4,SPSR @ Get SPSR
  130. TST R4,#CPSR_T_BIT @ Check Thumb Bit
  131. LDRNEH R4,[LR,#-2] @ Thumb: Load Halfword
  132. BICNE R4,R4,#0xFF00 @ Extract SVC Number
  133. LDREQ R4,[LR,#-4] @ ARM: Load Word
  134. BICEQ R4,R4,#0xFF000000 @ Extract SVC Number
  135. /* Lock out systick and re-enable interrupts */
  136. PUSH {R0-R3,R12,LR}
  137. AND R12, SP, #4 @ Ensure stack is 8-byte aligned
  138. SUB SP, SP, R12 @ Adjust stack
  139. PUSH {R12, LR} @ Store stack adjustment and dummy LR to SVC stack
  140. BLX rt_tsk_lock
  141. CPSIE i
  142. POP {R12, LR} @ Get stack adjustment & discard dummy LR
  143. ADD SP, SP, R12 @ Unadjust stack
  144. POP {R0-R3,R12,LR}
  145. CMP R4,#0
  146. BNE SVC_User
  147. MRS R4,SPSR
  148. PUSH {R4} @ Push R4 so we can use it as a temp
  149. AND R4, SP, #4 @ Ensure stack is 8-byte aligned
  150. SUB SP, SP, R4 @ Adjust stack
  151. PUSH {R4, LR} @ Store stack adjustment and dummy LR
  152. BLX R12
  153. POP {R4, LR} @ Get stack adjustment & discard dummy LR
  154. ADD SP, SP, R4 @ Unadjust stack
  155. POP {R4} @ Restore R4
  156. MSR SPSR_cxsf,R4
  157. /* Here we will be in SVC mode (even if coming in from PendSV_Handler or OS_Tick_Handler) */
  158. Sys_Switch:
  159. LDR LR,=os_tsk @ __cpp(&os_tsk)
  160. LDM LR,{R4,LR} @ os_tsk.run, os_tsk.new
  161. CMP R4,LR
  162. BNE switching
  163. PUSH {R0-R3,R12,LR}
  164. AND R12, SP, #4 @ Ensure stack is 8-byte aligned
  165. SUB SP, SP, R12 @ Adjust stack
  166. PUSH {R12, LR} @ Store stack adjustment and dummy LR to SVC stack
  167. CPSID i
  168. BLX rt_tsk_unlock
  169. POP {R12, LR} @ Get stack adjustment & discard dummy LR
  170. ADD SP, SP, R12 @ Unadjust stack
  171. POP {R0-R3,R12,LR}
  172. POP {R4}
  173. RFEFD SP! @ Return from exception, no task switch
  174. switching:
  175. CLREX
  176. CMP R4,#0
  177. ADDEQ SP,SP,#12 @ Original R4, LR & SPSR do not need to be popped when we are paging in a different task
  178. BEQ SVC_Next @ Runtask deleted?
  179. PUSH {R8-R11} @ R4 and LR already stacked
  180. MOV R10,R4 @ Preserve os_tsk.run
  181. MOV R11,LR @ Preserve os_tsk.new
  182. ADD R8,SP,#16 @ Unstack R4,LR
  183. LDMIA R8,{R4,LR}
  184. SUB SP,SP,#4 @ Make space on the stack for the next instn
  185. STMIA SP,{SP}^ @ Put User SP onto stack
  186. POP {R8} @ Pop User SP into R8
  187. MRS R9,SPSR
  188. STMDB R8!,{R9} @ User CPSR
  189. STMDB R8!,{LR} @ User PC
  190. STMDB R8,{LR}^ @ User LR
  191. SUB R8,R8,#4 @ No writeback for store of User LR
  192. STMDB R8!,{R0-R3,R12} @ User R0-R3,R12
  193. MOV R3,R10 @ os_tsk.run
  194. MOV LR,R11 @ os_tsk.new
  195. POP {R9-R12}
  196. ADD SP,SP,#12 @ Fix up SP for unstack of R4, LR & SPSR
  197. STMDB R8!,{R4-R7,R9-R12} @ User R4-R11
  198. @ If applicable, stack VFP state
  199. MRC p15,0,R1,c1,c0,2 @ VFP/NEON access enabled? (CPACR)
  200. AND R2,R1,#0x00F00000
  201. CMP R2,#0x00F00000
  202. BNE no_outgoing_vfp
  203. VMRS R2,FPSCR
  204. STMDB R8!,{R2,R4} @ Push FPSCR, maintain 8-byte alignment
  205. VSTMDB R8!,{S0-S31}
  206. LDRB R2,[R3,#TCB_STACKF] @ Record in TCB that VFP state is stacked
  207. ORR R2,#2
  208. STRB R2,[R3,#TCB_STACKF]
  209. no_outgoing_vfp:
  210. STR R8,[R3,#TCB_TSTACK]
  211. MOV R4,LR
  212. PUSH {R4} @ Push R4 so we can use it as a temp
  213. AND R4, SP, #4 @ Ensure stack is 8-byte aligned
  214. SUB SP, SP, R4 @ Adjust stack
  215. PUSH {R4, LR} @ Store stack adjustment and dummy LR to SVC stack
  216. BLX rt_stk_check
  217. POP {R4, LR} @ Get stack adjustment & discard dummy LR
  218. ADD SP, SP, R4 @ Unadjust stack
  219. POP {R4} @ Restore R4
  220. MOV LR,R4
  221. SVC_Next: @ R4 == os_tsk.run, LR == os_tsk.new, R0-R3, R5-R12 corruptible
  222. LDR R1,=os_tsk @ __cpp(&os_tsk), os_tsk.run = os_tsk.new
  223. STR LR,[R1]
  224. LDRB R1,[LR,#TCB_TID] @ os_tsk.run->task_id
  225. LSL R1,#8 @ Store PROCID
  226. MCR p15,0,R1,c13,c0,1 @ Write CONTEXTIDR
  227. LDR R0,[LR,#TCB_TSTACK] @ os_tsk.run->tsk_stack
  228. @ Does incoming task have VFP state in stack?
  229. LDRB R3,[LR,#TCB_STACKF]
  230. TST R3,#0x2
  231. MRC p15,0,R1,c1,c0,2 @ Read CPACR
  232. ANDEQ R1,R1,#0xFF0FFFFF @ Disable VFP access if incoming task does not have stacked VFP state
  233. ORRNE R1,R1,#0x00F00000 @ Enable VFP access if incoming task does have stacked VFP state
  234. MCR p15,0,R1,c1,c0,2 @ Write CPACR
  235. BEQ no_incoming_vfp
  236. ISB @ We only need the sync if we enabled, otherwise we will context switch before next VFP instruction anyway
  237. VLDMIA R0!,{S0-S31}
  238. LDR R2,[R0]
  239. VMSR FPSCR,R2
  240. ADD R0,R0,#8
  241. no_incoming_vfp:
  242. LDR R1,[R0,#60] @ Restore User CPSR
  243. MSR SPSR_cxsf,R1
  244. LDMIA R0!,{R4-R11} @ Restore User R4-R11
  245. ADD R0,R0,#4 @ Restore User R1-R3,R12
  246. LDMIA R0!,{R1-R3,R12}
  247. LDMIA R0,{LR}^ @ Restore User LR
  248. ADD R0,R0,#4 @ No writeback for load to user LR
  249. LDMIA R0!,{LR} @ Restore User PC
  250. ADD R0,R0,#4 @ Correct User SP for unstacked user CPSR
  251. PUSH {R0} @ Push R0 onto stack
  252. LDMIA SP,{SP}^ @ Get R0 off stack into User SP
  253. ADD SP,SP,#4 @ Put SP back
  254. LDR R0,[R0,#-32] @ Restore R0
  255. PUSH {R0-R3,R12,LR}
  256. AND R12, SP, #4 @ Ensure stack is 8-byte aligned
  257. SUB SP, SP, R12 @ Adjust stack
  258. PUSH {R12, LR} @ Store stack adjustment and dummy LR to SVC stack
  259. CPSID i
  260. BLX rt_tsk_unlock
  261. POP {R12, LR} @ Get stack adjustment & discard dummy LR
  262. ADD SP, SP, R12 @ Unadjust stack
  263. POP {R0-R3,R12,LR}
  264. MOVS PC,LR @ Return from exception
  265. /*------------------- User SVC -------------------------------*/
  266. SVC_User:
  267. LDR R12,=SVC_Count
  268. LDR R12,[R12]
  269. CMP R4,R12 @ Check for overflow
  270. BHI SVC_Done
  271. LDR R12,=SVC_Table-4
  272. LDR R12,[R12,R4,LSL #2] @ Load SVC Function Address
  273. MRS R4,SPSR @ Save SPSR
  274. PUSH {R4} @ Push R4 so we can use it as a temp
  275. AND R4, SP, #4 @ Ensure stack is 8-byte aligned
  276. SUB SP, SP, R4 @ Adjust stack
  277. PUSH {R4, LR} @ Store stack adjustment and dummy LR
  278. BLX R12 @ Call SVC Function
  279. POP {R4, LR} @ Get stack adjustment & discard dummy LR
  280. ADD SP, SP, R4 @ Unadjust stack
  281. POP {R4} @ Restore R4
  282. MSR SPSR_cxsf,R4 @ Restore SPSR
  283. SVC_Done:
  284. PUSH {R0-R3,R12,LR}
  285. PUSH {R4} @ Push R4 so we can use it as a temp
  286. AND R4, SP, #4 @ Ensure stack is 8-byte aligned
  287. SUB SP, SP, R4 @ Adjust stack
  288. PUSH {R4, LR} @ Store stack adjustment and dummy LR
  289. CPSID i
  290. BLX rt_tsk_unlock
  291. POP {R4, LR} @ Get stack adjustment & discard dummy LR
  292. ADD SP, SP, R4 @ Unadjust stack
  293. POP {R4} @ Restore R4
  294. POP {R0-R3,R12,LR}
  295. POP {R4}
  296. RFEFD SP! @ Return from exception
  297. @ }
  298. @ #pragma pop
  299. @ #pragma push
  300. @ #pragma arm
  301. @ __asm void PendSV_Handler (U32 IRQn) {
  302. PendSV_Handler:
  303. .arm
  304. .extern rt_tsk_lock
  305. .extern IRQNestLevel
  306. ADD SP,SP,#8 @ fix up stack pointer (R0 has been pushed and will never be popped, R1 was pushed for stack alignment)
  307. @ Disable systick interrupts, then write EOIR. We want interrupts disabled before we enter the context switcher.
  308. PUSH {R0, R1}
  309. BLX rt_tsk_lock
  310. POP {R0, R1}
  311. LDR R1, =GICInterface_BASE @ __cpp(&GICInterface_BASE)
  312. LDR R1, [R1, #0]
  313. STR R0, [R1, #0x10]
  314. LDR R0, =IRQNestLevel @ Get address of nesting counter
  315. LDR R1, [R0]
  316. SUB R1, R1, #1 @ Decrement nesting counter
  317. STR R1, [R0]
  318. BLX rt_pop_req @ __cpp(rt_pop_req)
  319. POP {R1, LR} @ Get stack adjustment & discard dummy LR
  320. ADD SP, SP, R1 @ Unadjust stack
  321. LDR R0,[SP,#24]
  322. MSR SPSR_cxsf,R0
  323. POP {R0-R3,R12} @ Leave SPSR & LR on the stack
  324. PUSH {R4}
  325. B Sys_Switch
  326. @ }
  327. @ #pragma pop
  328. @ #pragma push
  329. @ #pragma arm
  330. @ __asm void OS_Tick_Handler (U32 IRQn) {
  331. OS_Tick_Handler:
  332. .arm
  333. ADD SP,SP,#8 @ fix up stack pointer (R0 has been pushed and will never be popped, R1 was pushed for stack alignment)
  334. PUSH {R0, R1}
  335. BLX rt_tsk_lock
  336. POP {R0, R1}
  337. LDR R1, =GICInterface_BASE @ __cpp(&GICInterface_BASE)
  338. LDR R1, [R1, #0]
  339. STR R0, [R1, #0x10]
  340. LDR R0, =IRQNestLevel @ Get address of nesting counter
  341. LDR R1, [R0]
  342. SUB R1, R1, #1 @ Decrement nesting counter
  343. STR R1, [R0]
  344. BLX os_tick_irqack @ __cpp(os_tick_irqack)
  345. BLX rt_systick @ __cpp(rt_systick)
  346. POP {R1, LR} @ Get stack adjustment & discard dummy LR
  347. ADD SP, SP, R1 @ Unadjust stack
  348. LDR R0,[SP,#24]
  349. MSR SPSR_cxsf,R0
  350. POP {R0-R3,R12} @ Leave SPSR & LR on the stack
  351. PUSH {R4}
  352. B Sys_Switch
  353. @ }
  354. @ #pragma pop
  355. .global __set_PSP
  356. @ __STATIC_ASM void __set_PSP(uint32_t topOfProcStack)
  357. @ {
  358. __set_PSP:
  359. @ PRESERVE8
  360. .arm
  361. BIC R0, R0, #7 @ensure stack is 8-byte aligned
  362. MRS R1, CPSR
  363. CPS #MODE_SYS @no effect in USR mode
  364. MOV SP, R0
  365. MSR CPSR_c, R1 @no effect in USR mode
  366. ISB
  367. BX LR
  368. @ }
  369. .global __set_CPS_USR
  370. @ __STATIC_ASM void __set_CPS_USR(void)
  371. @ {
  372. __set_CPS_USR:
  373. .arm
  374. CPS #MODE_USR
  375. BX LR
  376. @ }
  377. .END
  378. /*----------------------------------------------------------------------------
  379. * end of file
  380. *---------------------------------------------------------------------------*/