-
Notifications
You must be signed in to change notification settings - Fork 1.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
RISCV Add FPU context save #1250
Conversation
Matching PR FreeRTOS/FreeRTOS#1329 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @cubidesj.
Sorry for the late response.
Overall I think this looks fine. My main hang up is I am not sure if we should add directly to base RISC V implementation or use the already existing portasmSAVE_ADDITIONAL_REGISTERS and portasmADDITIONAL_CONTEXT_SIZE. I am not terribly familiar with the implementations of this port, but that would make sense to me. I would like to get the opinion of someone more familiar.
portable/GCC/RISC-V/portContext.h
Outdated
@@ -66,6 +97,97 @@ | |||
.extern pxCriticalNesting | |||
/*-----------------------------------------------------------*/ | |||
|
|||
.macro portcontexSAVE_FPU_CONTEXT_INTERNAL |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
.macro portcontexSAVE_FPU_CONTEXT_INTERNAL | |
.macro portcontextSAVE_FPU_CONTEXT_INTERNAL |
portable/GCC/RISC-V/portContext.h
Outdated
.endm | ||
/*-----------------------------------------------------------*/ | ||
|
||
.macro portasmRESTORE_FPU_CONTEXT_INTERNAL |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
portasm macros should be in portASM.S
.macro portasmRESTORE_FPU_CONTEXT_INTERNAL | |
.macro portcontextRESTORE_FPU_CONTEXT_INTERNAL |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
for consistency / its declared in this file not the asm file.
portable/GCC/RISC-V/portContext.h
Outdated
@@ -105,6 +227,9 @@ store_x t0, portCRITICAL_NESTING_OFFSET * portWORD_SIZE( sp ) /* Store the criti | |||
|
|||
csrr t0, mstatus /* Required for MPIE bit. */ | |||
store_x t0, portMSTATUS_OFFSET * portWORD_SIZE( sp ) | |||
#ifdef portasmSTORE_FPU_CONTEXT | |||
portcontexSAVE_FPU_CONTEXT_INTERNAL |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
portcontexSAVE_FPU_CONTEXT_INTERNAL | |
portcontextSAVE_FPU_CONTEXT_INTERNAL |
portable/GCC/RISC-V/portContext.h
Outdated
@@ -145,6 +270,10 @@ csrw mepc, t0 | |||
/* Defined in freertos_risc_v_chip_specific_extensions.h to restore any registers unique to the RISC-V implementation. */ | |||
portasmRESTORE_ADDITIONAL_REGISTERS | |||
|
|||
#ifdef portasmSTORE_FPU_CONTEXT | |||
portasmRESTORE_FPU_CONTEXT_INTERNAL |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
portasmRESTORE_FPU_CONTEXT_INTERNAL | |
portcontextRESTORE_FPU_CONTEXT_INTERNAL |
portable/GCC/RISC-V/portASM.S
Outdated
@@ -196,8 +197,9 @@ definitions. */ | |||
* pxCode | |||
*/ | |||
pxPortInitialiseStack: | |||
addi a0, a0, -portFPUCONTEXT_SIZE |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why are we adding it like this instead of just making use of portasmADDITIONAL_CONTEXT_SIZE?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This will probably be solved as part of #1250 (comment) but if we don't use the portasm*_ADDITIONAL_REGISTERS
macros we cannot use portasmADDITIONAL_CONTEXT_SIZE
as the "additional context" is not in the same place as the FPU registers.
#define portasmADDITIONAL_CONTEXT_SIZE 0 /* Must be even number on 32-bit cores. */ | ||
#define portasmSTORE_FPU_CONTEXT 1 | ||
|
||
.macro portasmSAVE_ADDITIONAL_REGISTERS |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we just store the FPU registers using this macro?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @jasonpcarroll, this is an interesting question.
We could call the portcontext*_FPU_CONTEXT_INTERNAL
from the freertos_risc_v_chip_specific_extensions.h
but I still think the definition of the macro should be available in the "main" part of the port so it can be used by multiple freertos_risc_v_chip_specific_extensions.h
instead of having multiple copies of that context save/restore.
I chose to put it directly as part of the main context macros as the FPU is a standard extension in the CPU rather than for other parts of the chip but I also hesitated with this.
I agree that it is a judgment call so I will do whatever you recommend. Keep in mind that I plan to create a follow up adding Vector context save support that should follow a very similar structure to what we do for the FPU and that having FPU and having Vectors is orthogonal.
portable/GCC/RISC-V/portASM.S
Outdated
@@ -196,8 +197,9 @@ definitions. */ | |||
* pxCode | |||
*/ | |||
pxPortInitialiseStack: | |||
addi a0, a0, -portFPUCONTEXT_SIZE | |||
csrr t0, mstatus /* Obtain current mstatus value. */ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not blocking but would it be more consistent to perform the add for the FPU context size before reading the mstatus registers? Looks like most stack sizing operations occur before this register access.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @kstribrnAmzn, I am not sure that I fully understand the question but this instruction can be moved a bit later but no later than line 207 unless I move the FPU context to some other place (this includes using the portasmSAVE_ADDITIONAL_REGISTERS
as in another discussion).
Assuming that it remains as is, this function calculates the value to store, stores in the context (using a0
as pointer) and decreases a0
. In order to keep mstatus
at the right offset I need to decrease a0
by the size of the FPU context before mstatus
is stored (at line 208 store_x t0, 0(a0) /* mstatus onto the stack. */
.
Did I understand your concern correctly?
I have the following 2 suggestions:
Both my suggestions are in this patch. Note that I am yet to build and test this patch but I just wanted to share the idea with you. |
Here is the patch that I have tested to ensure that it correctly stores and restores FPU registers only for the tasks which use FPU - 0001-Code-review-suggestions.patch. You'll need to apply the following patch to your demo Makefile - Makefile.patch. Please apply these patches and refresh this branch from upstream main so that we can merge this PR. |
OK, thanks. I will do that, test and push the changes back. |
5c824e5
to
dec30a8
Compare
|
Add FPU context switch support for the RISC-V port
Description
Current context switch in the RISC-V port only supports the integer (
x
) registers. Which means that having two tasks using FPU code will corrupt one another's work.This PR
portasmSTORE_FPU_CONTEXT
) to identify if the FPU context should be saved and does it when the FPU is present and the flag is set. To that have a platform that can tested I also added aRV32IF_CLINT
underrisc_v_chip_specific_extensions
portFPUCONTEXT_SIZE
macro, that is defined as 0 when the FPU support is not present or is not enabledportasmSTORE_FPU_CONTEXT
is enabledTest Steps
A Demo using QEMU was created. That demo also adds a couple of tasks modifying the FPU registers to check that they values are maintained across context switch. This is part of another PR in the FreeRTOS repo
Checklist:
By submitting this pull request, I confirm that you can use, modify, copy, and redistribute this contribution, under the terms of your choice.