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 .text 30 31 /* Variables and functions. */ 32 .extern ullMaxAPIPriorityMask 33 .extern pxCurrentTCB 34 .extern vTaskSwitchContext 35 .extern vApplicationIRQHandler 36 .extern ullPortInterruptNesting 37 .extern ullPortTaskHasFPUContext 38 .extern ullCriticalNesting 39 .extern ullPortYieldRequired 40 .extern _freertos_vector_table 41 42 .global FreeRTOS_IRQ_Handler 43 .global FreeRTOS_SWI_Handler 44 .global vPortRestoreTaskContext 45 46 47.macro portSAVE_CONTEXT 48 49 /* Switch to use the EL0 stack pointer. */ 50 MSR SPSEL, #0 51 52 /* Save the entire context. */ 53 STP X0, X1, [SP, #-0x10]! 54 STP X2, X3, [SP, #-0x10]! 55 STP X4, X5, [SP, #-0x10]! 56 STP X6, X7, [SP, #-0x10]! 57 STP X8, X9, [SP, #-0x10]! 58 STP X10, X11, [SP, #-0x10]! 59 STP X12, X13, [SP, #-0x10]! 60 STP X14, X15, [SP, #-0x10]! 61 STP X16, X17, [SP, #-0x10]! 62 STP X18, X19, [SP, #-0x10]! 63 STP X20, X21, [SP, #-0x10]! 64 STP X22, X23, [SP, #-0x10]! 65 STP X24, X25, [SP, #-0x10]! 66 STP X26, X27, [SP, #-0x10]! 67 STP X28, X29, [SP, #-0x10]! 68 STP X30, XZR, [SP, #-0x10]! 69 70 /* Save the SPSR. */ 71#if defined( GUEST ) 72 MRS X3, SPSR_EL1 73 MRS X2, ELR_EL1 74#else 75 MRS X3, SPSR_EL3 76 /* Save the ELR. */ 77 MRS X2, ELR_EL3 78#endif 79 80 STP X2, X3, [SP, #-0x10]! 81 82 /* Save the critical section nesting depth. */ 83 LDR X0, ullCriticalNestingConst 84 LDR X3, [X0] 85 86 /* Save the FPU context indicator. */ 87 LDR X0, ullPortTaskHasFPUContextConst 88 LDR X2, [X0] 89 90 /* Save the FPU context, if any (32 128-bit registers). */ 91 CMP X2, #0 92 B.EQ 1f 93 STP Q0, Q1, [SP,#-0x20]! 94 STP Q2, Q3, [SP,#-0x20]! 95 STP Q4, Q5, [SP,#-0x20]! 96 STP Q6, Q7, [SP,#-0x20]! 97 STP Q8, Q9, [SP,#-0x20]! 98 STP Q10, Q11, [SP,#-0x20]! 99 STP Q12, Q13, [SP,#-0x20]! 100 STP Q14, Q15, [SP,#-0x20]! 101 STP Q16, Q17, [SP,#-0x20]! 102 STP Q18, Q19, [SP,#-0x20]! 103 STP Q20, Q21, [SP,#-0x20]! 104 STP Q22, Q23, [SP,#-0x20]! 105 STP Q24, Q25, [SP,#-0x20]! 106 STP Q26, Q27, [SP,#-0x20]! 107 STP Q28, Q29, [SP,#-0x20]! 108 STP Q30, Q31, [SP,#-0x20]! 109 1101: 111 /* Store the critical nesting count and FPU context indicator. */ 112 STP X2, X3, [SP, #-0x10]! 113 114 LDR X0, pxCurrentTCBConst 115 LDR X1, [X0] 116 MOV X0, SP /* Move SP into X0 for saving. */ 117 STR X0, [X1] 118 119 /* Switch to use the ELx stack pointer. */ 120 MSR SPSEL, #1 121 122 .endm 123 124; /**********************************************************************/ 125 126.macro portRESTORE_CONTEXT 127 128 /* Switch to use the EL0 stack pointer. */ 129 MSR SPSEL, #0 130 131 /* Set the SP to point to the stack of the task being restored. */ 132 LDR X0, pxCurrentTCBConst 133 LDR X1, [X0] 134 LDR X0, [X1] 135 MOV SP, X0 136 137 LDP X2, X3, [SP], #0x10 /* Critical nesting and FPU context. */ 138 139 /* Set the PMR register to be correct for the current critical nesting 140 depth. */ 141 LDR X0, ullCriticalNestingConst /* X0 holds the address of ullCriticalNesting. */ 142 MOV X1, #255 /* X1 holds the unmask value. */ 143 CMP X3, #0 144 B.EQ 1f 145 LDR X6, ullMaxAPIPriorityMaskConst 146 LDR X1, [X6] /* X1 holds the mask value. */ 1471: 148 MSR s3_0_c4_c6_0, X1 /* Write the mask value to ICCPMR. s3_0_c4_c6_0 is ICC_PMR_EL1. */ 149 DSB SY /* _RB_Barriers probably not required here. */ 150 ISB SY 151 STR X3, [X0] /* Restore the task's critical nesting count. */ 152 153 /* Restore the FPU context indicator. */ 154 LDR X0, ullPortTaskHasFPUContextConst 155 STR X2, [X0] 156 157 /* Restore the FPU context, if any. */ 158 CMP X2, #0 159 B.EQ 1f 160 LDP Q30, Q31, [SP], #0x20 161 LDP Q28, Q29, [SP], #0x20 162 LDP Q26, Q27, [SP], #0x20 163 LDP Q24, Q25, [SP], #0x20 164 LDP Q22, Q23, [SP], #0x20 165 LDP Q20, Q21, [SP], #0x20 166 LDP Q18, Q19, [SP], #0x20 167 LDP Q16, Q17, [SP], #0x20 168 LDP Q14, Q15, [SP], #0x20 169 LDP Q12, Q13, [SP], #0x20 170 LDP Q10, Q11, [SP], #0x20 171 LDP Q8, Q9, [SP], #0x20 172 LDP Q6, Q7, [SP], #0x20 173 LDP Q4, Q5, [SP], #0x20 174 LDP Q2, Q3, [SP], #0x20 175 LDP Q0, Q1, [SP], #0x20 1761: 177 LDP X2, X3, [SP], #0x10 /* SPSR and ELR. */ 178 179#if defined( GUEST ) 180 /* Restore the SPSR. */ 181 MSR SPSR_EL1, X3 182 /* Restore the ELR. */ 183 MSR ELR_EL1, X2 184#else 185 /* Restore the SPSR. */ 186 MSR SPSR_EL3, X3 /*_RB_ Assumes started in EL3. */ 187 /* Restore the ELR. */ 188 MSR ELR_EL3, X2 189#endif 190 191 LDP X30, XZR, [SP], #0x10 192 LDP X28, X29, [SP], #0x10 193 LDP X26, X27, [SP], #0x10 194 LDP X24, X25, [SP], #0x10 195 LDP X22, X23, [SP], #0x10 196 LDP X20, X21, [SP], #0x10 197 LDP X18, X19, [SP], #0x10 198 LDP X16, X17, [SP], #0x10 199 LDP X14, X15, [SP], #0x10 200 LDP X12, X13, [SP], #0x10 201 LDP X10, X11, [SP], #0x10 202 LDP X8, X9, [SP], #0x10 203 LDP X6, X7, [SP], #0x10 204 LDP X4, X5, [SP], #0x10 205 LDP X2, X3, [SP], #0x10 206 LDP X0, X1, [SP], #0x10 207 208 /* Switch to use the ELx stack pointer. _RB_ Might not be required. */ 209 MSR SPSEL, #1 210 211 ERET 212 213 .endm 214 215 216/****************************************************************************** 217 * FreeRTOS_SWI_Handler handler is used to perform a context switch. 218 *****************************************************************************/ 219.align 8 220.type FreeRTOS_SWI_Handler, %function 221FreeRTOS_SWI_Handler: 222 /* Save the context of the current task and select a new task to run. */ 223 portSAVE_CONTEXT 224#if defined( GUEST ) 225 MRS X0, ESR_EL1 226#else 227 MRS X0, ESR_EL3 228#endif 229 230 LSR X1, X0, #26 231 232#if defined( GUEST ) 233 CMP X1, #0x15 /* 0x15 = SVC instruction. */ 234#else 235 CMP X1, #0x17 /* 0x17 = SMC instruction. */ 236#endif 237 B.NE FreeRTOS_Abort 238 BL vTaskSwitchContext 239 240 portRESTORE_CONTEXT 241 242FreeRTOS_Abort: 243 /* Full ESR is in X0, exception class code is in X1. */ 244 B . 245 246/****************************************************************************** 247 * vPortRestoreTaskContext is used to start the scheduler. 248 *****************************************************************************/ 249.align 8 250.type vPortRestoreTaskContext, %function 251vPortRestoreTaskContext: 252.set freertos_vector_base, _freertos_vector_table 253 254 /* Install the FreeRTOS interrupt handlers. */ 255 LDR X1, =freertos_vector_base 256#if defined( GUEST ) 257 MSR VBAR_EL1, X1 258#else 259 MSR VBAR_EL3, X1 260#endif 261 DSB SY 262 ISB SY 263 264 /* Start the first task. */ 265 portRESTORE_CONTEXT 266 267 268/****************************************************************************** 269 * FreeRTOS_IRQ_Handler handles IRQ entry and exit. 270 271 * This handler is supposed to be used only for IRQs and never for FIQs. Per ARM 272 * GIC documentation [1], Group 0 interrupts are always signaled as FIQs. Since 273 * this handler is only for IRQs, We can safely assume Group 1 while accessing 274 * Interrupt Acknowledge and End Of Interrupt registers and therefore, use 275 * ICC_IAR1_EL1 and ICC_EOIR1_EL1. 276 * 277 * [1] https://developer.arm.com/documentation/198123/0300/Arm-CoreLink-GIC-fundamentals 278 *****************************************************************************/ 279.align 8 280.type FreeRTOS_IRQ_Handler, %function 281FreeRTOS_IRQ_Handler: 282 /* Save volatile registers. */ 283 STP X0, X1, [SP, #-0x10]! 284 STP X2, X3, [SP, #-0x10]! 285 STP X4, X5, [SP, #-0x10]! 286 STP X6, X7, [SP, #-0x10]! 287 STP X8, X9, [SP, #-0x10]! 288 STP X10, X11, [SP, #-0x10]! 289 STP X12, X13, [SP, #-0x10]! 290 STP X14, X15, [SP, #-0x10]! 291 STP X16, X17, [SP, #-0x10]! 292 STP X18, X19, [SP, #-0x10]! 293 STP X29, X30, [SP, #-0x10]! 294 295 /* Save the SPSR and ELR. */ 296#if defined( GUEST ) 297 MRS X3, SPSR_EL1 298 MRS X2, ELR_EL1 299#else 300 MRS X3, SPSR_EL3 301 MRS X2, ELR_EL3 302#endif 303 STP X2, X3, [SP, #-0x10]! 304 305 /* Increment the interrupt nesting counter. */ 306 LDR X5, ullPortInterruptNestingConst 307 LDR X1, [X5] /* Old nesting count in X1. */ 308 ADD X6, X1, #1 309 STR X6, [X5] /* Address of nesting count variable in X5. */ 310 311 /* Maintain the interrupt nesting information across the function call. */ 312 STP X1, X5, [SP, #-0x10]! 313 314 /* Read interrupt ID from the interrupt acknowledge register and store it 315 in X0 for future parameter and interrupt clearing use. */ 316 MRS X0, S3_0_C12_C12_0 /* S3_0_C12_C12_0 is ICC_IAR1_EL1. */ 317 318 /* Maintain the interrupt ID value across the function call. */ 319 STP X0, X1, [SP, #-0x10]! 320 321 /* Call the C handler. */ 322 BL vApplicationIRQHandler 323 324 /* Disable interrupts. */ 325 MSR DAIFSET, #2 326 DSB SY 327 ISB SY 328 329 /* Restore the interrupt ID value. */ 330 LDP X0, X1, [SP], #0x10 331 332 /* End IRQ processing by writing interrupt ID value to the EOI register. */ 333 MSR S3_0_C12_C12_1, X0 /* S3_0_C12_C12_1 is ICC_EOIR1_EL1. */ 334 335 /* Restore the critical nesting count. */ 336 LDP X1, X5, [SP], #0x10 337 STR X1, [X5] 338 339 /* Has interrupt nesting unwound? */ 340 CMP X1, #0 341 B.NE Exit_IRQ_No_Context_Switch 342 343 /* Is a context switch required? */ 344 LDR X0, ullPortYieldRequiredConst 345 LDR X1, [X0] 346 CMP X1, #0 347 B.EQ Exit_IRQ_No_Context_Switch 348 349 /* Reset ullPortYieldRequired to 0. */ 350 MOV X2, #0 351 STR X2, [X0] 352 353 /* Restore volatile registers. */ 354 LDP X4, X5, [SP], #0x10 /* SPSR and ELR. */ 355#if defined( GUEST ) 356 MSR SPSR_EL1, X5 357 MSR ELR_EL1, X4 358#else 359 MSR SPSR_EL3, X5 /*_RB_ Assumes started in EL3. */ 360 MSR ELR_EL3, X4 361#endif 362 DSB SY 363 ISB SY 364 365 LDP X29, X30, [SP], #0x10 366 LDP X18, X19, [SP], #0x10 367 LDP X16, X17, [SP], #0x10 368 LDP X14, X15, [SP], #0x10 369 LDP X12, X13, [SP], #0x10 370 LDP X10, X11, [SP], #0x10 371 LDP X8, X9, [SP], #0x10 372 LDP X6, X7, [SP], #0x10 373 LDP X4, X5, [SP], #0x10 374 LDP X2, X3, [SP], #0x10 375 LDP X0, X1, [SP], #0x10 376 377 /* Save the context of the current task and select a new task to run. */ 378 portSAVE_CONTEXT 379 BL vTaskSwitchContext 380 portRESTORE_CONTEXT 381 382Exit_IRQ_No_Context_Switch: 383 /* Restore volatile registers. */ 384 LDP X4, X5, [SP], #0x10 /* SPSR and ELR. */ 385#if defined( GUEST ) 386 MSR SPSR_EL1, X5 387 MSR ELR_EL1, X4 388#else 389 MSR SPSR_EL3, X5 /*_RB_ Assumes started in EL3. */ 390 MSR ELR_EL3, X4 391#endif 392 DSB SY 393 ISB SY 394 395 LDP X29, X30, [SP], #0x10 396 LDP X18, X19, [SP], #0x10 397 LDP X16, X17, [SP], #0x10 398 LDP X14, X15, [SP], #0x10 399 LDP X12, X13, [SP], #0x10 400 LDP X10, X11, [SP], #0x10 401 LDP X8, X9, [SP], #0x10 402 LDP X6, X7, [SP], #0x10 403 LDP X4, X5, [SP], #0x10 404 LDP X2, X3, [SP], #0x10 405 LDP X0, X1, [SP], #0x10 406 407 ERET 408 409 410 411 412.align 8 413pxCurrentTCBConst: .dword pxCurrentTCB 414ullCriticalNestingConst: .dword ullCriticalNesting 415ullPortTaskHasFPUContextConst: .dword ullPortTaskHasFPUContext 416 417ullMaxAPIPriorityMaskConst: .dword ullMaxAPIPriorityMask 418ullPortInterruptNestingConst: .dword ullPortInterruptNesting 419ullPortYieldRequiredConst: .dword ullPortYieldRequired 420 421.end 422