Skip to content
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

Merged
merged 9 commits into from
Mar 6, 2025
Merged

Conversation

cubidesj
Copy link
Contributor

@cubidesj cubidesj commented Feb 21, 2025

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

  • Creates a chip_specific_extension define (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 a RV32IF_CLINT under risc_v_chip_specific_extensions
  • Create portFPUCONTEXT_SIZE macro, that is defined as 0 when the FPU support is not present or is not enabled
  • Creates a couple of macros to save/restore the FPU context and uses it when portasmSTORE_FPU_CONTEXT is enabled

Test 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:

  • I have tested my changes. No regression in existing tests.
  • I have modified and/or added unit-tests to cover the code changes in this Pull Request.

By submitting this pull request, I confirm that you can use, modify, copy, and redistribute this contribution, under the terms of your choice.

@cubidesj
Copy link
Contributor Author

Matching PR FreeRTOS/FreeRTOS#1329

Copy link
Member

@jasonpcarroll jasonpcarroll left a 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.

@@ -66,6 +97,97 @@
.extern pxCriticalNesting
/*-----------------------------------------------------------*/

.macro portcontexSAVE_FPU_CONTEXT_INTERNAL
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
.macro portcontexSAVE_FPU_CONTEXT_INTERNAL
.macro portcontextSAVE_FPU_CONTEXT_INTERNAL

.endm
/*-----------------------------------------------------------*/

.macro portasmRESTORE_FPU_CONTEXT_INTERNAL
Copy link
Member

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

Suggested change
.macro portasmRESTORE_FPU_CONTEXT_INTERNAL
.macro portcontextRESTORE_FPU_CONTEXT_INTERNAL

Copy link
Member

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.

@@ -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
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
portcontexSAVE_FPU_CONTEXT_INTERNAL
portcontextSAVE_FPU_CONTEXT_INTERNAL

@@ -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
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
portasmRESTORE_FPU_CONTEXT_INTERNAL
portcontextRESTORE_FPU_CONTEXT_INTERNAL

@@ -196,8 +197,9 @@ definitions. */
* pxCode
*/
pxPortInitialiseStack:
addi a0, a0, -portFPUCONTEXT_SIZE
Copy link
Member

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?

Copy link
Contributor Author

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
Copy link
Member

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?

Copy link
Contributor Author

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.

@@ -196,8 +197,9 @@ definitions. */
* pxCode
*/
pxPortInitialiseStack:
addi a0, a0, -portFPUCONTEXT_SIZE
csrr t0, mstatus /* Obtain current mstatus value. */
Copy link
Member

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.

Copy link
Contributor Author

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?

@aggarg
Copy link
Member

aggarg commented Mar 4, 2025

I have the following 2 suggestions:

  1. Use a config macro configENABLE_FPU to enable FPU support. This would be consistent with other ports.
  2. The current implementation reserves space for FPU registers in the task context regardless of whether or not the uses FPU. I think we can only reserve space for FPU registers if the task uses FPU.

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.

@aggarg
Copy link
Member

aggarg commented Mar 5, 2025

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.

@cubidesj
Copy link
Contributor Author

cubidesj commented Mar 6, 2025

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.

@cubidesj cubidesj force-pushed the s5/jcu/add_fpu_support branch from 5c824e5 to dec30a8 Compare March 6, 2025 09:53
Copy link

sonarqubecloud bot commented Mar 6, 2025

@xuelix xuelix merged commit 4d9cd90 into FreeRTOS:main Mar 6, 2025
17 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants