2020-09-30 16:53:55 +02:00

354 lines
7.2 KiB
C

/*++
Copyright (c) 1991 Microsoft Corporation
Module Name:
vdmuserrefs.c
Abstract:
This module contains routines that probe and fetch from the
instruction stream to make the vdm bop support solid.
Author:
Neill Clift (NeillC) 27-Jun-2001
Revision History:
--*/
#include "vdmp.h"
#pragma alloc_text (PAGE, VdmFetchBop1)
#pragma alloc_text (PAGE, VdmFetchBop4)
#pragma alloc_text (PAGE, VdmDispatchOpcodeV86_try)
#pragma alloc_text (PAGE, VdmTibPass1)
#pragma alloc_text (PAGE, VdmDispatchBop)
#pragma alloc_text (PAGE, VdmFetchULONG)
VOID
NTFastDOSIO (
PKTRAP_FRAME TrapFrame,
ULONG IoType
);
ULONG
VdmFetchBop4 (
IN PVOID Pc
)
/*++
Routine Description:
This routine reads up to 4 bytes of bop instruction data
Arguments:
Pc - Program counter fetched from the faulting instruction's trap frame.
Return Value:
ULONG - Up to 4 bytes of instruction stream. Unfetchable bytes are zeroed.
--*/
{
ULONG Value;
ULONG i;
BOOLEAN DidProbe;
DidProbe = FALSE;
try {
ProbeForReadSmallStructure (Pc, sizeof (UCHAR), sizeof (UCHAR));
DidProbe = TRUE;
return *(PULONG)Pc;
} except (EXCEPTION_EXECUTE_HANDLER) {
if (DidProbe == FALSE) {
return 0;
}
}
Value = 0;
try {
for (i = 0; i < sizeof (ULONG); i++) {
Value += (((PUCHAR)Pc)[i])<<(i*8);
}
} except (EXCEPTION_EXECUTE_HANDLER) {
}
return Value;
}
ULONG
VdmFetchULONG (
IN PVOID Pc
)
/*++
Routine Description:
This routine reads 4 bytes from the user address space
Arguments:
Pc - Program counter fetched from the faulting instruction's trap frame.
Return Value:
ULONG - 4 bytes of user mode data
--*/
{
try {
ProbeForReadSmallStructure (Pc, sizeof (ULONG), sizeof (UCHAR));
return *(PULONG)Pc;
} except (EXCEPTION_EXECUTE_HANDLER) {
return 0;
}
}
ULONG
VdmFetchBop1 (
IN PVOID Pc
)
/*++
Routine Description:
This routine reads a single byte of bop instruction data.
Arguments:
Pc - Program counter fetched from the faulting instruction's trap frame
Return Value:
ULONG - 1 byte of instruction stream or zero if unreadable
--*/
{
try {
ProbeForReadSmallStructure (Pc, sizeof (UCHAR), sizeof (UCHAR));
return *(PUCHAR)Pc;
} except (EXCEPTION_EXECUTE_HANDLER) {
return 0;
}
}
ULONG
VdmDispatchOpcodeV86_try (
IN PKTRAP_FRAME TrapFrame
)
/*++
Routine Description:
This routine is just a shell around trap.asm code to handle faulting
instruction stream references. This routines is called at APC_LEVEL
to prevent NtSetContextThread from changing the EIP after its been probed
earlier.
Arguments:
Pc - Program counter fetched from the faulting instructions trap frame
Return Value:
ULONG - 1 byte of instruction stream or zero if unreadable
--*/
{
try {
return Ki386DispatchOpcodeV86 (TrapFrame);
} except (EXCEPTION_EXECUTE_HANDLER) {
return 0;
}
}
ULONG
VdmDispatchOpcode_try (
IN PKTRAP_FRAME TrapFrame
)
/*++
Routine Description:
This routine is just a shell around trap.asm code to handle faulting
instruction stream references. This routines is called at APC_LEVEL
to prevent NtSetContextThread from changing the EIP after its been probed
earlier.
Arguments:
Pc - Program counter fetched from the faulting instructions trap frame
Return Value:
ULONG - 1 byte of instruction stream or zero if unreadable
--*/
{
try {
return Ki386DispatchOpcode (TrapFrame);
} except (EXCEPTION_EXECUTE_HANDLER) {
return 0;
}
}
PVOID
VdmTibPass1 (
IN ULONG Cs,
IN ULONG Eip,
IN ULONG Ebx
)
{
PVDM_TIB VdmTib;
//
// Copy the specified registers to the VDM Tib communication area,
// using proper probing and exception handling.
//
try {
VdmTib = NtCurrentTeb()->Vdm;
ProbeForWrite (VdmTib, sizeof(VDM_TIB), sizeof(UCHAR));
VdmTib->VdmContext.Ebx = Ebx;
VdmTib->VdmContext.Eip = Eip;
VdmTib->VdmContext.SegCs = Cs;
} except (EXCEPTION_EXECUTE_HANDLER) {
return NULL;
}
return VdmTib;
}
#define BOP_INSTRUCTION 0xC4C4
#define SVC_DEMFASTREAD 0x42
#define SVC_DEMFASTWRITE 0x43
#define DOS_BOP 0x50
extern ULONG VdmBopCount;
LOGICAL
VdmDispatchBop (
IN PKTRAP_FRAME TrapFrame
)
/*++
Routine Description:
This routine attempts to decode and execute the user instruction. If
this cannot be done, FALSE is returned and the ntvdm monitor must handle
it.
Arguments:
TrapFrame - Supplies a pointer to register trapframe.
Return Value:
TRUE if the opcode was handled here.
FALSE if not (ie: the caller must reflect this instruction to ntvdm
to handle on behalf of the 16-bit app).
Environment:
Kernel mode, APC_LEVEL.
--*/
{
LOGICAL RetVal;
BOOLEAN GotSelector;
PVDM_TIB VdmTib;
ULONG SegCs;
ULONG LinearEIP;
ULONG Flags;
ULONG Base;
ULONG Limit;
ULONG IoType = 0;
LOGICAL DoFastIo;
if (TrapFrame->EFlags & EFLAGS_V86_MASK) {
SegCs = TrapFrame->SegCs & 0xFFFF;
LinearEIP = (SegCs << 4) + (TrapFrame->Eip & 0xffff);
}
else {
GotSelector = Ki386GetSelectorParameters ((USHORT) TrapFrame->SegCs,
&Flags,
&Base,
&Limit);
if (GotSelector == FALSE) {
return TRUE;
}
LinearEIP = Base + TrapFrame->Eip;
}
DoFastIo = FALSE;
RetVal = TRUE;
try {
ProbeForReadSmallStructure (LinearEIP, sizeof (UCHAR), sizeof (UCHAR));
if (*(PUSHORT)LinearEIP != BOP_INSTRUCTION) {
RetVal = FALSE;
leave;
}
//
// Check the BOP number.
//
if (*(PUCHAR)(LinearEIP + 2) == DOS_BOP) {
if ((*(PUCHAR)(LinearEIP + 3) == SVC_DEMFASTREAD) ||
(*(PUCHAR)(LinearEIP + 3) == SVC_DEMFASTWRITE)) {
//
// Take the fast I/O path.
//
IoType = (ULONG)(*(PUCHAR)(LinearEIP + 3));
DoFastIo = TRUE;
leave;
}
}
VdmBopCount += 1;
VdmTib = NtCurrentTeb()->Vdm;
ProbeForWrite (VdmTib, sizeof(VDM_TIB), sizeof(UCHAR));
VdmTib->EventInfo.Event = VdmBop;
VdmTib->EventInfo.BopNumber = *(PUCHAR)(LinearEIP + 2);
VdmTib->EventInfo.InstructionSize = 3;
VdmEndExecution (TrapFrame, VdmTib);
} except (EXCEPTION_EXECUTE_HANDLER) {
RetVal = FALSE;
NOTHING; // Fall through
}
if (DoFastIo) {
NTFastDOSIO (TrapFrame, IoType);
}
return RetVal;
}