1;/*
2; * FreeRTOS Kernel V11.1.0
3; * Copyright (C) 2021 Amazon.com, Inc. or its affiliates. All Rights Reserved.
4; *
5; * SPDX-License-Identifier: MIT
6; *
7; * Permission is hereby granted, free of charge, to any person obtaining a copy of
8; * this software and associated documentation files (the "Software"), to deal in
9; * the Software without restriction, including without limitation the rights to
10; * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
11; * the Software, and to permit persons to whom the Software is furnished to do so,
12; * subject to the following conditions:
13; *
14; * The above copyright notice and this permission notice shall be included in all
15; * copies or substantial portions of the Software.
16; *
17; * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18; * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
19; * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
20; * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
21; * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
22; * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23; *
24; * https://www.FreeRTOS.org
25; * https://github.com/FreeRTOS
26; *
27; */
28
29#include "FreeRTOSConfig.h"
30
31    SECTION .text:CODE:ROOT(2)
32    arm
33
34    /* Variables and functions. */
35    EXTERN pxCurrentTCB
36    EXTERN vTaskSwitchContext
37    EXTERN vApplicationIRQHandler
38    EXTERN ulPortInterruptNesting
39    EXTERN ulPortTaskHasFPUContext
40    EXTERN ulPortYieldRequired
41    EXTERN ulCriticalNesting
42
43    PUBLIC FreeRTOS_IRQ_Handler
44    PUBLIC FreeRTOS_SVC_Handler
45    PUBLIC vPortRestoreTaskContext
46
47SYS_MODE            EQU     0x1f
48SVC_MODE            EQU     0x13
49IRQ_MODE            EQU     0x12
50
51portSAVE_CONTEXT MACRO
52
53    /* Save the LR and SPSR onto the system mode stack before switching to
54    system mode to save the remaining system mode registers. */
55    SRSDB   sp!, #SYS_MODE
56    CPS     #SYS_MODE
57    PUSH    {R0-R12, R14}
58
59    /* Push the critical nesting count. */
60    LDR     R2, =ulCriticalNesting
61    LDR     R1, [R2]
62    PUSH    {R1}
63
64    /* Does the task have a floating point context that needs saving?  If
65    ulPortTaskHasFPUContext is 0 then no. */
66    LDR     R2, =ulPortTaskHasFPUContext
67    LDR     R3, [R2]
68    CMP     R3, #0
69
70    /* Save the floating point context, if any. */
71    FMRXNE  R1,  FPSCR
72    VPUSHNE {D0-D15}
73#if configFPU_D32 == 1
74    VPUSHNE {D16-D31}
75#endif /* configFPU_D32 */
76    PUSHNE  {R1}
77
78    /* Save ulPortTaskHasFPUContext itself. */
79    PUSH    {R3}
80
81    /* Save the stack pointer in the TCB. */
82    LDR     R0, =pxCurrentTCB
83    LDR     R1, [R0]
84    STR     SP, [R1]
85
86    ENDM
87
88; /**********************************************************************/
89
90portRESTORE_CONTEXT MACRO
91
92    /* Set the SP to point to the stack of the task being restored. */
93    LDR     R0, =pxCurrentTCB
94    LDR     R1, [R0]
95    LDR     SP, [R1]
96
97    /* Is there a floating point context to restore?  If the restored
98    ulPortTaskHasFPUContext is zero then no. */
99    LDR     R0, =ulPortTaskHasFPUContext
100    POP     {R1}
101    STR     R1, [R0]
102    CMP     R1, #0
103
104    /* Restore the floating point context, if any. */
105    POPNE   {R0}
106#if configFPU_D32 == 1
107    VPOPNE  {D16-D31}
108#endif /* configFPU_D32 */
109    VPOPNE  {D0-D15}
110    VMSRNE  FPSCR, R0
111
112    /* Restore the critical section nesting depth. */
113    LDR     R0, =ulCriticalNesting
114    POP     {R1}
115    STR     R1, [R0]
116
117    /* Restore all system mode registers other than the SP (which is already
118    being used). */
119    POP     {R0-R12, R14}
120
121    /* Return to the task code, loading CPSR on the way. */
122    RFEIA   sp!
123
124    ENDM
125
126
127
128
129/******************************************************************************
130 * SVC handler is used to yield.
131 *****************************************************************************/
132FreeRTOS_SVC_Handler:
133    /* Save the context of the current task and select a new task to run. */
134    portSAVE_CONTEXT
135    LDR R0, =vTaskSwitchContext
136    BLX R0
137    portRESTORE_CONTEXT
138
139
140/******************************************************************************
141 * vPortRestoreTaskContext is used to start the scheduler.
142 *****************************************************************************/
143vPortRestoreTaskContext:
144    /* Switch to system mode. */
145    CPS     #SYS_MODE
146    portRESTORE_CONTEXT
147
148FreeRTOS_IRQ_Handler:
149    /* Return to the interrupted instruction. */
150    SUB     lr, lr, #4
151
152    /* Push the return address and SPSR. */
153    PUSH    {lr}
154    MRS     lr, SPSR
155    PUSH    {lr}
156
157    /* Change to supervisor mode to allow reentry. */
158    CPS     #SVC_MODE
159
160    /* Push used registers. */
161    PUSH    {r0-r3, r12}
162
163    /* Increment nesting count.  r3 holds the address of ulPortInterruptNesting
164    for future use.  r1 holds the original ulPortInterruptNesting value for
165    future use. */
166    LDR     r3, =ulPortInterruptNesting
167    LDR     r1, [r3]
168    ADD     r0, r1, #1
169    STR     r0, [r3]
170
171    /* Ensure bit 2 of the stack pointer is clear.  r2 holds the bit 2 value for
172    future use. */
173    MOV     r0, sp
174    AND     r2, r0, #4
175    SUB     sp, sp, r2
176
177    /* Call the interrupt handler. */
178    PUSH    {r0-r3, lr}
179    LDR     r1, =vApplicationIRQHandler
180    BLX     r1
181    POP     {r0-r3, lr}
182    ADD     sp, sp, r2
183
184    CPSID   i
185    DSB
186    ISB
187
188    /* Write to the EOI register. */
189    LDR     r2, =configEOI_ADDRESS
190    STR     r0, [r2]
191
192    /* Restore the old nesting count. */
193    STR     r1, [r3]
194
195    /* A context switch is never performed if the nesting count is not 0. */
196    CMP     r1, #0
197    BNE     exit_without_switch
198
199    /* Did the interrupt request a context switch?  r1 holds the address of
200    ulPortYieldRequired and r0 the value of ulPortYieldRequired for future
201    use. */
202    LDR     r1, =ulPortYieldRequired
203    LDR     r0, [r1]
204    CMP     r0, #0
205    BNE     switch_before_exit
206
207exit_without_switch:
208    /* No context switch.  Restore used registers, LR_irq and SPSR before
209    returning. */
210    POP     {r0-r3, r12}
211    CPS     #IRQ_MODE
212    POP     {LR}
213    MSR     SPSR_cxsf, LR
214    POP     {LR}
215    MOVS    PC, LR
216
217switch_before_exit:
218    /* A context switch is to be performed.  Clear the context switch pending
219    flag. */
220    MOV     r0, #0
221    STR     r0, [r1]
222
223    /* Restore used registers, LR-irq and SPSR before saving the context
224    to the task stack. */
225    POP     {r0-r3, r12}
226    CPS     #IRQ_MODE
227    POP     {LR}
228    MSR     SPSR_cxsf, LR
229    POP     {LR}
230    portSAVE_CONTEXT
231
232    /* Call the function that selects the new task to execute.
233    vTaskSwitchContext() if vTaskSwitchContext() uses LDRD or STRD
234    instructions, or 8 byte aligned stack allocated data.  LR does not need
235    saving as a new LR will be loaded by portRESTORE_CONTEXT anyway. */
236    LDR     R0, =vTaskSwitchContext
237    BLX     R0
238
239    /* Restore the context of, and branch to, the task selected to execute
240    next. */
241    portRESTORE_CONTEXT
242
243    END
244