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.file "portASM.S"
30#include "FreeRTOSConfig.h"
31#include "ISR_Support.h"
32
33    .extern pxCurrentTCB
34    .extern vTaskSwitchContext
35    .extern vPortCentralInterruptHandler
36    .extern xTaskIncrementTick
37    .extern vPortAPICErrorHandler
38    .extern pucPortTaskFPUContextBuffer
39    .extern ulPortYieldPending
40
41    .global vPortStartFirstTask
42    .global vPortCentralInterruptWrapper
43    .global vPortAPICErrorHandlerWrapper
44    .global vPortTimerHandler
45    .global vPortYieldCall
46    .global vPortAPICSpuriousHandler
47
48    .text
49
50/*-----------------------------------------------------------*/
51
52.align 4
53.func vPortYieldCall
54vPortYieldCall:
55    /* Save general purpose registers. */
56    pusha
57
58    .if configSUPPORT_FPU == 1
59
60        /* If the task has a buffer allocated to save the FPU context then save
61        the FPU context now. */
62        movl    pucPortTaskFPUContextBuffer, %eax
63        test    %eax, %eax
64        je      1f
65        fnsave  ( %eax )
66        fwait
67
68        1:
69
70        /* Save the address of the FPU context, if any. */
71        push    pucPortTaskFPUContextBuffer
72
73    .endif /* configSUPPORT_FPU */
74
75    /* Find the TCB. */
76    movl    pxCurrentTCB, %eax
77
78    /* Stack location is first item in the TCB. */
79    movl    %esp, (%eax)
80
81    call vTaskSwitchContext
82
83    /* Find the location of pxCurrentTCB again - a callee saved register could
84    be used in place of eax to prevent this second load, but that then relies
85    on the compiler and other asm code. */
86    movl    pxCurrentTCB, %eax
87    movl    (%eax), %esp
88
89    .if configSUPPORT_FPU == 1
90
91        /* Restore address of task's FPU context buffer. */
92        pop     pucPortTaskFPUContextBuffer
93
94        /* If the task has a buffer allocated in which its FPU context is saved,
95        then restore it now. */
96        movl    pucPortTaskFPUContextBuffer, %eax
97        test    %eax, %eax
98        je      1f
99        frstor  ( %eax )
100        1:
101    .endif
102
103    popa
104    iret
105
106.endfunc
107/*-----------------------------------------------------------*/
108
109.align 4
110.func vPortStartFirstTask
111vPortStartFirstTask:
112
113    /* Find the TCB. */
114    movl    pxCurrentTCB, %eax
115
116    /* Stack location is first item in the TCB. */
117    movl    (%eax), %esp
118
119    /* Restore FPU context flag. */
120    .if configSUPPORT_FPU == 1
121
122        pop     pucPortTaskFPUContextBuffer
123
124    .endif /* configSUPPORT_FPU */
125
126    /* Restore general purpose registers. */
127    popa
128    iret
129.endfunc
130/*-----------------------------------------------------------*/
131
132.align 4
133.func vPortAPICErrorHandlerWrapper
134vPortAPICErrorHandlerWrapper:
135    pusha
136    call    vPortAPICErrorHandler
137    popa
138    /* EOI. */
139    movl    $0x00, (0xFEE000B0)
140    iret
141.endfunc
142/*-----------------------------------------------------------*/
143
144.align 4
145.func vPortTimerHandler
146vPortTimerHandler:
147
148    /* Save general purpose registers. */
149    pusha
150
151    /* Interrupts are not nested, so save the rest of the task context. */
152    .if configSUPPORT_FPU == 1
153
154        /* If the task has a buffer allocated to save the FPU context then save the
155        FPU context now. */
156        movl    pucPortTaskFPUContextBuffer, %eax
157        test    %eax, %eax
158        je      1f
159        fnsave  ( %eax ) /* Save FLOP context into ucTempFPUBuffer array. */
160        fwait
161
162        1:
163        /* Save the address of the FPU context, if any. */
164        push    pucPortTaskFPUContextBuffer
165
166    .endif /* configSUPPORT_FPU */
167
168    /* Find the TCB. */
169    movl    pxCurrentTCB, %eax
170
171    /* Stack location is first item in the TCB. */
172    movl    %esp, (%eax)
173
174    /* Switch stacks. */
175    movl    ulTopOfSystemStack, %esp
176    movl    %esp, %ebp
177
178    /* Increment nesting count. */
179    add     $1, ulInterruptNesting
180
181    call    xTaskIncrementTick
182
183    sti
184
185    /* Is a switch to another task required? */
186    test    %eax, %eax
187    je      _skip_context_switch
188    cli
189    call    vTaskSwitchContext
190
191_skip_context_switch:
192    cli
193
194    /* Decrement the variable used to determine if a switch to a system
195    stack is necessary. */
196    sub     $1, ulInterruptNesting
197
198    /* Stack location is first item in the TCB. */
199    movl    pxCurrentTCB, %eax
200    movl    (%eax), %esp
201
202    .if configSUPPORT_FPU == 1
203
204        /* Restore address of task's FPU context buffer. */
205        pop     pucPortTaskFPUContextBuffer
206
207        /* If the task has a buffer allocated in which its FPU context is saved,
208        then restore it now. */
209        movl    pucPortTaskFPUContextBuffer, %eax
210        test    %eax, %eax
211        je      1f
212        frstor  ( %eax )
213        1:
214    .endif
215
216    popa
217
218    /* EOI. */
219    movl    $0x00, (0xFEE000B0)
220    iret
221
222.endfunc
223/*-----------------------------------------------------------*/
224
225.if configUSE_COMMON_INTERRUPT_ENTRY_POINT == 1
226
227    .align 4
228    .func vPortCentralInterruptWrapper
229    vPortCentralInterruptWrapper:
230
231        portFREERTOS_INTERRUPT_ENTRY
232
233        movl $0xFEE00170, %eax          /* Highest In Service Register (ISR) long word. */
234        movl $8, %ecx                   /* Loop counter. */
235
236    next_isr_long_word:
237        test %ecx, %ecx                 /* Loop counter reached 0? */
238        je wrapper_epilogue             /* Looked at all ISR registers without finding a bit set. */
239        sub $1, %ecx                    /* Sub 1 from loop counter. */
240        movl (%eax), %ebx               /* Load next ISR long word. */
241        sub $0x10, %eax                 /* Point to next ISR long word in case no bits are set in the current long word. */
242        test %ebx, %ebx                 /* Are there any bits set? */
243        je next_isr_long_word           /* Look at next ISR long word if no bits were set. */
244        sti
245        bsr %ebx, %ebx                  /* A bit was set, which one? */
246        movl $32, %eax                  /* Destination operand for following multiplication. */
247        mul %ecx                        /* Calculate base vector for current register, 32 vectors per register. */
248        add %ebx, %eax                  /* Add bit offset into register to get final vector number. */
249        push %eax                       /* Vector number is function parameter. */
250        call vPortCentralInterruptHandler
251        pop %eax                        /* Remove parameter. */
252
253    wrapper_epilogue:
254        portFREERTOS_INTERRUPT_EXIT
255
256    .endfunc
257
258.endif /* configUSE_COMMON_INTERRUPT_ENTRY_POINT */
259/*-----------------------------------------------------------*/
260
261.align 4
262.func vPortAPISpuriousHandler
263vPortAPICSpuriousHandler:
264    iret
265
266.endfunc
267
268.end
269