mirror of
https://github.com/zephyrproject-rtos/zephyr
synced 2025-09-07 08:32:57 +00:00
This patch reworks the current ARM __swap() function into a C function. Due to some issues with using svc calls withing fault handlers, we needed to change the way we initiate a swap by removing the dependence on svc #0. Before __swap() is called, the system has already done an irq_lock(). Upon return from __swap(), the equivalent of an irq_lock() is done due to restoration of the key value from the irq_lock preceeding the call. For ARM V6M (M0/M0+), the pendsv bit is toggled and the irqs are enabled. There is no priority masking in v6m, so it's just a global enable. For ARM V7M, the priority mask has to be set to 0x0 to allow for the pendsv IRQ to be taken. This is done for both via a call to irq_unlock(0). After this unlock, a pendsv irq will be taken, either at the tail end of the current irq handling if we are in handler mode, or immediately due to the pendsv being asserted (no other outstanding irqs). The next thread will be scheduled. Upon return from the context switch to the original thread, the priority mask will already be correct due to the pendsv processing. Signed-off-by: Andy Gross <andy.gross@linaro.org>
70 lines
2.2 KiB
C
70 lines
2.2 KiB
C
/*
|
|
* Copyright (c) 2018 Linaro, Limited
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include <kernel.h>
|
|
#include <toolchain.h>
|
|
#include <kernel_structs.h>
|
|
|
|
#ifdef CONFIG_EXECUTION_BENCHMARKING
|
|
extern void read_timer_start_of_swap(void);
|
|
#endif
|
|
extern const int _k_neg_eagain;
|
|
|
|
/**
|
|
*
|
|
* @brief Initiate a cooperative context switch
|
|
*
|
|
* The __swap() routine is invoked by various kernel services to effect
|
|
* a cooperative context context switch. Prior to invoking __swap(), the caller
|
|
* disables interrupts via irq_lock() and the return 'key' is passed as a
|
|
* parameter to __swap(). The 'key' actually represents the BASEPRI register
|
|
* prior to disabling interrupts via the BASEPRI mechanism.
|
|
*
|
|
* __swap() itself does not do much.
|
|
*
|
|
* It simply stores the intlock key (the BASEPRI value) parameter into
|
|
* current->basepri, and then triggers a PendSV exception, which does
|
|
* the heavy lifting of context switching.
|
|
|
|
* This is the only place we have to save BASEPRI since the other paths to
|
|
* __pendsv all come from handling an interrupt, which means we know the
|
|
* interrupts were not locked: in that case the BASEPRI value is 0.
|
|
*
|
|
* Given that __swap() is called to effect a cooperative context switch,
|
|
* only the caller-saved integer registers need to be saved in the thread of the
|
|
* outgoing thread. This is all performed by the hardware, which stores it in
|
|
* its exception stack frame, created when handling the svc exception.
|
|
*
|
|
* On ARMv6-M, the intlock key is represented by the PRIMASK register,
|
|
* as BASEPRI is not available.
|
|
*
|
|
* @return may contain a return value setup by a call to
|
|
* _set_thread_return_value()
|
|
*
|
|
*/
|
|
unsigned int __swap(int key)
|
|
{
|
|
#ifdef CONFIG_USERSPACE
|
|
/* Save off current privilege mode */
|
|
_current->arch.mode = __get_CONTROL() & CONTROL_nPRIV_Msk;
|
|
#endif
|
|
#ifdef CONFIG_EXECUTION_BENCHMARKING
|
|
read_timer_start_of_swap();
|
|
#endif
|
|
|
|
/* store off key and return value */
|
|
_current->arch.basepri = key;
|
|
_current->arch.swap_return_value = _k_neg_eagain;
|
|
|
|
/* set pending bit to make sure we will take a PendSV exception */
|
|
SCB->ICSR |= SCB_ICSR_PENDSVSET_Msk;
|
|
|
|
/* clear mask or enable all irqs to take a pendsv */
|
|
irq_unlock(0);
|
|
|
|
return _current->arch.swap_return_value;
|
|
}
|