HAL_CM4.s 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. /*----------------------------------------------------------------------------
  2. * CMSIS-RTOS - RTX
  3. *----------------------------------------------------------------------------
  4. * Name: HAL_CM4.S
  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. NAME HAL_CM4.S
  26. #define TCB_STACKF 37
  27. #define TCB_TSTACK 40
  28. EXTERN os_flags
  29. EXTERN os_tsk
  30. EXTERN rt_alloc_box
  31. EXTERN rt_free_box
  32. EXTERN rt_stk_check
  33. EXTERN rt_pop_req
  34. EXTERN rt_systick
  35. EXTERN os_tick_irqack
  36. EXTERN SVC_Table
  37. EXTERN SVC_Count
  38. /*----------------------------------------------------------------------------
  39. * Functions
  40. *---------------------------------------------------------------------------*/
  41. SECTION .text:CODE:NOROOT(2)
  42. THUMB
  43. /*--------------------------- rt_set_PSP ------------------------------------*/
  44. ; void rt_set_PSP (U32 stack);
  45. PUBLIC rt_set_PSP
  46. rt_set_PSP:
  47. MSR PSP,R0
  48. BX LR
  49. /*--------------------------- rt_get_PSP ------------------------------------*/
  50. ; U32 rt_get_PSP (void);
  51. PUBLIC rt_get_PSP
  52. rt_get_PSP:
  53. MRS R0,PSP
  54. BX LR
  55. /*--------------------------- os_set_env ------------------------------------*/
  56. ; void os_set_env (void);
  57. /* Switch to Unprivileged/Privileged Thread mode, use PSP. */
  58. PUBLIC os_set_env
  59. os_set_env:
  60. MOV R0,SP /* PSP = MSP */
  61. MSR PSP,R0
  62. LDR R0,=os_flags
  63. LDRB R0,[R0]
  64. LSLS R0,#31
  65. ITE NE
  66. MOVNE R0,#0x02 /* Privileged Thread mode, use PSP */
  67. MOVEQ R0,#0x03 /* Unprivileged Thread mode, use PSP */
  68. MSR CONTROL,R0
  69. BX LR
  70. /*--------------------------- _alloc_box ------------------------------------*/
  71. ; void *_alloc_box (void *box_mem);
  72. /* Function wrapper for Unprivileged/Privileged mode. */
  73. PUBLIC _alloc_box
  74. _alloc_box:
  75. LDR R12,=rt_alloc_box
  76. MRS R3,IPSR
  77. LSLS R3,#24
  78. IT NE
  79. BXNE R12
  80. MRS R3,CONTROL
  81. LSLS R3,#31
  82. IT EQ
  83. BXEQ R12
  84. SVC 0
  85. BX LR
  86. /*--------------------------- _free_box -------------------------------------*/
  87. ; U32 _free_box (void *box_mem, void *box);
  88. /* Function wrapper for Unprivileged/Privileged mode. */
  89. PUBLIC _free_box
  90. _free_box:
  91. LDR R12,=rt_free_box
  92. MRS R3,IPSR
  93. LSLS R3,#24
  94. IT NE
  95. BXNE R12
  96. MRS R3,CONTROL
  97. LSLS R3,#31
  98. IT EQ
  99. BXEQ R12
  100. SVC 0
  101. BX LR
  102. /*-------------------------- SVC_Handler ------------------------------------*/
  103. ; void SVC_Handler (void);
  104. PUBLIC SVC_Handler
  105. SVC_Handler:
  106. MRS R0,PSP /* Read PSP */
  107. LDR R1,[R0,#24] /* Read Saved PC from Stack */
  108. LDRB R1,[R1,#-2] /* Load SVC Number */
  109. CBNZ R1,SVC_User
  110. LDM R0,{R0-R3,R12} /* Read R0-R3,R12 from stack */
  111. PUSH {R4,LR} /* Save EXC_RETURN */
  112. BLX R12 /* Call SVC Function */
  113. POP {R4,LR} /* Restore EXC_RETURN */
  114. MRS R12,PSP /* Read PSP */
  115. STM R12,{R0-R2} /* Store return values */
  116. LDR R3,=os_tsk
  117. LDM R3,{R1,R2} /* os_tsk.run, os_tsk.next */
  118. CMP R1,R2
  119. IT EQ
  120. BXEQ LR /* RETI, no task switch */
  121. CBNZ R1,SVC_ContextSave /* Runtask not deleted? */
  122. TST LR,#0x10 /* is it extended frame? */
  123. BNE SVC_ContextRestore
  124. LDR R1,=0xE000EF34
  125. LDR R0,[R1] /* Load FPCCR */
  126. BIC R0,R0,#1 /* Clear LSPACT (Lazy state) */
  127. STR R0,[R1] /* Store FPCCR */
  128. B SVC_ContextRestore
  129. SVC_ContextSave:
  130. TST LR,#0x10 /* is it extended frame? */
  131. ITTE EQ
  132. VSTMDBEQ R12!,{S16-S31} /* yes, stack also VFP hi-regs */
  133. MOVEQ R0,#0x01 /* os_tsk->stack_frame val */
  134. MOVNE R0,#0x00
  135. STRB R0,[R1,#TCB_STACKF] /* os_tsk.run->stack_frame = val */
  136. STMDB R12!,{R4-R11} /* Save Old context */
  137. STR R12,[R1,#TCB_TSTACK] /* Update os_tsk.run->tsk_stack */
  138. PUSH {R2,R3}
  139. BL rt_stk_check /* Check for Stack overflow */
  140. POP {R2,R3}
  141. SVC_ContextRestore:
  142. STR R2,[R3] /* os_tsk.run = os_tsk.next */
  143. LDR R12,[R2,#TCB_TSTACK] /* os_tsk.next->tsk_stack */
  144. LDMIA R12!,{R4-R11} /* Restore New Context */
  145. LDRB R0,[R2,#TCB_STACKF] /* Stack Frame */
  146. CMP R0,#0 /* Basic/Extended Stack Frame */
  147. ITEE EQ
  148. MVNEQ LR,#~0xFFFFFFFD /* set EXC_RETURN value */
  149. MVNNE LR,#~0xFFFFFFED
  150. VLDMIANE R12!,{S16-S31} /* restore VFP hi-registers */
  151. MSR PSP,R12 /* Write PSP */
  152. SVC_Exit:
  153. BX LR
  154. /*------------------- User SVC ------------------------------*/
  155. SVC_User:
  156. PUSH {R4,LR} /* Save Registers */
  157. LDR R2,=SVC_Count
  158. LDR R2,[R2]
  159. CMP R1,R2
  160. BHI SVC_Done /* Overflow */
  161. LDR R4,=SVC_Table-4
  162. LDR R4,[R4,R1,LSL #2] /* Load SVC Function Address */
  163. LDM R0,{R0-R3,R12} /* Read R0-R3,R12 from stack */
  164. BLX R4 /* Call SVC Function */
  165. MRS R12,PSP
  166. STM R12,{R0-R3} /* Function return values */
  167. SVC_Done:
  168. POP {R4,PC} /* RETI */
  169. /*-------------------------- PendSV_Handler ---------------------------------*/
  170. ; void PendSV_Handler (void);
  171. PUBLIC PendSV_Handler
  172. PendSV_Handler:
  173. PUSH {R4,LR} /* Save EXC_RETURN */
  174. BL rt_pop_req
  175. Sys_Switch:
  176. POP {R4,LR} /* Restore EXC_RETURN */
  177. LDR R3,=os_tsk
  178. LDM R3,{R1,R2} /* os_tsk.run, os_tsk.next */
  179. CMP R1,R2
  180. IT EQ
  181. BXEQ LR /* RETI, no task switch */
  182. MRS R12,PSP /* Read PSP */
  183. TST LR,#0x10 /* is it extended frame? */
  184. ITTE EQ
  185. VSTMDBEQ R12!,{S16-S31} /* yes, stack also VFP hi-regs */
  186. MOVEQ R0,#0x01 /* os_tsk->stack_frame val */
  187. MOVNE R0,#0x00
  188. STRB R0,[R1,#TCB_STACKF] /* os_tsk.run->stack_frame = val */
  189. STMDB R12!,{R4-R11} /* Save Old context */
  190. STR R12,[R1,#TCB_TSTACK] /* Update os_tsk.run->tsk_stack */
  191. PUSH {R2,R3}
  192. BL rt_stk_check /* Check for Stack overflow */
  193. POP {R2,R3}
  194. STR R2,[R3] /* os_tsk.run = os_tsk.next */
  195. LDR R12,[R2,#TCB_TSTACK] /* os_tsk.next->tsk_stack */
  196. LDMIA R12!,{R4-R11} /* Restore New Context */
  197. LDRB R0,[R2,#TCB_STACKF] /* Stack Frame */
  198. CMP R0,#0 /* Basic/Extended Stack Frame */
  199. ITEE EQ
  200. MVNEQ LR,#~0xFFFFFFFD /* set EXC_RETURN value */
  201. MVNNE LR,#~0xFFFFFFED
  202. VLDMIANE R12!,{S16-S31} /* restore VFP hi-registers */
  203. MSR PSP,R12 /* Write PSP */
  204. Sys_Exit:
  205. BX LR /* Return to Thread Mode */
  206. /*-------------------------- SysTick_Handler --------------------------------*/
  207. ; void SysTick_Handler (void);
  208. PUBLIC SysTick_Handler
  209. SysTick_Handler:
  210. PUSH {R4,LR} /* Save EXC_RETURN */
  211. BL rt_systick
  212. B Sys_Switch
  213. /*-------------------------- OS_Tick_Handler --------------------------------*/
  214. ; void OS_Tick_Handler (void);
  215. PUBLIC OS_Tick_Handler
  216. OS_Tick_Handler:
  217. PUSH {R4,LR} /* Save EXC_RETURN */
  218. BL os_tick_irqack
  219. BL rt_systick
  220. B Sys_Switch
  221. END
  222. /*----------------------------------------------------------------------------
  223. * end of file
  224. *---------------------------------------------------------------------------*/