2026 lines
50 KiB
C
2026 lines
50 KiB
C
/*++
|
|
|
|
Copyright (c) 1991 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
vdmints.c
|
|
|
|
Abstract:
|
|
|
|
Vdm kernel Virtual interrupt support
|
|
|
|
Author:
|
|
|
|
13-Oct-1993 Jonathan Lew (Jonle)
|
|
|
|
Notes:
|
|
|
|
|
|
Revision History:
|
|
|
|
|
|
--*/
|
|
|
|
|
|
#include <ntos.h>
|
|
#include <zwapi.h>
|
|
#include "vdmp.h"
|
|
|
|
// thread priority boost for vdm hardware interrupt.
|
|
#define VDM_HWINT_INCREMENT EVENT_INCREMENT
|
|
|
|
|
|
// internal function prototypes
|
|
|
|
|
|
|
|
VOID
|
|
VdmpQueueIntApcRoutine (
|
|
IN PKAPC Apc,
|
|
IN PKNORMAL_ROUTINE *NormalRoutine,
|
|
IN PVOID *NormalContext,
|
|
IN PVOID *SystemArgument1,
|
|
IN PVOID *SystemArgument2
|
|
);
|
|
|
|
VOID
|
|
VdmpQueueIntNormalRoutine (
|
|
IN PVOID NormalContext,
|
|
IN PVOID SystemArgument1,
|
|
IN PVOID SystemArgument2
|
|
);
|
|
|
|
|
|
VOID
|
|
VdmpDelayIntDpcRoutine (
|
|
IN PKDPC Dpc,
|
|
IN PVOID DeferredContext,
|
|
IN PVOID SystemArgument1,
|
|
IN PVOID SystemArgument2
|
|
);
|
|
|
|
|
|
VOID
|
|
VdmpDelayIntApcRoutine (
|
|
IN PKAPC Apc,
|
|
IN PKNORMAL_ROUTINE *NormalRoutine,
|
|
IN PVOID *NormalContext,
|
|
IN PVOID *SystemArgument1,
|
|
IN PVOID *SystemArgument2
|
|
);
|
|
|
|
int
|
|
RestartDelayedInterrupts(
|
|
PVDMICAUSERDATA pIcaUserData
|
|
);
|
|
|
|
int
|
|
IcaScan(
|
|
PVDMICAUSERDATA pIcaUserData,
|
|
PVDMVIRTUALICA pIcaAdapter
|
|
);
|
|
|
|
int
|
|
IcaAccept(
|
|
PVDMICAUSERDATA pIcaUserData,
|
|
PVDMVIRTUALICA pIcaAdapter
|
|
);
|
|
|
|
ULONG
|
|
GetIretHookAddress(
|
|
PKTRAP_FRAME TrapFrame,
|
|
PVDMICAUSERDATA pIcaUserData,
|
|
int IrqNum
|
|
);
|
|
|
|
VOID
|
|
PushRmInterrupt(
|
|
PKTRAP_FRAME TrapFrame,
|
|
ULONG IretHookAddress,
|
|
PVDM_TIB VdmTib,
|
|
ULONG InterruptNumber
|
|
);
|
|
|
|
NTSTATUS
|
|
PushPmInterrupt(
|
|
PKTRAP_FRAME TrapFrame,
|
|
ULONG IretHookAddress,
|
|
PVDM_TIB VdmTib,
|
|
ULONG InterruptNumber
|
|
);
|
|
|
|
VOID
|
|
VdmpNullRundownRoutine(
|
|
IN PKAPC Apc
|
|
);
|
|
|
|
|
|
int
|
|
VdmpExceptionHandler(
|
|
IN PEXCEPTION_POINTERS ExceptionInfo
|
|
);
|
|
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(PAGE, VdmpQueueInterrupt)
|
|
#pragma alloc_text(PAGE, VdmpQueueIntApcRoutine)
|
|
#pragma alloc_text(PAGE, VdmpQueueIntNormalRoutine)
|
|
#pragma alloc_text(PAGE, VdmDispatchInterrupts)
|
|
#pragma alloc_text(PAGE, RestartDelayedInterrupts)
|
|
#pragma alloc_text(PAGE, IcaScan)
|
|
#pragma alloc_text(PAGE, IcaAccept)
|
|
#pragma alloc_text(PAGE, GetIretHookAddress)
|
|
#pragma alloc_text(PAGE, PushRmInterrupt)
|
|
#pragma alloc_text(PAGE, PushPmInterrupt)
|
|
#pragma alloc_text(PAGE, VdmpDispatchableIntPending)
|
|
#pragma alloc_text(PAGE, VdmpIsThreadTerminating)
|
|
#pragma alloc_text(PAGE, VdmpNullRundownRoutine)
|
|
#pragma alloc_text(PAGE, VdmpExceptionHandler)
|
|
#endif
|
|
|
|
extern POBJECT_TYPE ExSemaphoreObjectType;
|
|
extern POBJECT_TYPE ExEventObjectType;
|
|
|
|
NTSTATUS
|
|
VdmpQueueInterrupt(
|
|
IN HANDLE ThreadHandle
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Queues a user mode APC to the specifed application thread
|
|
which will dispatch an interrupt.
|
|
|
|
if APC is already queued to specified thread
|
|
does nothing
|
|
|
|
if APC is queued to the wrong thread
|
|
dequeue it
|
|
|
|
Reset the user APC for the specifed thread
|
|
|
|
Insert the APC in the queue for the specifed thread
|
|
|
|
Arguments:
|
|
|
|
ThreadHandle - handle of thread to insert QueueIntApcRoutine
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS.
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
PEPROCESS Process;
|
|
PETHREAD Thread;
|
|
NTSTATUS Status;
|
|
KPROCESSOR_MODE PrevMode;
|
|
PVDM_PROCESS_OBJECTS pVdmObjects;
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
PrevMode = KeGetPreviousMode();
|
|
|
|
Status = ObReferenceObjectByHandle(ThreadHandle,
|
|
THREAD_QUERY_INFORMATION,
|
|
PsThreadType,
|
|
PrevMode,
|
|
&Thread,
|
|
NULL
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
|
|
Process = PsGetCurrentProcess();
|
|
if (Process != Thread->ThreadsProcess || Process->Pcb.VdmFlag != TRUE)
|
|
{
|
|
Status = STATUS_INVALID_PARAMETER_1;
|
|
}
|
|
else {
|
|
pVdmObjects = Process->VdmObjects;
|
|
|
|
if (!Ke386VdmInsertQueueApc(&pVdmObjects->QueuedIntApc,
|
|
&Thread->Tcb,
|
|
KernelMode,
|
|
VdmpQueueIntApcRoutine,
|
|
VdmpNullRundownRoutine, // rundown
|
|
VdmpQueueIntNormalRoutine, // normal routine
|
|
(PVOID)KernelMode, // NormalContext
|
|
0
|
|
))
|
|
{
|
|
Status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
else {
|
|
Status = STATUS_SUCCESS;
|
|
}
|
|
}
|
|
|
|
ObDereferenceObject(Thread);
|
|
return Status;
|
|
}
|
|
|
|
|
|
VOID
|
|
VdmpQueueIntApcRoutine (
|
|
IN PKAPC Apc,
|
|
IN PKNORMAL_ROUTINE *NormalRoutine,
|
|
IN PVOID *NormalContext,
|
|
IN PVOID *SystemArgument1,
|
|
IN PVOID *SystemArgument2
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Kernel and User mode Special Apc routine to dispatchvirtual
|
|
interrupts to the vdm.
|
|
|
|
For KernelMode routine:
|
|
if vdm is running in application mode
|
|
queue a UserModeApc to the same thread
|
|
else do nothing
|
|
|
|
For UserMode routine
|
|
if vdm is running in application mode dispatch virtual interrupts
|
|
else do nothing
|
|
|
|
Arguments:
|
|
|
|
Apc -
|
|
Supplies a pointer to the APC object used to invoke this routine.
|
|
|
|
NormalRoutine -
|
|
Supplies a pointer to a pointer to the normal routine
|
|
function that was specified when the APC was initialized.
|
|
|
|
NormalContext - Supplies a pointer to the processor mode
|
|
specifying that this is a Kernel Mode or UserMode apc
|
|
|
|
SystemArgument1 -
|
|
|
|
SystemArgument2 - NOT USED
|
|
Supplies a set of two pointers to two arguments that contain
|
|
untyped data.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
PVDM_PROCESS_OBJECTS pVdmObjects;
|
|
NTSTATUS Status;
|
|
PETHREAD Thread;
|
|
PKTRAP_FRAME TrapFrame;
|
|
PVDM_TIB VdmTib;
|
|
|
|
PAGED_CODE();
|
|
|
|
Ke386VdmClearApcObject(Apc);
|
|
|
|
|
|
try {
|
|
|
|
|
|
// Get the trap frame for the current thread
|
|
|
|
Thread = PsGetCurrentThread();
|
|
|
|
|
|
// If the thread is dying, bail out
|
|
|
|
if (PsIsThreadTerminating(Thread))
|
|
return;
|
|
|
|
|
|
// if no pending interrupts, ignore this APC.
|
|
|
|
if (!(*pNtVDMState & VDM_INTERRUPT_PENDING)) {
|
|
return;
|
|
}
|
|
|
|
TrapFrame = VdmGetTrapFrame(&Thread->Tcb);
|
|
|
|
if (VdmpDispatchableIntPending(TrapFrame->EFlags))
|
|
{
|
|
pVdmObjects = PsGetCurrentProcess()->VdmObjects;
|
|
|
|
// if we are in v86 mode or segmented protected mode
|
|
// then queue the UserMode Apc, which will dispatch
|
|
// the hardware interrupt
|
|
|
|
if ((TrapFrame->EFlags & EFLAGS_V86_MASK) ||
|
|
TrapFrame->SegCs != (KGDT_R3_CODE | RPL_MASK))
|
|
{
|
|
if ((KPROCESSOR_MODE)*NormalContext == KernelMode) {
|
|
|
|
Ke386VdmInsertQueueApc(
|
|
&pVdmObjects->QueuedIntUserApc,
|
|
&Thread->Tcb,
|
|
UserMode,
|
|
VdmpQueueIntApcRoutine,
|
|
VdmpNullRundownRoutine,// rundown
|
|
NULL, // normal routine
|
|
(PVOID)UserMode, // NormalContext
|
|
(*pNtVDMState & VDM_INT_HARDWARE) // PrBoost
|
|
? VDM_HWINT_INCREMENT : 0
|
|
);
|
|
|
|
}
|
|
else {
|
|
ASSERT(*NormalContext == (PVOID)UserMode);
|
|
|
|
Status = VdmpGetVdmTib(&VdmTib, VDMTIB_KPROBE);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return;
|
|
}
|
|
|
|
|
|
//VdmTib = (PsGetCurrentProcess()->VdmObjects)->VdmTib;
|
|
// VdmTib =
|
|
// ((PVDM_PROCESS_OBJECTS)(PsGetCurrentProcess()->VdmObjects))->VdmTib;
|
|
|
|
|
|
// If there are no hardware ints, dispatch timer ints
|
|
// else dispatch hw interrupts
|
|
|
|
if (*pNtVDMState & VDM_INT_TIMER &&
|
|
!(*pNtVDMState & VDM_INT_HARDWARE))
|
|
{
|
|
VdmTib->EventInfo.Event = VdmIntAck;
|
|
VdmTib->EventInfo.InstructionSize = 0;
|
|
VdmTib->EventInfo.IntAckInfo = 0;
|
|
VdmEndExecution(TrapFrame, VdmTib);
|
|
}
|
|
else {
|
|
VdmDispatchInterrupts(TrapFrame, VdmTib);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// If we are not in application mode and wow is all blocked
|
|
// then Wake up WowExec by setting the wow idle event
|
|
|
|
else if (*NormalRoutine &&
|
|
!(*pNtVDMState & VDM_WOWBLOCKED))
|
|
{
|
|
*NormalRoutine = NULL;
|
|
}
|
|
|
|
}
|
|
// WARNING this may set VIP for flat if VPI is ever set in CR4
|
|
else if (((KeI386VirtualIntExtensions & V86_VIRTUAL_INT_EXTENSIONS) &&
|
|
(TrapFrame->EFlags & EFLAGS_V86_MASK)) ||
|
|
((KeI386VirtualIntExtensions & PM_VIRTUAL_INT_EXTENSIONS) &&
|
|
!(TrapFrame->EFlags & EFLAGS_V86_MASK)) )
|
|
{
|
|
// The CPU traps EVERY instruction if VIF and VIP are both ON.
|
|
// Make sure that you set VIP ON only when there are pending
|
|
// interrupts, i.e. (*pNtVDMState & VDM_INTERRUPT_PENDING) != 0.
|
|
ASSERT(*pNtVDMState & VDM_INTERRUPT_PENDING);
|
|
|
|
TrapFrame->EFlags |= EFLAGS_VIP;
|
|
}
|
|
}
|
|
except(VdmpExceptionHandler(GetExceptionInformation())) {
|
|
VdmDispatchException(TrapFrame,
|
|
GetExceptionCode(),
|
|
(PVOID)TrapFrame->Eip,
|
|
0,0,0,0 // no parameters
|
|
);
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
VdmpQueueIntNormalRoutine (
|
|
IN PVOID NormalContext,
|
|
IN PVOID SystemArgument1,
|
|
IN PVOID SystemArgument2
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
PETHREAD Thread;
|
|
PKEVENT Event;
|
|
NTSTATUS Status;
|
|
PKTRAP_FRAME TrapFrame;
|
|
PVDM_PROCESS_OBJECTS pVdmObjects;
|
|
|
|
|
|
try {
|
|
|
|
|
|
// Wake up WowExec by setting the wow idle event
|
|
|
|
|
|
pVdmObjects = PsGetCurrentProcess()->VdmObjects;
|
|
|
|
Status = ObReferenceObjectByHandle(
|
|
*pVdmObjects->pIcaUserData->phWowIdleEvent,
|
|
EVENT_MODIFY_STATE,
|
|
ExEventObjectType,
|
|
UserMode,
|
|
&Event,
|
|
NULL
|
|
);
|
|
|
|
if (NT_SUCCESS(Status)) {
|
|
KeSetEvent(Event, EVENT_INCREMENT, FALSE);
|
|
ObDereferenceObject(Event);
|
|
}
|
|
|
|
}
|
|
except(VdmpExceptionHandler(GetExceptionInformation())) {
|
|
Thread = PsGetCurrentThread();
|
|
TrapFrame = VdmGetTrapFrame(&Thread->Tcb);
|
|
VdmDispatchException(TrapFrame,
|
|
GetExceptionCode(),
|
|
(PVOID)TrapFrame->Eip,
|
|
0,0,0,0 // no parameters
|
|
);
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
NTSTATUS
|
|
VdmDispatchInterrupts(
|
|
PKTRAP_FRAME TrapFrame,
|
|
PVDM_TIB VdmTib
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine dispatches interrupts to the vdm.
|
|
Assumes that we are in application mode and NOT MONITOR context.
|
|
This routine may switch from application context back to monitor
|
|
context, if it cannot handle the interrupt (Ica in AEOI, or timer
|
|
int pending).
|
|
|
|
Arguments:
|
|
|
|
TrapFrame address of current trapframe
|
|
VdmTib address of current vdm tib
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS Status;
|
|
ULONG IretHookAddress;
|
|
ULONG InterruptNumber;
|
|
int IrqLineNum;
|
|
USHORT IcaRotate = 0;
|
|
PVDMICAUSERDATA pIcaUserData;
|
|
PVDMVIRTUALICA pIcaAdapter;
|
|
VDMEVENTCLASS VdmEvent = VdmMaxEvent;
|
|
|
|
PAGED_CODE();
|
|
|
|
try {
|
|
|
|
|
|
// Probe pointers in IcaUserData which will be touched
|
|
|
|
pIcaUserData = ((PVDM_PROCESS_OBJECTS)PsGetCurrentProcess()->VdmObjects)
|
|
->pIcaUserData;
|
|
|
|
|
|
ProbeForWrite(pIcaUserData->pAddrIretBopTable,
|
|
(TrapFrame->EFlags & EFLAGS_V86_MASK)
|
|
? VDM_RM_IRETBOPSIZE*16 : VDM_PM_IRETBOPSIZE*16,
|
|
sizeof(ULONG)
|
|
);
|
|
|
|
|
|
// Take the Ica Lock, if this fails raise status as we can't
|
|
// safely recover the critical section state
|
|
|
|
Status = VdmpEnterIcaLock(pIcaUserData->pIcaLock);
|
|
if (!NT_SUCCESS(Status)) {
|
|
ExRaiseStatus(Status);
|
|
}
|
|
|
|
|
|
if (*pIcaUserData->pUndelayIrq)
|
|
RestartDelayedInterrupts(pIcaUserData);
|
|
VDIretry:
|
|
|
|
|
|
|
|
// Clear the VIP bit
|
|
|
|
if (((KeI386VirtualIntExtensions & V86_VIRTUAL_INT_EXTENSIONS) &&
|
|
(TrapFrame->EFlags & EFLAGS_V86_MASK)) ||
|
|
((KeI386VirtualIntExtensions & PM_VIRTUAL_INT_EXTENSIONS) &&
|
|
!(TrapFrame->EFlags & EFLAGS_V86_MASK)) )
|
|
{
|
|
TrapFrame->EFlags &= ~EFLAGS_VIP;
|
|
}
|
|
|
|
|
|
|
|
// Mark the vdm state as hw int dispatched. Must use the lock as
|
|
// kernel mode DelayedIntApcRoutine changes the bit as well
|
|
|
|
_asm {
|
|
mov eax,FIXED_NTVDMSTATE_LINEAR
|
|
lock and dword ptr [eax], ~VDM_INT_HARDWARE
|
|
}
|
|
|
|
pIcaAdapter = pIcaUserData->pIcaMaster;
|
|
IrqLineNum = IcaAccept(pIcaUserData, pIcaAdapter);
|
|
if (IrqLineNum >= 0) {
|
|
UCHAR bit = 1 << IrqLineNum;
|
|
|
|
if (pIcaUserData->pIcaMaster->ica_ssr & bit) {
|
|
pIcaAdapter = pIcaUserData->pIcaSlave;
|
|
IrqLineNum = IcaAccept(pIcaUserData, pIcaAdapter);
|
|
if (IrqLineNum < 0) {
|
|
pIcaUserData->pIcaMaster->ica_isr &= ~bit;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// Skip spurious ints
|
|
|
|
if (IrqLineNum < 0) {
|
|
|
|
// Check for delayed interrupts which need to be restarted
|
|
|
|
if (*pIcaUserData->pUndelayIrq &&
|
|
RestartDelayedInterrupts(pIcaUserData) != -1)
|
|
{
|
|
goto VDIretry;
|
|
}
|
|
|
|
Status = VdmpLeaveIcaLock(pIcaUserData->pIcaLock);
|
|
if (!NT_SUCCESS(Status)) {
|
|
ExRaiseStatus(Status);
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
|
|
|
|
// Capture the AutoEoi mode case for special handling
|
|
|
|
if (pIcaAdapter->ica_mode & ICA_AEOI) {
|
|
VdmEvent = VdmIntAck;
|
|
VdmTib->EventInfo.IntAckInfo = (ULONG)IcaRotate | VDMINTACK_AEOI;
|
|
if (pIcaAdapter == pIcaUserData->pIcaSlave) {
|
|
VdmTib->EventInfo.IntAckInfo |= VDMINTACK_SLAVE;
|
|
}
|
|
}
|
|
|
|
|
|
InterruptNumber = IrqLineNum + pIcaAdapter->ica_base;
|
|
|
|
|
|
// Get the IretHookAddress ... if any
|
|
|
|
if (pIcaAdapter == pIcaUserData->pIcaSlave) {
|
|
IrqLineNum += 8;
|
|
}
|
|
|
|
|
|
IretHookAddress = GetIretHookAddress( TrapFrame,
|
|
pIcaUserData,
|
|
IrqLineNum
|
|
);
|
|
|
|
if (*pNtVDMState & VDM_TRACE_HISTORY) {
|
|
VdmTraceEvent(VDMTR_KERNEL_HW_INT,
|
|
(USHORT)InterruptNumber,
|
|
0,
|
|
TrapFrame);
|
|
}
|
|
|
|
|
|
// Push the interrupt frames
|
|
|
|
if (TrapFrame->EFlags & EFLAGS_V86_MASK) {
|
|
PushRmInterrupt(TrapFrame,
|
|
IretHookAddress,
|
|
VdmTib,
|
|
InterruptNumber
|
|
);
|
|
}
|
|
else {
|
|
Status = PushPmInterrupt(
|
|
TrapFrame,
|
|
IretHookAddress,
|
|
VdmTib,
|
|
InterruptNumber
|
|
);
|
|
|
|
if (!NT_SUCCESS(Status)) {
|
|
VdmpLeaveIcaLock(pIcaUserData->pIcaLock);
|
|
ExRaiseStatus(Status);
|
|
}
|
|
}
|
|
|
|
|
|
// Disable interrupts and the trap flag
|
|
|
|
|
|
|
|
if (((KeI386VirtualIntExtensions & V86_VIRTUAL_INT_EXTENSIONS) &&
|
|
(TrapFrame->EFlags & EFLAGS_V86_MASK)) ||
|
|
((KeI386VirtualIntExtensions & PM_VIRTUAL_INT_EXTENSIONS) &&
|
|
!(TrapFrame->EFlags & EFLAGS_V86_MASK)) )
|
|
{
|
|
TrapFrame->EFlags &= ~EFLAGS_VIF;
|
|
}
|
|
else if (!KeI386VdmIoplAllowed ||
|
|
!(TrapFrame->EFlags & EFLAGS_V86_MASK))
|
|
{
|
|
*pNtVDMState &= ~VDM_VIRTUAL_INTERRUPTS;
|
|
}
|
|
else {
|
|
TrapFrame->EFlags &= ~EFLAGS_INTERRUPT_MASK;
|
|
}
|
|
|
|
TrapFrame->EFlags &= ~(EFLAGS_NT_MASK | EFLAGS_TF_MASK);
|
|
|
|
KeBoostPriorityThread(KeGetCurrentThread(), VDM_HWINT_INCREMENT);
|
|
|
|
|
|
|
|
// Release the ica lock
|
|
|
|
Status = VdmpLeaveIcaLock(pIcaUserData->pIcaLock);
|
|
if (!NT_SUCCESS(Status)) {
|
|
ExRaiseStatus(Status);
|
|
}
|
|
|
|
|
|
// check to see if we are supposed to switch back to monitor context
|
|
|
|
if (VdmEvent != VdmMaxEvent) {
|
|
VdmTib->EventInfo.Event = VdmIntAck;
|
|
VdmTib->EventInfo.InstructionSize = 0;
|
|
VdmEndExecution(TrapFrame, VdmTib);
|
|
}
|
|
}
|
|
except(VdmpExceptionHandler(GetExceptionInformation())) {
|
|
Status = GetExceptionCode();
|
|
VdmDispatchException(TrapFrame,
|
|
Status,
|
|
(PVOID)TrapFrame->Eip,
|
|
0,0,0,0 // no parameters
|
|
);
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
|
|
int
|
|
RestartDelayedInterrupts(
|
|
PVDMICAUSERDATA pIcaUserData
|
|
)
|
|
{
|
|
int line;
|
|
|
|
|
|
PAGED_CODE();
|
|
|
|
*pIcaUserData->pUndelayIrq = 0;
|
|
|
|
line = IcaScan(pIcaUserData, pIcaUserData->pIcaSlave);
|
|
if (line != -1) {
|
|
// set the slave
|
|
pIcaUserData->pIcaSlave->ica_int_line = line;
|
|
pIcaUserData->pIcaSlave->ica_cpu_int = TRUE;
|
|
|
|
// set the master cascade
|
|
line = pIcaUserData->pIcaSlave->ica_ssr;
|
|
pIcaUserData->pIcaMaster->ica_irr |= 1 << line;
|
|
pIcaUserData->pIcaMaster->ica_count[line]++;
|
|
}
|
|
|
|
line = IcaScan(pIcaUserData, pIcaUserData->pIcaMaster);
|
|
if (line != -1) {
|
|
pIcaUserData->pIcaMaster->ica_cpu_int = TRUE;
|
|
pIcaUserData->pIcaMaster->ica_int_line = TRUE;
|
|
}
|
|
|
|
return line;
|
|
}
|
|
|
|
|
|
|
|
int
|
|
IcaScan(
|
|
PVDMICAUSERDATA pIcaUserData,
|
|
PVDMVIRTUALICA pIcaAdapter
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Similar to softpc\base\system\ica.c - scan_irr(),
|
|
|
|
Check the IRR, the IMR and the ISR to determine which interrupt
|
|
should be delivered.
|
|
|
|
A bit set in the IRR will generate an interrupt if:
|
|
IMR bit, DelayIret bit, DelayIrq bit AND ISR higher priority bits
|
|
are clear (unless Special Mask Mode, in which case ISR is ignored)
|
|
|
|
If there is no bit set, then return -1
|
|
|
|
|
|
Arguments:
|
|
PVDMICAUSERDATA pIcaUserData - addr of ica userdata
|
|
PVDMVIRTUALICA pIcaAdapter - addr of ica adapter
|
|
|
|
|
|
Return Value:
|
|
|
|
int IrqLineNum for the specific adapter (0 to 7)
|
|
-1 for none
|
|
|
|
--*/
|
|
{
|
|
int i,line;
|
|
UCHAR bit;
|
|
ULONG IrrImrDelay;
|
|
ULONG ActiveIsr;
|
|
|
|
PAGED_CODE();
|
|
|
|
IrrImrDelay = *pIcaUserData->pDelayIrq | *pIcaUserData->pDelayIret;
|
|
if (pIcaAdapter == pIcaUserData->pIcaSlave) {
|
|
IrrImrDelay >>= 8;
|
|
}
|
|
|
|
IrrImrDelay = pIcaAdapter->ica_irr & ~(pIcaAdapter->ica_imr | (UCHAR)IrrImrDelay);
|
|
|
|
if (IrrImrDelay) {
|
|
|
|
/*
|
|
* Does the current mode require the ica to prevent
|
|
* interrupts if that line is still active (ie in the isr)?
|
|
|
|
* Normal Case: Used by DOS and Win3.1/S the isr prevents interrupts.
|
|
* Special Mask Mode, Special Fully Nested Mode do not block
|
|
* interrupts using bits in the isr. SMM is the mode used
|
|
* by Windows95 and Win3.1/E.
|
|
|
|
*/
|
|
ActiveIsr = (pIcaAdapter->ica_mode & (ICA_SMM|ICA_SFNM))
|
|
? 0 : pIcaAdapter->ica_isr;
|
|
|
|
for(i = 0; i < 8; i++) {
|
|
line = (pIcaAdapter->ica_hipri + i) & 7;
|
|
bit = 1 << line;
|
|
if (ActiveIsr & bit) {
|
|
break; /* No nested interrupt possible */
|
|
}
|
|
|
|
if (IrrImrDelay & bit) {
|
|
return line;
|
|
}
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
IcaAccept(
|
|
PVDMICAUSERDATA pIcaUserData,
|
|
PVDMVIRTUALICA pIcaAdapter
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Does the equivalent of a cpu IntAck cycle retrieving the Irql Line Num
|
|
for interrupt dispatch, and setting the ica state to reflect that
|
|
the interrupt is in service.
|
|
|
|
Similar to softpc\base\system\ica.c - ica_accept() scan_irr(),
|
|
except that this code rejects interrupt dispatching if the ica
|
|
is in Auto-EOI as this may involve a new interrupt cycle, and
|
|
eoi hooks to be activated.
|
|
|
|
Arguments:
|
|
PVDMICAUSERDATA pIcaUserData - addr of ica userdata
|
|
PVDMVIRTUALICA pIcaAdapter - addr of ica adapter
|
|
|
|
|
|
Return Value:
|
|
|
|
ULONG IrqLineNum for the specific adapter (0 to 7)
|
|
returns -1 if there are no interrupts to generate (spurious ints
|
|
are normally done on line 7
|
|
|
|
--*/
|
|
{
|
|
int line;
|
|
UCHAR bit;
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
// Drop the INT line, and scan the ica
|
|
|
|
pIcaAdapter->ica_cpu_int = FALSE;
|
|
|
|
line = IcaScan(pIcaUserData, pIcaAdapter);
|
|
|
|
if (line < 0) {
|
|
return -1;
|
|
}
|
|
|
|
bit = (1 << line);
|
|
pIcaAdapter->ica_isr |= bit;
|
|
|
|
|
|
// decrement the count and clear the IRR bit
|
|
// ensure the count doesn't wrap past zero.
|
|
|
|
if (--(pIcaAdapter->ica_count[line]) <= 0) {
|
|
pIcaAdapter->ica_irr &= ~bit;
|
|
pIcaAdapter->ica_count[line] = 0;
|
|
}
|
|
|
|
|
|
return(line);
|
|
}
|
|
|
|
|
|
|
|
ULONG
|
|
GetIretHookAddress(
|
|
PKTRAP_FRAME TrapFrame,
|
|
PVDMICAUSERDATA pIcaUserData,
|
|
int IrqNum
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Retrieves the IretHookAddress from the real mode\protect mode
|
|
iret hook bop table. This function is equivalent to
|
|
softpc\base\system\ica.c - ica_iret_hook_needed()
|
|
|
|
Arguments:
|
|
|
|
TrapFrame - address of current trapframe
|
|
pIcaUserData - addr of ica data
|
|
IrqNum - IrqLineNum
|
|
|
|
Return Value:
|
|
|
|
ULONG IretHookAddress. seg:offset or sel:offset Iret Hook,
|
|
0 if none
|
|
--*/
|
|
{
|
|
ULONG IrqMask;
|
|
ULONG AddrBopTable;
|
|
int IretBopSize;
|
|
|
|
PAGED_CODE();
|
|
|
|
IrqMask = 1 << IrqNum;
|
|
if (!(IrqMask & *pIcaUserData->pIretHooked) ||
|
|
!*pIcaUserData->pAddrIretBopTable )
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (TrapFrame->EFlags & EFLAGS_V86_MASK) {
|
|
AddrBopTable = *pIcaUserData->pAddrIretBopTable;
|
|
IretBopSize = VDM_RM_IRETBOPSIZE;
|
|
}
|
|
else {
|
|
AddrBopTable = (VDM_PM_IRETBOPSEG << 16) | VDM_PM_IRETBOPOFF;
|
|
IretBopSize = VDM_PM_IRETBOPSIZE;
|
|
}
|
|
|
|
*pIcaUserData->pDelayIret |= IrqMask;
|
|
|
|
return AddrBopTable + IretBopSize * IrqNum;
|
|
}
|
|
|
|
|
|
|
|
|
|
VOID
|
|
PushRmInterrupt(
|
|
PKTRAP_FRAME TrapFrame,
|
|
ULONG IretHookAddress,
|
|
PVDM_TIB VdmTib,
|
|
ULONG InterruptNumber
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Pushes RealMode interrupt frame onto the UserMode stack in the TrapFrame
|
|
|
|
Arguments:
|
|
|
|
TrapFrame - address of current trapframe
|
|
IretHookAddress - address of Iret Hook, 0 if none
|
|
VdmTib - address of current vdm tib
|
|
InterruptNumber - interrupt number to reflect
|
|
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
ULONG UserSS;
|
|
USHORT UserSP;
|
|
USHORT NewCS;
|
|
USHORT NewIP;
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
// Get pointers to current stack
|
|
|
|
UserSS = TrapFrame->HardwareSegSs << 4;
|
|
UserSP = (USHORT) TrapFrame->HardwareEsp;
|
|
|
|
|
|
// load interrupt stack frame, pushing flags, Cs and ip
|
|
|
|
UserSP -= 2;
|
|
*(PUSHORT)(UserSS + UserSP) = (USHORT)TrapFrame->EFlags;
|
|
UserSP -= 2;
|
|
*(PUSHORT)(UserSS + UserSP) = (USHORT)TrapFrame->SegCs;
|
|
UserSP -= 2;
|
|
*(PUSHORT)(UserSS + UserSP) = (USHORT)TrapFrame->Eip;
|
|
|
|
|
|
|
|
// load IretHook stack frame if one exists
|
|
|
|
if (IretHookAddress) {
|
|
UserSP -= 2;
|
|
*(PUSHORT)(UserSS + UserSP) = (USHORT)(TrapFrame->EFlags & ~EFLAGS_TF_MASK);
|
|
UserSP -= 2;
|
|
*(PUSHORT)(UserSS + UserSP) = (USHORT)(IretHookAddress >> 16);
|
|
UserSP -= 2;
|
|
*(PUSHORT)(UserSS + UserSP) = (USHORT)IretHookAddress;
|
|
}
|
|
|
|
|
|
// Set new sp, ip, and cs.
|
|
|
|
|
|
if (VdmTib->VdmInterruptHandlers[InterruptNumber].Flags & VDM_INT_HOOKED) {
|
|
NewCS = (USHORT) (VdmTib->DpmiInfo.DosxRmReflector >> 16);
|
|
NewIP = (USHORT) VdmTib->DpmiInfo.DosxRmReflector;
|
|
|
|
// now encode the interrupt number into CS
|
|
|
|
NewCS -= (USHORT) InterruptNumber;
|
|
NewIP += (USHORT) (InterruptNumber*16);
|
|
|
|
} else {
|
|
PUSHORT pIvtEntry = (PUSHORT) (InterruptNumber * 4);
|
|
|
|
NewIP = *pIvtEntry++;
|
|
NewCS = *pIvtEntry;
|
|
}
|
|
|
|
TrapFrame->HardwareEsp = UserSP;
|
|
TrapFrame->Eip = NewIP;
|
|
TrapFrame->SegCs = NewCS;
|
|
}
|
|
|
|
|
|
|
|
|
|
NTSTATUS
|
|
PushPmInterrupt(
|
|
PKTRAP_FRAME TrapFrame,
|
|
ULONG IretHookAddress,
|
|
PVDM_TIB VdmTib,
|
|
ULONG InterruptNumber
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Pushes ProtectMode interrupt frame onto the UserMode stack in the TrapFrame
|
|
Raises an exception if an invalid stack is found
|
|
|
|
Arguments:
|
|
|
|
TrapFrame - address of current trapframe
|
|
IretHookAddress - address of Iret Hook, 0 if none
|
|
VdmTib - address of current vdm tib
|
|
InterruptNumber - interrupt number to reflect
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
ULONG Flags,Base,Limit;
|
|
ULONG VdmSp, VdmSpOrg;
|
|
PUSHORT VdmStackPointer;
|
|
ULONG StackOffset;
|
|
BOOLEAN Frame32 = (BOOLEAN) VdmTib->DpmiInfo.Flags;
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
// Switch to "locked" dpmi stack if lock count is zero
|
|
// This emulates the win3.1 Begin_Use_Locked_PM_Stack function.
|
|
|
|
|
|
if (!VdmTib->DpmiInfo.LockCount++) {
|
|
VdmTib->DpmiInfo.SaveEsp = TrapFrame->HardwareEsp;
|
|
VdmTib->DpmiInfo.SaveEip = TrapFrame->Eip;
|
|
VdmTib->DpmiInfo.SaveSsSelector = (USHORT) TrapFrame->HardwareSegSs;
|
|
TrapFrame->HardwareEsp = 0x1000;
|
|
TrapFrame->HardwareSegSs = (ULONG) VdmTib->DpmiInfo.SsSelector | 0x7;
|
|
}
|
|
|
|
|
|
// Use Sp or Esp ?
|
|
|
|
if (!Ki386GetSelectorParameters((USHORT)TrapFrame->HardwareSegSs,
|
|
&Flags, &Base, &Limit))
|
|
{
|
|
return STATUS_ACCESS_VIOLATION;
|
|
}
|
|
|
|
|
|
// Adjust the limit for page granularity
|
|
|
|
Limit++;
|
|
if (Flags & SEL_TYPE_2GIG) {
|
|
Limit = (Limit << 12) | 0xfff;
|
|
}
|
|
|
|
VdmSp = (Flags & SEL_TYPE_BIG) ? TrapFrame->HardwareEsp
|
|
: (USHORT)TrapFrame->HardwareEsp;
|
|
|
|
|
|
// Get pointer to current stack
|
|
|
|
VdmStackPointer = (PUSHORT)(Base + VdmSp);
|
|
|
|
|
|
|
|
// Create enough room for iret hook frame
|
|
|
|
VdmSpOrg = VdmSp;
|
|
if (IretHookAddress) {
|
|
if (Frame32) {
|
|
VdmSp -= 3*sizeof(ULONG);
|
|
} else {
|
|
VdmSp -= 3*sizeof(USHORT);
|
|
}
|
|
}
|
|
|
|
|
|
// Create enough room for 2 iret frames
|
|
|
|
|
|
if (Frame32) {
|
|
VdmSp -= 6*sizeof(ULONG);
|
|
} else {
|
|
VdmSp -= 6*sizeof(USHORT);
|
|
}
|
|
|
|
|
|
// Set Final Value of Sp\Esp, do this before checking stack
|
|
// limits so that invalid esp is visible to debuggers
|
|
|
|
if (Flags & SEL_TYPE_BIG) {
|
|
TrapFrame->HardwareEsp = VdmSp;
|
|
}
|
|
else {
|
|
(USHORT)TrapFrame->HardwareEsp = (USHORT)VdmSp;
|
|
}
|
|
|
|
|
|
|
|
// Check stack limits
|
|
// If any of the following conditions are TRUE
|
|
// - New stack pointer wraps (not enuf space)
|
|
// - If normal stack and Sp not below limit
|
|
// - If Expand Down stack and Sp not above limit
|
|
|
|
// Then raise a Stack Fault
|
|
|
|
if ( VdmSp >= VdmSpOrg ||
|
|
!(Flags & SEL_TYPE_ED) && VdmSpOrg > Limit ||
|
|
(Flags & SEL_TYPE_ED) && VdmSp < Limit )
|
|
{
|
|
return STATUS_ACCESS_VIOLATION;
|
|
}
|
|
|
|
|
|
// Build the Hw Int iret frame
|
|
|
|
|
|
if (Frame32) {
|
|
|
|
*(--(PULONG)VdmStackPointer) = TrapFrame->EFlags;
|
|
*(PUSHORT)(--(PULONG)VdmStackPointer) = (USHORT)TrapFrame->SegCs;
|
|
*(--(PULONG)VdmStackPointer) = TrapFrame->Eip;
|
|
*(--(PULONG)VdmStackPointer) = TrapFrame->EFlags & ~EFLAGS_TF_MASK;
|
|
*(--(PULONG)VdmStackPointer) = VdmTib->DpmiInfo.DosxIntIretD >> 16;
|
|
*(--(PULONG)VdmStackPointer) = VdmTib->DpmiInfo.DosxIntIretD & 0xffff;
|
|
|
|
} else {
|
|
|
|
*(--(PUSHORT)VdmStackPointer) = (USHORT)TrapFrame->EFlags;
|
|
*(--(PUSHORT)VdmStackPointer) = (USHORT)TrapFrame->SegCs;
|
|
*(--(PUSHORT)VdmStackPointer) = (USHORT)TrapFrame->Eip;
|
|
*(--(PUSHORT)VdmStackPointer) = (USHORT)(TrapFrame->EFlags & ~EFLAGS_TF_MASK);
|
|
*(--(PULONG)VdmStackPointer) = VdmTib->DpmiInfo.DosxIntIret;
|
|
|
|
}
|
|
|
|
|
|
// Point cs and ip at interrupt handler
|
|
|
|
TrapFrame->SegCs = VdmTib->VdmInterruptHandlers[InterruptNumber].CsSelector | 0x7;
|
|
TrapFrame->Eip = VdmTib->VdmInterruptHandlers[InterruptNumber].Eip;
|
|
|
|
|
|
// Turn off trace bit so we don't trace the iret hook
|
|
|
|
TrapFrame->EFlags &= ~EFLAGS_TF_MASK;
|
|
|
|
|
|
// Build the Irethook Iret frame, if one exists
|
|
|
|
if (IretHookAddress) {
|
|
ULONG SegCs, Eip;
|
|
|
|
|
|
// Point cs and eip at the iret hook, so when we build
|
|
// the frame below, the correct contents are set
|
|
|
|
SegCs = IretHookAddress >> 16;
|
|
Eip = IretHookAddress & 0xFFFF;
|
|
|
|
if (Frame32) {
|
|
|
|
*(--(PULONG)VdmStackPointer) = TrapFrame->EFlags;
|
|
*(PUSHORT)(--(PULONG)VdmStackPointer) = (USHORT)SegCs;
|
|
*(--(PULONG)VdmStackPointer) = Eip;
|
|
|
|
} else {
|
|
|
|
*(--(PUSHORT)VdmStackPointer) = (USHORT)TrapFrame->EFlags;
|
|
*(--(PUSHORT)VdmStackPointer) = (USHORT)SegCs;
|
|
*(--(PUSHORT)VdmStackPointer) = (USHORT)Eip;
|
|
|
|
}
|
|
}
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
VdmpDelayInterrupt(
|
|
PVDMDELAYINTSDATA pdsd
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sets a timer to dispatch the delayed interrupt through KeSetTimer.
|
|
When the timer fires a user mode APC is queued to queue the interrupt.
|
|
|
|
This function uses lazy allocation routines to allocate internal
|
|
data structures (nonpaged pool) on a per Irq basis, and needs to
|
|
be notified when specific Irq Lines no longer need Delayed
|
|
Interrupt services.
|
|
|
|
The caller must own the IcaLock to synchronize access to the
|
|
Irq lists.
|
|
|
|
WARNING: - Until the Delayed interrupt fires or is cancelled,
|
|
the specific Irq line will not generate any interrupts.
|
|
|
|
- The APC routine, does not take the HostIca lock, when
|
|
unblocking the IrqLine. Devices which use delayed Interrupts
|
|
should not queue ANY additional interrupts for the same IRQ
|
|
line until the delayed interrupt has fired or been cancelled.
|
|
|
|
Arguments:
|
|
|
|
pdsd.Delay Delay Interval in usecs
|
|
if Delay is 0xFFFFFFFF then per Irq Line nonpaged
|
|
data structures are freed. No Timers are set.
|
|
else the Delay is used as the timer delay.
|
|
|
|
pdsd.DelayIrqLine IrqLine Number
|
|
|
|
pdsd.hThread Thread Handle of CurrentMonitorTeb
|
|
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS.
|
|
|
|
--*/
|
|
|
|
{
|
|
PVDM_PROCESS_OBJECTS pVdmObjects;
|
|
PLIST_ENTRY Next;
|
|
PEPROCESS Process;
|
|
PDELAYINTIRQ pDelayIntIrq;
|
|
PETHREAD Thread, MainThread;
|
|
NTSTATUS Status;
|
|
KIRQL OldIrql;
|
|
ULONG IrqLine;
|
|
ULONG Delay;
|
|
PULONG pDelayIrq;
|
|
PULONG pUndelayIrq;
|
|
LARGE_INTEGER liDelay;
|
|
BOOLEAN FreeIrqLine, AlreadyInUse;
|
|
|
|
|
|
|
|
|
|
// Get a pointer to pVdmObjects
|
|
|
|
Process = PsGetCurrentProcess();
|
|
pVdmObjects = Process->VdmObjects;
|
|
if (Process->Pcb.VdmFlag != TRUE || !pVdmObjects) {
|
|
return STATUS_INVALID_PARAMETER_1;
|
|
}
|
|
|
|
ExAcquireFastMutex(&pVdmObjects->DelayIntFastMutex);
|
|
|
|
Status = STATUS_SUCCESS;
|
|
Thread = MainThread = NULL;
|
|
FreeIrqLine = TRUE;
|
|
AlreadyInUse = FALSE;
|
|
|
|
try {
|
|
|
|
|
|
// Probe the parameters
|
|
|
|
ProbeForRead(pdsd, sizeof(VDMDELAYINTSDATA), sizeof(ULONG));
|
|
|
|
|
|
// Form a BitMask for the IrqLine Number
|
|
|
|
IrqLine = 1 << pdsd->DelayIrqLine;
|
|
if (!IrqLine) {
|
|
Status = STATUS_INVALID_PARAMETER_2;
|
|
goto VidEarlyExit;
|
|
}
|
|
|
|
pDelayIrq = pVdmObjects->pIcaUserData->pDelayIrq;
|
|
ProbeForWriteUlong(pDelayIrq);
|
|
pUndelayIrq = pVdmObjects->pIcaUserData->pUndelayIrq;
|
|
ProbeForWriteUlong(pUndelayIrq);
|
|
|
|
|
|
|
|
// Copy out the Delay parameter, and convert hundreths nanosecs
|
|
|
|
Delay = pdsd->Delay;
|
|
|
|
|
|
// Check to see if we need to reset the timer resolution
|
|
|
|
if (Delay == 0xFFFFFFFF) {
|
|
ZwSetTimerResolution(KeMaximumIncrement, FALSE, &Delay);
|
|
goto VidEarlyExit;
|
|
}
|
|
|
|
|
|
FreeIrqLine = FALSE;
|
|
|
|
// convert delay to hundreths of nanosecs
|
|
// and ensure min delay of 1 msec
|
|
|
|
Delay = Delay < 1000 ? 10000 : Delay * 10;
|
|
|
|
|
|
// If the delay time is close to the system's clock rate
|
|
// then adjust the system's clock rate and if needed
|
|
// the delay time so that the timer will fire before the
|
|
// the due time.
|
|
|
|
if (Delay < 150000) {
|
|
ULONG ul = Delay >> 1;
|
|
|
|
if (ul < KeTimeIncrement && KeTimeIncrement > KeMinimumIncrement) {
|
|
ZwSetTimerResolution(ul, TRUE, (PULONG)&liDelay.LowPart);
|
|
}
|
|
|
|
if (Delay < KeTimeIncrement) {
|
|
// can't set system clock rate low enuf, so use half delay
|
|
Delay >>= 1;
|
|
}
|
|
else if (Delay < (KeTimeIncrement << 1)) {
|
|
// Real close to the system clock rate, lower delay
|
|
// proportionally, to avoid missing clock cycles.
|
|
Delay -= KeTimeIncrement >> 1;
|
|
}
|
|
}
|
|
|
|
|
|
// Reference the Target Thread
|
|
|
|
Status = ObReferenceObjectByHandle(
|
|
pdsd->hThread,
|
|
THREAD_QUERY_INFORMATION,
|
|
PsThreadType,
|
|
KeGetPreviousMode(),
|
|
&Thread,
|
|
NULL
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
Thread = NULL;
|
|
goto VidEarlyExit;
|
|
}
|
|
|
|
|
|
Status = ObReferenceObjectByPointer(
|
|
pVdmObjects->MainThread,
|
|
THREAD_QUERY_INFORMATION,
|
|
PsThreadType,
|
|
KernelMode
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
MainThread = pVdmObjects->MainThread;
|
|
}
|
|
else {
|
|
goto VidEarlyExit;
|
|
}
|
|
|
|
VidEarlyExit:;
|
|
}
|
|
except(EXCEPTION_EXECUTE_HANDLER) {
|
|
Status = GetExceptionCode();
|
|
}
|
|
|
|
if (!NT_SUCCESS(Status)) {
|
|
ExReleaseFastMutex(&pVdmObjects->DelayIntFastMutex);
|
|
if (Thread) {
|
|
ObDereferenceObject(Thread);
|
|
}
|
|
|
|
if (MainThread) {
|
|
ObDereferenceObject(MainThread);
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
|
|
|
|
KeAcquireSpinLock(&pVdmObjects->DelayIntSpinLock, &OldIrql);
|
|
|
|
try {
|
|
|
|
|
|
// Search the DelayedIntList for a matching Irq Line.
|
|
|
|
Next = pVdmObjects->DelayIntListHead.Flink;
|
|
while (Next != &pVdmObjects->DelayIntListHead) {
|
|
pDelayIntIrq = CONTAINING_RECORD(Next, DELAYINTIRQ, DelayIntListEntry);
|
|
if (pDelayIntIrq->IrqLine == IrqLine) {
|
|
break;
|
|
}
|
|
Next = Next->Flink;
|
|
}
|
|
|
|
if (Next == &pVdmObjects->DelayIntListHead) {
|
|
pDelayIntIrq = NULL;
|
|
}
|
|
|
|
|
|
if (!pDelayIntIrq) {
|
|
if (FreeIrqLine) {
|
|
goto VidExit;
|
|
}
|
|
|
|
|
|
// If a DelayIntIrq does not exist for this irql, allocate from nonpaged
|
|
// pool and initialize it
|
|
pDelayIntIrq = ExAllocatePoolWithTag(NonPagedPool, sizeof(DELAYINTIRQ), ' MDV');
|
|
if (!pDelayIntIrq) {
|
|
Status = STATUS_NO_MEMORY;
|
|
goto VidExit;
|
|
}
|
|
|
|
try {
|
|
PsChargePoolQuota(Process, NonPagedPool, sizeof(DELAYINTIRQ));
|
|
}
|
|
except(EXCEPTION_EXECUTE_HANDLER) {
|
|
Status = GetExceptionCode();
|
|
ExFreePool(pDelayIntIrq);
|
|
goto VidExit;
|
|
}
|
|
RtlZeroMemory(pDelayIntIrq, sizeof(DELAYINTIRQ));
|
|
pDelayIntIrq->IrqLine = IrqLine;
|
|
|
|
KeInitializeTimer(&pDelayIntIrq->Timer);
|
|
|
|
KeInitializeDpc(&pDelayIntIrq->Dpc,
|
|
VdmpDelayIntDpcRoutine,
|
|
Process
|
|
);
|
|
|
|
InsertTailList(&pVdmObjects->DelayIntListHead,
|
|
&pDelayIntIrq->DelayIntListEntry
|
|
);
|
|
}
|
|
|
|
|
|
if (Delay == 0xFFFFFFFF) {
|
|
if (pDelayIntIrq->InUse == VDMDELAY_KTIMER) {
|
|
pDelayIntIrq->InUse = VDMDELAY_NOTINUSE;
|
|
pDelayIntIrq = NULL;
|
|
}
|
|
}
|
|
else if (pDelayIntIrq->InUse == VDMDELAY_NOTINUSE) {
|
|
liDelay = RtlEnlargedIntegerMultiply(Delay, -1);
|
|
KeSetTimer(&pDelayIntIrq->Timer, liDelay, &pDelayIntIrq->Dpc);
|
|
ObReferenceObject(Process);
|
|
}
|
|
|
|
VidExit:;
|
|
}
|
|
except(EXCEPTION_EXECUTE_HANDLER) {
|
|
Status = GetExceptionCode();
|
|
}
|
|
|
|
if (pDelayIntIrq && !pDelayIntIrq->InUse) {
|
|
|
|
if (NT_SUCCESS(Status)) {
|
|
|
|
// Save PETHREAD of Target thread for the dpc routine
|
|
// the DPC routine will deref the threads.
|
|
|
|
pDelayIntIrq->InUse = VDMDELAY_KTIMER;
|
|
pDelayIntIrq->Thread = Thread;
|
|
Thread = NULL;
|
|
pDelayIntIrq->MainThread = MainThread;
|
|
MainThread = NULL;
|
|
}
|
|
else {
|
|
pDelayIntIrq->InUse = VDMDELAY_NOTINUSE;
|
|
pDelayIntIrq->Thread = NULL;
|
|
FreeIrqLine = TRUE;
|
|
}
|
|
}
|
|
else {
|
|
AlreadyInUse = TRUE;
|
|
}
|
|
|
|
|
|
|
|
KeReleaseSpinLock(&pVdmObjects->DelayIntSpinLock, OldIrql);
|
|
|
|
try {
|
|
if (FreeIrqLine) {
|
|
*pDelayIrq &= ~IrqLine;
|
|
_asm {
|
|
mov eax, pUndelayIrq
|
|
mov ebx, IrqLine
|
|
lock or [eax], ebx
|
|
}
|
|
}
|
|
else if (!AlreadyInUse) { // TakeIrqLine
|
|
*pDelayIrq |= IrqLine;
|
|
_asm {
|
|
mov eax, pUndelayIrq
|
|
mov ebx, IrqLine
|
|
not ebx
|
|
lock and [eax], ebx
|
|
}
|
|
}
|
|
}
|
|
except(EXCEPTION_EXECUTE_HANDLER) {
|
|
Status = GetExceptionCode();
|
|
}
|
|
|
|
ExReleaseFastMutex(&pVdmObjects->DelayIntFastMutex);
|
|
|
|
if (Thread) {
|
|
ObDereferenceObject(Thread);
|
|
}
|
|
|
|
if (MainThread) {
|
|
ObDereferenceObject(MainThread);
|
|
}
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VOID
|
|
VdmpDelayIntDpcRoutine (
|
|
IN PKDPC Dpc,
|
|
IN PVOID DeferredContext,
|
|
IN PVOID SystemArgument1,
|
|
IN PVOID SystemArgument2
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function is the DPC routine that is called when a DelayedInterrupt
|
|
timer expires. Its function is to insert the associated APC into the
|
|
target thread's APC queue.
|
|
|
|
Arguments:
|
|
|
|
Dpc - Supplies a pointer to a control object of type DPC.
|
|
|
|
DeferredContext - Supplies a pointer to the Target EProcess
|
|
|
|
SystemArgument1, SystemArgument2 - Supplies a set of two pointers to
|
|
two arguments that contain untyped data that are
|
|
NOT USED.
|
|
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
PVDM_PROCESS_OBJECTS pVdmObjects;
|
|
PEPROCESS Process;
|
|
PETHREAD Thread, MainThread;
|
|
PLIST_ENTRY Next;
|
|
PDELAYINTIRQ pDelayIntIrq;
|
|
KIRQL OldIrql;
|
|
|
|
|
|
// Get address of Process VdmObjects
|
|
|
|
Process = (PEPROCESS)DeferredContext;
|
|
pVdmObjects = (PVDM_PROCESS_OBJECTS)Process->VdmObjects;
|
|
|
|
|
|
KeAcquireSpinLock(&pVdmObjects->DelayIntSpinLock, &OldIrql);
|
|
|
|
|
|
|
|
// Search the DelayedIntList for the matching Dpc.
|
|
|
|
Next = pVdmObjects->DelayIntListHead.Flink;
|
|
while (Next != &pVdmObjects->DelayIntListHead) {
|
|
pDelayIntIrq = CONTAINING_RECORD(Next,DELAYINTIRQ,DelayIntListEntry);
|
|
if (&pDelayIntIrq->Dpc == Dpc) {
|
|
break;
|
|
}
|
|
Next = Next->Flink;
|
|
}
|
|
|
|
if (Next == &pVdmObjects->DelayIntListHead) {
|
|
pDelayIntIrq = NULL;
|
|
MainThread = Thread = NULL;
|
|
}
|
|
else {
|
|
Thread = pDelayIntIrq->Thread;
|
|
pDelayIntIrq->Thread = NULL;
|
|
MainThread = pDelayIntIrq->MainThread;
|
|
pDelayIntIrq->MainThread = NULL;
|
|
}
|
|
|
|
|
|
if (pDelayIntIrq && pDelayIntIrq->InUse) {
|
|
if ((Thread &&
|
|
Ke386VdmInsertQueueApc(&pDelayIntIrq->Apc,
|
|
&Thread->Tcb,
|
|
KernelMode,
|
|
VdmpDelayIntApcRoutine,
|
|
VdmpNullRundownRoutine, // rundown
|
|
VdmpQueueIntNormalRoutine, // normal routine
|
|
NULL, // NormalContext
|
|
VDM_HWINT_INCREMENT
|
|
))
|
|
||
|
|
(MainThread &&
|
|
Ke386VdmInsertQueueApc(&pDelayIntIrq->Apc,
|
|
&MainThread->Tcb,
|
|
KernelMode,
|
|
VdmpDelayIntApcRoutine,
|
|
VdmpNullRundownRoutine, // rundown
|
|
VdmpQueueIntNormalRoutine, // normal routine
|
|
NULL, // NormalContext
|
|
VDM_HWINT_INCREMENT
|
|
)))
|
|
{
|
|
pDelayIntIrq->InUse = VDMDELAY_KAPC;
|
|
}
|
|
else {
|
|
// This hwinterrupt line is blocked forever.
|
|
pDelayIntIrq->InUse = VDMDELAY_NOTINUSE;
|
|
}
|
|
}
|
|
|
|
|
|
KeReleaseSpinLock(&pVdmObjects->DelayIntSpinLock, OldIrql);
|
|
|
|
if (Thread) {
|
|
ObDereferenceObject(Thread);
|
|
}
|
|
|
|
if (MainThread) {
|
|
ObDereferenceObject(MainThread);
|
|
}
|
|
|
|
ObDereferenceObject(Process);
|
|
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
VdmpDelayIntApcRoutine (
|
|
IN PKAPC Apc,
|
|
IN PKNORMAL_ROUTINE *NormalRoutine,
|
|
IN PVOID *NormalContext,
|
|
IN PVOID *SystemArgument1,
|
|
IN PVOID *SystemArgument2
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function is the special APC routine that is called to
|
|
dispatch a delayed interupt. This routines clears the IrqLine
|
|
bit. VdmpQueueIntApcRoutine will restart interrupts.
|
|
|
|
Arguments:
|
|
|
|
Apc - Supplies a pointer to the APC object used to invoke this routine.
|
|
|
|
NormalRoutine - Supplies a pointer to a pointer to an optional
|
|
normal routine, which is executed when wow is blocked.
|
|
|
|
NormalContext - Supplies a pointer to a pointer to an arbitrary data
|
|
structure that was specified when the APC was initialized and is
|
|
NOT USED.
|
|
|
|
SystemArgument1, SystemArgument2 - Supplies a set of two pointers to
|
|
two arguments that contain untyped data that are
|
|
NOT USED.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
KIRQL OldIrql;
|
|
PLIST_ENTRY Next;
|
|
PDELAYINTIRQ pDelayIntIrq;
|
|
PVDM_PROCESS_OBJECTS pVdmObjects;
|
|
KPROCESSOR_MODE ProcessorMode;
|
|
PULONG pDelayIrq;
|
|
PULONG pUndelayIrq;
|
|
PULONG pDelayIret;
|
|
ULONG IrqLine;
|
|
BOOLEAN FreeIrqLine;
|
|
|
|
|
|
|
|
// Get a pointer to pVdmObjects
|
|
|
|
pVdmObjects = PsGetCurrentProcess()->VdmObjects;
|
|
ExAcquireFastMutex(&pVdmObjects->DelayIntFastMutex);
|
|
KeAcquireSpinLock(&pVdmObjects->DelayIntSpinLock, &OldIrql);
|
|
|
|
Ke386VdmClearApcObject(Apc);
|
|
|
|
FreeIrqLine = FALSE;
|
|
|
|
try {
|
|
|
|
|
|
// Search the DelayedIntList for the pDelayIntIrq.
|
|
|
|
Next = pVdmObjects->DelayIntListHead.Flink;
|
|
while (Next != &pVdmObjects->DelayIntListHead) {
|
|
pDelayIntIrq = CONTAINING_RECORD(Next,DELAYINTIRQ,DelayIntListEntry);
|
|
if (&pDelayIntIrq->Apc == Apc) {
|
|
break;
|
|
}
|
|
Next = Next->Flink;
|
|
}
|
|
|
|
if (Next == &pVdmObjects->DelayIntListHead) {
|
|
pDelayIntIrq = NULL;
|
|
}
|
|
|
|
|
|
// If we found the IrqLine in the DelayedIntList,
|
|
// restart interrupts.
|
|
|
|
if (pDelayIntIrq && pDelayIntIrq->InUse) {
|
|
pDelayIntIrq->InUse = VDMDELAY_NOTINUSE;
|
|
IrqLine = pDelayIntIrq->IrqLine;
|
|
FreeIrqLine = TRUE;
|
|
}
|
|
|
|
}
|
|
except(VdmpExceptionHandler(GetExceptionInformation())) {
|
|
; // fall thru
|
|
}
|
|
|
|
|
|
KeReleaseSpinLock(&pVdmObjects->DelayIntSpinLock, OldIrql);
|
|
|
|
|
|
try {
|
|
|
|
if (!FreeIrqLine) {
|
|
leave;
|
|
}
|
|
|
|
pDelayIrq = pVdmObjects->pIcaUserData->pDelayIrq;
|
|
pUndelayIrq = pVdmObjects->pIcaUserData->pUndelayIrq;
|
|
pDelayIret = pVdmObjects->pIcaUserData->pDelayIret;
|
|
|
|
|
|
// These variables are being modified without holding the
|
|
// ICA lock. This should be OK because none of the ntvdm
|
|
// devices (timer, mouse etc. should ever do a delayed int
|
|
// while a previous delayed interrupt is still pending.
|
|
|
|
|
|
*pDelayIrq &= ~IrqLine;
|
|
_asm {
|
|
mov eax, pUndelayIrq
|
|
mov ebx, IrqLine
|
|
lock or [eax], ebx
|
|
}
|
|
|
|
|
|
// If we are waiting for an iret hook we have nothing left to do
|
|
// since the iret hook will restart interrupts.
|
|
|
|
if (!(IrqLine & *pDelayIret)) {
|
|
|
|
|
|
// set hardware int pending
|
|
|
|
_asm {
|
|
mov eax,FIXED_NTVDMSTATE_LINEAR
|
|
lock or dword ptr [eax], VDM_INT_HARDWARE
|
|
}
|
|
|
|
|
|
// Queue a UserModeApc to dispatch interrupts
|
|
|
|
if (NormalRoutine) {
|
|
ProcessorMode = KernelMode;
|
|
VdmpQueueIntApcRoutine(Apc,
|
|
NormalRoutine,
|
|
(PVOID *)&ProcessorMode,
|
|
SystemArgument1,
|
|
SystemArgument2
|
|
);
|
|
}
|
|
}
|
|
}
|
|
except(VdmpExceptionHandler(GetExceptionInformation())) {
|
|
; // fall thru
|
|
}
|
|
|
|
ExReleaseFastMutex(&pVdmObjects->DelayIntFastMutex);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
BOOLEAN
|
|
VdmpDispatchableIntPending(
|
|
ULONG EFlags
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine determines whether or not there is a dispatchable
|
|
virtual interrupt to dispatch.
|
|
|
|
Arguments:
|
|
|
|
EFlags -- supplies a pointer to the EFlags to be checked
|
|
|
|
Return Value:
|
|
|
|
True -- a virtual interrupt should be dispatched
|
|
False -- no virtual interrupt should be dispatched
|
|
|
|
--*/
|
|
{
|
|
PAGED_CODE();
|
|
|
|
// Insure that we are not trying to run with IOPL and pentium extensions
|
|
|
|
ASSERT((!(KeI386VdmIoplAllowed &&
|
|
(KeI386VirtualIntExtensions & (V86_VIRTUAL_INT_EXTENSIONS |
|
|
PM_VIRTUAL_INT_EXTENSIONS)))));
|
|
|
|
if (EFlags & EFLAGS_V86_MASK) {
|
|
if (KeI386VirtualIntExtensions & V86_VIRTUAL_INT_EXTENSIONS) {
|
|
return(0 != (EFlags & EFLAGS_VIF));
|
|
} else if (KeI386VdmIoplAllowed) {
|
|
return (0 != (EFlags & EFLAGS_INTERRUPT_MASK));
|
|
} else {
|
|
return (0 != (*pNtVDMState & VDM_VIRTUAL_INTERRUPTS));
|
|
}
|
|
} else {
|
|
if (KeI386VirtualIntExtensions & PM_VIRTUAL_INT_EXTENSIONS) {
|
|
return(0 != (EFlags & EFLAGS_VIF));
|
|
} else {
|
|
return (0 != (*pNtVDMState & VDM_VIRTUAL_INTERRUPTS));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
NTSTATUS
|
|
VdmpIsThreadTerminating(
|
|
HANDLE ThreadId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine determines if the specified thread is terminating or not.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
True --
|
|
False -
|
|
|
|
--*/
|
|
{
|
|
CLIENT_ID Cid;
|
|
PETHREAD Thread;
|
|
NTSTATUS Status;
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
// If the owning thread juest exited the IcaLock the
|
|
// OwningThread Tid may be NULL, return success, since
|
|
// we don't know what the owning threads state was.
|
|
|
|
if (!ThreadId) {
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
Cid.UniqueProcess = NtCurrentTeb()->ClientId.UniqueProcess;
|
|
Cid.UniqueThread = ThreadId;
|
|
|
|
Status = PsLookupProcessThreadByCid(&Cid, NULL, &Thread);
|
|
if (NT_SUCCESS(Status)) {
|
|
Status = PsIsThreadTerminating(Thread) ? STATUS_THREAD_IS_TERMINATING
|
|
: STATUS_SUCCESS;
|
|
ObDereferenceObject(Thread);
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
VOID
|
|
VdmpNullRundownRoutine(
|
|
IN PKAPC Apc
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is used as a rundown routine for our APC.
|
|
Attempts to clear the delayed int state
|
|
|
|
Arguments:
|
|
|
|
Apc - Supplies a pointer to the Apc to run down.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
VdmpDelayIntApcRoutine( Apc, NULL, NULL, NULL, NULL);
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
VdmpExceptionHandler(
|
|
IN PEXCEPTION_POINTERS ExceptionInfo
|
|
)
|
|
{
|
|
#if DBG
|
|
PEXCEPTION_RECORD ExceptionRecord;
|
|
PCONTEXT ContextRecord;
|
|
ULONG NumberParameters;
|
|
PULONG ExceptionInformation;
|
|
#endif
|
|
|
|
PAGED_CODE();
|
|
|
|
#if DBG
|
|
|
|
ExceptionRecord = ExceptionInfo->ExceptionRecord;
|
|
DbgPrint("VdmExRecord ExCode %x Flags %x Address %x\n",
|
|
ExceptionRecord->ExceptionCode,
|
|
ExceptionRecord->ExceptionFlags,
|
|
ExceptionRecord->ExceptionAddress
|
|
);
|
|
|
|
NumberParameters = ExceptionRecord->NumberParameters;
|
|
if (NumberParameters) {
|
|
DbgPrint("VdmExRecord Parameters:\n");
|
|
|
|
ExceptionInformation = ExceptionRecord->ExceptionInformation;
|
|
while (NumberParameters--) {
|
|
DbgPrint("\t%x\n", *ExceptionInformation);
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
return EXCEPTION_EXECUTE_HANDLER;
|
|
}
|