2020-09-30 17:12:29 +02:00

472 lines
7.8 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1994 Microsoft Corporation
Module Name:
ctrlops.c
Abstract:
This module implements the code to emulate call, retunr, and various
control operations.
Author:
David N. Cutler (davec) 10-Nov-1994
Environment:
Kernel mode only.
Revision History:
Scott Geranen 5-Mar-1996 Added hooks for System BIOS emulation
--*/
#include "nthal.h"
#include "emulate.h"
#include "sysbios.h"
VOID
XmCallOp (
PRXM_CONTEXT P
)
/*++
Routine Description:
This function emulates a call opcode.
Arguments:
P - Supplies a pointer to an emulator context structure.
Return Value:
None.
--*/
{
ULONG Target;
ULONG Source;
//
// Save the target address, push the current segment, if required, and
// push the current IP, set the destination segment, if required, and
// set the new IP.
//
Target = P->DstValue.Long;
if (P->OpsizePrefixActive != FALSE) {
P->DataType = LONG_DATA;
} else {
P->DataType = WORD_DATA;
}
if ((P->CurrentOpcode == 0x9a) || (P->FunctionIndex != X86_CALL_OP)) {
XmPushStack(P, P->SegmentRegister[CS]);
XmPushStack(P, P->Eip);
P->SegmentRegister[CS] = P->DstSegment;
} else {
XmPushStack(P, P->Eip);
}
P->Eip = Target;
XmTraceJumps(P);
return;
}
VOID
XmEnterOp (
PRXM_CONTEXT P
)
/*++
Routine Description:
This function emulates an enter opcode.
Arguments:
P - Supplies a pointer to an emulator context structure.
Return Value:
None.
--*/
{
ULONG Allocate;
ULONG Frame;
ULONG Number;
//
// set the number of bytes to allocate on the stack and the number
// of nesting levels.
//
Allocate = P->SrcValue.Long;
Number = P->DstValue.Long;
//
// Set the data type and save the frame pointer on the stack.
//
if (P->OpsizePrefixActive != FALSE) {
P->DataType = LONG_DATA;
XmPushStack(P, P->Gpr[EBP].Exx);
Frame = P->Gpr[ESP].Exx;
} else {
P->DataType = WORD_DATA;
XmPushStack(P, P->Gpr[BP].Xx);
Frame = P->Gpr[SP].Xx;
}
//
// Save the current stack pointer and push parameters on the stack.
//
if (Number != 0) {
//
// If the level number is not one, then raise an exception.
//
// N.B. Level numbers greater than one are not supported.
//
if (Number != 1) {
longjmp(&P->JumpBuffer[0], XM_ILLEGAL_LEVEL_NUMBER);
}
XmPushStack(P, Frame);
}
//
// Allocate local storage on stack.
//
if (P->OpsizePrefixActive != FALSE) {
P->Gpr[EBP].Exx = Frame;
P->Gpr[ESP].Exx = P->Gpr[ESP].Exx - Allocate;
} else {
P->Gpr[BP].Xx = (USHORT)Frame;
P->Gpr[SP].Xx = (USHORT)(P->Gpr[SP].Xx - Allocate);
}
return;
}
VOID
XmHltOp (
PRXM_CONTEXT P
)
/*++
Routine Description:
This function emulates a hlt opcode.
Arguments:
P - Supplies a pointer to an emulator context structure.
Return Value:
None.
--*/
{
//
// Halt instructions are not supported by the emulator.
//
longjmp(&P->JumpBuffer[0], XM_HALT_INSTRUCTION);
return;
}
VOID
XmIntOp (
PRXM_CONTEXT P
)
/*++
Routine Description:
This function emulates an int opcode.
Arguments:
P - Supplies a pointer to an emulator context structure.
Return Value:
None.
--*/
{
ULONG Number;
PULONG Vector;
//
// If the int instruction is an int 3, then set the interrupt vector
// to 3. Otherwise, if the int instruction is an into, then set the
// vector to 4 if OF is set. use the source interrupt vector.
//
if (P->OpsizePrefixActive != FALSE) {
P->DataType = LONG_DATA;
} else {
P->DataType = WORD_DATA;
}
if (P->CurrentOpcode == 0xcc) {
Number = 3;
} else if (P->CurrentOpcode == 0xce) {
if (P->Eflags.OF == 0) {
return;
}
Number = 4;
} else {
Number = P->SrcValue.Byte;
}
//
// If the vector number is 0x42, then nop the interrupt. This is the
// standard EGA video driver entry point in a PC's motherboard BIOS
// for which there is no code.
//
#if !defined(_PURE_EMULATION_)
if (Number == 0x42) {
return;
}
#endif
//
// Try to emulate a system BIOS call first.
//
if (!HalpEmulateSystemBios(P, Number)) {
//
// Either it isn't a BIOS call or it isn't supported.
// Emulate the x86 stream instead. Note that this
// doesn't support chained handlers.
//
// Push the current flags, code segment, and EIP on the stack.
//
XmPushStack(P, P->AllFlags);
XmPushStack(P, P->SegmentRegister[CS]);
XmPushStack(P, P->Eip);
//
// Set the new coded segment and IP from the specified interrupt
// vector.
//
Vector = (PULONG)(P->TranslateAddress)(0, 0);
P->SegmentRegister[CS] = (USHORT)(Vector[Number] >> 16);
P->Eip = (USHORT)(Vector[Number] & 0xffff);
XmTraceJumps(P);
}
return;
}
VOID
XmIretOp (
PRXM_CONTEXT P
)
/*++
Routine Description:
This function emulates an iret opcode.
Arguments:
P - Supplies a pointer to an emulator context structure.
Return Value:
None.
--*/
{
//
// Set the data type and restore the return address, code segment,
// and flags.
//
if (P->OpsizePrefixActive != FALSE) {
P->DataType = LONG_DATA;
} else {
P->DataType = WORD_DATA;
}
P->Eip = XmPopStack(P);
P->SegmentRegister[CS] = (USHORT)XmPopStack(P);
P->AllFlags = XmPopStack(P);
XmTraceJumps(P);
//
// Check for emulator exit conditions.
//
if ((P->Eip == 0xffff) && (P->SegmentRegister[CS] == 0xffff)) {
longjmp(&P->JumpBuffer[0], XM_SUCCESS);
}
return;
}
VOID
XmLeaveOp (
PRXM_CONTEXT P
)
/*++
Routine Description:
This function emulates a leave opcode.
Arguments:
P - Supplies a pointer to an emulator context structure.
Return Value:
None.
--*/
{
//
// Set the data type, restore the stack pointer, and restore the frame
// pointer.
//
if (P->OpsizePrefixActive != FALSE) {
P->DataType = LONG_DATA;
P->Gpr[ESP].Exx = P->Gpr[EBP].Exx;
P->Gpr[EBP].Exx = XmPopStack(P);
} else {
P->DataType = WORD_DATA;
P->Gpr[SP].Xx = P->Gpr[BP].Xx;
P->Gpr[BP].Xx = (USHORT)XmPopStack(P);
}
return;
}
VOID
XmRetOp (
PRXM_CONTEXT P
)
/*++
Routine Description:
This function emulates a ret opcode.
Arguments:
P - Supplies a pointer to an emulator context structure.
Return Value:
None.
--*/
{
ULONG Adjust;
//
// Compute the number of bytes that are to be removed from the stack
// after having removed the return address and optionally the new CS
// segment value.
//
if ((P->CurrentOpcode & 0x1) == 0) {
Adjust = XmGetWordImmediate(P);
} else {
Adjust = 0;
}
//
// Remove the return address from the stack and set the new IP.
//
if (P->OpsizePrefixActive != FALSE) {
P->DataType = LONG_DATA;
} else {
P->DataType = WORD_DATA;
}
P->Eip = XmPopStack(P);
//
// If the current opcode is a far return, then remove the new CS segment
// value from the stack.
//
if ((P->CurrentOpcode & 0x8) != 0) {
P->SegmentRegister[CS] = (USHORT)XmPopStack(P);
}
//
// Remove the specified number of bytes from the stack.
//
P->Gpr[ESP].Exx += Adjust;
XmTraceJumps(P);
//
// Check for emulator exit conditions.
//
if ((P->Eip == 0xffff) && (P->SegmentRegister[CS] == 0xffff)) {
longjmp(&P->JumpBuffer[0], XM_SUCCESS);
}
return;
}