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/* Including FreeRTOSConfig.h here will cause build errors if the header file
29contains code not understood by the assembler - for example the 'extern' keyword.
30To avoid errors place any such code inside a #ifdef __ICCARM__/#endif block so
31the code is included in C files but excluded by the preprocessor in assembly
32files (__ICCARM__ is defined by the IAR C compiler but not by the IAR assembler. */
33#include "FreeRTOSConfig.h"
34
35/* System call numbers includes. */
36#include "mpu_syscall_numbers.h"
37
38#ifndef configUSE_MPU_WRAPPERS_V1
39    #define configUSE_MPU_WRAPPERS_V1 0
40#endif
41
42    EXTERN pxCurrentTCB
43    EXTERN xSecureContext
44    EXTERN vTaskSwitchContext
45    EXTERN vPortSVCHandler_C
46    EXTERN SecureContext_SaveContext
47    EXTERN SecureContext_LoadContext
48#if ( ( configENABLE_MPU == 1 ) && ( configUSE_MPU_WRAPPERS_V1 == 0 ) )
49    EXTERN vSystemCallEnter
50    EXTERN vSystemCallExit
51#endif
52
53    PUBLIC xIsPrivileged
54    PUBLIC vResetPrivilege
55    PUBLIC vPortAllocateSecureContext
56    PUBLIC vRestoreContextOfFirstTask
57    PUBLIC vRaisePrivilege
58    PUBLIC vStartFirstTask
59    PUBLIC ulSetInterruptMask
60    PUBLIC vClearInterruptMask
61    PUBLIC PendSV_Handler
62    PUBLIC SVC_Handler
63    PUBLIC vPortFreeSecureContext
64/*-----------------------------------------------------------*/
65
66/*---------------- Unprivileged Functions -------------------*/
67
68/*-----------------------------------------------------------*/
69
70    SECTION .text:CODE:NOROOT(2)
71    THUMB
72/*-----------------------------------------------------------*/
73
74xIsPrivileged:
75    mrs r0, control                         /* r0 = CONTROL. */
76    tst r0, #1                              /* Perform r0 & 1 (bitwise AND) and update the conditions flag. */
77    ite ne
78    movne r0, #0                            /* CONTROL[0]!=0. Return false to indicate that the processor is not privileged. */
79    moveq r0, #1                            /* CONTROL[0]==0. Return true to indicate that the processor is not privileged. */
80    bx lr                                   /* Return. */
81/*-----------------------------------------------------------*/
82
83vResetPrivilege:
84    mrs r0, control                         /* r0 = CONTROL. */
85    orr r0, r0, #1                          /* r0 = r0 | 1. */
86    msr control, r0                         /* CONTROL = r0. */
87    bx lr                                   /* Return to the caller. */
88/*-----------------------------------------------------------*/
89
90vPortAllocateSecureContext:
91    svc 100                                 /* Secure context is allocated in the supervisor call. portSVC_ALLOCATE_SECURE_CONTEXT = 100. */
92    bx lr                                   /* Return. */
93/*-----------------------------------------------------------*/
94
95/*----------------- Privileged Functions --------------------*/
96
97/*-----------------------------------------------------------*/
98
99    SECTION privileged_functions:CODE:NOROOT(2)
100    THUMB
101/*-----------------------------------------------------------*/
102
103#if ( configENABLE_MPU == 1 )
104
105vRestoreContextOfFirstTask:
106    program_mpu_first_task:
107        ldr r3, =pxCurrentTCB               /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
108        ldr r0, [r3]                        /* r0 = pxCurrentTCB. */
109
110        dmb                                 /* Complete outstanding transfers before disabling MPU. */
111        ldr r1, =0xe000ed94                 /* r1 = 0xe000ed94 [Location of MPU_CTRL]. */
112        ldr r2, [r1]                        /* Read the value of MPU_CTRL. */
113        bic r2, #1                          /* r2 = r2 & ~1 i.e. Clear the bit 0 in r2. */
114        str r2, [r1]                        /* Disable MPU. */
115
116        adds r0, #4                         /* r0 = r0 + 4. r0 now points to MAIR0 in TCB. */
117        ldr r1, [r0]                        /* r1 = *r0 i.e. r1 = MAIR0. */
118        ldr r2, =0xe000edc0                 /* r2 = 0xe000edc0 [Location of MAIR0]. */
119        str r1, [r2]                        /* Program MAIR0. */
120
121        adds r0, #4                         /* r0 = r0 + 4. r0 now points to first RBAR in TCB. */
122        ldr r1, =0xe000ed98                 /* r1 = 0xe000ed98 [Location of RNR]. */
123        ldr r2, =0xe000ed9c                 /* r2 = 0xe000ed9c [Location of RBAR]. */
124
125        movs r3, #4                         /* r3 = 4. */
126        str r3, [r1]                        /* Program RNR = 4. */
127        ldmia r0!, {r4-r11}                 /* Read 4 set of RBAR/RLAR registers from TCB. */
128        stmia r2, {r4-r11}                  /* Write 4 set of RBAR/RLAR registers using alias registers. */
129
130    #if ( configTOTAL_MPU_REGIONS == 16 )
131        movs r3, #8                         /* r3 = 8. */
132        str r3, [r1]                        /* Program RNR = 8. */
133        ldmia r0!, {r4-r11}                 /* Read 4 set of RBAR/RLAR registers from TCB. */
134        stmia r2, {r4-r11}                  /* Write 4 set of RBAR/RLAR registers using alias registers. */
135        movs r3, #12                        /* r3 = 12. */
136        str r3, [r1]                        /* Program RNR = 12. */
137        ldmia r0!, {r4-r11}                 /* Read 4 set of RBAR/RLAR registers from TCB. */
138        stmia r2, {r4-r11}                  /* Write 4 set of RBAR/RLAR registers using alias registers. */
139    #endif /* configTOTAL_MPU_REGIONS == 16 */
140
141        ldr r1, =0xe000ed94                 /* r1 = 0xe000ed94 [Location of MPU_CTRL]. */
142        ldr r2, [r1]                        /* Read the value of MPU_CTRL. */
143        orr r2, #1                          /* r2 = r1 | 1 i.e. Set the bit 0 in r2. */
144        str r2, [r1]                        /* Enable MPU. */
145        dsb                                 /* Force memory writes before continuing. */
146
147    restore_context_first_task:
148        ldr r3, =pxCurrentTCB               /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
149        ldr r1, [r3]                        /* r1 = pxCurrentTCB.*/
150        ldr r2, [r1]                        /* r2 = Location of saved context in TCB. */
151
152    restore_special_regs_first_task:
153        ldmdb r2!, {r0, r3-r5, lr}          /* r0 = xSecureContext, r3 = original PSP, r4 = PSPLIM, r5 = CONTROL, LR restored. */
154        msr psp, r3
155        msr psplim, r4
156        msr control, r5
157        ldr r4, =xSecureContext             /* Read the location of xSecureContext i.e. &( xSecureContext ). */
158        str r0, [r4]                        /* Restore xSecureContext. */
159
160    restore_general_regs_first_task:
161        ldmdb r2!, {r4-r11}                 /* r4-r11 contain hardware saved context. */
162        stmia r3!, {r4-r11}                 /* Copy the hardware saved context on the task stack. */
163        ldmdb r2!, {r4-r11}                 /* r4-r11 restored. */
164
165    restore_context_done_first_task:
166        str r2, [r1]                        /* Save the location where the context should be saved next as the first member of TCB. */
167        mov r0, #0
168        msr basepri, r0                     /* Ensure that interrupts are enabled when the first task starts. */
169        bx lr
170
171#else /* configENABLE_MPU */
172
173vRestoreContextOfFirstTask:
174    ldr  r2, =pxCurrentTCB                  /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
175    ldr  r3, [r2]                           /* Read pxCurrentTCB. */
176    ldr  r0, [r3]                           /* Read top of stack from TCB - The first item in pxCurrentTCB is the task top of stack. */
177
178    ldm  r0!, {r1-r3}                       /* Read from stack - r1 = xSecureContext, r2 = PSPLIM and r3 = EXC_RETURN. */
179    ldr  r4, =xSecureContext
180    str  r1, [r4]                           /* Set xSecureContext to this task's value for the same. */
181    msr  psplim, r2                         /* Set this task's PSPLIM value. */
182    movs r1, #2                             /* r1 = 2. */
183    msr  CONTROL, r1                        /* Switch to use PSP in the thread mode. */
184    adds r0, #32                            /* Discard everything up to r0. */
185    msr  psp, r0                            /* This is now the new top of stack to use in the task. */
186    isb
187    mov  r0, #0
188    msr  basepri, r0                        /* Ensure that interrupts are enabled when the first task starts. */
189    bx   r3                                 /* Finally, branch to EXC_RETURN. */
190
191#endif /* configENABLE_MPU */
192/*-----------------------------------------------------------*/
193
194vRaisePrivilege:
195    mrs  r0, control                        /* Read the CONTROL register. */
196    bic r0, r0, #1                          /* Clear the bit 0. */
197    msr  control, r0                        /* Write back the new CONTROL value. */
198    bx lr                                   /* Return to the caller. */
199/*-----------------------------------------------------------*/
200
201vStartFirstTask:
202    ldr r0, =0xe000ed08                     /* Use the NVIC offset register to locate the stack. */
203    ldr r0, [r0]                            /* Read the VTOR register which gives the address of vector table. */
204    ldr r0, [r0]                            /* The first entry in vector table is stack pointer. */
205    msr msp, r0                             /* Set the MSP back to the start of the stack. */
206    cpsie i                                 /* Globally enable interrupts. */
207    cpsie f
208    dsb
209    isb
210    svc 102                                 /* System call to start the first task. portSVC_START_SCHEDULER = 102. */
211/*-----------------------------------------------------------*/
212
213ulSetInterruptMask:
214    mrs r0, basepri                         /* r0 = basepri. Return original basepri value. */
215    mov r1, #configMAX_SYSCALL_INTERRUPT_PRIORITY
216    msr basepri, r1                         /* Disable interrupts upto configMAX_SYSCALL_INTERRUPT_PRIORITY. */
217    dsb
218    isb
219    bx lr                                   /* Return. */
220/*-----------------------------------------------------------*/
221
222vClearInterruptMask:
223    msr basepri, r0                         /* basepri = ulMask. */
224    dsb
225    isb
226    bx lr                                   /* Return. */
227/*-----------------------------------------------------------*/
228
229#if ( configENABLE_MPU == 1 )
230
231PendSV_Handler:
232    ldr r3, =xSecureContext                 /* Read the location of xSecureContext i.e. &( xSecureContext ). */
233    ldr r0, [r3]                            /* Read xSecureContext - Value of xSecureContext must be in r0 as it is used as a parameter later. */
234    ldr r3, =pxCurrentTCB                   /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
235    ldr r1, [r3]                            /* Read pxCurrentTCB - Value of pxCurrentTCB must be in r1 as it is used as a parameter later. */
236    ldr r2, [r1]                            /* r2 = Location in TCB where the context should be saved. */
237
238    cbz r0, save_ns_context                 /* No secure context to save. */
239    save_s_context:
240        push {r0-r2, lr}
241        bl SecureContext_SaveContext        /* Params are in r0 and r1. r0 = xSecureContext and r1 = pxCurrentTCB. */
242        pop {r0-r2, lr}
243
244    save_ns_context:
245        mov r3, lr                          /* r3 = LR (EXC_RETURN). */
246        lsls r3, r3, #25                    /* r3 = r3 << 25. Bit[6] of EXC_RETURN is 1 if secure stack was used, 0 if non-secure stack was used to store stack frame. */
247        bmi save_special_regs               /* r3 < 0 ==> Bit[6] in EXC_RETURN is 1 ==> secure stack was used to store the stack frame. */
248
249    save_general_regs:
250        mrs r3, psp
251
252    #if ( ( configENABLE_FPU == 1 ) || ( configENABLE_MVE == 1 ) )
253        add r3, r3, #0x20                   /* Move r3 to location where s0 is saved. */
254        tst lr, #0x10
255        ittt eq
256        vstmiaeq r2!, {s16-s31}             /* Store s16-s31. */
257        vldmiaeq r3, {s0-s16}               /* Copy hardware saved FP context into s0-s16. */
258        vstmiaeq r2!, {s0-s16}              /* Store hardware saved FP context. */
259        sub r3, r3, #0x20                   /* Set r3 back to the location of hardware saved context. */
260    #endif /* configENABLE_FPU || configENABLE_MVE */
261
262        stmia r2!, {r4-r11}                 /* Store r4-r11. */
263        ldmia r3, {r4-r11}                  /* Copy the hardware saved context into r4-r11. */
264        stmia r2!, {r4-r11}                 /* Store the hardware saved context. */
265
266    save_special_regs:
267        mrs r3, psp                         /* r3 = PSP. */
268        mrs r4, psplim                      /* r4 = PSPLIM. */
269        mrs r5, control                     /* r5 = CONTROL. */
270        stmia r2!, {r0, r3-r5, lr}          /* Store xSecureContext, original PSP (after hardware has saved context), PSPLIM, CONTROL and LR. */
271        str r2, [r1]                        /* Save the location from where the context should be restored as the first member of TCB. */
272
273    select_next_task:
274        mov r0, #configMAX_SYSCALL_INTERRUPT_PRIORITY
275        msr basepri, r0                     /* Disable interrupts upto configMAX_SYSCALL_INTERRUPT_PRIORITY. */
276        dsb
277        isb
278        bl vTaskSwitchContext
279        mov r0, #0                          /* r0 = 0. */
280        msr basepri, r0                     /* Enable interrupts. */
281
282    program_mpu:
283        ldr r3, =pxCurrentTCB               /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
284        ldr r0, [r3]                        /* r0 = pxCurrentTCB.*/
285
286        dmb                                 /* Complete outstanding transfers before disabling MPU. */
287        ldr r1, =0xe000ed94                 /* r1 = 0xe000ed94 [Location of MPU_CTRL]. */
288        ldr r2, [r1]                        /* Read the value of MPU_CTRL. */
289        bic r2, #1                          /* r2 = r2 & ~1 i.e. Clear the bit 0 in r2. */
290        str r2, [r1]                        /* Disable MPU. */
291
292        adds r0, #4                         /* r0 = r0 + 4. r0 now points to MAIR0 in TCB. */
293        ldr r1, [r0]                        /* r1 = *r0 i.e. r1 = MAIR0. */
294        ldr r2, =0xe000edc0                 /* r2 = 0xe000edc0 [Location of MAIR0]. */
295        str r1, [r2]                        /* Program MAIR0. */
296
297        adds r0, #4                         /* r0 = r0 + 4. r0 now points to first RBAR in TCB. */
298        ldr r1, =0xe000ed98                 /* r1 = 0xe000ed98 [Location of RNR]. */
299        ldr r2, =0xe000ed9c                 /* r2 = 0xe000ed9c [Location of RBAR]. */
300
301        movs r3, #4                         /* r3 = 4. */
302        str r3, [r1]                        /* Program RNR = 4. */
303        ldmia r0!, {r4-r11}                 /* Read 4 sets of RBAR/RLAR registers from TCB. */
304        stmia r2, {r4-r11}                  /* Write 4 set of RBAR/RLAR registers using alias registers. */
305
306    #if ( configTOTAL_MPU_REGIONS == 16 )
307        movs r3, #8                         /* r3 = 8. */
308        str r3, [r1]                        /* Program RNR = 8. */
309        ldmia r0!, {r4-r11}                 /* Read 4 sets of RBAR/RLAR registers from TCB. */
310        stmia r2, {r4-r11}                  /* Write 4 set of RBAR/RLAR registers using alias registers. */
311        movs r3, #12                        /* r3 = 12. */
312        str r3, [r1]                        /* Program RNR = 12. */
313        ldmia r0!, {r4-r11}                 /* Read 4 sets of RBAR/RLAR registers from TCB. */
314        stmia r2, {r4-r11}                  /* Write 4 set of RBAR/RLAR registers using alias registers. */
315    #endif /* configTOTAL_MPU_REGIONS == 16 */
316
317       ldr r1, =0xe000ed94                  /* r1 = 0xe000ed94 [Location of MPU_CTRL]. */
318       ldr r2, [r1]                         /* Read the value of MPU_CTRL. */
319       orr r2, #1                           /* r2 = r2 | 1 i.e. Set the bit 0 in r2. */
320       str r2, [r1]                         /* Enable MPU. */
321       dsb                                  /* Force memory writes before continuing. */
322
323    restore_context:
324        ldr r3, =pxCurrentTCB               /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
325        ldr r1, [r3]                        /* r1 = pxCurrentTCB.*/
326        ldr r2, [r1]                        /* r2 = Location of saved context in TCB. */
327
328    restore_special_regs:
329        ldmdb r2!, {r0, r3-r5, lr}          /* r0 = xSecureContext, r3 = original PSP, r4 = PSPLIM, r5 = CONTROL, LR restored. */
330        msr psp, r3
331        msr psplim, r4
332        msr control, r5
333        ldr r4, =xSecureContext             /* Read the location of xSecureContext i.e. &( xSecureContext ). */
334        str r0, [r4]                        /* Restore xSecureContext. */
335        cbz r0, restore_ns_context          /* No secure context to restore. */
336
337    restore_s_context:
338        push {r1-r3, lr}
339        bl SecureContext_LoadContext        /* Params are in r0 and r1. r0 = xSecureContext and r1 = pxCurrentTCB. */
340        pop {r1-r3, lr}
341
342    restore_ns_context:
343        mov r0, lr                          /* r0 = LR (EXC_RETURN). */
344        lsls r0, r0, #25                    /* r0 = r0 << 25. Bit[6] of EXC_RETURN is 1 if secure stack was used, 0 if non-secure stack was used to store stack frame. */
345        bmi restore_context_done            /* r0 < 0 ==> Bit[6] in EXC_RETURN is 1 ==> secure stack was used to store the stack frame. */
346
347    restore_general_regs:
348        ldmdb r2!, {r4-r11}                 /* r4-r11 contain hardware saved context. */
349        stmia r3!, {r4-r11}                 /* Copy the hardware saved context on the task stack. */
350        ldmdb r2!, {r4-r11}                 /* r4-r11 restored. */
351
352    #if ( ( configENABLE_FPU == 1 ) || ( configENABLE_MVE == 1 ) )
353        tst lr, #0x10
354        ittt eq
355        vldmdbeq r2!, {s0-s16}              /* s0-s16 contain hardware saved FP context. */
356        vstmiaeq r3!, {s0-s16}              /* Copy hardware saved FP context on the task stack. */
357        vldmdbeq r2!, {s16-s31}             /* Restore s16-s31. */
358    #endif /* configENABLE_FPU || configENABLE_MVE */
359
360    restore_context_done:
361        str r2, [r1]                        /* Save the location where the context should be saved next as the first member of TCB. */
362        bx lr
363
364#else /* configENABLE_MPU */
365
366PendSV_Handler:
367    ldr r3, =xSecureContext                 /* Read the location of xSecureContext i.e. &( xSecureContext ). */
368    ldr r0, [r3]                            /* Read xSecureContext - Value of xSecureContext must be in r0 as it is used as a parameter later. */
369    ldr r3, =pxCurrentTCB                   /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
370    ldr r1, [r3]                            /* Read pxCurrentTCB - Value of pxCurrentTCB must be in r1 as it is used as a parameter later. */
371    mrs r2, psp                             /* Read PSP in r2. */
372
373    cbz r0, save_ns_context                 /* No secure context to save. */
374    push {r0-r2, r14}
375    bl SecureContext_SaveContext            /* Params are in r0 and r1. r0 = xSecureContext and r1 = pxCurrentTCB. */
376    pop {r0-r3}                             /* LR is now in r3. */
377    mov lr, r3                              /* LR = r3. */
378    lsls r1, r3, #25                        /* r1 = r3 << 25. Bit[6] of EXC_RETURN is 1 if secure stack was used, 0 if non-secure stack was used to store stack frame. */
379    bpl save_ns_context                     /* bpl - branch if positive or zero. If r1 >= 0 ==> Bit[6] in EXC_RETURN is 0 i.e. non-secure stack was used. */
380
381    ldr r3, =pxCurrentTCB                   /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
382    ldr r1, [r3]                            /* Read pxCurrentTCB. */
383    subs r2, r2, #12                        /* Make space for xSecureContext, PSPLIM and LR on the stack. */
384    str r2, [r1]                            /* Save the new top of stack in TCB. */
385    mrs r1, psplim                          /* r1 = PSPLIM. */
386    mov r3, lr                              /* r3 = LR/EXC_RETURN. */
387    stmia r2!, {r0, r1, r3}                 /* Store xSecureContext, PSPLIM and LR on the stack. */
388    b select_next_task
389
390    save_ns_context:
391        ldr r3, =pxCurrentTCB               /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
392        ldr r1, [r3]                        /* Read pxCurrentTCB. */
393    #if ( ( configENABLE_FPU == 1 ) || ( configENABLE_MVE == 1 ) )
394        tst lr, #0x10                       /* Test Bit[4] in LR. Bit[4] of EXC_RETURN is 0 if the Extended Stack Frame is in use. */
395        it eq
396        vstmdbeq r2!, {s16-s31}             /* Store the additional FP context registers which are not saved automatically. */
397    #endif /* configENABLE_FPU || configENABLE_MVE */
398        subs r2, r2, #44                    /* Make space for xSecureContext, PSPLIM, LR and the remaining registers on the stack. */
399        str r2, [r1]                        /* Save the new top of stack in TCB. */
400        adds r2, r2, #12                    /* r2 = r2 + 12. */
401        stm r2, {r4-r11}                    /* Store the registers that are not saved automatically. */
402        mrs r1, psplim                      /* r1 = PSPLIM. */
403        mov r3, lr                          /* r3 = LR/EXC_RETURN. */
404        subs r2, r2, #12                    /* r2 = r2 - 12. */
405        stmia r2!, {r0, r1, r3}             /* Store xSecureContext, PSPLIM and LR on the stack. */
406
407    select_next_task:
408        mov r0, #configMAX_SYSCALL_INTERRUPT_PRIORITY
409        msr basepri, r0                     /* Disable interrupts upto configMAX_SYSCALL_INTERRUPT_PRIORITY. */
410        dsb
411        isb
412        bl vTaskSwitchContext
413        mov r0, #0                          /* r0 = 0. */
414        msr basepri, r0                     /* Enable interrupts. */
415
416        ldr r3, =pxCurrentTCB               /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
417        ldr r1, [r3]                        /* Read pxCurrentTCB. */
418        ldr r2, [r1]                        /* The first item in pxCurrentTCB is the task top of stack. r2 now points to the top of stack. */
419
420        ldmia r2!, {r0, r1, r4}             /* Read from stack - r0 = xSecureContext, r1 = PSPLIM and r4 = LR. */
421        msr psplim, r1                      /* Restore the PSPLIM register value for the task. */
422        mov lr, r4                          /* LR = r4. */
423        ldr r3, =xSecureContext             /* Read the location of xSecureContext i.e. &( xSecureContext ). */
424        str r0, [r3]                        /* Restore the task's xSecureContext. */
425        cbz r0, restore_ns_context          /* If there is no secure context for the task, restore the non-secure context. */
426        ldr r3, =pxCurrentTCB               /* Read the location of pxCurrentTCB i.e. &( pxCurrentTCB ). */
427        ldr r1, [r3]                        /* Read pxCurrentTCB. */
428        push {r2, r4}
429        bl SecureContext_LoadContext        /* Restore the secure context. Params are in r0 and r1. r0 = xSecureContext and r1 = pxCurrentTCB. */
430        pop {r2, r4}
431        mov lr, r4                          /* LR = r4. */
432        lsls r1, r4, #25                    /* r1 = r4 << 25. Bit[6] of EXC_RETURN is 1 if secure stack was used, 0 if non-secure stack was used to store stack frame. */
433        bpl restore_ns_context              /* bpl - branch if positive or zero. If r1 >= 0 ==> Bit[6] in EXC_RETURN is 0 i.e. non-secure stack was used. */
434        msr psp, r2                         /* Remember the new top of stack for the task. */
435        bx lr
436
437    restore_ns_context:
438        ldmia r2!, {r4-r11}                 /* Restore the registers that are not automatically restored. */
439    #if ( ( configENABLE_FPU == 1 ) || ( configENABLE_MVE == 1 ) )
440        tst lr, #0x10                       /* Test Bit[4] in LR. Bit[4] of EXC_RETURN is 0 if the Extended Stack Frame is in use. */
441        it eq
442        vldmiaeq r2!, {s16-s31}             /* Restore the additional FP context registers which are not restored automatically. */
443    #endif /* configENABLE_FPU || configENABLE_MVE */
444        msr psp, r2                         /* Remember the new top of stack for the task. */
445        bx lr
446
447#endif /* configENABLE_MPU */
448/*-----------------------------------------------------------*/
449
450#if ( ( configENABLE_MPU == 1 ) && ( configUSE_MPU_WRAPPERS_V1 == 0 ) )
451
452SVC_Handler:
453    tst lr, #4
454    ite eq
455    mrseq r0, msp
456    mrsne r0, psp
457
458    ldr r1, [r0, #24]
459    ldrb r2, [r1, #-2]
460    cmp r2, #NUM_SYSTEM_CALLS
461    blt syscall_enter
462    cmp r2, #104            /* portSVC_SYSTEM_CALL_EXIT. */
463    beq syscall_exit
464    b vPortSVCHandler_C
465
466    syscall_enter:
467        mov r1, lr
468        b vSystemCallEnter
469
470    syscall_exit:
471        mov r1, lr
472        b vSystemCallExit
473
474#else /* ( configENABLE_MPU == 1 ) && ( configUSE_MPU_WRAPPERS_V1 == 0 ) */
475
476SVC_Handler:
477    tst lr, #4
478    ite eq
479    mrseq r0, msp
480    mrsne r0, psp
481    b vPortSVCHandler_C
482
483#endif /* ( configENABLE_MPU == 1 ) && ( configUSE_MPU_WRAPPERS_V1 == 0 ) */
484/*-----------------------------------------------------------*/
485
486vPortFreeSecureContext:
487    /* r0 = uint32_t *pulTCB. */
488    ldr r2, [r0]                            /* The first item in the TCB is the top of the stack. */
489    ldr r1, [r2]                            /* The first item on the stack is the task's xSecureContext. */
490    cmp r1, #0                              /* Raise svc if task's xSecureContext is not NULL. */
491    it ne
492    svcne 101                               /* Secure context is freed in the supervisor call. portSVC_FREE_SECURE_CONTEXT = 101. */
493    bx lr                                   /* Return. */
494/*-----------------------------------------------------------*/
495
496    END
497