mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
Merge pull request #20633 from dylad/pr/cpu/cortexm33/add_splim_support
cpu/cortexm: add stack limit support for Cortex-M33
This commit is contained in:
commit
2e9ce4d6b4
@ -285,7 +285,7 @@ kernel_pid_t thread_create(char *stack, int stacksize, uint8_t priority,
|
|||||||
thread->sp = thread_stack_init(function, arg, stack, stacksize);
|
thread->sp = thread_stack_init(function, arg, stack, stacksize);
|
||||||
|
|
||||||
#if defined(DEVELHELP) || IS_ACTIVE(SCHED_TEST_STACK) || \
|
#if defined(DEVELHELP) || IS_ACTIVE(SCHED_TEST_STACK) || \
|
||||||
defined(MODULE_MPU_STACK_GUARD)
|
defined(MODULE_MPU_STACK_GUARD) || defined(MODULE_CORTEXM_STACK_LIMIT)
|
||||||
thread->stack_start = stack;
|
thread->stack_start = stack;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -36,6 +36,7 @@ else ifeq ($(CPU_CORE),cortex-m3)
|
|||||||
RUST_TARGET = thumbv7m-none-eabi
|
RUST_TARGET = thumbv7m-none-eabi
|
||||||
else ifeq ($(CPU_CORE),cortex-m33)
|
else ifeq ($(CPU_CORE),cortex-m33)
|
||||||
CPU_ARCH := armv8m
|
CPU_ARCH := armv8m
|
||||||
|
FEATURES_PROVIDED += cortexm_stack_limit
|
||||||
#RUST_TARGET = thumbv8m.main-none-eabi
|
#RUST_TARGET = thumbv8m.main-none-eabi
|
||||||
else ifeq ($(CPU_CORE),cortex-m4)
|
else ifeq ($(CPU_CORE),cortex-m4)
|
||||||
CPU_ARCH := armv7m
|
CPU_ARCH := armv7m
|
||||||
|
@ -297,6 +297,13 @@ void NORETURN cpu_switch_context_exit(void)
|
|||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef MODULE_CORTEXM_STACK_LIMIT
|
||||||
|
static void* __attribute__((used)) _get_new_stacksize(unsigned int *args) {
|
||||||
|
thread_t* t = (thread_t*) args;
|
||||||
|
return thread_get_stackstart(t);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
#if CPU_CORE_CORTEXM_FULL_THUMB
|
#if CPU_CORE_CORTEXM_FULL_THUMB
|
||||||
void __attribute__((naked)) __attribute__((used)) isr_pendsv(void) {
|
void __attribute__((naked)) __attribute__((used)) isr_pendsv(void) {
|
||||||
__asm__ volatile (
|
__asm__ volatile (
|
||||||
@ -337,8 +344,14 @@ void __attribute__((naked)) __attribute__((used)) isr_pendsv(void) {
|
|||||||
|
|
||||||
/* current thread context is now saved */
|
/* current thread context is now saved */
|
||||||
"restore_context: \n" /* Label to skip thread state saving */
|
"restore_context: \n" /* Label to skip thread state saving */
|
||||||
|
#ifdef MODULE_CORTEXM_STACK_LIMIT
|
||||||
"ldr r0, [r0] \n" /* load tcb->sp to register 1 */
|
"mov r4, r0 \n" /* Save content of R0 into R4*/
|
||||||
|
"bl _get_new_stacksize \n" /* Get the new lower limit stack in R0 */
|
||||||
|
"msr psplim, r0 \n" /* Set the PSP lower limit stack */
|
||||||
|
"ldr r0, [r4] \n" /* Load tcb->sp to register 0 */
|
||||||
|
#else
|
||||||
|
"ldr r0, [r0] \n" /* load tcb->sp to register 0 */
|
||||||
|
#endif
|
||||||
"ldmia r0!, {r4-r11,lr} \n" /* restore other registers, including lr */
|
"ldmia r0!, {r4-r11,lr} \n" /* restore other registers, including lr */
|
||||||
#ifdef MODULE_CORTEXM_FPU
|
#ifdef MODULE_CORTEXM_FPU
|
||||||
"tst lr, #0x10 \n"
|
"tst lr, #0x10 \n"
|
||||||
|
@ -99,6 +99,11 @@ void reset_handler_default(void)
|
|||||||
uint32_t *dst;
|
uint32_t *dst;
|
||||||
const uint32_t *src = &_etext;
|
const uint32_t *src = &_etext;
|
||||||
|
|
||||||
|
#ifdef __ARM_ARCH_8M_MAIN__
|
||||||
|
/* Set the lower limit of the exception stack into MSPLIM register */
|
||||||
|
__set_MSPLIM((uint32_t)&_sstack);
|
||||||
|
#endif
|
||||||
|
|
||||||
cortexm_init_fpu();
|
cortexm_init_fpu();
|
||||||
|
|
||||||
#ifdef MODULE_PUF_SRAM
|
#ifdef MODULE_PUF_SRAM
|
||||||
|
@ -79,6 +79,8 @@ groups:
|
|||||||
- name: cpu_check_address
|
- name: cpu_check_address
|
||||||
help: The @ref cpu_check_address can be used to check if
|
help: The @ref cpu_check_address can be used to check if
|
||||||
accessing a given address would cause a bus fault.
|
accessing a given address would cause a bus fault.
|
||||||
|
- name: cortexm_stack_limit
|
||||||
|
help: ARM Cortex-M PSPLIM/MSPLIM registers are available.
|
||||||
- name: cortexm_svc
|
- name: cortexm_svc
|
||||||
help: ARM Cortex-M Supervisor Calls are available.
|
help: ARM Cortex-M Supervisor Calls are available.
|
||||||
- name: cortexm_fpu
|
- name: cortexm_fpu
|
||||||
|
@ -46,6 +46,7 @@ FEATURES_EXISTING := \
|
|||||||
can_rx_mailbox \
|
can_rx_mailbox \
|
||||||
cortexm_fpu \
|
cortexm_fpu \
|
||||||
cortexm_mpu \
|
cortexm_mpu \
|
||||||
|
cortexm_stack_limit \
|
||||||
cortexm_svc \
|
cortexm_svc \
|
||||||
cpp \
|
cpp \
|
||||||
cpu_arm7tdmi_gba \
|
cpu_arm7tdmi_gba \
|
||||||
|
@ -78,6 +78,10 @@ ifneq (,$(filter periph_rtc,$(USEMODULE)))
|
|||||||
USEMODULE += rtc_utils
|
USEMODULE += rtc_utils
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
# select cortexm_stack_limit pseudomodule if the corresponding
|
||||||
|
# feature is used
|
||||||
|
USEMODULE += $(filter cortexm_stack_limit, $(FEATURES_USED))
|
||||||
|
|
||||||
# select cortexm_svc pseudomodule if the corresponding feature is used
|
# select cortexm_svc pseudomodule if the corresponding feature is used
|
||||||
USEMODULE += $(filter cortexm_svc, $(FEATURES_USED))
|
USEMODULE += $(filter cortexm_svc, $(FEATURES_USED))
|
||||||
|
|
||||||
|
@ -35,6 +35,16 @@ PSEUDOMODULES += board_software_reset
|
|||||||
|
|
||||||
PSEUDOMODULES += arduino_pwm
|
PSEUDOMODULES += arduino_pwm
|
||||||
PSEUDOMODULES += arduino_serial_stdio
|
PSEUDOMODULES += arduino_serial_stdio
|
||||||
|
## @defgroup pseudomodule_arm_stack_limit arm_stack_limit
|
||||||
|
## @{
|
||||||
|
## @brief Set MSP/PSP stack lower limit
|
||||||
|
##
|
||||||
|
## Use PSPLIM and MSPLIM ARM registers to set the lower limit of a stack
|
||||||
|
## This is a protection mechanism to catch stack overflow early before it
|
||||||
|
## can corrupt adjacent memory. Only available on ARMv8-M architecture.
|
||||||
|
PSEUDOMODULES += cortexm_stack_limit
|
||||||
|
## @}
|
||||||
|
|
||||||
PSEUDOMODULES += can_mbox
|
PSEUDOMODULES += can_mbox
|
||||||
PSEUDOMODULES += can_pm
|
PSEUDOMODULES += can_pm
|
||||||
PSEUDOMODULES += can_raw
|
PSEUDOMODULES += can_raw
|
||||||
|
11
tests/cpu/cortexm_stack_limit/Makefile
Normal file
11
tests/cpu/cortexm_stack_limit/Makefile
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
BOARD ?= nrf5340dk-app
|
||||||
|
|
||||||
|
include ../Makefile.cpu_common
|
||||||
|
|
||||||
|
FEATURES_REQUIRED += cortexm_stack_limit
|
||||||
|
|
||||||
|
include $(RIOTBASE)/Makefile.include
|
||||||
|
|
||||||
|
ifeq (llvm,$(TOOLCHAIN))
|
||||||
|
CFLAGS += -Wno-infinite-recursion
|
||||||
|
endif
|
99
tests/cpu/cortexm_stack_limit/main.c
Normal file
99
tests/cpu/cortexm_stack_limit/main.c
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2024 Mesotic SAS
|
||||||
|
*
|
||||||
|
* This file is subject to the terms and conditions of the GNU Lesser
|
||||||
|
* General Public License v2.1. See the file LICENSE in the top level
|
||||||
|
* directory for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @ingroup tests
|
||||||
|
* @{
|
||||||
|
*
|
||||||
|
* @file
|
||||||
|
* @brief Test application for the cortexm_stack_limit pseudo-module
|
||||||
|
*
|
||||||
|
* @author Dylan Laduranty <dylan.laduranty@mesotic.com>
|
||||||
|
*
|
||||||
|
* @}
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include "cpu.h"
|
||||||
|
#include "thread.h"
|
||||||
|
#include "board.h"
|
||||||
|
|
||||||
|
#define CANARY_VALUE 0xdeadbeef
|
||||||
|
|
||||||
|
static struct {
|
||||||
|
unsigned int canary;
|
||||||
|
char stack[THREAD_STACKSIZE_MAIN];
|
||||||
|
} buf;
|
||||||
|
|
||||||
|
/* Tell modern GCC (12.x) to not complain that this infinite recursion is
|
||||||
|
* bound to overflow the stack - this is exactly what this test wants to do :)
|
||||||
|
*
|
||||||
|
* Also, tell older versions of GCC that do not know about -Winfinit-recursion
|
||||||
|
* that it is safe to ignore `GCC diagnostics ignored "-Winfinit-recursion"`.
|
||||||
|
* They behave as intended in this case :)
|
||||||
|
*/
|
||||||
|
#pragma GCC diagnostic push
|
||||||
|
#pragma GCC diagnostic ignored "-Wpragmas"
|
||||||
|
#pragma GCC diagnostic ignored "-Winfinite-recursion"
|
||||||
|
static int recurse(int counter)
|
||||||
|
{
|
||||||
|
if (buf.canary != CANARY_VALUE) {
|
||||||
|
#ifdef LED0_ON
|
||||||
|
LED0_ON;
|
||||||
|
#endif
|
||||||
|
#ifdef LED1_ON
|
||||||
|
LED1_ON;
|
||||||
|
#endif
|
||||||
|
#ifdef LED2_ON
|
||||||
|
LED2_ON;
|
||||||
|
#endif
|
||||||
|
#ifdef LED3_ON
|
||||||
|
LED3_ON;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
thread_sleep();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
counter++;
|
||||||
|
/* Recursing twice here prevents the compiler from optimizing-out the recursion. */
|
||||||
|
return recurse(counter) + recurse(counter);
|
||||||
|
}
|
||||||
|
#pragma GCC diagnostic pop
|
||||||
|
|
||||||
|
static void *thread(void *arg)
|
||||||
|
{
|
||||||
|
(void) arg;
|
||||||
|
|
||||||
|
recurse(0);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(void)
|
||||||
|
{
|
||||||
|
puts("\nCortex-M Stack limit test\n");
|
||||||
|
|
||||||
|
puts("If the test fails, all onboard LEDs will be on");
|
||||||
|
|
||||||
|
#ifdef MODULE_CORTEXM_STACK_LIMIT
|
||||||
|
puts("The cortexm_stack_limit module is present. Expect the board to crash"\
|
||||||
|
" without onboard LEDs on.\n");
|
||||||
|
#else
|
||||||
|
puts("The cortexm_stack_limit module is missing! Expect the test to crash"\
|
||||||
|
" with all onboard LEDs on\n");
|
||||||
|
#endif
|
||||||
|
|
||||||
|
buf.canary = CANARY_VALUE;
|
||||||
|
thread_create(buf.stack, sizeof(buf.stack), THREAD_PRIORITY_MAIN + 1,
|
||||||
|
0, thread, NULL, "thread");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user