89455bc9a1
Signed-off-by: buxiasen <buxiasen@xiaomi.com>
484 lines
13 KiB
C
484 lines
13 KiB
C
/****************************************************************************
|
|
* arch/xtensa/include/irq.h
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
* this work for additional information regarding copyright ownership. The
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance with the
|
|
* License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
* License for the specific language governing permissions and limitations
|
|
* under the License.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/* This file should never be included directly but, rather, only indirectly
|
|
* through nuttx/irq.h
|
|
*/
|
|
|
|
#ifndef __ARCH_XTENSA_INCLUDE_IRQ_H
|
|
#define __ARCH_XTENSA_INCLUDE_IRQ_H
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
/* Include NuttX-specific IRQ definitions */
|
|
|
|
#include <nuttx/config.h>
|
|
#include <nuttx/irq.h>
|
|
|
|
#include <sys/types.h>
|
|
#ifndef __ASSEMBLY__
|
|
# include <stdbool.h>
|
|
# include <arch/syscall.h>
|
|
#endif
|
|
|
|
#include <arch/types.h>
|
|
#include <arch/chip/tie.h>
|
|
#include <arch/chip/core-isa.h>
|
|
#include <arch/xtensa/core.h>
|
|
|
|
#include <arch/xtensa/xtensa_specregs.h>
|
|
#include <arch/xtensa/xtensa_corebits.h>
|
|
#include <arch/xtensa/xtensa_coproc.h>
|
|
|
|
/* Include chip-specific IRQ definitions (including IRQ numbers) */
|
|
|
|
#include <arch/chip/irq.h>
|
|
|
|
/* Include architecture-specific IRQ definitions */
|
|
|
|
#ifdef CONFIG_ARCH_FAMILY_LX6
|
|
# include <arch/lx6/irq.h>
|
|
|
|
#elif CONFIG_ARCH_FAMILY_LX7
|
|
# include <arch/lx7/irq.h>
|
|
|
|
#else
|
|
# error Unknown XTENSA architecture
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Pre-processor Definitions
|
|
****************************************************************************/
|
|
|
|
/* IRQ Stack Frame Format. Each value is a uint32_t register index */
|
|
|
|
#define REG_PC (0) /* Return PC */
|
|
#define REG_PS (1) /* Return PS */
|
|
#define REG_A0 (2)
|
|
#define REG_A1 (3) /* Stack pointer before interrupt */
|
|
#define REG_A2 (4)
|
|
#define REG_A3 (5)
|
|
#define REG_A4 (6)
|
|
#define REG_A5 (7)
|
|
#define REG_A6 (8)
|
|
#define REG_A7 (9)
|
|
#define REG_A8 (10)
|
|
#define REG_A9 (11)
|
|
#define REG_A10 (12)
|
|
#define REG_A11 (13)
|
|
#define REG_A12 (14)
|
|
#define REG_A13 (15)
|
|
#define REG_A14 (16)
|
|
#define REG_A15 (17)
|
|
#define REG_SAR (18)
|
|
#define REG_EXCCAUSE (19)
|
|
#define REG_EXCVADDR (20)
|
|
#define REG_THREADPTR (21)
|
|
|
|
#define _REG_EXTRA_START (22)
|
|
|
|
#if XCHAL_HAVE_S32C1I != 0
|
|
# define REG_SCOMPARE1 (_REG_EXTRA_START + 0)
|
|
# define _REG_LOOPS_START (_REG_EXTRA_START + 1)
|
|
#else
|
|
# define _REG_LOOPS_START _REG_EXTRA_START
|
|
#endif
|
|
|
|
#if XCHAL_HAVE_LOOPS != 0
|
|
# define REG_LBEG (_REG_LOOPS_START + 0)
|
|
# define REG_LEND (_REG_LOOPS_START + 1)
|
|
# define REG_LCOUNT (_REG_LOOPS_START + 2)
|
|
# define _REG_WINDOW_TMPS (_REG_LOOPS_START + 3)
|
|
#else
|
|
# define _REG_WINDOW_TMPS _REG_LOOPS_START
|
|
#endif
|
|
|
|
#ifndef __XTENSA_CALL0_ABI__
|
|
/* Temporary space for saving stuff during window spill. */
|
|
|
|
# define REG_TMP0 (_REG_WINDOW_TMPS + 0)
|
|
# define _REG_INT_CTX_START (_REG_WINDOW_TMPS + 1)
|
|
#else
|
|
# define _REG_INT_CTX_START _REG_WINDOW_TMPS
|
|
#endif
|
|
|
|
#ifndef CONFIG_BUILD_FLAT
|
|
/* Temporary space for saving Interrupt Context information */
|
|
|
|
# define REG_INT_CTX (_REG_INT_CTX_START + 0)
|
|
# define _REG_OVLY_START (_REG_INT_CTX_START + 1)
|
|
#else
|
|
# define _REG_OVLY_START _REG_INT_CTX_START
|
|
#endif
|
|
|
|
#ifdef CONFIG_XTENSA_USE_OVLY
|
|
/* Storage for overlay state */
|
|
|
|
# error Overlays not supported
|
|
# define _REG_CP_START _REG_OVLY_START
|
|
#else
|
|
# define _REG_CP_START _REG_OVLY_START
|
|
#endif
|
|
|
|
#if XCHAL_CP_NUM > 0
|
|
/* FPU first address must align to CP align size.
|
|
* ESP32 3rd party redefine the ALIGN_UP, so define a new macro XALIGN_UP()
|
|
* instead use ALGIN_UP() in nuttx/nuttx.h
|
|
*/
|
|
|
|
# define XALIGN_UP(x,a) (((x) + ((a) - 1)) & ~((a) - 1))
|
|
# define COMMON_CTX_REGS XALIGN_UP(_REG_CP_START, XCHAL_TOTAL_SA_ALIGN / 4)
|
|
# define COPROC_CTX_REGS (XTENSA_CP_SA_SIZE / 4)
|
|
# define RESERVE_REGS 8
|
|
# define XCPTCONTEXT_REGS (COMMON_CTX_REGS + COPROC_CTX_REGS + RESERVE_REGS)
|
|
#else
|
|
# define COMMON_CTX_REGS _REG_CP_START
|
|
# define RESERVE_REGS 8
|
|
# define XCPTCONTEXT_REGS (COMMON_CTX_REGS + RESERVE_REGS)
|
|
#endif
|
|
|
|
#define XCPTCONTEXT_SIZE (4 * XCPTCONTEXT_REGS)
|
|
|
|
/****************************************************************************
|
|
* Public Types
|
|
****************************************************************************/
|
|
|
|
#ifndef __ASSEMBLY__
|
|
|
|
#ifdef CONFIG_LIB_SYSCALL
|
|
/* This structure represents the return state from a system call */
|
|
|
|
struct xcpt_syscall_s
|
|
{
|
|
uintptr_t sysreturn; /* The return PC */
|
|
#ifndef CONFIG_BUILD_FLAT
|
|
uintptr_t int_ctx; /* Interrupt context */
|
|
#endif
|
|
};
|
|
#endif
|
|
|
|
/* This struct defines the way the registers are stored. */
|
|
|
|
struct xcptcontext
|
|
{
|
|
/* These are saved copies of registers used during signal processing.
|
|
*
|
|
* REVISIT: Because there is only one copy of these save areas,
|
|
* only a single signal handler can be active. This precludes
|
|
* queuing of signal actions. As a result, signals received while
|
|
* another signal handler is executing will be ignored!
|
|
*/
|
|
|
|
uint32_t *saved_regs;
|
|
|
|
/* Register save area */
|
|
|
|
uint32_t *regs;
|
|
|
|
#ifndef CONFIG_BUILD_FLAT
|
|
/* This is the saved address to use when returning from a user-space
|
|
* signal handler.
|
|
*/
|
|
|
|
uintptr_t sigreturn;
|
|
#endif
|
|
|
|
#ifdef CONFIG_LIB_SYSCALL
|
|
/* The following array holds the return address and the exc_return value
|
|
* needed to return from each nested system call.
|
|
*/
|
|
|
|
uint8_t nsyscalls;
|
|
struct xcpt_syscall_s syscall[CONFIG_SYS_NNEST];
|
|
#endif
|
|
};
|
|
|
|
/****************************************************************************
|
|
* Inline functions
|
|
****************************************************************************/
|
|
|
|
/* Return the current value of the PS register */
|
|
|
|
static inline_function uint32_t xtensa_getps(void)
|
|
{
|
|
uint32_t ps;
|
|
|
|
__asm__ __volatile__
|
|
(
|
|
"rsr %0, PS" : "=r"(ps)
|
|
);
|
|
|
|
return ps;
|
|
}
|
|
|
|
/* Set the value of the PS register */
|
|
|
|
noinstrument_function static inline_function
|
|
void xtensa_setps(uint32_t ps)
|
|
{
|
|
__asm__ __volatile__
|
|
(
|
|
"wsr %0, PS \n"
|
|
"rsync \n"
|
|
:
|
|
: "r"(ps)
|
|
: "memory"
|
|
);
|
|
}
|
|
|
|
/* Return the current value of the stack pointer */
|
|
|
|
static inline_function uint32_t up_getsp(void)
|
|
{
|
|
register uint32_t sp;
|
|
|
|
__asm__ __volatile__
|
|
(
|
|
"mov %0, sp\n"
|
|
: "=r" (sp)
|
|
);
|
|
|
|
return sp;
|
|
}
|
|
|
|
/* Restore the value of the PS register */
|
|
|
|
noinstrument_function static inline_function
|
|
void up_irq_restore(uint32_t ps)
|
|
{
|
|
__asm__ __volatile__
|
|
(
|
|
"wsr %0, PS\n"
|
|
"rsync \n"
|
|
:
|
|
: "r"(ps)
|
|
: "memory"
|
|
);
|
|
}
|
|
|
|
/* Disable interrupts and return the previous value of the PS register */
|
|
|
|
noinstrument_function static inline_function uint32_t up_irq_save(void)
|
|
{
|
|
uint32_t ps;
|
|
|
|
/* Disable all low- and medium-priority interrupts. High priority
|
|
* interrupts should not interfere with ongoing RTOS operations and
|
|
* are not disabled.
|
|
*/
|
|
|
|
__asm__ __volatile__
|
|
(
|
|
"rsil %0, %1" : "=r"(ps) : "i"(XCHAL_IRQ_LEVEL)
|
|
);
|
|
|
|
/* Return the previous PS value so that it can be restored with
|
|
* up_irq_restore().
|
|
*/
|
|
|
|
return ps;
|
|
}
|
|
|
|
/* Enable interrupts at all levels */
|
|
|
|
static inline_function void up_irq_enable(void)
|
|
{
|
|
#ifdef __XTENSA_CALL0_ABI__
|
|
xtensa_setps(PS_INTLEVEL(0) | PS_UM);
|
|
#else
|
|
xtensa_setps(PS_INTLEVEL(0) | PS_UM | PS_WOE);
|
|
#endif
|
|
}
|
|
|
|
/* Disable low- and medium- priority interrupts */
|
|
|
|
noinstrument_function static inline_function void up_irq_disable(void)
|
|
{
|
|
#ifdef __XTENSA_CALL0_ABI__
|
|
xtensa_setps(PS_INTLEVEL(XCHAL_EXCM_LEVEL) | PS_UM);
|
|
#else
|
|
xtensa_setps(PS_INTLEVEL(XCHAL_EXCM_LEVEL) | PS_UM | PS_WOE);
|
|
#endif
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: xtensa_disable_all
|
|
****************************************************************************/
|
|
|
|
static inline_function void xtensa_disable_all(void)
|
|
{
|
|
__asm__ __volatile__
|
|
(
|
|
"movi a2, 0\n"
|
|
"xsr a2, INTENABLE\n"
|
|
"rsync\n"
|
|
: : : "a2"
|
|
);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: xtensa_intclear
|
|
****************************************************************************/
|
|
|
|
static inline_function void xtensa_intclear(uint32_t mask)
|
|
{
|
|
__asm__ __volatile__
|
|
(
|
|
"wsr %0, INTCLEAR\n"
|
|
"rsync\n"
|
|
:
|
|
: "r"(mask)
|
|
:
|
|
);
|
|
}
|
|
|
|
#ifdef __cplusplus
|
|
#define EXTERN extern "C"
|
|
extern "C"
|
|
{
|
|
#else
|
|
#define EXTERN extern
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Public Data
|
|
****************************************************************************/
|
|
|
|
#ifndef __ASSEMBLY__
|
|
/* g_interrupt_context store irq status */
|
|
|
|
extern volatile bool g_interrupt_context[CONFIG_SMP_NCPUS];
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Public Function Prototypes
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: xtensa_enable_interrupts
|
|
*
|
|
* Description:
|
|
* Enables a set of interrupts. Does not simply set INTENABLE directly,
|
|
* but computes it as a function of the current virtual priority.
|
|
* Can be called from interrupt handlers.
|
|
*
|
|
****************************************************************************/
|
|
|
|
irqstate_t xtensa_enable_interrupts(irqstate_t mask);
|
|
|
|
/****************************************************************************
|
|
* Name: xtensa_disable_interrupts
|
|
*
|
|
* Description:
|
|
* Disables a set of interrupts. Does not simply clear INTENABLE directly,
|
|
* but computes it as a function of the current virtual priority.
|
|
* Can be called from interrupt handlers.
|
|
*
|
|
****************************************************************************/
|
|
|
|
irqstate_t xtensa_disable_interrupts(irqstate_t mask);
|
|
|
|
/****************************************************************************
|
|
* Name: up_cpu_index
|
|
*
|
|
* Description:
|
|
* Return the real core number regardless CONFIG_SMP setting
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifdef CONFIG_ARCH_HAVE_MULTICPU
|
|
int up_cpu_index(void) noinstrument_function;
|
|
#endif /* CONFIG_ARCH_HAVE_MULTICPU */
|
|
|
|
/****************************************************************************
|
|
* Name: up_set_interrupt_context
|
|
*
|
|
* Description:
|
|
* Set the interrupt handler context.
|
|
*
|
|
****************************************************************************/
|
|
|
|
noinstrument_function
|
|
static inline_function void up_set_interrupt_context(bool flag)
|
|
{
|
|
#ifdef CONFIG_SMP
|
|
g_interrupt_context[up_cpu_index()] = flag;
|
|
#else
|
|
g_interrupt_context[0] = flag;
|
|
#endif
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: up_interrupt_context
|
|
*
|
|
* Description:
|
|
* Return true is we are currently executing in the interrupt
|
|
* handler context.
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifndef __ASSEMBLY__
|
|
noinstrument_function static inline_function bool up_interrupt_context(void)
|
|
{
|
|
#ifdef CONFIG_SMP
|
|
irqstate_t flags = up_irq_save();
|
|
bool ret = g_interrupt_context[up_cpu_index()];
|
|
up_irq_restore(flags);
|
|
return ret;
|
|
#else
|
|
return g_interrupt_context[0];
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
#define up_switch_context(tcb, rtcb) \
|
|
do { \
|
|
if (!up_interrupt_context()) \
|
|
{ \
|
|
sys_call0(SYS_switch_context); \
|
|
} \
|
|
UNUSED(rtcb); \
|
|
} while (0)
|
|
|
|
/****************************************************************************
|
|
* Name: up_getusrpc
|
|
****************************************************************************/
|
|
|
|
#define up_getusrpc(regs) \
|
|
(((uint32_t *)((regs) ? (regs) : running_regs()))[REG_PC])
|
|
|
|
/****************************************************************************
|
|
* Name: up_getusrsp
|
|
****************************************************************************/
|
|
|
|
#define up_getusrsp(regs) \
|
|
(((uintptr_t*)(regs))[REG_A1])
|
|
|
|
#undef EXTERN
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* __ASSEMBLY__ */
|
|
#endif /* __ARCH_XTENSA_INCLUDE_IRQ_H */
|