50,6 → 50,7 |
#include <arch/ddi/ddi.h> |
#include <ipc/sysipc.h> |
#include <interrupt.h> |
#include <ddi/irq.h> |
|
/* |
* Interrupt and exception dispatching. |
59,7 → 60,7 |
void (* enable_irqs_function)(uint16_t irqmask) = NULL; |
void (* eoi_function)(void) = NULL; |
|
void PRINT_INFO_ERRCODE(istate_t *istate) |
void decode_istate(istate_t *istate) |
{ |
char *symbol = get_symtab_entry(istate->eip); |
|
82,16 → 83,25 |
printf(" %#x, %#x, %#x, %#x\n", istate->stack[4], istate->stack[5], istate->stack[6], istate->stack[7]); |
} |
|
void null_interrupt(int n, istate_t *istate) |
static void trap_virtual_eoi(void) |
{ |
if (eoi_function) |
eoi_function(); |
else |
panic("no eoi_function\n"); |
|
} |
|
static void null_interrupt(int n, istate_t *istate) |
{ |
fault_if_from_uspace(istate, "unserviced interrupt: %d", n); |
|
PRINT_INFO_ERRCODE(istate); |
decode_istate(istate); |
panic("unserviced interrupt: %d\n", n); |
} |
|
/** General Protection Fault. */ |
void gp_fault(int n, istate_t *istate) |
static void gp_fault(int n, istate_t *istate) |
{ |
if (TASK) { |
count_t ver; |
114,19 → 124,19 |
fault_if_from_uspace(istate, "general protection fault"); |
} |
|
PRINT_INFO_ERRCODE(istate); |
decode_istate(istate); |
panic("general protection fault\n"); |
} |
|
void ss_fault(int n, istate_t *istate) |
static void ss_fault(int n, istate_t *istate) |
{ |
fault_if_from_uspace(istate, "stack fault"); |
|
PRINT_INFO_ERRCODE(istate); |
decode_istate(istate); |
panic("stack fault\n"); |
} |
|
void simd_fp_exception(int n, istate_t *istate) |
static void simd_fp_exception(int n, istate_t *istate) |
{ |
uint32_t mxcsr; |
asm |
137,12 → 147,12 |
fault_if_from_uspace(istate, "SIMD FP exception(19), MXCSR: %#zx", |
(unative_t)mxcsr); |
|
PRINT_INFO_ERRCODE(istate); |
decode_istate(istate); |
printf("MXCSR: %#zx\n",(unative_t)(mxcsr)); |
panic("SIMD FP exception(19)\n"); |
} |
|
void nm_fault(int n, istate_t *istate) |
static void nm_fault(int n, istate_t *istate) |
{ |
#ifdef CONFIG_FPU_LAZY |
scheduler_fpu_lazy_request(); |
152,17 → 162,63 |
#endif |
} |
|
void syscall(int n, istate_t *istate) |
#ifdef CONFIG_SMP |
static void tlb_shootdown_ipi(int n, istate_t *istate) |
{ |
panic("Obsolete syscall handler."); |
trap_virtual_eoi(); |
tlb_shootdown_ipi_recv(); |
} |
#endif |
|
void tlb_shootdown_ipi(int n, istate_t *istate) |
/** Handler of IRQ exceptions */ |
static void irq_interrupt(int n, istate_t *istate) |
{ |
ASSERT(n >= IVT_IRQBASE); |
|
int inum = n - IVT_IRQBASE; |
ASSERT(inum < IRQ_COUNT); |
ASSERT((inum != IRQ_PIC_SPUR) && (inum != IRQ_PIC1)); |
|
irq_t *irq = irq_dispatch_and_lock(inum); |
if (irq) { |
/* |
* The IRQ handler was found. |
*/ |
irq->handler(irq, irq->arg); |
spinlock_unlock(&irq->lock); |
} else { |
/* |
* Spurious interrupt. |
*/ |
#ifdef CONFIG_DEBUG |
printf("cpu%d: spurious interrupt (inum=%d)\n", CPU->id, inum); |
#endif |
} |
trap_virtual_eoi(); |
tlb_shootdown_ipi_recv(); |
} |
|
void interrupt_init(void) |
{ |
int i; |
|
for (i = 0; i < IVT_ITEMS; i++) |
exc_register(i, "null", (iroutine) null_interrupt); |
|
for (i = 0; i < IRQ_COUNT; i++) { |
if ((i != IRQ_PIC_SPUR) && (i != IRQ_PIC1)) |
exc_register(IVT_IRQBASE + i, "irq", (iroutine) irq_interrupt); |
} |
|
exc_register(7, "nm_fault", (iroutine) nm_fault); |
exc_register(12, "ss_fault", (iroutine) ss_fault); |
exc_register(13, "gp_fault", (iroutine) gp_fault); |
exc_register(19, "simd_fp", (iroutine) simd_fp_exception); |
|
#ifdef CONFIG_SMP |
exc_register(VECTOR_TLB_SHOOTDOWN_IPI, "tlb_shootdown", (iroutine) tlb_shootdown_ipi); |
#endif |
} |
|
void trap_virtual_enable_irqs(uint16_t irqmask) |
{ |
if (enable_irqs_function) |
179,14 → 235,5 |
panic("no disable_irqs_function\n"); |
} |
|
void trap_virtual_eoi(void) |
{ |
if (eoi_function) |
eoi_function(); |
else |
panic("no eoi_function\n"); |
|
} |
|
/** @} |
*/ |