2015-04-11 01:44:37 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2014 Wind River Systems, Inc.
|
|
|
|
*
|
2017-01-19 02:01:01 +01:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2015-04-11 01:44:37 +02:00
|
|
|
*/
|
|
|
|
|
2015-12-04 16:09:39 +01:00
|
|
|
/**
|
|
|
|
* @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.
|
2015-07-01 23:22:39 +02:00
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2016-11-08 16:36:50 +01:00
|
|
|
#include <kernel_structs.h>
|
|
|
|
#include <offsets_short.h>
|
2015-04-11 01:44:37 +02:00
|
|
|
#include <toolchain.h>
|
2019-10-24 17:08:21 +02:00
|
|
|
#include <linker/sections.h>
|
2015-05-28 19:56:47 +02:00
|
|
|
#include <arch/cpu.h>
|
2016-11-01 14:07:34 +01:00
|
|
|
#include <swap_macros.h>
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
GTEXT(_rirq_enter)
|
|
|
|
GTEXT(_rirq_exit)
|
2020-02-03 14:07:19 +01:00
|
|
|
GTEXT(_rirq_newthread_switch)
|
arc: trap handler, used by irq_offload, now handles thread switch
It was found that the test latency_measure, when compiled
for microkernel, would fail on the ARC. This because the
trap handler, used by irq_offload, wasn't supporting thread switching.
This submission adds the code to do that, and the code size is
bigger only when CONFIG_MICROKERNEL is defined.
To keep code a bit smaller, there is a trick exploited here where
the AE bit is cleared in the STATUS32 register and in AUX_IRQ_ACT,
bit 1 is set, to make it appear as if the machine has interrupted
at priority 1 level. It then can jump into some common interrupt
exit code for regular interrupts and perform an RTIE instruction
to switch into the new thread.
test/latency_measure/microkernel now passes.
Change-Id: I1872a80bb09a259814540567f51721203201679a
Signed-off-by: Chuck Jordan <cjordan@synopsys.com>
2016-05-26 19:39:20 +02:00
|
|
|
|
2019-07-25 06:13:13 +02:00
|
|
|
|
2016-10-29 00:27:11 +02:00
|
|
|
#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.
|
|
|
|
*/
|
|
|
|
|
2019-08-29 22:18:37 +02:00
|
|
|
/*
|
|
|
|
|
|
|
|
===========================================================
|
|
|
|
RETURN FROM INTERRUPT TO COOPERATIVE THREAD
|
|
|
|
===========================================================
|
|
|
|
|
|
|
|
That's a special case because:
|
|
|
|
1. We return from IRQ handler to a cooperative thread
|
|
|
|
2. During IRQ handling context switch did happen
|
|
|
|
3. Returning to a thread which previously gave control
|
|
|
|
to another thread because of:
|
|
|
|
- Calling k_sleep()
|
|
|
|
- Explicitly yielding
|
|
|
|
- Bumping into locked sync primitive etc
|
|
|
|
|
|
|
|
What (3) means is before passing control to another thread our thread
|
|
|
|
in question:
|
|
|
|
a. Stashed all precious caller-saved registers on its stack
|
|
|
|
b. Pushed return address to the top of the stack as well
|
|
|
|
|
|
|
|
That's how thread's stack looks like right before jumping to another thread:
|
|
|
|
----------------------------->8---------------------------------
|
|
|
|
PRE-CONTEXT-SWITCH STACK
|
|
|
|
|
|
|
|
lower_addr, let's say: 0x1000
|
|
|
|
|
|
|
|
--------------------------------------
|
|
|
|
SP -> | Return address; PC (Program Counter), in fact value taken from
|
2019-11-07 21:43:29 +01:00
|
|
|
| BLINK register in arch_switch()
|
2019-08-29 22:18:37 +02:00
|
|
|
--------------------------------------
|
|
|
|
| STATUS32 value, we explicitly save it here for later usage, read-on
|
|
|
|
--------------------------------------
|
|
|
|
| Caller-saved registers: some of R0-R12
|
|
|
|
--------------------------------------
|
|
|
|
|...
|
|
|
|
|...
|
|
|
|
|
|
|
|
higher_addr, let's say: 0x2000
|
|
|
|
----------------------------->8---------------------------------
|
|
|
|
|
|
|
|
When context gets switched the kernel saves callee-saved registers in the
|
|
|
|
thread's stack right on top of pre-switch contents so that's what we have:
|
|
|
|
----------------------------->8---------------------------------
|
|
|
|
POST-CONTEXT-SWITCH STACK
|
|
|
|
|
|
|
|
lower_addr, let's say: 0x1000
|
|
|
|
|
|
|
|
--------------------------------------
|
|
|
|
SP -> | Callee-saved registers: see struct _callee_saved_stack{}
|
|
|
|
| |- R13
|
|
|
|
| |- R14
|
|
|
|
| | ...
|
|
|
|
| \- FP
|
|
|
|
| ...
|
|
|
|
--------------------------------------
|
|
|
|
| Return address; PC (Program Counter)
|
|
|
|
--------------------------------------
|
|
|
|
| STATUS32 value
|
|
|
|
--------------------------------------
|
|
|
|
| Caller-saved registers: some of R0-R12
|
|
|
|
--------------------------------------
|
|
|
|
|...
|
|
|
|
|...
|
|
|
|
|
|
|
|
higher_addr, let's say: 0x2000
|
|
|
|
----------------------------->8---------------------------------
|
|
|
|
|
|
|
|
So how do we return in such a complex scenario.
|
|
|
|
|
|
|
|
First we restore callee-saved regs with help of _load_callee_saved_regs().
|
|
|
|
Now we're back to PRE-CONTEXT-SWITCH STACK (see above).
|
|
|
|
|
|
|
|
Logically our next step is to load return address from the top of the stack
|
|
|
|
and jump to that address to continue execution of the desired thread, but
|
|
|
|
we're still in interrupt handling mode and the only way to return to normal
|
|
|
|
execution mode is to execute "rtie" instruction. And here we need to deal
|
|
|
|
with peculiarities of return from IRQ on ARCv2 cores.
|
|
|
|
|
|
|
|
Instead of simple jump to a return address stored in the tip of thread's stack
|
|
|
|
(with subsequent interrupt enable) ARCv2 core additionally automatically
|
|
|
|
restores some registers from stack. Most important ones are
|
|
|
|
PC ("Program Counter") which holds address of the next instruction to execute
|
|
|
|
and STATUS32 which holds imortant flags including global interrupt enable,
|
|
|
|
zero, carry etc.
|
|
|
|
|
|
|
|
To make things worse depending on ARC core configuration and run-time setup
|
|
|
|
of certain features different set of registers will be restored.
|
|
|
|
|
|
|
|
Typically those same registers are automatically saved on stack on entry to
|
|
|
|
an interrupt, but remember we're returning to the thread which was
|
|
|
|
not interrupted by interrupt and so on its stack there're no automatically
|
|
|
|
saved registers, still inevitably on RTIE execution register restoration
|
|
|
|
will happen. So if we do nothing special we'll end-up with that:
|
|
|
|
----------------------------->8---------------------------------
|
|
|
|
lower_addr, let's say: 0x1000
|
|
|
|
|
|
|
|
--------------------------------------
|
|
|
|
# | Return address; PC (Program Counter)
|
|
|
|
| --------------------------------------
|
|
|
|
| | STATUS32 value
|
|
|
|
| --------------------------------------
|
|
|
|
|
|
|
|
|
sizeof(_irq_stack_frame)
|
|
|
|
|
|
|
|
|
| | Caller-saved registers: R0-R12
|
|
|
|
V --------------------------------------
|
|
|
|
|...
|
|
|
|
SP -> | < Some data on thread's stack>
|
|
|
|
|...
|
|
|
|
|
|
|
|
higher_addr, let's say: 0x2000
|
|
|
|
----------------------------->8---------------------------------
|
|
|
|
|
|
|
|
I.e. we'll go much deeper down the stack over needed return address, read
|
|
|
|
some value from unexpected location in stack and will try to jump there.
|
|
|
|
Nobody knows were we end-up then.
|
|
|
|
|
|
|
|
To work-around that problem we need to mimic existance of IRQ stack frame
|
|
|
|
of which we really only need return address obviously to return where we
|
|
|
|
need to. For that we just shift SP so that it points sizeof(_irq_stack_frame)
|
|
|
|
above like that:
|
|
|
|
----------------------------->8---------------------------------
|
|
|
|
lower_addr, let's say: 0x1000
|
|
|
|
|
|
|
|
SP -> |
|
|
|
|
A | < Some unrelated data >
|
|
|
|
| |
|
|
|
|
|
|
|
|
|
sizeof(_irq_stack_frame)
|
|
|
|
|
|
|
|
|
| --------------------------------------
|
|
|
|
| | Return address; PC (Program Counter)
|
|
|
|
| --------------------------------------
|
|
|
|
# | STATUS32 value
|
|
|
|
--------------------------------------
|
|
|
|
| Caller-saved registers: R0-R12
|
|
|
|
--------------------------------------
|
|
|
|
|...
|
|
|
|
| < Some data on thread's stack>
|
|
|
|
|...
|
|
|
|
|
|
|
|
higher_addr, let's say: 0x2000
|
|
|
|
----------------------------->8---------------------------------
|
|
|
|
|
|
|
|
Indeed R0-R13 "restored" from IRQ stack frame will contain garbage but
|
|
|
|
it makes no difference because we're returning to execution of code as if
|
|
|
|
we're returning from yet another function call and so we will restore
|
|
|
|
all needed registers from the stack.
|
|
|
|
|
|
|
|
One other important remark here is R13.
|
|
|
|
|
|
|
|
CPU hardware automatically save/restore registers in pairs and since we
|
|
|
|
wanted to save/restore R12 in IRQ stack frame as a caller-saved register we
|
|
|
|
just happen to do that for R13 as well. But given compiler treats it as
|
|
|
|
a callee-saved register we save/restore it separately in _callee_saved_stack
|
|
|
|
structure. And when we restore callee-saved registers from stack we among
|
|
|
|
other registers recover R13. But later on return from IRQ with RTIE
|
|
|
|
instruction, R13 will be "restored" again from fake IRQ stack frame and
|
|
|
|
if we don't copy correct R13 value to fake IRQ stack frame R13 value
|
|
|
|
will be corrupted.
|
|
|
|
|
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2015-07-01 23:22:39 +02:00
|
|
|
/**
|
|
|
|
*
|
2015-07-01 23:51:40 +02:00
|
|
|
* @brief Work to be done before handing control to an IRQ ISR
|
2015-07-01 23:22:39 +02:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2015-07-01 23:29:04 +02:00
|
|
|
* @return N/A
|
2015-07-01 23:22:39 +02:00
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
SECTION_FUNC(TEXT, _rirq_enter)
|
|
|
|
|
2017-07-11 04:39:54 +02:00
|
|
|
|
2016-03-11 18:29:14 +01:00
|
|
|
#ifdef CONFIG_ARC_STACK_CHECKING
|
2019-08-01 06:39:35 +02:00
|
|
|
#ifdef CONFIG_ARC_SECURE_FIRMWARE
|
2018-07-19 10:59:21 +02:00
|
|
|
lr r2, [_ARC_V2_SEC_STAT]
|
|
|
|
bclr r2, r2, _ARC_V2_SEC_STAT_SSC_BIT
|
2019-08-01 06:39:35 +02:00
|
|
|
sflag r2
|
|
|
|
|
2018-07-19 10:59:21 +02:00
|
|
|
#else
|
2016-03-11 18:29:14 +01:00
|
|
|
/* disable stack checking */
|
|
|
|
lr r2, [_ARC_V2_STATUS32]
|
|
|
|
bclr r2, r2, _ARC_V2_STATUS32_SC_BIT
|
|
|
|
kflag r2
|
2018-07-19 10:59:21 +02:00
|
|
|
#endif
|
2016-03-11 18:29:14 +01:00
|
|
|
#endif
|
2017-07-11 04:39:54 +02:00
|
|
|
clri
|
|
|
|
|
2019-08-13 18:45:13 +02:00
|
|
|
/* check whether irq stack is used */
|
|
|
|
_check_and_inc_int_nest_counter r0, r1
|
2019-07-03 17:43:31 +02:00
|
|
|
|
|
|
|
bne.d rirq_nest
|
2019-08-29 12:52:04 +02:00
|
|
|
mov_s r0, sp
|
2017-07-11 04:39:54 +02:00
|
|
|
|
2019-07-25 06:13:13 +02:00
|
|
|
_get_curr_cpu_irq_stack sp
|
2017-07-11 04:39:54 +02:00
|
|
|
rirq_nest:
|
|
|
|
push_s r0
|
|
|
|
|
|
|
|
seti
|
2015-04-11 01:44:37 +02:00
|
|
|
j _isr_demux
|
|
|
|
|
|
|
|
|
2015-07-01 23:22:39 +02:00
|
|
|
/**
|
|
|
|
*
|
2015-07-01 23:51:40 +02:00
|
|
|
* @brief Work to be done exiting an IRQ
|
2015-07-01 23:22:39 +02:00
|
|
|
*
|
2015-07-01 23:29:04 +02:00
|
|
|
* @return N/A
|
2015-07-01 23:22:39 +02:00
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
SECTION_FUNC(TEXT, _rirq_exit)
|
2017-07-11 04:39:54 +02:00
|
|
|
clri
|
|
|
|
|
|
|
|
pop sp
|
|
|
|
|
2019-08-13 18:45:13 +02:00
|
|
|
_dec_int_nest_counter r0, r1
|
|
|
|
|
2019-07-03 17:43:31 +02:00
|
|
|
_check_nest_int_by_irq_act r0, r1
|
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
jne _rirq_no_switch
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
/* sp is struct k_thread **old of z_arc_switch_in_isr
|
|
|
|
* which is a wrapper of z_get_next_switch_handle.
|
|
|
|
* r0 contains the 1st thread in ready queue. if
|
|
|
|
* it equals _current(r2) ,then do swap, or no swap.
|
2015-04-11 01:44:37 +02:00
|
|
|
*/
|
2020-02-03 14:07:19 +01:00
|
|
|
_get_next_switch_handle
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
cmp r0, r2
|
|
|
|
beq _rirq_no_switch
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2019-08-13 13:58:03 +02:00
|
|
|
#ifdef CONFIG_ARC_SECURE_FIRMWARE
|
|
|
|
/* here need to remember SEC_STAT.IRM bit */
|
|
|
|
lr r3, [_ARC_V2_SEC_STAT]
|
2019-08-29 12:52:04 +02:00
|
|
|
push_s r3
|
2019-08-13 13:58:03 +02:00
|
|
|
#endif
|
2020-02-01 12:37:51 +01:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
/* r2 is old thread */
|
|
|
|
_irq_store_old_thread_callee_regs
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2016-11-08 16:36:50 +01:00
|
|
|
st _CAUSE_RIRQ, [r2, _thread_offset_to_relinquish_cause]
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
/* mov new thread (r0) to r2 */
|
|
|
|
mov r2, r0
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
/* _rirq_newthread_switch required by exception handling */
|
2016-11-03 21:11:13 +01:00
|
|
|
.balign 4
|
2020-02-03 14:07:19 +01:00
|
|
|
_rirq_newthread_switch:
|
2019-07-01 09:48:20 +02:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
_load_new_thread_callee_regs
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
breq r3, _CAUSE_RIRQ, _rirq_switch_from_rirq
|
2019-08-29 12:52:04 +02:00
|
|
|
nop_s
|
2020-02-03 14:07:19 +01:00
|
|
|
breq r3, _CAUSE_FIRQ, _rirq_switch_from_firq
|
2019-08-29 12:52:04 +02:00
|
|
|
nop_s
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
/* fall through */
|
|
|
|
|
|
|
|
.balign 4
|
2020-02-03 14:07:19 +01:00
|
|
|
_rirq_switch_from_coop:
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
_set_misc_regs_irq_switch_from_coop
|
2019-08-29 22:18:37 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* See verbose explanation of
|
|
|
|
* RETURN FROM INTERRUPT TO COOPERATIVE THREAD above
|
|
|
|
*/
|
|
|
|
|
2015-04-11 01:44:37 +02:00
|
|
|
/* carve fake stack */
|
2018-03-07 19:48:48 +01:00
|
|
|
sub sp, sp, ___isf_t_pc_OFFSET
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
|
2019-02-04 13:22:51 +01:00
|
|
|
/* reset zero-overhead loops */
|
|
|
|
st 0, [sp, ___isf_t_lp_end_OFFSET]
|
2019-01-14 14:16:10 +01:00
|
|
|
|
2015-04-11 01:44:37 +02:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
2016-11-08 16:36:50 +01:00
|
|
|
st_s r13, [sp, ___isf_t_r13_OFFSET]
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2019-08-13 13:58:03 +02:00
|
|
|
/* stack now has the IRQ stack frame layout, pointing to sp */
|
2015-04-11 01:44:37 +02:00
|
|
|
/* rtie will pop the rest from the stack */
|
2019-08-13 13:58:03 +02:00
|
|
|
rtie
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
.balign 4
|
2020-02-03 14:07:19 +01:00
|
|
|
_rirq_switch_from_firq:
|
|
|
|
_rirq_switch_from_rirq:
|
|
|
|
|
|
|
|
_set_misc_regs_irq_switch_from_irq
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2020-02-03 14:07:19 +01:00
|
|
|
_rirq_no_switch:
|
2015-04-11 01:44:37 +02:00
|
|
|
rtie
|