cca39204c2
* it's based on ARC SecureShield * add basic secure service in arch/arc/core/secureshield * necesssary changes in arch level * thread switch * irq/exception handling * initialization * add secure time support Signed-off-by: Wayne Ren <wei.ren@synopsys.com>
238 lines
5.2 KiB
ArmAsm
238 lines
5.2 KiB
ArmAsm
/*
|
|
* Copyright (c) 2014 Wind River Systems, Inc.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/**
|
|
* @file
|
|
* @brief Handling of transitions to-and-from regular IRQs (RIRQ)
|
|
*
|
|
* This module implements the code for handling entry to and exit from regular
|
|
* IRQs.
|
|
*
|
|
* See isr_wrapper.S for details.
|
|
*/
|
|
|
|
#include <kernel_structs.h>
|
|
#include <offsets_short.h>
|
|
#include <toolchain.h>
|
|
#include <arch/cpu.h>
|
|
#include <swap_macros.h>
|
|
|
|
GTEXT(_rirq_enter)
|
|
GTEXT(_rirq_exit)
|
|
GTEXT(_rirq_common_interrupt_swap)
|
|
|
|
|
|
#if 0 /* TODO: when FIRQ is not present, all would be regular */
|
|
#define NUM_REGULAR_IRQ_PRIO_LEVELS CONFIG_NUM_IRQ_PRIO_LEVELS
|
|
#else
|
|
#define NUM_REGULAR_IRQ_PRIO_LEVELS (CONFIG_NUM_IRQ_PRIO_LEVELS-1)
|
|
#endif
|
|
/* note: the above define assumes that prio 0 IRQ is for FIRQ, and
|
|
* that all others are regular interrupts.
|
|
* TODO: Revist this if FIRQ becomes configurable.
|
|
*/
|
|
|
|
|
|
/**
|
|
*
|
|
* @brief Work to be done before handing control to an IRQ ISR
|
|
*
|
|
* The processor pushes automatically all registers that need to be saved.
|
|
* However, since the processor always runs at kernel privilege there is no
|
|
* automatic switch to the IRQ stack: this must be done in software.
|
|
*
|
|
* Assumption by _isr_demux: r3 is untouched by _rirq_enter.
|
|
*
|
|
* @return N/A
|
|
*/
|
|
|
|
SECTION_FUNC(TEXT, _rirq_enter)
|
|
|
|
|
|
#ifdef CONFIG_ARC_STACK_CHECKING
|
|
#ifdef CONFIG_ARC_SECURE_FIRMWARE
|
|
lr r2, [_ARC_V2_SEC_STAT]
|
|
bclr r2, r2, _ARC_V2_SEC_STAT_SSC_BIT
|
|
sflag r2
|
|
|
|
#else
|
|
/* disable stack checking */
|
|
lr r2, [_ARC_V2_STATUS32]
|
|
bclr r2, r2, _ARC_V2_STATUS32_SC_BIT
|
|
kflag r2
|
|
#endif
|
|
#endif
|
|
clri
|
|
|
|
_check_nest_int_by_irq_act r0, r1
|
|
|
|
bne.d rirq_nest
|
|
mov r0, sp
|
|
|
|
_get_curr_cpu_irq_stack sp
|
|
rirq_nest:
|
|
push_s r0
|
|
|
|
seti
|
|
j _isr_demux
|
|
|
|
|
|
/**
|
|
*
|
|
* @brief Work to be done exiting an IRQ
|
|
*
|
|
* @return N/A
|
|
*/
|
|
|
|
SECTION_FUNC(TEXT, _rirq_exit)
|
|
clri
|
|
|
|
pop sp
|
|
|
|
_check_nest_int_by_irq_act r0, r1
|
|
|
|
jne _rirq_return_from_rirq
|
|
|
|
#ifdef CONFIG_STACK_SENTINEL
|
|
bl z_check_stack_sentinel
|
|
#endif
|
|
|
|
#ifdef CONFIG_PREEMPT_ENABLED
|
|
|
|
#ifdef CONFIG_SMP
|
|
bl z_arch_smp_switch_in_isr
|
|
/* r0 points to new thread, r1 points to old thread */
|
|
cmp r0, 0
|
|
beq _rirq_no_reschedule
|
|
mov r2, r1
|
|
#else
|
|
mov r1, _kernel
|
|
ld_s r2, [r1, _kernel_offset_to_current]
|
|
|
|
/*
|
|
* Lock interrupts to ensure kernel queues do not change from this
|
|
* point on until return from interrupt.
|
|
*/
|
|
|
|
/*
|
|
* Both (a)reschedule and (b)non-reschedule cases need to load the
|
|
* current thread's stack, but don't have to use it until the decision
|
|
* is taken: load the delay slots with the 'load stack pointer'
|
|
* instruction.
|
|
*
|
|
* a) needs to load it to save outgoing context.
|
|
* b) needs to load it to restore the interrupted context.
|
|
*/
|
|
|
|
/* check if the current thread needs to be rescheduled */
|
|
ld_s r0, [r1, _kernel_offset_to_ready_q_cache]
|
|
cmp_s r0, r2
|
|
beq _rirq_no_reschedule
|
|
|
|
/* cached thread to run is in r0, fall through */
|
|
#endif
|
|
.balign 4
|
|
_rirq_reschedule:
|
|
|
|
/* _save_callee_saved_regs expects outgoing thread in r2 */
|
|
_save_callee_saved_regs
|
|
|
|
st _CAUSE_RIRQ, [r2, _thread_offset_to_relinquish_cause]
|
|
|
|
#ifdef CONFIG_SMP
|
|
mov r2, r0
|
|
#else
|
|
/* incoming thread is in r0: it becomes the new 'current' */
|
|
mov r2, r0
|
|
st_s r2, [r1, _kernel_offset_to_current]
|
|
#endif
|
|
|
|
.balign 4
|
|
_rirq_common_interrupt_swap:
|
|
/* r2 contains pointer to new thread */
|
|
|
|
#ifdef CONFIG_ARC_STACK_CHECKING
|
|
_load_stack_check_regs
|
|
#endif
|
|
/*
|
|
* _load_callee_saved_regs expects incoming thread in r2.
|
|
* _load_callee_saved_regs restores the stack pointer.
|
|
*/
|
|
_load_callee_saved_regs
|
|
|
|
#if defined(CONFIG_MPU_STACK_GUARD) || defined(CONFIG_USERSPACE)
|
|
push_s r2
|
|
mov r0, r2
|
|
bl configure_mpu_thread
|
|
pop_s r2
|
|
#endif
|
|
|
|
#if defined(CONFIG_USERSPACE)
|
|
/*
|
|
* when USERSPACE is enabled, according to ARCv2 ISA, SP will be switched
|
|
* if interrupt comes out in user mode, and will be recorded in bit 31
|
|
* (U bit) of IRQ_ACT. when interrupt exits, SP will be switched back
|
|
* according to U bit.
|
|
*
|
|
* For the case that context switches in interrupt, the target sp must be
|
|
* thread's kernel stack, no need to do hardware sp switch. so, U bit should
|
|
* be cleared.
|
|
*/
|
|
lr r0, [_ARC_V2_AUX_IRQ_ACT]
|
|
bclr r0, r0, 31
|
|
sr r0, [_ARC_V2_AUX_IRQ_ACT]
|
|
#endif
|
|
|
|
ld r3, [r2, _thread_offset_to_relinquish_cause]
|
|
|
|
breq r3, _CAUSE_RIRQ, _rirq_return_from_rirq
|
|
nop
|
|
breq r3, _CAUSE_FIRQ, _rirq_return_from_firq
|
|
nop
|
|
|
|
/* fall through */
|
|
|
|
.balign 4
|
|
_rirq_return_from_coop:
|
|
|
|
#ifdef CONFIG_ARC_SECURE_FIRMWARE
|
|
/* must return to secure mode, so set IRM bit to 1 */
|
|
lr r0, [_ARC_V2_SEC_STAT]
|
|
bset r0, r0, _ARC_V2_SEC_STAT_IRM_BIT
|
|
sflag r0
|
|
#endif
|
|
/* carve fake stack */
|
|
sub sp, sp, ___isf_t_pc_OFFSET
|
|
|
|
|
|
/* reset zero-overhead loops */
|
|
st 0, [sp, ___isf_t_lp_end_OFFSET]
|
|
|
|
/*
|
|
* r13 is part of both the callee and caller-saved register sets because
|
|
* the processor is only able to save registers in pair in the regular
|
|
* IRQ prologue. r13 thus has to be set to its correct value in the IRQ
|
|
* stack frame.
|
|
*/
|
|
st_s r13, [sp, ___isf_t_r13_OFFSET]
|
|
|
|
/* stack now has the IRQ stack frame layout, pointing to r0 */
|
|
|
|
/* fall through to rtie instruction */
|
|
|
|
/* rtie will pop the rest from the stack */
|
|
|
|
/* fall through to rtie instruction */
|
|
|
|
#endif /* CONFIG_PREEMPT_ENABLED */
|
|
|
|
.balign 4
|
|
_rirq_return_from_firq:
|
|
_rirq_return_from_rirq:
|
|
_rirq_no_reschedule:
|
|
|
|
rtie
|