3d1ce144df
since all other special register operation in irq.h Signed-off-by: Xiang Xiao <xiaoxiang@xiaomi.com>
337 lines
10 KiB
C
337 lines
10 KiB
C
/****************************************************************************
|
||
* arch/x86/include/i486/irq.h
|
||
*
|
||
* 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_X86_INCLUDE_I486_IRQ_H
|
||
#define __ARCH_X86_INCLUDE_I486_IRQ_H
|
||
|
||
/****************************************************************************
|
||
* Included Files
|
||
****************************************************************************/
|
||
|
||
#ifndef __ASSEMBLY__
|
||
# include <stdint.h>
|
||
# include <stdbool.h>
|
||
# include <arch/arch.h>
|
||
#endif
|
||
|
||
/****************************************************************************
|
||
* Pre-processor Definitions
|
||
****************************************************************************/
|
||
|
||
/* ISR and IRQ numbers */
|
||
|
||
#define ISR0 0 /* Division by zero exception */
|
||
#define ISR1 1 /* Debug exception */
|
||
#define ISR2 2 /* Non maskable interrupt */
|
||
#define ISR3 3 /* Breakpoint exception */
|
||
#define ISR4 4 /* 'Into detected overflow' */
|
||
#define ISR5 5 /* Out of bounds exception */
|
||
#define ISR6 6 /* Invalid opcode exception */
|
||
#define ISR7 7 /* No coprocessor exception */
|
||
#define ISR8 8 /* Double fault (pushes an error code) */
|
||
#define ISR9 9 /* Coprocessor segment overrun */
|
||
#define ISR10 10 /* Bad TSS (pushes an error code) */
|
||
#define ISR11 11 /* Segment not present (pushes an error code) */
|
||
#define ISR12 12 /* Stack fault (pushes an error code) */
|
||
#define ISR13 13 /* General protection fault (pushes an error code) */
|
||
#define ISR14 14 /* Page fault (pushes an error code) */
|
||
#define ISR15 15 /* Unknown interrupt exception */
|
||
#define ISR16 16 /* Coprocessor fault */
|
||
#define ISR17 17 /* Alignment check exception */
|
||
#define ISR18 18 /* Machine check exception */
|
||
#define ISR19 19 /* Reserved */
|
||
#define ISR20 20 /* Reserved */
|
||
#define ISR21 21 /* Reserved */
|
||
#define ISR22 22 /* Reserved */
|
||
#define ISR23 23 /* Reserved */
|
||
#define ISR24 24 /* Reserved */
|
||
#define ISR25 25 /* Reserved */
|
||
#define ISR26 26 /* Reserved */
|
||
#define ISR27 27 /* Reserved */
|
||
#define ISR28 28 /* Reserved */
|
||
#define ISR29 29 /* Reserved */
|
||
#define ISR30 30 /* Reserved */
|
||
#define ISR31 31 /* Reserved */
|
||
|
||
#define IRQ0 32 /* System timer (cannot be changed) */
|
||
#define IRQ1 33 /* Keyboard controller (cannot be changed) */
|
||
#define IRQ2 34 /* Cascaded signals from IRQs 8<>15 */
|
||
#define IRQ3 35 /* Serial port controller for COM2/4 */
|
||
#define IRQ4 36 /* serial port controller for COM1/3 */
|
||
#define IRQ5 37 /* LPT port 2 or sound card */
|
||
#define IRQ6 38 /* Floppy disk controller */
|
||
#define IRQ7 39 /* LPT port 1 or sound card */
|
||
#define IRQ8 40 /* Real time clock (RTC) */
|
||
#define IRQ9 41 /* Open interrupt/available or SCSI host adapter */
|
||
#define IRQ10 42 /* Open interrupt/available or SCSI or NIC */
|
||
#define IRQ11 43 /* Open interrupt/available or SCSI or NIC */
|
||
#define IRQ12 44 /* Mouse on PS/2 connector */
|
||
#define IRQ13 45 /* Math coprocessor */
|
||
#define IRQ14 46 /* Primary ATA channel */
|
||
#define IRQ15 47 /* Secondary ATA channel */
|
||
|
||
#define NR_IRQS 48
|
||
|
||
/* Common register save structure created by up_saveusercontext() and by
|
||
* ISR/IRQ interrupt processing.
|
||
*/
|
||
|
||
#define REG_DS (0) /* Data segment selector */
|
||
#define REG_EDI (1) /* Saved by pusha */
|
||
#define REG_ESI (2) /* " " "" " " */
|
||
#define REG_EBP (3) /* " " "" " " */
|
||
#define REG_ESP (4) /* " " "" " " (NOTE 1)*/
|
||
#define REG_EBX (5) /* " " "" " " */
|
||
#define REG_EDX (6) /* " " "" " " */
|
||
#define REG_ECX (7) /* " " "" " " */
|
||
#define REG_EAX (8) /* " " "" " " */
|
||
#define REG_IRQNO (9) /* Interrupt number (NOTE 2) */
|
||
#define REG_ERRCODE (10) /* Error code (NOTE 2) */
|
||
#define REG_EIP (11) /* Pushed by process on interrupt processing */
|
||
#define REG_CS (12) /* " " "" " " "" " " " " */
|
||
#define REG_EFLAGS (13) /* " " "" " " "" " " " " */
|
||
#define REG_SP (14) /* " " "" " " "" " " " " */
|
||
#define REG_SS (15) /* " " "" " " "" " " " " */
|
||
|
||
/* NOTE 1: Two versions of the ESP are saved: One from the interrupt
|
||
* processing and one from pusha. Only the interrupt ESP (REG_SP) is used.
|
||
* NOTE 2: This is not really state data. Rather, this is just a convenient
|
||
* way to pass parameters from the interrupt handler to C cod.
|
||
*/
|
||
|
||
#define XCPTCONTEXT_REGS (16)
|
||
#define XCPTCONTEXT_SIZE (4 * XCPTCONTEXT_REGS)
|
||
|
||
/* Some special landmarks in the stack frame:
|
||
*
|
||
* TOP_PUSHA - The offset (in 32-bit words) from the beginning of the
|
||
* save area on the stack to the value that should be in REG_ESP.
|
||
* BOTTOM_PUSHA - The offset (in 32-bit words) from the stack position before
|
||
* the interrupt occurred to the value that should be in REG_ESP.
|
||
* save area on the stack to the value that should be in REG_ESP.
|
||
* OFFSET_PRIO - The offset from the value of REG_ESP to the value of the
|
||
* stack pointer before the interrupt occurred (assuming that a priority
|
||
* change occurred.
|
||
* OFFSET_PRIO - The offset from the value of REG_ESP to the value of the
|
||
* stack pointer before the interrupt occurred (assuming that NO priority
|
||
* change occurred.
|
||
*/
|
||
|
||
#define TOP_PUSHA REG_IRQNO
|
||
#define BOTTOM_PRIO (XCPTCONTEXT_REGS-REG_IRQNO)
|
||
#define BOTTOM_NOPRIO (REG_SP-REG_IRQNO)
|
||
|
||
/****************************************************************************
|
||
* Public Types
|
||
****************************************************************************/
|
||
|
||
/* This struct defines the way the registers are stored */
|
||
|
||
#ifndef __ASSEMBLY__
|
||
struct xcptcontext
|
||
{
|
||
/* The following function pointer is non-zero if there are pending signals
|
||
* to be processed.
|
||
*/
|
||
|
||
void *sigdeliver; /* Actual type is sig_deliver_t */
|
||
|
||
/* These are saved copies of instruction pointer and EFLAGS 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_eip;
|
||
uint32_t saved_eflags;
|
||
|
||
/* Register save area */
|
||
|
||
uint32_t regs[XCPTCONTEXT_REGS];
|
||
};
|
||
#endif
|
||
|
||
/****************************************************************************
|
||
* Inline functions
|
||
****************************************************************************/
|
||
|
||
#ifndef __ASSEMBLY__
|
||
|
||
/* Return stack pointer */
|
||
|
||
static inline uint32_t up_getsp(void)
|
||
{
|
||
uint32_t regval;
|
||
|
||
asm volatile(
|
||
"\tmovl %%esp, %0\n"
|
||
: "=rm" (regval)
|
||
:
|
||
: "memory");
|
||
return regval;
|
||
}
|
||
|
||
/* Get segment registers */
|
||
|
||
static inline uint32_t up_getds(void)
|
||
{
|
||
uint32_t regval;
|
||
|
||
asm volatile(
|
||
"\tmov %%ds, %0\n"
|
||
: "=rm" (regval)
|
||
:
|
||
: "memory");
|
||
return regval;
|
||
}
|
||
|
||
static inline uint32_t up_getcs(void)
|
||
{
|
||
uint32_t regval;
|
||
|
||
asm volatile(
|
||
"\tmov %%cs, %0\n"
|
||
: "=rm" (regval)
|
||
:
|
||
: "memory");
|
||
return regval;
|
||
}
|
||
|
||
static inline uint32_t up_getss(void)
|
||
{
|
||
uint32_t regval;
|
||
|
||
asm volatile(
|
||
"\tmov %%ss, %0\n"
|
||
: "=rm" (regval)
|
||
:
|
||
: "memory");
|
||
return regval;
|
||
}
|
||
|
||
/* Name: up_irq_save, up_irq_restore, and friends.
|
||
*
|
||
* NOTE: This function should never be called from application code and,
|
||
* as a general rule unless you really know what you are doing, this
|
||
* function should not be called directly from operation system code either:
|
||
* Typically, the wrapper functions, enter_critical_section() and
|
||
* leave_critical section(), are probably what you really want.
|
||
*/
|
||
|
||
/* Get the current FLAGS register contents */
|
||
|
||
static inline irqstate_t irqflags()
|
||
{
|
||
irqstate_t flags;
|
||
|
||
asm volatile(
|
||
"\tpushf\n"
|
||
"\tpop %0\n"
|
||
: "=rm" (flags)
|
||
:
|
||
: "memory");
|
||
return flags;
|
||
}
|
||
|
||
/* Get a sample of the FLAGS register, determine if interrupts are disabled.
|
||
* If the X86_FLAGS_IF is cleared by cli, then interrupts are disabled. If
|
||
* if the X86_FLAGS_IF is set by sti, then interrupts are enable.
|
||
*/
|
||
|
||
static inline bool up_irq_disabled(irqstate_t flags)
|
||
{
|
||
return ((flags & X86_FLAGS_IF) == 0);
|
||
}
|
||
|
||
static inline bool up_irq_enabled(irqstate_t flags)
|
||
{
|
||
return ((flags & X86_FLAGS_IF) != 0);
|
||
}
|
||
|
||
/* Disable interrupts unconditionally */
|
||
|
||
static inline void up_irq_disable(void)
|
||
{
|
||
asm volatile("cli": : :"memory");
|
||
}
|
||
|
||
/* Enable interrupts unconditionally */
|
||
|
||
static inline void up_irq_enable(void)
|
||
{
|
||
asm volatile("sti": : :"memory");
|
||
}
|
||
|
||
/* Disable interrupts, but return previous interrupt state */
|
||
|
||
static inline irqstate_t up_irq_save(void)
|
||
{
|
||
irqstate_t flags = irqflags();
|
||
up_irq_disable();
|
||
return flags;
|
||
}
|
||
|
||
/* Conditionally disable interrupts */
|
||
|
||
static inline void up_irq_restore(irqstate_t flags)
|
||
{
|
||
if (up_irq_enabled(flags))
|
||
{
|
||
up_irq_enable();
|
||
}
|
||
}
|
||
|
||
static inline void system_call3(unsigned int nbr, uintptr_t parm1,
|
||
uintptr_t parm2, uintptr_t parm3)
|
||
{
|
||
/* To be provided */
|
||
}
|
||
|
||
/****************************************************************************
|
||
* Public Data
|
||
****************************************************************************/
|
||
|
||
/****************************************************************************
|
||
* Public Function Prototypes
|
||
****************************************************************************/
|
||
|
||
#ifdef __cplusplus
|
||
#define EXTERN extern "C"
|
||
extern "C"
|
||
{
|
||
#else
|
||
#define EXTERN extern
|
||
#endif
|
||
|
||
#undef EXTERN
|
||
#ifdef __cplusplus
|
||
}
|
||
#endif
|
||
|
||
#endif /* __ASSEMBLY__ */
|
||
#endif /* __ARCH_X86_INCLUDE_I486_IRQ_H */
|