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