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

440 lines
12 KiB
C

/*++
Copyright (c) 1991 Microsoft Corporation
Module Name:
vdmprint.c
Abstract:
This module contains the support for printing ports which could be
handled in kernel without going to ntvdm.exe
Author:
Sudeep Bharati (sudeepb) 16-Jan-1993
Revision History:
William Hsieh (williamh) 31-May-1996
rewrote for Dongle support
--*/
#include "vdmp.h"
#include "vdmprint.h"
#include <i386.h>
#include <v86emul.h>
#include "..\..\..\..\inc\ntddvdm.h"
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, VdmPrinterStatus)
#pragma alloc_text(PAGE, VdmPrinterWriteData)
#pragma alloc_text(PAGE, VdmFlushPrinterWriteData)
#pragma alloc_text(PAGE, VdmpPrinterDirectIoOpen)
#pragma alloc_text(PAGE, VdmpPrinterDirectIoClose)
#endif
BOOLEAN
VdmPrinterStatus(
ULONG iPort,
ULONG cbInstructionSize,
PKTRAP_FRAME TrapFrame
)
/*++
Routine Description:
This routine handles the read operation on the printer status port
Arguments:
iPort - port on which the io was trapped
cbInstructionSize - Instruction size to update TsEip
TrapFrame - Trap Frame
Return Value:
True if successfull, False otherwise
--*/
{
UCHAR PrtMode;
USHORT adapter;
ULONG * printer_status;
KIRQL OldIrql;
PIO_STATUS_BLOCK IoStatusBlock;
PVDM_PRINTER_INFO PrtInfo;
PVDM_TIB VdmTib;
NTSTATUS Status = STATUS_SUCCESS;
PAGED_CODE();
// why we have printer stuff in TIB? It would be more appropriate
// if we have them in the process object. The main reason(I think)
// is that we can get rid of synchronization.
Status = VdmpGetVdmTib(&VdmTib, VDMTIB_KMODE);
if (!NT_SUCCESS(Status)) {
return(FALSE);
}
PrtInfo = &VdmTib->PrinterInfo;
IoStatusBlock = (PIO_STATUS_BLOCK) &VdmTib->TempArea1;
printer_status = &VdmTib->PrinterInfo.prt_Scratch;
*pNtVDMState |= VDM_IDLEACTIVITY;
try {
// first, figure out which PRT we are dealing with. The
// port addresses in the PrinterInfo are base address of each
// PRT sorted in the adapter order.
if ((USHORT)iPort == PrtInfo->prt_PortAddr[0] + STATUS_PORT_OFFSET)
adapter = 0;
else if ((USHORT)iPort == PrtInfo->prt_PortAddr[1] + STATUS_PORT_OFFSET)
adapter = 1;
else if ((USHORT)iPort == PrtInfo->prt_PortAddr[2] + STATUS_PORT_OFFSET)
adapter = 2;
else {
// something must be wrong in our code, better check it out
ASSERT(FALSE);
return FALSE;
}
PrtMode = PrtInfo->prt_Mode[adapter];
if(PRT_MODE_SIMULATE_STATUS_PORT == PrtMode) {
// we are simulating printer status read.
// get the current status from softpc.
if (!(get_status(adapter) & NOTBUSY) &&
!(host_lpt_status(adapter) & HOST_LPT_BUSY)) {
if (get_control(adapter) & IRQ)
return FALSE;
set_status(adapter, get_status(adapter) | NOTBUSY);
}
*printer_status = (ULONG)(get_status(adapter) | STATUS_REG_MASK);
}
else if (PRT_MODE_DIRECT_IO == PrtMode) {
// we have to read the i/o directly(of course, through file system
// which in turn goes to the driver).
// Before performing read, flush out all pending output data in our
// buffer. This is done because the status we are about to read
// may depend on those pending output data.
if (PrtInfo->prt_BytesInBuffer[adapter]) {
Status = VdmFlushPrinterWriteData(adapter);
#ifdef DBG
if (!NT_SUCCESS(Status)) {
DbgPrint("VdmPrintStatus: failed to flush buffered data, status = %ls\n", Status);
}
#endif
}
OldIrql = KeGetCurrentIrql();
// lower irql to passive before doing any IO
KeLowerIrql(PASSIVE_LEVEL);
Status = NtDeviceIoControlFile(PrtInfo->prt_Handle[adapter],
NULL, // notification event
NULL, // APC routine
NULL, // Apc Context
IoStatusBlock,
IOCTL_VDM_PAR_READ_STATUS_PORT,
NULL,
0,
printer_status,
sizeof(ULONG)
);
if (!NT_SUCCESS(Status) || !NT_SUCCESS(IoStatusBlock->Status)) {
// fake a status to make it looks like the port is not connected
// to a printer.
*printer_status = 0x7F;
#ifdef DBG
DbgPrint("VdmPrinterStatus: failed to get status from printer, status = %lx\n", Status);
#endif
// always tell the caller that we have simulated the operation.
Status = STATUS_SUCCESS;
}
KeRaiseIrql(OldIrql, &OldIrql);
}
else
// we don't simulate it here
return FALSE;
TrapFrame->Eax &= 0xffffff00;
TrapFrame->Eax |= (UCHAR)*printer_status;
TrapFrame->Eip += cbInstructionSize;
} except(EXCEPTION_EXECUTE_HANDLER) {
Status = GetExceptionCode();
}
return (NT_SUCCESS(Status));
}
BOOLEAN VdmPrinterWriteData(
ULONG iPort,
ULONG cbInstructionSize,
PKTRAP_FRAME TrapFrame
)
{
UCHAR PrtMode;
PVDM_PRINTER_INFO PrtInfo;
USHORT adapter;
PVDM_TIB VdmTib;
NTSTATUS Status = STATUS_SUCCESS;
PAGED_CODE();
Status = VdmpGetVdmTib(&VdmTib, VDMTIB_KMODE);
if (!NT_SUCCESS(Status)) {
return(FALSE);
}
PrtInfo = &VdmTib->PrinterInfo;
*pNtVDMState |= VDM_IDLEACTIVITY;
try {
// first, figure out which PRT we are dealing with. The
// port addresses in the PrinterInfo are base address of each
// PRT sorted in the adapter order
if ((USHORT)iPort == PrtInfo->prt_PortAddr[0] + DATA_PORT_OFFSET)
adapter = 0;
else if ((USHORT)iPort == PrtInfo->prt_PortAddr[1] + DATA_PORT_OFFSET)
adapter = 1;
else if ((USHORT)iPort == PrtInfo->prt_PortAddr[2] + DATA_PORT_OFFSET)
adapter = 2;
else {
// something must be wrong in our code, better check it out
ASSERT(FALSE);
return FALSE;
}
if (PRT_MODE_DIRECT_IO == PrtInfo->prt_Mode[adapter]){
PrtInfo->prt_Buffer[adapter][PrtInfo->prt_BytesInBuffer[adapter]] = (UCHAR)TrapFrame->Eax;
// buffer full, then flush it out
if (++PrtInfo->prt_BytesInBuffer[adapter] >= PRT_DATA_BUFFER_SIZE){
VdmFlushPrinterWriteData(adapter);
}
TrapFrame->Eip += cbInstructionSize;
}
else
Status = STATUS_ILLEGAL_INSTRUCTION;
} except(EXCEPTION_EXECUTE_HANDLER) {
Status = GetExceptionCode();
}
return(NT_SUCCESS(Status));
}
NTSTATUS
VdmFlushPrinterWriteData(USHORT adapter)
{
KIRQL OldIrql;
PVDM_TIB VdmTib;
PVDM_PRINTER_INFO PrtInfo;
PIO_STATUS_BLOCK IoStatusBlock;
NTSTATUS Status;
PAGED_CODE();
Status = VdmpGetVdmTib(&VdmTib, VDMTIB_KMODE);
if (!NT_SUCCESS(Status)) {
return(FALSE);
}
PrtInfo = &VdmTib->PrinterInfo;
IoStatusBlock = (PIO_STATUS_BLOCK)&VdmTib->TempArea1;
try {
if (PrtInfo->prt_Handle[adapter] &&
PrtInfo->prt_BytesInBuffer[adapter] &&
PRT_MODE_DIRECT_IO == PrtInfo->prt_Mode[adapter]) {
OldIrql = KeGetCurrentIrql();
KeLowerIrql(PASSIVE_LEVEL);
Status = NtDeviceIoControlFile(PrtInfo->prt_Handle[adapter],
NULL, // notification event
NULL, // APC routine
NULL, // APC context
IoStatusBlock,
IOCTL_VDM_PAR_WRITE_DATA_PORT,
&PrtInfo->prt_Buffer[adapter][0],
PrtInfo->prt_BytesInBuffer[adapter],
NULL,
0
);
PrtInfo->prt_BytesInBuffer[adapter] = 0;
KeRaiseIrql(OldIrql, &OldIrql);
if (!NT_SUCCESS(Status)) {
#ifdef DBG
DbgPrint("IOCTL_VDM_PAR_WRITE_DATA_PORT failed %lx %x\n",
Status, IoStatusBlock->Status);
#endif
Status = IoStatusBlock->Status;
}
}
else
Status = STATUS_INVALID_PARAMETER;
} except (EXCEPTION_EXECUTE_HANDLER) {
Status = GetExceptionCode();
}
return Status;
}
NTSTATUS
VdmpPrinterInitialize(
PVOID ServiceData
)
/*++
Routine Description:
This routine probes and caches the data associated with kernel
mode printer emulation.
Arguments:
ServiceData - Not used.
Return Value:
--*/
{
PUCHAR State, PrtStatus, Control, HostState;
PVDM_TIB VdmTib;
PVDM_PROCESS_OBJECTS VdmObjects;
NTSTATUS Status;
Status = STATUS_SUCCESS;
// Note: We only support two printers in the kernel.
Status = VdmpGetVdmTib(&VdmTib, VDMTIB_KMODE);
if (!NT_SUCCESS(Status)) {
return(FALSE);
}
try {
State = VdmTib->PrinterInfo.prt_State;
PrtStatus = VdmTib->PrinterInfo.prt_Status;
Control = VdmTib->PrinterInfo.prt_Control;
HostState = VdmTib->PrinterInfo.prt_HostState;
// Probe the locations for two printers
ProbeForWrite(
State,
2 * sizeof(UCHAR),
sizeof(UCHAR)
);
ProbeForWrite(
PrtStatus,
2 * sizeof(UCHAR),
sizeof(UCHAR)
);
ProbeForWrite(
Control,
2 * sizeof(UCHAR),
sizeof(UCHAR)
);
ProbeForWrite(
HostState,
2 * sizeof(UCHAR),
sizeof(UCHAR)
);
} except (EXCEPTION_EXECUTE_HANDLER) {
Status = GetExceptionCode();
}
if (NT_SUCCESS(Status)) {
VdmObjects = PsGetCurrentProcess()->VdmObjects;
VdmObjects->PrinterState = State;
VdmObjects->PrinterStatus = PrtStatus;
VdmObjects->PrinterControl = Control;
VdmObjects->PrinterHostState = HostState;
}
return Status;
}
NTSTATUS
VdmpPrinterDirectIoOpen(PVOID ServiceData)
{
PAGED_CODE();
return STATUS_SUCCESS;
}
NTSTATUS
VdmpPrinterDirectIoClose(PVOID ServiceData)
{
NTSTATUS Status;
PVDM_PRINTER_INFO PrtInfo;
USHORT Adapter;
PVDM_TIB VdmTib;
PAGED_CODE();
Status = STATUS_SUCCESS;
// first we fetch vdm tib and do some damage control in case
// this is bad user-mode memory
// PrtInfo points to a stricture
try {
VdmTib = NtCurrentTeb()->Vdm;
ProbeForWrite(VdmTib, sizeof(VDM_TIB), sizeof(UCHAR));
// now verify that servicedata ptr is valid
ProbeForRead(ServiceData, sizeof(USHORT), sizeof(UCHAR));
Adapter = *(USHORT *)ServiceData;
} except (ExSystemExceptionFilter()) {
Status = GetExceptionCode();
}
if (NULL == VdmTib || NULL == ServiceData) {
Status = STATUS_ACCESS_VIOLATION;
}
if (!NT_SUCCESS(Status)) {
return(Status);
}
PrtInfo =&VdmTib->PrinterInfo;
try {
if (Adapter < VDM_NUMBER_OF_LPT) {
if (PRT_MODE_DIRECT_IO == PrtInfo->prt_Mode[Adapter] &&
PrtInfo->prt_BytesInBuffer[Adapter]) {
Status = VdmFlushPrinterWriteData(Adapter);
}
}
else {
Status = STATUS_INVALID_PARAMETER;
}
} except (EXCEPTION_EXECUTE_HANDLER) {
Status = GetExceptionCode();
}
return Status;
}