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/* FreeRTOS includes. */ 30#include "FreeRTOSConfig.h" 31#include "ISR_Support.h" 32 33/* Microchip includes. */ 34#include <xc.h> 35#include <sys/asm.h> 36 37 .extern pxCurrentTCB 38 .extern vTaskSwitchContext 39 .extern vPortIncrementTick 40 .extern xISRStackTop 41 42 PORT_CPP_JTVIC_BASE = 0xBFFFC000 43 PORT_CCP_JTVIC_GIRQ24_SRC = 0xBFFFC100 44 45 .global vPortStartFirstTask .text 46 .global vPortYieldISR .text 47 .global vPortTickInterruptHandler .text 48 49 50/******************************************************************/ 51 52 53/*************************************************************** 54* The following is needed to locate the 55* vPortTickInterruptHandler function into the correct vector 56* MEC14xx - This ISR will only be used if HW timers' interrupts 57* in GIRQ23 are disaggregated. 58* 59***************************************************************/ 60 61 .set noreorder 62 .set noat 63 .set micromips 64 65 .section .text, code 66 .ent vPortTickInterruptHandler 67 68#if configTIMERS_DISAGGREGATED_ISRS == 0 69 70 .globl girq23_isr 71 72girq23_isr: 73vPortTickInterruptHandler: 74 75 portSAVE_CONTEXT 76 77 jal girq23_handler 78 nop 79 80 portRESTORE_CONTEXT 81 82.end vPortTickInterruptHandler 83 84#else 85 86 .globl girq23_b4 87 88girq23_b4: 89vPortTickInterruptHandler: 90 91 portSAVE_CONTEXT 92 93 jal vPortIncrementTick 94 nop 95 96 portRESTORE_CONTEXT 97 98.end vPortTickInterruptHandler 99 100#endif /* #if configTIMERS_DISAGGREGATED_ISRS == 0 */ 101 102/******************************************************************/ 103 104 .set micromips 105 .set noreorder 106 .set noat 107 108 .section .text, code 109 .ent vPortStartFirstTask 110 111vPortStartFirstTask: 112 113 /* Simply restore the context of the highest priority task that has 114 been created so far. */ 115 portRESTORE_CONTEXT 116 117.end vPortStartFirstTask 118 119 120 121/*******************************************************************/ 122 123/*************************************************************** 124* The following is needed to locate the vPortYieldISR function into the correct 125* vector. 126***************************************************************/ 127 128 .set micromips 129 .set noreorder 130 .set noat 131 132 .section .text, code 133 134 .global vPortYieldISR 135 136 137#if configCPU_DISAGGREGATED_ISRS == 0 138 .global girq24_isr 139 .ent girq24_isr 140girq24_isr: 141 la k0, PORT_CPP_JTVIC_BASE 142 lw k0, 0x10C(k0) 143 andi k1, k0, 0x2 144 bgtz k1, vPortYieldISR 145 nop 146 147 portSAVE_CONTEXT 148 149 jal girq24_b_0_2 150 151 portRESTORE_CONTEXT 152 153 .end girq24_isr 154 155#else 156 .global girq24_b1 157girq24_b1: 158#endif 159 .ent vPortYieldISR 160vPortYieldISR: 161 162 /* Make room for the context. First save the current status so it can be 163 manipulated, and the cause and EPC registers so thier original values 164 are captured. */ 165 addiu sp, sp, -portCONTEXT_SIZE 166 mfc0 k1, _CP0_STATUS 167 168 /* Also save s6 and s5 so they can be used. Any nesting interrupts should 169 maintain the values of these registers across the ISR. */ 170 sw s6, 44(sp) 171 sw s5, 40(sp) 172 sw k1, portSTATUS_STACK_LOCATION(sp) 173 174 /* Prepare to re-enable interrupts above the kernel priority. */ 175 ins k1, zero, 10, 7 /* Clear IPL bits 0:6. */ 176 ins k1, zero, 18, 1 /* Clear IPL bit 7 */ 177 ori k1, k1, ( configMAX_SYSCALL_INTERRUPT_PRIORITY << 10 ) 178 ins k1, zero, 1, 4 /* Clear EXL, ERL and UM. */ 179 180 /* s5 is used as the frame pointer. */ 181 add s5, zero, sp 182 183 /* Swap to the system stack. This is not conditional on the nesting 184 count as this interrupt is always the lowest priority and therefore 185 the nesting is always 0. */ 186 la sp, xISRStackTop 187 lw sp, (sp) 188 189 /* Set the nesting count. */ 190 la k0, uxInterruptNesting 191 addiu s6, zero, 1 192 sw s6, 0(k0) 193 194 /* s6 holds the EPC value, this is saved with the rest of the context 195 after interrupts are enabled. */ 196 mfc0 s6, _CP0_EPC 197 198 /* Re-enable interrupts above configMAX_SYSCALL_INTERRUPT_PRIORITY. */ 199 mtc0 k1, _CP0_STATUS 200 201 /* Save the context into the space just created. s6 is saved again 202 here as it now contains the EPC value. */ 203 sw ra, 120(s5) 204 sw s8, 116(s5) 205 sw t9, 112(s5) 206 sw t8, 108(s5) 207 sw t7, 104(s5) 208 sw t6, 100(s5) 209 sw t5, 96(s5) 210 sw t4, 92(s5) 211 sw t3, 88(s5) 212 sw t2, 84(s5) 213 sw t1, 80(s5) 214 sw t0, 76(s5) 215 sw a3, 72(s5) 216 sw a2, 68(s5) 217 sw a1, 64(s5) 218 sw a0, 60(s5) 219 sw v1, 56(s5) 220 sw v0, 52(s5) 221 sw s7, 48(s5) 222 sw s6, portEPC_STACK_LOCATION(s5) 223 /* s5 and s6 has already been saved. */ 224 sw s4, 36(s5) 225 sw s3, 32(s5) 226 sw s2, 28(s5) 227 sw s1, 24(s5) 228 sw s0, 20(s5) 229 sw $1, 16(s5) 230 231 /* s7 is used as a scratch register as this should always be saved acro ss 232 nesting interrupts. */ 233 mfhi s7 234 sw s7, 12(s5) 235 mflo s7 236 sw s7, 8(s5) 237 238 /* Save the stack pointer to the task. */ 239 la s7, pxCurrentTCB 240 lw s7, (s7) 241 sw s5, (s7) 242 243 /* Set the interrupt mask to the max priority that can use the API. 244 The yield handler will only be called at configKERNEL_INTERRUPT_PRIORITY 245 which is below configMAX_SYSCALL_INTERRUPT_PRIORITY - so this can only 246 ever raise the IPL value and never lower it. */ 247 di 248 ehb 249 mfc0 s7, _CP0_STATUS 250 ins s7, zero, 10, 7 251 ins s7, zero, 18, 1 252 ori s6, s7, ( configMAX_SYSCALL_INTERRUPT_PRIORITY << 10 ) | 1 253 254 /* This mtc0 re-enables interrupts, but only above 255 configMAX_SYSCALL_INTERRUPT_PRIORITY. */ 256 mtc0 s6, _CP0_STATUS 257 ehb 258 259 /* Clear the software interrupt in the core. */ 260 mfc0 s6, _CP0_CAUSE 261 ins s6, zero, 8, 1 262 mtc0 s6, _CP0_CAUSE 263 ehb 264 265 /* Clear the interrupt in the interrupt controller. 266 MEC14xx GIRQ24 Source bit[1] = 1 to clear */ 267 la s6, PORT_CCP_JTVIC_GIRQ24_SRC 268 addiu s4, zero, 2 269 sw s4, (s6) 270 jal vTaskSwitchContext 271 nop 272 273 /* Clear the interrupt mask again. The saved status value is still in s7 */ 274 mtc0 s7, _CP0_STATUS 275 ehb 276 277 /* Restore the stack pointer from the TCB. */ 278 la s0, pxCurrentTCB 279 lw s0, (s0) 280 lw s5, (s0) 281 282 /* Restore the rest of the context. */ 283 lw s0, 8(s5) 284 mtlo s0 285 lw s0, 12(s5) 286 mthi s0 287 288 lw $1, 16(s5) 289 lw s0, 20(s5) 290 lw s1, 24(s5) 291 lw s2, 28(s5) 292 lw s3, 32(s5) 293 lw s4, 36(s5) 294 295 /* s5 is loaded later. */ 296 lw s6, 44(s5) 297 lw s7, 48(s5) 298 lw v0, 52(s5) 299 lw v1, 56(s5) 300 lw a0, 60(s5) 301 lw a1, 64(s5) 302 lw a2, 68(s5) 303 lw a3, 72(s5) 304 lw t0, 76(s5) 305 lw t1, 80(s5) 306 lw t2, 84(s5) 307 lw t3, 88(s5) 308 lw t4, 92(s5) 309 lw t5, 96(s5) 310 lw t6, 100(s5) 311 lw t7, 104(s5) 312 lw t8, 108(s5) 313 lw t9, 112(s5) 314 lw s8, 116(s5) 315 lw ra, 120(s5) 316 317 /* Protect access to the k registers, and others. */ 318 di 319 ehb 320 321 /* Set nesting back to zero. As the lowest priority interrupt this 322 interrupt cannot have nested. */ 323 la k0, uxInterruptNesting 324 sw zero, 0(k0) 325 326 /* Switch back to use the real stack pointer. */ 327 add sp, zero, s5 328 329 /* Restore the real s5 value. */ 330 lw s5, 40(sp) 331 332 /* Pop the status and epc values. */ 333 lw k1, portSTATUS_STACK_LOCATION(sp) 334 lw k0, portEPC_STACK_LOCATION(sp) 335 336 /* Remove stack frame. */ 337 addiu sp, sp, portCONTEXT_SIZE 338 339 mtc0 k1, _CP0_STATUS 340 mtc0 k0, _CP0_EPC 341 ehb 342 eret 343 nop 344 345.end vPortYieldISR 346