You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
291 lines
7.9 KiB
291 lines
7.9 KiB
/*++
|
|
|
|
Copyright (c) 1992 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
rdwr.c
|
|
|
|
Abstract:
|
|
|
|
This module contains routines for read and write for NTDOS. These
|
|
routines saves the switch to user mode. The BOP is handled in the
|
|
kernel for performance reasons. These routines are called only for
|
|
files. Local DOS devices and named pipe operations never come here.
|
|
|
|
Author:
|
|
|
|
Sudeep Bharati (Sudeepb) 04-Mar-1993
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
#include "vdmp.h"
|
|
|
|
VOID
|
|
NTFastDOSIO (
|
|
PKTRAP_FRAME TrapFrame,
|
|
ULONG IoType
|
|
);
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(PAGE,NTFastDOSIO)
|
|
#endif
|
|
|
|
#define EFLAGS_CF 0x1
|
|
#define EFLAGS_ZF 0x40
|
|
#define GETFILEPOINTER(hi,lo) (((ULONG)hi << 16) + (ULONG)lo)
|
|
#define GETHANDLE(hi,lo) (HANDLE)(((ULONG)hi << 16) + (ULONG)lo)
|
|
#define GETBUFFER(hi,lo) (((ULONG)hi << 4) + lo)
|
|
#define SVC_DEMFASTREAD 0x42
|
|
#define SVC_DEMFASTWRITE 0x43
|
|
#define CONSOLE_HANDLE_SIGNATURE 0x00000003
|
|
#define CONSOLE_HANDLE(HANDLE) (((ULONG)(HANDLE) & CONSOLE_HANDLE_SIGNATURE) == CONSOLE_HANDLE_SIGNATURE)
|
|
#define STD_INPUT_HANDLE (ULONG)-10
|
|
#define STD_OUTPUT_HANDLE (ULONG)-11
|
|
#define STD_ERROR_HANDLE (ULONG)-12
|
|
|
|
VOID
|
|
NTFastDOSIO (
|
|
PKTRAP_FRAME TrapFrame,
|
|
ULONG IoType
|
|
)
|
|
{
|
|
HANDLE hFile;
|
|
PVOID lpBuf;
|
|
ULONG ulBX,ulSI;
|
|
LARGE_INTEGER Large;
|
|
PIO_STATUS_BLOCK IoStatusBlock;
|
|
PFILE_POSITION_INFORMATION CurrentPosition;
|
|
NTSTATUS Status;
|
|
ULONG CountToIO;
|
|
PFILE_END_OF_FILE_INFORMATION EndOfFile;
|
|
PVDM_TIB VdmTib;
|
|
KIRQL OldIrql;
|
|
|
|
PAGED_CODE();
|
|
|
|
//
|
|
// Clear CF flag and assume success
|
|
//
|
|
|
|
TrapFrame->EFlags &= ~EFLAGS_CF;
|
|
|
|
//
|
|
// Validate the request
|
|
//
|
|
|
|
if (IoType != SVC_DEMFASTREAD && IoType != SVC_DEMFASTWRITE) {
|
|
TrapFrame->EFlags |= EFLAGS_CF;
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Signal softpc that we are doing disk io for idle detection.
|
|
//
|
|
|
|
try {
|
|
*FIXED_NTVDMSTATE_LINEAR_PC_AT |= VDM_IDLEACTIVITY;
|
|
}
|
|
except (EXCEPTION_EXECUTE_HANDLER) {
|
|
ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
|
|
TrapFrame->EFlags |= EFLAGS_CF;
|
|
return;
|
|
}
|
|
|
|
Status = VdmpGetVdmTib(&VdmTib);
|
|
|
|
if (!NT_SUCCESS(Status)) { // vdmtib is bad
|
|
ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
|
|
TrapFrame->EFlags |= EFLAGS_CF;
|
|
return;
|
|
}
|
|
|
|
IoStatusBlock = (PIO_STATUS_BLOCK) &VdmTib->TempArea1;
|
|
CurrentPosition = (PFILE_POSITION_INFORMATION) &VdmTib->TempArea2;
|
|
EndOfFile = (PFILE_END_OF_FILE_INFORMATION) CurrentPosition;
|
|
|
|
try {
|
|
ProbeForWrite (IoStatusBlock, sizeof (IO_STATUS_BLOCK), sizeof (UCHAR));
|
|
ProbeForWrite (CurrentPosition, sizeof (FILE_END_OF_FILE_INFORMATION), sizeof (UCHAR));
|
|
}
|
|
except (EXCEPTION_EXECUTE_HANDLER) {
|
|
ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
|
|
TrapFrame->EFlags |= EFLAGS_CF;
|
|
return;
|
|
}
|
|
|
|
// Get the NT handle
|
|
hFile = GETHANDLE((TrapFrame->Eax & 0x0000ffff),(TrapFrame->Ebp & 0x0000ffff));
|
|
|
|
// advance ip past the bop instruction
|
|
// clear carry flag, assuming success
|
|
TrapFrame->Eip += 4;
|
|
ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
|
|
TrapFrame->EFlags &= ~EFLAGS_CF;
|
|
|
|
if (CONSOLE_HANDLE(hFile) ||
|
|
hFile == (HANDLE) STD_INPUT_HANDLE ||
|
|
hFile == (HANDLE) STD_OUTPUT_HANDLE ||
|
|
hFile == (HANDLE) STD_ERROR_HANDLE )
|
|
{
|
|
TrapFrame->EFlags |= EFLAGS_CF;
|
|
return;
|
|
}
|
|
|
|
// Get the IO buffer
|
|
lpBuf = (PVOID) GETBUFFER(TrapFrame->V86Ds, (TrapFrame->Edx & 0x0000ffff));
|
|
|
|
// Get the Count
|
|
CountToIO = TrapFrame->Ecx & 0x0000ffff;
|
|
|
|
// Get Seek Parameters
|
|
ulBX = TrapFrame->Ebx & 0x0000ffff;
|
|
ulSI = TrapFrame->Esi & 0x0000ffff;
|
|
|
|
|
|
//
|
|
// Lower Irql to PASSIVE_LEVEL for io system
|
|
//
|
|
|
|
OldIrql = KeGetCurrentIrql();
|
|
|
|
KeLowerIrql (PASSIVE_LEVEL);
|
|
|
|
//
|
|
// Check if we need to seek
|
|
//
|
|
|
|
if (!(TrapFrame->EFlags & EFLAGS_ZF)) {
|
|
|
|
Large = RtlConvertUlongToLargeInteger(GETFILEPOINTER(ulBX,ulSI));
|
|
|
|
try {
|
|
CurrentPosition->CurrentByteOffset = Large;
|
|
}
|
|
except(EXCEPTION_EXECUTE_HANDLER) {
|
|
goto ErrorExit; // we have caught an exception, error exit
|
|
}
|
|
|
|
Status = NtSetInformationFile (hFile,
|
|
IoStatusBlock,
|
|
CurrentPosition,
|
|
sizeof(FILE_POSITION_INFORMATION),
|
|
FilePositionInformation);
|
|
|
|
if (!NT_SUCCESS(Status)) {
|
|
goto ErrorExit;
|
|
}
|
|
|
|
try {
|
|
if (CurrentPosition->CurrentByteOffset.LowPart == -1) {
|
|
goto ErrorExit;
|
|
}
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
if (IoType == SVC_DEMFASTREAD) {
|
|
|
|
Status = NtReadFile (hFile,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
IoStatusBlock,
|
|
(PVOID)lpBuf,
|
|
CountToIO,
|
|
NULL,
|
|
NULL);
|
|
}
|
|
else {
|
|
|
|
if (CountToIO == 0) {
|
|
|
|
Status = NtQueryInformationFile (hFile,
|
|
IoStatusBlock,
|
|
CurrentPosition,
|
|
sizeof(FILE_POSITION_INFORMATION),
|
|
FilePositionInformation);
|
|
|
|
if (!NT_SUCCESS(Status)) {
|
|
goto ErrorExit;
|
|
}
|
|
|
|
try {
|
|
EndOfFile->EndOfFile = CurrentPosition->CurrentByteOffset;
|
|
}
|
|
except(EXCEPTION_EXECUTE_HANDLER) {
|
|
goto ErrorExit; // we have caught an exception, error exit
|
|
}
|
|
|
|
Status = NtSetInformationFile (hFile,
|
|
IoStatusBlock,
|
|
EndOfFile,
|
|
sizeof(FILE_END_OF_FILE_INFORMATION),
|
|
FileEndOfFileInformation);
|
|
|
|
if (NT_SUCCESS(Status)) {
|
|
KeRaiseIrql(OldIrql, &OldIrql);
|
|
return;
|
|
}
|
|
|
|
goto ErrorExit;
|
|
}
|
|
|
|
Status = NtWriteFile (hFile,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
IoStatusBlock,
|
|
(PVOID)lpBuf,
|
|
CountToIO,
|
|
NULL,
|
|
NULL);
|
|
}
|
|
|
|
if (Status == STATUS_PENDING) {
|
|
|
|
//
|
|
// Operation must complete before return & IoStatusBlock destroyed
|
|
//
|
|
|
|
Status = NtWaitForSingleObject (hFile, FALSE, NULL);
|
|
|
|
if ( NT_SUCCESS(Status)) {
|
|
try {
|
|
Status = IoStatusBlock->Status;
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
NOTHING;
|
|
}
|
|
}
|
|
}
|
|
|
|
KeRaiseIrql(OldIrql, &OldIrql);
|
|
|
|
if ( NT_SUCCESS(Status) ) {
|
|
TrapFrame->Eax &= 0xffff0000;
|
|
try {
|
|
TrapFrame->Eax |= (USHORT) IoStatusBlock->Information;
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
NOTHING;
|
|
}
|
|
}
|
|
else if (IoType == SVC_DEMFASTREAD && Status == STATUS_END_OF_FILE) {
|
|
TrapFrame->Eax &= 0xffff0000;
|
|
}
|
|
else {
|
|
ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
|
|
TrapFrame->EFlags |= EFLAGS_CF;
|
|
}
|
|
|
|
return;
|
|
|
|
|
|
ErrorExit:
|
|
KeRaiseIrql(OldIrql, &OldIrql);
|
|
ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
|
|
TrapFrame->EFlags |= EFLAGS_CF;
|
|
|
|
return;
|
|
}
|