HAL_CM4.c 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. /*----------------------------------------------------------------------------
  2. * CMSIS-RTOS - RTX
  3. *----------------------------------------------------------------------------
  4. * Name: HAL_CM4.C
  5. * Purpose: Hardware Abstraction Layer for Cortex-M4
  6. * Rev.: V4.79
  7. *----------------------------------------------------------------------------
  8. *
  9. * Copyright (c) 1999-2009 KEIL, 2009-2017 ARM Germany GmbH. All rights reserved.
  10. *
  11. * SPDX-License-Identifier: Apache-2.0
  12. *
  13. * Licensed under the Apache License, Version 2.0 (the License); you may
  14. * not use this file except in compliance with the License.
  15. * You may obtain a copy of the License at
  16. *
  17. * www.apache.org/licenses/LICENSE-2.0
  18. *
  19. * Unless required by applicable law or agreed to in writing, software
  20. * distributed under the License is distributed on an AS IS BASIS, WITHOUT
  21. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  22. * See the License for the specific language governing permissions and
  23. * limitations under the License.
  24. *---------------------------------------------------------------------------*/
  25. #include "rt_TypeDef.h"
  26. #include "RTX_Config.h"
  27. #include "rt_System.h"
  28. #include "rt_HAL_CM.h"
  29. #include "rt_Task.h"
  30. #include "rt_MemBox.h"
  31. /*----------------------------------------------------------------------------
  32. * Functions
  33. *---------------------------------------------------------------------------*/
  34. /*--------------------------- rt_set_PSP ------------------------------------*/
  35. __asm void rt_set_PSP (U32 stack) {
  36. MSR PSP,R0
  37. BX LR
  38. }
  39. /*--------------------------- rt_get_PSP ------------------------------------*/
  40. __asm U32 rt_get_PSP (void) {
  41. MRS R0,PSP
  42. BX LR
  43. }
  44. /*--------------------------- os_set_env ------------------------------------*/
  45. __asm void os_set_env (void) {
  46. /* Switch to Unprivileged/Privileged Thread mode, use PSP. */
  47. MOV R0,SP ; PSP = MSP
  48. MSR PSP,R0
  49. LDR R0,=__cpp(&os_flags)
  50. LDRB R0,[R0]
  51. LSLS R0,#31
  52. MOVNE R0,#0x02 ; Privileged Thread mode, use PSP
  53. MOVEQ R0,#0x03 ; Unprivileged Thread mode, use PSP
  54. MSR CONTROL,R0
  55. BX LR
  56. ALIGN
  57. }
  58. /*--------------------------- _alloc_box ------------------------------------*/
  59. __asm void *_alloc_box (void *box_mem) {
  60. /* Function wrapper for Unprivileged/Privileged mode. */
  61. LDR R12,=__cpp(rt_alloc_box)
  62. MRS R3,IPSR
  63. LSLS R3,#24
  64. BXNE R12
  65. MRS R3,CONTROL
  66. LSLS R3,#31
  67. BXEQ R12
  68. SVC 0
  69. BX LR
  70. ALIGN
  71. }
  72. /*--------------------------- _free_box -------------------------------------*/
  73. __asm U32 _free_box (void *box_mem, void *box) {
  74. /* Function wrapper for Unprivileged/Privileged mode. */
  75. LDR R12,=__cpp(rt_free_box)
  76. MRS R3,IPSR
  77. LSLS R3,#24
  78. BXNE R12
  79. MRS R3,CONTROL
  80. LSLS R3,#31
  81. BXEQ R12
  82. SVC 0
  83. BX LR
  84. ALIGN
  85. }
  86. /*-------------------------- SVC_Handler ------------------------------------*/
  87. __asm void SVC_Handler (void) {
  88. PRESERVE8
  89. IMPORT SVC_Count
  90. IMPORT SVC_Table
  91. IMPORT rt_stk_check
  92. #ifdef IFX_XMC4XXX
  93. EXPORT SVC_Handler_Veneer
  94. SVC_Handler_Veneer
  95. #endif
  96. MRS R0,PSP ; Read PSP
  97. LDR R1,[R0,#24] ; Read Saved PC from Stack
  98. LDRB R1,[R1,#-2] ; Load SVC Number
  99. CBNZ R1,SVC_User
  100. LDM R0,{R0-R3,R12} ; Read R0-R3,R12 from stack
  101. PUSH {R4,LR} ; Save EXC_RETURN
  102. BLX R12 ; Call SVC Function
  103. POP {R4,LR} ; Restore EXC_RETURN
  104. MRS R12,PSP ; Read PSP
  105. STM R12,{R0-R2} ; Store return values
  106. LDR R3,=__cpp(&os_tsk)
  107. LDM R3,{R1,R2} ; os_tsk.run, os_tsk.next
  108. CMP R1,R2
  109. #ifdef IFX_XMC4XXX
  110. PUSHEQ {LR}
  111. POPEQ {PC}
  112. #else
  113. BXEQ LR ; RETI, no task switch
  114. #endif
  115. CBNZ R1,SVC_ContextSave ; Runtask not deleted?
  116. TST LR,#0x10 ; is it extended frame?
  117. BNE SVC_ContextRestore
  118. LDR R1,=0xE000EF34
  119. LDR R0,[R1] ; Load FPCCR
  120. BIC R0,#1 ; Clear LSPACT (Lazy state)
  121. STR R0,[R1] ; Store FPCCR
  122. B SVC_ContextRestore
  123. SVC_ContextSave
  124. TST LR,#0x10 ; is it extended frame?
  125. VSTMDBEQ R12!,{S16-S31} ; yes, stack also VFP hi-regs
  126. MOVEQ R0,#0x01 ; os_tsk->stack_frame val
  127. MOVNE R0,#0x00
  128. STRB R0,[R1,#TCB_STACKF] ; os_tsk.run->stack_frame = val
  129. STMDB R12!,{R4-R11} ; Save Old context
  130. STR R12,[R1,#TCB_TSTACK] ; Update os_tsk.run->tsk_stack
  131. PUSH {R2,R3}
  132. BL rt_stk_check ; Check for Stack overflow
  133. POP {R2,R3}
  134. SVC_ContextRestore
  135. STR R2,[R3] ; os_tsk.run = os_tsk.next
  136. LDR R12,[R2,#TCB_TSTACK] ; os_tsk.next->tsk_stack
  137. LDMIA R12!,{R4-R11} ; Restore New Context
  138. LDRB R0,[R2,#TCB_STACKF] ; Stack Frame
  139. CMP R0,#0 ; Basic/Extended Stack Frame
  140. MVNEQ LR,#:NOT:0xFFFFFFFD ; set EXC_RETURN value
  141. MVNNE LR,#:NOT:0xFFFFFFED
  142. VLDMIANE R12!,{S16-S31} ; restore VFP hi-registers
  143. MSR PSP,R12 ; Write PSP
  144. SVC_Exit
  145. #ifdef IFX_XMC4XXX
  146. PUSH {LR}
  147. POP {PC}
  148. #else
  149. BX LR
  150. #endif
  151. /*------------------- User SVC ------------------------------*/
  152. SVC_User
  153. PUSH {R4,LR} ; Save Registers
  154. LDR R2,=SVC_Count
  155. LDR R2,[R2]
  156. CMP R1,R2
  157. BHI SVC_Done ; Overflow
  158. LDR R4,=SVC_Table-4
  159. LDR R4,[R4,R1,LSL #2] ; Load SVC Function Address
  160. LDM R0,{R0-R3,R12} ; Read R0-R3,R12 from stack
  161. BLX R4 ; Call SVC Function
  162. MRS R12,PSP
  163. STM R12,{R0-R3} ; Function return values
  164. SVC_Done
  165. POP {R4,PC} ; RETI
  166. ALIGN
  167. }
  168. /*-------------------------- PendSV_Handler ---------------------------------*/
  169. __asm void PendSV_Handler (void) {
  170. PRESERVE8
  171. #ifdef IFX_XMC4XXX
  172. EXPORT PendSV_Handler_Veneer
  173. PendSV_Handler_Veneer
  174. #endif
  175. PUSH {R4,LR} ; Save EXC_RETURN
  176. BL __cpp(rt_pop_req)
  177. Sys_Switch
  178. POP {R4,LR} ; Restore EXC_RETURN
  179. LDR R3,=__cpp(&os_tsk)
  180. LDM R3,{R1,R2} ; os_tsk.run, os_tsk.next
  181. CMP R1,R2
  182. #ifdef IFX_XMC4XXX
  183. PUSHEQ {LR}
  184. POPEQ {PC}
  185. #else
  186. BXEQ LR ; RETI, no task switch
  187. #endif
  188. MRS R12,PSP ; Read PSP
  189. TST LR,#0x10 ; is it extended frame?
  190. VSTMDBEQ R12!,{S16-S31} ; yes, stack also VFP hi-regs
  191. MOVEQ R0,#0x01 ; os_tsk->stack_frame val
  192. MOVNE R0,#0x00
  193. STRB R0,[R1,#TCB_STACKF] ; os_tsk.run->stack_frame = val
  194. STMDB R12!,{R4-R11} ; Save Old context
  195. STR R12,[R1,#TCB_TSTACK] ; Update os_tsk.run->tsk_stack
  196. PUSH {R2,R3}
  197. BL rt_stk_check ; Check for Stack overflow
  198. POP {R2,R3}
  199. STR R2,[R3] ; os_tsk.run = os_tsk.next
  200. LDR R12,[R2,#TCB_TSTACK] ; os_tsk.next->tsk_stack
  201. LDMIA R12!,{R4-R11} ; Restore New Context
  202. LDRB R0,[R2,#TCB_STACKF] ; Stack Frame
  203. CMP R0,#0 ; Basic/Extended Stack Frame
  204. MVNEQ LR,#:NOT:0xFFFFFFFD ; set EXC_RETURN value
  205. MVNNE LR,#:NOT:0xFFFFFFED
  206. VLDMIANE R12!,{S16-S31} ; restore VFP hi-regs
  207. MSR PSP,R12 ; Write PSP
  208. Sys_Exit
  209. #ifdef IFX_XMC4XXX
  210. PUSH {LR}
  211. POP {PC}
  212. #else
  213. BX LR ; Return to Thread Mode
  214. #endif
  215. ALIGN
  216. }
  217. /*-------------------------- SysTick_Handler --------------------------------*/
  218. __asm void SysTick_Handler (void) {
  219. PRESERVE8
  220. #ifdef IFX_XMC4XXX
  221. EXPORT SysTick_Handler_Veneer
  222. SysTick_Handler_Veneer
  223. #endif
  224. PUSH {R4,LR} ; Save EXC_RETURN
  225. BL __cpp(rt_systick)
  226. B Sys_Switch
  227. ALIGN
  228. }
  229. /*-------------------------- OS_Tick_Handler --------------------------------*/
  230. __asm void OS_Tick_Handler (void) {
  231. PRESERVE8
  232. PUSH {R4,LR} ; Save EXC_RETURN
  233. BL __cpp(os_tick_irqack)
  234. BL __cpp(rt_systick)
  235. B Sys_Switch
  236. ALIGN
  237. }
  238. /*----------------------------------------------------------------------------
  239. * end of file
  240. *---------------------------------------------------------------------------*/