2008-02-13 05:16:14 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS Kernel
|
2008-06-29 02:58:05 +00:00
|
|
|
* LICENSE: BSD - See COPYING.ARM in the top level directory
|
2008-02-13 05:16:14 +00:00
|
|
|
* FILE: ntoskrnl/ke/arm/exp.c
|
|
|
|
* PURPOSE: Implements exception helper routines for ARM machines
|
|
|
|
* PROGRAMMERS: ReactOS Portable Systems Group
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES *******************************************************************/
|
|
|
|
|
|
|
|
#include <ntoskrnl.h>
|
2008-07-19 06:13:10 +00:00
|
|
|
#include <internal/arm/ksarm.h>
|
2008-02-13 05:16:14 +00:00
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
/* FUNCTIONS ******************************************************************/
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KeContextToTrapFrame(IN PCONTEXT Context,
|
|
|
|
IN OUT PKEXCEPTION_FRAME ExceptionFrame,
|
|
|
|
IN OUT PKTRAP_FRAME TrapFrame,
|
|
|
|
IN ULONG ContextFlags,
|
|
|
|
IN KPROCESSOR_MODE PreviousMode)
|
|
|
|
{
|
2008-07-14 00:04:53 +00:00
|
|
|
KIRQL OldIrql;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Do this at APC_LEVEL
|
|
|
|
//
|
|
|
|
OldIrql = KeGetCurrentIrql();
|
|
|
|
if (OldIrql < APC_LEVEL) KeRaiseIrql(APC_LEVEL, &OldIrql);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Start with the Control flags
|
|
|
|
//
|
|
|
|
if ((Context->ContextFlags & CONTEXT_CONTROL) == CONTEXT_CONTROL)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// So this basically means all the special stuff
|
|
|
|
//
|
|
|
|
if (PreviousMode == UserMode)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// ARM has register banks
|
|
|
|
//
|
|
|
|
TrapFrame->UserSp = Context->Sp;
|
|
|
|
TrapFrame->UserLr = Context->Lr;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// ARM has register banks
|
|
|
|
//
|
|
|
|
TrapFrame->SvcSp = Context->Sp;
|
|
|
|
TrapFrame->SvcLr = Context->Lr;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// The rest is already in the right mode
|
|
|
|
//
|
|
|
|
TrapFrame->Pc = Context->Pc;
|
|
|
|
TrapFrame->Spsr = Context->Psr;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Now do the integers
|
|
|
|
//
|
|
|
|
if ((Context->ContextFlags & CONTEXT_INTEGER) == CONTEXT_INTEGER)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// Basically everything else but FPU
|
|
|
|
//
|
|
|
|
TrapFrame->R0 = Context->R0;
|
|
|
|
TrapFrame->R1 = Context->R1;
|
|
|
|
TrapFrame->R2 = Context->R2;
|
|
|
|
TrapFrame->R3 = Context->R3;
|
|
|
|
TrapFrame->R4 = Context->R4;
|
|
|
|
TrapFrame->R5 = Context->R5;
|
|
|
|
TrapFrame->R6 = Context->R6;
|
|
|
|
TrapFrame->R7 = Context->R7;
|
|
|
|
TrapFrame->R8 = Context->R8;
|
|
|
|
TrapFrame->R0 = Context->R9;
|
|
|
|
TrapFrame->R10 = Context->R10;
|
|
|
|
TrapFrame->R11 = Context->R11;
|
|
|
|
TrapFrame->R12 = Context->R12;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Restore IRQL
|
|
|
|
//
|
|
|
|
if (OldIrql < APC_LEVEL) KeLowerIrql(OldIrql);
|
2008-02-13 05:16:14 +00:00
|
|
|
}
|
2008-07-13 23:46:50 +00:00
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KeTrapFrameToContext(IN PKTRAP_FRAME TrapFrame,
|
|
|
|
IN PKEXCEPTION_FRAME ExceptionFrame,
|
|
|
|
IN OUT PCONTEXT Context)
|
|
|
|
{
|
2008-07-13 23:58:47 +00:00
|
|
|
KIRQL OldIrql;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Do this at APC_LEVEL
|
|
|
|
//
|
|
|
|
OldIrql = KeGetCurrentIrql();
|
|
|
|
if (OldIrql < APC_LEVEL) KeRaiseIrql(APC_LEVEL, &OldIrql);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Start with the Control flags
|
|
|
|
//
|
|
|
|
if ((Context->ContextFlags & CONTEXT_CONTROL) == CONTEXT_CONTROL)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// So this basically means all the special stuff
|
|
|
|
//
|
2008-07-14 00:04:53 +00:00
|
|
|
if (KiGetPreviousMode(TrapFrame) == UserMode)
|
2008-07-13 23:58:47 +00:00
|
|
|
{
|
|
|
|
//
|
|
|
|
// ARM has register banks
|
|
|
|
//
|
|
|
|
Context->Sp = TrapFrame->UserSp;
|
|
|
|
Context->Lr = TrapFrame->UserLr;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// ARM has register banks
|
|
|
|
//
|
|
|
|
Context->Sp = TrapFrame->SvcSp;
|
|
|
|
Context->Lr = TrapFrame->SvcLr;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// The rest is already in the right mode
|
|
|
|
//
|
|
|
|
Context->Pc = TrapFrame->Pc;
|
|
|
|
Context->Psr = TrapFrame->Spsr;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Now do the integers
|
|
|
|
//
|
|
|
|
if ((Context->ContextFlags & CONTEXT_INTEGER) == CONTEXT_INTEGER)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// Basically everything else but FPU
|
|
|
|
//
|
|
|
|
Context->R0 = TrapFrame->R0;
|
|
|
|
Context->R1 = TrapFrame->R1;
|
|
|
|
Context->R2 = TrapFrame->R2;
|
|
|
|
Context->R3 = TrapFrame->R3;
|
|
|
|
Context->R4 = TrapFrame->R4;
|
|
|
|
Context->R5 = TrapFrame->R5;
|
|
|
|
Context->R6 = TrapFrame->R6;
|
|
|
|
Context->R7 = TrapFrame->R7;
|
|
|
|
Context->R8 = TrapFrame->R8;
|
|
|
|
Context->R0 = TrapFrame->R9;
|
|
|
|
Context->R10 = TrapFrame->R10;
|
|
|
|
Context->R11 = TrapFrame->R11;
|
|
|
|
Context->R12 = TrapFrame->R12;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Restore IRQL
|
|
|
|
//
|
|
|
|
if (OldIrql < APC_LEVEL) KeLowerIrql(OldIrql);
|
2008-07-13 23:46:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KiDispatchException(IN PEXCEPTION_RECORD ExceptionRecord,
|
|
|
|
IN PKEXCEPTION_FRAME ExceptionFrame,
|
|
|
|
IN PKTRAP_FRAME TrapFrame,
|
|
|
|
IN KPROCESSOR_MODE PreviousMode,
|
|
|
|
IN BOOLEAN FirstChance)
|
|
|
|
{
|
|
|
|
CONTEXT Context;
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* Increase number of Exception Dispatches */
|
2008-07-13 23:46:50 +00:00
|
|
|
KeGetCurrentPrcb()->KeExceptionDispatchCount++;
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* Set the context flags */
|
2008-07-13 23:46:50 +00:00
|
|
|
Context.ContextFlags = CONTEXT_FULL;
|
|
|
|
|
2010-02-09 02:00:52 +00:00
|
|
|
/* Check if User Mode or if the kernel debugger is enabled */
|
|
|
|
if ((PreviousMode == UserMode) || (KeGetPcr()->KdVersionBlock))
|
|
|
|
{
|
|
|
|
/* FIXME-V6: VFP Support */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get a Context */
|
2008-07-13 23:46:50 +00:00
|
|
|
KeTrapFrameToContext(TrapFrame, ExceptionFrame, &Context);
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* Look at our exception code */
|
2008-07-13 23:46:50 +00:00
|
|
|
switch (ExceptionRecord->ExceptionCode)
|
|
|
|
{
|
2010-02-09 02:00:52 +00:00
|
|
|
/* Breakpoint */
|
2008-07-13 23:46:50 +00:00
|
|
|
case STATUS_BREAKPOINT:
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* Decrement PC by four */
|
2008-07-14 00:24:33 +00:00
|
|
|
Context.Pc -= sizeof(ULONG);
|
2008-07-13 23:46:50 +00:00
|
|
|
break;
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* Internal exception */
|
2008-07-13 23:46:50 +00:00
|
|
|
case KI_EXCEPTION_ACCESS_VIOLATION:
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* Set correct code */
|
2008-07-13 23:46:50 +00:00
|
|
|
ExceptionRecord->ExceptionCode = STATUS_ACCESS_VIOLATION;
|
2010-02-09 02:00:52 +00:00
|
|
|
if (PreviousMode == UserMode)
|
|
|
|
{
|
|
|
|
/* FIXME: Handle no execute */
|
|
|
|
}
|
2008-07-13 23:46:50 +00:00
|
|
|
break;
|
|
|
|
}
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* Handle kernel-mode first, it's simpler */
|
2008-07-13 23:46:50 +00:00
|
|
|
if (PreviousMode == KernelMode)
|
|
|
|
{
|
2010-02-09 02:00:52 +00:00
|
|
|
/* Check if this is a first-chance exception */
|
2008-07-13 23:46:50 +00:00
|
|
|
if (FirstChance == TRUE)
|
|
|
|
{
|
2010-02-09 02:00:52 +00:00
|
|
|
/* Break into the debugger for the first time */
|
2008-07-13 23:46:50 +00:00
|
|
|
if (KiDebugRoutine(TrapFrame,
|
|
|
|
ExceptionFrame,
|
|
|
|
ExceptionRecord,
|
|
|
|
&Context,
|
|
|
|
PreviousMode,
|
|
|
|
FALSE))
|
|
|
|
{
|
2010-02-09 02:00:52 +00:00
|
|
|
/* Exception was handled */
|
2008-07-13 23:46:50 +00:00
|
|
|
goto Handled;
|
|
|
|
}
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* If the Debugger couldn't handle it, dispatch the exception */
|
2008-07-13 23:46:50 +00:00
|
|
|
if (RtlDispatchException(ExceptionRecord, &Context)) goto Handled;
|
|
|
|
}
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* This is a second-chance exception, only for the debugger */
|
2008-07-13 23:46:50 +00:00
|
|
|
if (KiDebugRoutine(TrapFrame,
|
|
|
|
ExceptionFrame,
|
|
|
|
ExceptionRecord,
|
|
|
|
&Context,
|
|
|
|
PreviousMode,
|
|
|
|
TRUE))
|
|
|
|
{
|
2010-02-09 02:00:52 +00:00
|
|
|
/* Exception was handled */
|
2008-07-13 23:46:50 +00:00
|
|
|
goto Handled;
|
|
|
|
}
|
2010-02-09 02:00:52 +00:00
|
|
|
|
|
|
|
/* Third strike; you're out */
|
2008-07-13 23:46:50 +00:00
|
|
|
KeBugCheckEx(KMODE_EXCEPTION_NOT_HANDLED,
|
|
|
|
ExceptionRecord->ExceptionCode,
|
|
|
|
(ULONG_PTR)ExceptionRecord->ExceptionAddress,
|
|
|
|
(ULONG_PTR)TrapFrame,
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-02-09 02:00:52 +00:00
|
|
|
/* FIXME: TODO */
|
|
|
|
/* 3rd strike, kill the process */
|
|
|
|
DPRINT1("Kill %.16s, ExceptionCode: %lx, ExceptionAddress: %lx\n",
|
|
|
|
PsGetCurrentProcess()->ImageFileName,
|
|
|
|
ExceptionRecord->ExceptionCode,
|
|
|
|
ExceptionRecord->ExceptionAddress);
|
|
|
|
|
|
|
|
ZwTerminateProcess(NtCurrentProcess(), ExceptionRecord->ExceptionCode);
|
|
|
|
KeBugCheckEx(KMODE_EXCEPTION_NOT_HANDLED,
|
|
|
|
ExceptionRecord->ExceptionCode,
|
|
|
|
(ULONG_PTR)ExceptionRecord->ExceptionAddress,
|
|
|
|
(ULONG_PTR)TrapFrame,
|
|
|
|
0);
|
2008-07-13 23:46:50 +00:00
|
|
|
}
|
2010-02-09 02:00:52 +00:00
|
|
|
|
2008-07-13 23:46:50 +00:00
|
|
|
Handled:
|
2010-02-09 02:00:52 +00:00
|
|
|
/* Convert the context back into Trap/Exception Frames */
|
2008-07-13 23:46:50 +00:00
|
|
|
KeContextToTrapFrame(&Context,
|
|
|
|
ExceptionFrame,
|
|
|
|
TrapFrame,
|
|
|
|
Context.ContextFlags,
|
|
|
|
PreviousMode);
|
2010-02-09 02:00:52 +00:00
|
|
|
return;
|
2008-07-13 23:46:50 +00:00
|
|
|
}
|