|
|
/*++
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; }
|