]> begriffs open source - freertos/blob - portable/GCC/RISC-V/portContext.h
[AUTO][RELEASE]: Bump file header version to "10.5.1"
[freertos] / portable / GCC / RISC-V / portContext.h
1 /*
2  * FreeRTOS Kernel V10.5.1
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 #ifndef PORTCONTEXT_H
30 #define PORTCONTEXT_H
31
32 #if __riscv_xlen == 64
33     #define portWORD_SIZE 8
34     #define store_x sd
35     #define load_x ld
36 #elif __riscv_xlen == 32
37     #define store_x sw
38     #define load_x lw
39     #define portWORD_SIZE 4
40 #else
41     #error Assembler did not define __riscv_xlen
42 #endif
43
44 #include "freertos_risc_v_chip_specific_extensions.h"
45
46 /* Only the standard core registers are stored by default.  Any additional
47  * registers must be saved by the portasmSAVE_ADDITIONAL_REGISTERS and
48  * portasmRESTORE_ADDITIONAL_REGISTERS macros - which can be defined in a chip
49  * specific version of freertos_risc_v_chip_specific_extensions.h.  See the
50  * notes at the top of portASM.S file. */
51 #ifdef __riscv_32e
52     #define portCONTEXT_SIZE ( 15 * portWORD_SIZE )
53     #define portCRITICAL_NESTING_OFFSET 13
54     #define portMSTATUS_OFFSET  14
55 #else
56     #define portCONTEXT_SIZE ( 31 * portWORD_SIZE )
57     #define portCRITICAL_NESTING_OFFSET 29
58     #define portMSTATUS_OFFSET  30
59 #endif
60
61 /*-----------------------------------------------------------*/
62
63 .extern pxCurrentTCB
64 .extern xISRStackTop
65 .extern xCriticalNesting
66 .extern pxCriticalNesting
67 /*-----------------------------------------------------------*/
68
69 .macro portcontextSAVE_CONTEXT_INTERNAL
70     addi sp, sp, -portCONTEXT_SIZE
71     store_x x1, 1 * portWORD_SIZE( sp )
72     store_x x5, 2 * portWORD_SIZE( sp )
73     store_x x6, 3 * portWORD_SIZE( sp )
74     store_x x7, 4 * portWORD_SIZE( sp )
75     store_x x8, 5 * portWORD_SIZE( sp )
76     store_x x9, 6 * portWORD_SIZE( sp )
77     store_x x10, 7 * portWORD_SIZE( sp )
78     store_x x11, 8 * portWORD_SIZE( sp )
79     store_x x12, 9 * portWORD_SIZE( sp )
80     store_x x13, 10 * portWORD_SIZE( sp )
81     store_x x14, 11 * portWORD_SIZE( sp )
82     store_x x15, 12 * portWORD_SIZE( sp )
83 #ifndef __riscv_32e
84     store_x x16, 13 * portWORD_SIZE( sp )
85     store_x x17, 14 * portWORD_SIZE( sp )
86     store_x x18, 15 * portWORD_SIZE( sp )
87     store_x x19, 16 * portWORD_SIZE( sp )
88     store_x x20, 17 * portWORD_SIZE( sp )
89     store_x x21, 18 * portWORD_SIZE( sp )
90     store_x x22, 19 * portWORD_SIZE( sp )
91     store_x x23, 20 * portWORD_SIZE( sp )
92     store_x x24, 21 * portWORD_SIZE( sp )
93     store_x x25, 22 * portWORD_SIZE( sp )
94     store_x x26, 23 * portWORD_SIZE( sp )
95     store_x x27, 24 * portWORD_SIZE( sp )
96     store_x x28, 25 * portWORD_SIZE( sp )
97     store_x x29, 26 * portWORD_SIZE( sp )
98     store_x x30, 27 * portWORD_SIZE( sp )
99     store_x x31, 28 * portWORD_SIZE( sp )
100 #endif
101
102     load_x  t0, xCriticalNesting         /* Load the value of xCriticalNesting into t0. */
103     store_x t0, portCRITICAL_NESTING_OFFSET * portWORD_SIZE( sp ) /* Store the critical nesting value to the stack. */
104
105
106     csrr t0, mstatus                     /* Required for MPIE bit. */
107     store_x t0, portMSTATUS_OFFSET * portWORD_SIZE( sp )
108
109
110     portasmSAVE_ADDITIONAL_REGISTERS     /* Defined in freertos_risc_v_chip_specific_extensions.h to save any registers unique to the RISC-V implementation. */
111
112     load_x  t0, pxCurrentTCB             /* Load pxCurrentTCB. */
113     store_x  sp, 0( t0 )                 /* Write sp to first TCB member. */
114
115     .endm
116 /*-----------------------------------------------------------*/
117
118 .macro portcontextSAVE_EXCEPTION_CONTEXT
119     portcontextSAVE_CONTEXT_INTERNAL
120     csrr a0, mcause
121     csrr a1, mepc
122     addi a1, a1, 4                      /* Synchronous so update exception return address to the instruction after the instruction that generated the exception. */
123     store_x a1, 0( sp )                 /* Save updated exception return address. */
124     load_x sp, xISRStackTop             /* Switch to ISR stack. */
125     .endm
126 /*-----------------------------------------------------------*/
127
128 .macro portcontextSAVE_INTERRUPT_CONTEXT
129     portcontextSAVE_CONTEXT_INTERNAL
130     csrr a0, mcause
131     csrr a1, mepc
132     store_x a1, 0( sp )                 /* Asynchronous interrupt so save unmodified exception return address. */
133     load_x sp, xISRStackTop             /* Switch to ISR stack. */
134     .endm
135 /*-----------------------------------------------------------*/
136
137 .macro portcontextRESTORE_CONTEXT
138     load_x  t1, pxCurrentTCB                /* Load pxCurrentTCB. */
139         load_x  sp, 0( t1 )                 /* Read sp from first TCB member. */
140
141     /* Load mepc with the address of the instruction in the task to run next. */
142     load_x t0, 0( sp )
143     csrw mepc, t0
144
145     /* Defined in freertos_risc_v_chip_specific_extensions.h to restore any registers unique to the RISC-V implementation. */
146     portasmRESTORE_ADDITIONAL_REGISTERS
147
148     /* Load mstatus with the interrupt enable bits used by the task. */
149     load_x  t0, portMSTATUS_OFFSET * portWORD_SIZE( sp )
150     csrw mstatus, t0                        /* Required for MPIE bit. */
151
152     load_x  t0, portCRITICAL_NESTING_OFFSET * portWORD_SIZE( sp )    /* Obtain xCriticalNesting value for this task from task's stack. */
153     load_x  t1, pxCriticalNesting           /* Load the address of xCriticalNesting into t1. */
154     store_x t0, 0( t1 )                     /* Restore the critical nesting value for this task. */
155
156     load_x  x1, 1 * portWORD_SIZE( sp )
157     load_x  x5, 2 * portWORD_SIZE( sp )
158     load_x  x6, 3 * portWORD_SIZE( sp )
159     load_x  x7, 4 * portWORD_SIZE( sp )
160     load_x  x8, 5 * portWORD_SIZE( sp )
161     load_x  x9, 6 * portWORD_SIZE( sp )
162     load_x  x10, 7 * portWORD_SIZE( sp )
163     load_x  x11, 8 * portWORD_SIZE( sp )
164     load_x  x12, 9 * portWORD_SIZE( sp )
165     load_x  x13, 10 * portWORD_SIZE( sp )
166     load_x  x14, 11 * portWORD_SIZE( sp )
167     load_x  x15, 12 * portWORD_SIZE( sp )
168 #ifndef __riscv_32e
169     load_x  x16, 13 * portWORD_SIZE( sp )
170     load_x  x17, 14 * portWORD_SIZE( sp )
171     load_x  x18, 15 * portWORD_SIZE( sp )
172     load_x  x19, 16 * portWORD_SIZE( sp )
173     load_x  x20, 17 * portWORD_SIZE( sp )
174     load_x  x21, 18 * portWORD_SIZE( sp )
175     load_x  x22, 19 * portWORD_SIZE( sp )
176     load_x  x23, 20 * portWORD_SIZE( sp )
177     load_x  x24, 21 * portWORD_SIZE( sp )
178     load_x  x25, 22 * portWORD_SIZE( sp )
179     load_x  x26, 23 * portWORD_SIZE( sp )
180     load_x  x27, 24 * portWORD_SIZE( sp )
181     load_x  x28, 25 * portWORD_SIZE( sp )
182     load_x  x29, 26 * portWORD_SIZE( sp )
183     load_x  x30, 27 * portWORD_SIZE( sp )
184     load_x  x31, 28 * portWORD_SIZE( sp )
185 #endif
186     addi sp, sp, portCONTEXT_SIZE
187
188     mret
189     .endm
190 /*-----------------------------------------------------------*/
191
192 #endif /* PORTCONTEXT_H */