Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

1268 lines
32 KiB

/*++
Copyright (c) 1989 Microsoft Corporation
Module Name:
NtFastIo.c
Abstract:
This module implements NT fastio routines.
Author:
Joe Linn [JoeLinn] 9-Nov-1994
Revision History:
--*/
#include "precomp.h"
#pragma hdrstop
//
// The local debug trace level
//
#define Dbg (DEBUG_TRACE_NTFASTIO)
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, RxFastIoRead)
#pragma alloc_text(PAGE, RxFastIoWrite)
#pragma alloc_text(PAGE, RxFastLock)
#pragma alloc_text(PAGE, RxFastUnlockAll)
#pragma alloc_text(PAGE, RxFastUnlockAllByKey)
#pragma alloc_text(PAGE, RxFastUnlockSingle)
#pragma alloc_text(PAGE, RxFastIoCheckIfPossible)
#pragma alloc_text(PAGE, RxFastQueryBasicInfo)
#pragma alloc_text(PAGE, RxFastQueryStdInfo)
#endif
//these declarations would be copied to fsrtl.h
BOOLEAN
FsRtlCopyRead2 (
IN PFILE_OBJECT FileObject,
IN PLARGE_INTEGER FileOffset,
IN ULONG Length,
IN BOOLEAN Wait,
IN ULONG LockKey,
OUT PVOID Buffer,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject,
IN ULONG_PTR TopLevelIrpValue
);
BOOLEAN
FsRtlCopyWrite2 (
IN PFILE_OBJECT FileObject,
IN PLARGE_INTEGER FileOffset,
IN ULONG Length,
IN BOOLEAN Wait,
IN ULONG LockKey,
IN PVOID Buffer,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject,
IN ULONG_PTR TopLevelIrpValue
);
BOOLEAN
RxFastIoRead (
IN PFILE_OBJECT FileObject,
IN PLARGE_INTEGER FileOffset,
IN ULONG Length,
IN BOOLEAN Wait,
IN ULONG LockKey,
OUT PVOID Buffer,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
{
BOOLEAN ReturnValue;
RX_TOPLEVELIRP_CONTEXT TopLevelContext;
PAGED_CODE();
RxDbgTrace(+1, Dbg, ("RxFastIoRead\n"));
RxLog(("FastRead %lx:%lx:%lx",FileObject,FileObject->FsContext,FileObject->FsContext2));
RxLog(("------>> %lx@%lx %lx",Length,FileOffset->LowPart,FileOffset->HighPart));
RxWmiLog(LOG,
RxFastIoRead_1,
LOGPTR(FileObject)
LOGPTR(FileObject->FsContext)
LOGPTR(FileObject->FsContext2)
LOGULONG(Length)
LOGULONG(FileOffset->LowPart)
LOGULONG(FileOffset->HighPart));
ASSERT(RxIsThisTheTopLevelIrp(NULL));
RxInitializeTopLevelIrpContext(
&TopLevelContext,
((PIRP)FSRTL_FAST_IO_TOP_LEVEL_IRP),
(PRDBSS_DEVICE_OBJECT)DeviceObject);
ReturnValue = FsRtlCopyRead2 (
FileObject,
FileOffset,
Length,
Wait,
LockKey,
Buffer,
IoStatus,
DeviceObject,
(ULONG_PTR)(&TopLevelContext)
);
RxDbgTrace(-1, Dbg, ("RxFastIoRead ReturnValue=%x\n", ReturnValue));
if (ReturnValue) {
RxLog(
("FastReadYes %lx ret %lx:%lx",
FileObject->FsContext2,IoStatus->Status,IoStatus->Information));
RxWmiLog(LOG,
RxFastIoRead_2,
LOGPTR(FileObject->FsContext2)
LOGULONG(IoStatus->Status)
LOGPTR(IoStatus->Information));
} else {
RxLog(("FastReadNo %lx",FileObject->FsContext2));
RxWmiLog(LOG,
RxFastIoRead_3,
LOGPTR(FileObject->FsContext2));
}
return ReturnValue;
}
BOOLEAN
RxFastIoWrite (
IN PFILE_OBJECT FileObject,
IN PLARGE_INTEGER FileOffset,
IN ULONG Length,
IN BOOLEAN Wait,
IN ULONG LockKey,
IN PVOID Buffer,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
{
BOOLEAN ReturnValue;
RX_TOPLEVELIRP_CONTEXT TopLevelContext;
PSRV_OPEN SrvOpen;
PAGED_CODE();
RxDbgTrace(+1, Dbg, ("RxFastIoWrite\n"));
SrvOpen = ((PFOBX)(FileObject->FsContext2))->SrvOpen;
if (FlagOn(SrvOpen->Flags,SRVOPEN_FLAG_DONTUSE_WRITE_CACHEING)) {
//if this flag is set, we have to treat this as an unbuffered Io....sigh.
RxDbgTrace(-1, Dbg, ("RxFastIoWrite DONTUSE_WRITE_CACHEING...failing\n"));
return FALSE;
}
ASSERT(RxIsThisTheTopLevelIrp(NULL));
RxInitializeTopLevelIrpContext(
&TopLevelContext,
((PIRP)FSRTL_FAST_IO_TOP_LEVEL_IRP),
(PRDBSS_DEVICE_OBJECT)DeviceObject);
ReturnValue = FsRtlCopyWrite2 (
FileObject,
FileOffset,
Length,
Wait,
LockKey,
Buffer,
IoStatus,
DeviceObject,
(ULONG_PTR)(&TopLevelContext)
);
RxDbgTrace(-1, Dbg, ("RxFastIoWrite ReturnValue=%x\n", ReturnValue));
if (ReturnValue) {
RxLog(
("FWY %lx OLP: %lx SLP: %lx IOSB %lx:%lx",
FileObject->FsContext2,
FileOffset->LowPart,
SrvOpen->pFcb->Header.FileSize.LowPart,
IoStatus->Status,
IoStatus->Information));
RxWmiLog(LOG,
RxFastIoWrite_1,
LOGPTR(FileObject->FsContext2)
LOGULONG(FileOffset->LowPart)
LOGULONG(SrvOpen->pFcb->Header.FileSize.LowPart)
LOGULONG(IoStatus->Status)
LOGPTR(IoStatus->Information));
} else {
RxLog(("FastWriteNo %lx",FileObject->FsContext2));
RxWmiLog(LOG,
RxFastIoWrite_2,
LOGPTR(FileObject->FsContext2));
}
return ReturnValue;
}
BOOLEAN
RxFastLock (
IN PFILE_OBJECT FileObject,
IN PLARGE_INTEGER FileOffset,
IN PLARGE_INTEGER Length,
PEPROCESS ProcessId,
ULONG Key,
BOOLEAN FailImmediately,
BOOLEAN ExclusiveLock,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This is a call back routine for doing the fast lock call.
Arguments:
FileObject - Supplies the file object used in this operation
FileOffset - Supplies the file offset used in this operation
Length - Supplies the length used in this operation
ProcessId - Supplies the process ID used in this operation
Key - Supplies the key used in this operation
FailImmediately - Indicates if the request should fail immediately
if the lock cannot be granted.
ExclusiveLock - Indicates if this is a request for an exclusive or
shared lock
IoStatus - Receives the Status if this operation is successful
Return Value:
BOOLEAN - TRUE if this operation completed and FALSE if caller
needs to take the long route.
--*/
{
BOOLEAN Results;
PFCB Fcb = (PFCB)(FileObject->FsContext); //need a macro
PAGED_CODE();
RxDbgTrace(+1, Dbg, ("RxFastLock\n", 0));
//
// Decode the type of file object we're being asked to process and make
// sure it is only a user file open.
//
if (NodeType(Fcb) != RDBSS_NTC_STORAGE_TYPE_FILE) {
IoStatus->Status = STATUS_INVALID_PARAMETER;
IoStatus->Information = 0;
RxDbgTrace(-1, Dbg, ("RxFastLock -> TRUE (RxStatus(INVALID_PARAMETER))\n", 0));
return TRUE;
}
RxDbgTrace(-1, Dbg, ("RxFastLock -> FALSE (fastlocks not yet implemented)\n", 0));
return FALSE; //stuff past here has been massaged but not tested
//
// Acquire exclusive access to the Fcb this operation can always wait; you need
// the resource to synchronize with oplock breaks
//
FsRtlEnterFileSystem();
ExAcquireResourceSharedLite( Fcb->Header.Resource, TRUE );
try {
//
// We check whether we can proceed
// based on the state of the file oplocks.
//
if (!FsRtlOplockIsFastIoPossible( &(Fcb)->Specific.Fcb.Oplock )) {
try_return( Results = FALSE );
}
//
// Now call the FsRtl routine to do the actual processing of the
// Lock request
//
if (Results = FsRtlFastLock(
&Fcb->Specific.Fcb.FileLock,
FileObject,
FileOffset,
Length,
ProcessId,
Key,
FailImmediately,
ExclusiveLock,
IoStatus,
NULL,
FALSE )) {
//
// Set the flag indicating if Fast I/O is possible
//
//Fcb->Header.IsFastIoPossible = RxIsFastIoPossible( Fcb );
}
try_exit: NOTHING;
} finally {
DebugUnwind( RxFastLock );
//
// Release the Fcb, and return to our caller
//
ExReleaseResourceLite( (Fcb)->Header.Resource );
FsRtlExitFileSystem();
RxDbgTrace(-1, Dbg, ("RxFastLock -> %08lx\n", Results));
}
return Results;
}
BOOLEAN
RxFastUnlockSingle (
IN PFILE_OBJECT FileObject,
IN PLARGE_INTEGER FileOffset,
IN PLARGE_INTEGER Length,
PEPROCESS ProcessId,
ULONG Key,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This is a call back routine for doing the fast unlock single call.
Arguments:
FileObject - Supplies the file object used in this operation
FileOffset - Supplies the file offset used in this operation
Length - Supplies the length used in this operation
ProcessId - Supplies the process ID used in this operation
Key - Supplies the key used in this operation
Status - Receives the Status if this operation is successful
Return Value:
BOOLEAN - TRUE if this operation completed and FALSE if caller
needs to take the long route.
--*/
{
BOOLEAN Results;
PFCB Fcb = (PFCB)(FileObject->FsContext); //need macro
PAGED_CODE();
RxDbgTrace(+1, Dbg, ("RxFastUnlockSingle\n", 0));
IoStatus->Information = 0;
//
// Decode the type of file object we're being asked to process and make sure
// it is only a user file open
//
if (NodeType(Fcb) != RDBSS_NTC_STORAGE_TYPE_FILE) {
IoStatus->Status = STATUS_INVALID_PARAMETER;
RxDbgTrace(-1, Dbg, ("RxFastUnlockSingle -> TRUE (RxStatus(INVALID_PARAMETER))\n", 0));
return TRUE;
}
RxDbgTrace(-1, Dbg, ("RxFastUnlockSingle -> FALSE (fastlocks not yet implemented)\n", 0));
return FALSE; //stuff past here has been massaged but not tested
//
// Acquire exclusive access to the Fcb this operation can always wait
//
FsRtlEnterFileSystem();
ExAcquireResourceSharedLite( Fcb->Header.Resource, TRUE );
try {
//
// We check whether we can proceed based on the state of the file oplocks.
//
if (!FsRtlOplockIsFastIoPossible( &(Fcb)->Specific.Fcb.Oplock )) {
try_return( Results = FALSE );
}
//
// Now call the FsRtl routine to do the actual processing of the
// Lock request. The call will always succeed.
//
Results = TRUE;
IoStatus->Status = FsRtlFastUnlockSingle(
&Fcb->Specific.Fcb.FileLock,
FileObject,
FileOffset,
Length,
ProcessId,
Key,
NULL,
FALSE );
//
// Set the flag indicating if Fast I/O is possible
//
//Fcb->Header.IsFastIoPossible = RxIsFastIoPossible( Fcb );
try_exit: NOTHING;
} finally {
DebugUnwind( RxFastUnlockSingle );
//
// Release the Fcb, and return to our caller
//
ExReleaseResourceLite( (Fcb)->Header.Resource );
FsRtlExitFileSystem();
RxDbgTrace(-1, Dbg, ("RxFastUnlockSingle -> %08lx\n", Results));
}
return Results;
}
BOOLEAN
RxFastUnlockAll (
IN PFILE_OBJECT FileObject,
PEPROCESS ProcessId,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This is a call back routine for doing the fast unlock all call.
Arguments:
FileObject - Supplies the file object used in this operation
ProcessId - Supplies the process ID used in this operation
Status - Receives the Status if this operation is successful
Return Value:
BOOLEAN - TRUE if this operation completed and FALSE if caller
needs to take the long route.
--*/
{
BOOLEAN Results;
PFCB Fcb = (PFCB)(FileObject->FsContext); //need macro
PAGED_CODE();
RxDbgTrace(+1, Dbg, ("RxFastUnlockAll\n", 0));
IoStatus->Information = 0;
//
// Decode the type of file object we're being asked to process and make sure
// it is only a user file open.
//
if (NodeType(Fcb) != RDBSS_NTC_STORAGE_TYPE_FILE) {
IoStatus->Status = STATUS_INVALID_PARAMETER;
RxDbgTrace(-1, Dbg, ("RxFastUnlockAll -> TRUE (RxStatus(INVALID_PARAMETER))\n", 0));
return TRUE;
}
RxDbgTrace(-1, Dbg, ("RxFastUnlockAll -> FALSE (fastlocks not yet implemented)\n", 0));
return FALSE; //stuff past here has been massaged but not tested
//
// Acquire exclusive access to the Fcb this operation can always wait
//
FsRtlEnterFileSystem();
(VOID) ExAcquireResourceSharedLite( Fcb->Header.Resource, TRUE );
try {
//
// We check whether we can proceed based on the state of the file oplocks.
//
if (!FsRtlOplockIsFastIoPossible( &(Fcb)->Specific.Fcb.Oplock )) {
try_return( Results = FALSE );
}
//
// Now call the FsRtl routine to do the actual processing of the
// Lock request. The call will always succeed.
//
Results = TRUE;
IoStatus->Status = FsRtlFastUnlockAll(
&Fcb->Specific.Fcb.FileLock,
FileObject,
ProcessId,
NULL );
//
// Set the flag indicating if Fast I/O is possible
//
//Fcb->Header.IsFastIoPossible = RxIsFastIoPossible( Fcb );
try_exit: NOTHING;
} finally {
DebugUnwind( RxFastUnlockAll );
//
// Release the Fcb, and return to our caller
//
ExReleaseResourceLite( (Fcb)->Header.Resource );
FsRtlExitFileSystem();
RxDbgTrace(-1, Dbg, ("RxFastUnlockAll -> %08lx\n", Results));
}
return Results;
}
BOOLEAN
RxFastUnlockAllByKey (
IN PFILE_OBJECT FileObject,
PVOID ProcessId,
ULONG Key,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This is a call back routine for doing the fast unlock all by key call.
Arguments:
FileObject - Supplies the file object used in this operation
ProcessId - Supplies the process ID used in this operation
Key - Supplies the key used in this operation
Status - Receives the Status if this operation is successful
Return Value:
BOOLEAN - TRUE if this operation completed and FALSE if caller
needs to take the long route.
--*/
{
BOOLEAN Results;
PFCB Fcb = (PFCB)(FileObject->FsContext); //need macro
PAGED_CODE();
RxDbgTrace(+1, Dbg, ("RxFastUnlockAllByKey\n", 0));
IoStatus->Information = 0;
//
// Decode the type of file object we're being asked to process and make sure
// it is only a user file open.
//
if (NodeType(Fcb) != RDBSS_NTC_STORAGE_TYPE_FILE) {
IoStatus->Status = STATUS_INVALID_PARAMETER;
RxDbgTrace(-1, Dbg, ("RxFastUnlockAll -> TRUE (RxStatus(INVALID_PARAMETER))\n", 0));
return TRUE;
}
RxDbgTrace(-1, Dbg, ("RxFastUnlockAll -> FALSE (fastlocks not yet implemented)\n", 0));
return FALSE; //stuff past here has been massaged but not tested
//
// Acquire exclusive access to the Fcb this operation can always wait
//
FsRtlEnterFileSystem();
(VOID) ExAcquireResourceSharedLite( Fcb->Header.Resource, TRUE );
try {
//
// We check whether we can proceed based on the state of the file oplocks.
//
if (!FsRtlOplockIsFastIoPossible( &(Fcb)->Specific.Fcb.Oplock )) {
try_return( Results = FALSE );
}
//
// Now call the FsRtl routine to do the actual processing of the
// Lock request. The call will always succeed.
//
Results = TRUE;
IoStatus->Status = FsRtlFastUnlockAllByKey(
&Fcb->Specific.Fcb.FileLock,
FileObject,
ProcessId,
Key,
NULL );
//
// Set the flag indicating if Fast I/O is possible
//
//Fcb->Header.IsFastIoPossible = RxIsFastIoPossible( Fcb );
try_exit: NOTHING;
} finally {
DebugUnwind( RxFastUnlockAllByKey );
//
// Release the Fcb, and return to our caller
//
ExReleaseResourceLite( (Fcb)->Header.Resource );
FsRtlExitFileSystem();
RxDbgTrace(-1, Dbg, ("RxFastUnlockAllByKey -> %08lx\n", Results));
}
return Results;
}
#define RxLogAndReturnFalse(x) { \
RxLog(("CheckFast fail %lx %s",FileObject,x)); \
RxWmiLog(LOG, \
RxFastIoCheckIfPossible, \
LOGPTR(FileObject) \
LOGARSTR(x)); \
return FALSE; \
}
BOOLEAN
RxFastIoCheckIfPossible (
IN PFILE_OBJECT FileObject,
IN PLARGE_INTEGER FileOffset,
IN ULONG Length,
IN BOOLEAN Wait,
IN ULONG LockKey,
IN BOOLEAN CheckForReadOperation,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This routine checks if fast i/o is possible for a read/write operation
Arguments:
FileObject - Supplies the file object used in the query
FileOffset - Supplies the starting byte offset for the read/write operation
Length - Supplies the length, in bytes, of the read/write operation
Wait - Indicates if we can wait
LockKey - Supplies the lock key
CheckForReadOperation - Indicates if this is a check for a read or write
operation
IoStatus - Receives the status of the operation if our return value is
FastIoReturnError
Return Value:
BOOLEAN - TRUE if fast I/O is possible and FALSE if the caller needs
to take the long route.
--*/
{
PFCB Fcb = (PFCB)(FileObject->FsContext);
PFOBX Fobx = (PFOBX)(FileObject->FsContext2);
PSRV_OPEN pSrvOpen = Fobx->SrvOpen;
LARGE_INTEGER LargeLength;
PAGED_CODE();
if (NodeType(Fcb) != RDBSS_NTC_STORAGE_TYPE_FILE) {
RxLogAndReturnFalse("notfile");
}
if (!FsRtlOplockIsFastIoPossible( &Fcb->Specific.Fcb.Oplock )) {
RxLogAndReturnFalse("cnd/oplock");
}
if (FileObject->DeletePending) {
RxLogAndReturnFalse("delpend");
}
if (Fcb->NonPaged->OutstandingAsyncWrites != 0) {
RxLogAndReturnFalse("asynW");
}
if (FlagOn(pSrvOpen->Flags,SRVOPEN_FLAG_ORPHANED)) {
RxLogAndReturnFalse("srvopen orphaned");
}
if (FlagOn(Fcb->FcbState,FCB_STATE_ORPHANED)) {
RxLogAndReturnFalse("orphaned");
}
if (BooleanFlagOn(
pSrvOpen->Flags,
SRVOPEN_FLAG_BUFFERING_STATE_CHANGE_PENDING)) {
RxLogAndReturnFalse("buf state change");
}
if (FlagOn(pSrvOpen->Flags,SRVOPEN_FLAG_FILE_RENAMED) ||
FlagOn(pSrvOpen->Flags,SRVOPEN_FLAG_FILE_DELETED)) {
RxLogAndReturnFalse("ren/del");
}
// Ensure that all pending buffering state change requests are processed
// before letting the operation through.
FsRtlEnterFileSystem();
RxProcessChangeBufferingStateRequestsForSrvOpen(pSrvOpen);
FsRtlExitFileSystem();
LargeLength.QuadPart = Length;
//
// Based on whether this is a read or write operation we call
// fsrtl check for read/write
//
if (CheckForReadOperation) {
if (!FlagOn(Fcb->FcbState,FCB_STATE_READCACHEING_ENABLED)) {
RxLogAndReturnFalse("notreadC");
}
if (!FsRtlFastCheckLockForRead(
&Fcb->Specific.Fcb.FileLock,
FileOffset,
&LargeLength,
LockKey,
FileObject,
PsGetCurrentProcess() )) {
RxLogAndReturnFalse("readlock");
}
} else {
if (!FlagOn(Fcb->FcbState,FCB_STATE_WRITECACHEING_ENABLED)) {
RxLogAndReturnFalse("notwriteC");
}
//
// Also check for a write-protected volume here.
//
if (!FsRtlFastCheckLockForWrite(
&Fcb->Specific.Fcb.FileLock,
FileOffset,
&LargeLength,
LockKey,
FileObject,
PsGetCurrentProcess() )) {
RxLogAndReturnFalse("writelock");
}
}
// RxLog(("IoPossible %lx",FileObject));
return TRUE;
}
BOOLEAN
RxFastIoDeviceControl(
IN struct _FILE_OBJECT *FileObject,
IN BOOLEAN Wait,
IN PVOID InputBuffer OPTIONAL,
IN ULONG InputBufferLength,
OUT PVOID OutputBuffer OPTIONAL,
IN ULONG OutputBufferLength,
IN ULONG IoControlCode,
OUT PIO_STATUS_BLOCK IoStatus,
IN struct _DEVICE_OBJECT *DeviceObject
)
/*++
Routine Description:
This routine is for the fast device control call.
Arguments:
FileObject - Supplies the file object used in this operation
Wait - Indicates if we are allowed to wait for the information
InputBuffer - Supplies the input buffer
InputBufferLength - the length of the input buffer
OutputBuffer - the output buffer
OutputBufferLength - the length of the output buffer
IoControlCode - the IO control code
IoStatus - Receives the final status of the operation
DeviceObject - the associated device object
Return Value:
BOOLEAN - TRUE if the operation succeeded and FALSE if the caller
needs to take the long route.
Notes:
The following IO control requests are handled in the first path
IOCTL_LMR_ARE_FILE_OBJECTS_ON_SAME_SERVER
InputBuffer - pointer to the other file object
InputBufferLength - length in bytes of a pointer.
OutputBuffer - not used
OutputBufferLength - not used
IoStatus --
IoStatus.Status set to STATUS_SUCCESS if both the file objects are
on the same server, otherwise set to STATUS_NOT_SAME_DEVICE
This is a kernel mode interface only.
--*/
{
BOOLEAN FastIoSucceeded;
switch (IoControlCode) {
case IOCTL_LMR_ARE_FILE_OBJECTS_ON_SAME_SERVER :
{
FastIoSucceeded = TRUE;
try {
if (InputBufferLength == sizeof(HANDLE)) {
PFCB pFcb1,pFcb2;
HANDLE hFile;
PFILE_OBJECT pFileObject2;
NTSTATUS Status;
pFcb1 = (PFCB)FileObject->FsContext;
RtlCopyMemory(
&hFile,
InputBuffer,
sizeof(HANDLE));
Status = ObReferenceObjectByHandle(
hFile,
FILE_ANY_ACCESS,
*IoFileObjectType,
UserMode,
&pFileObject2,
NULL);
if ((Status == STATUS_SUCCESS)) {
if(pFileObject2->DeviceObject == DeviceObject) {
pFcb2 = (PFCB)pFileObject2->FsContext;
if ((pFcb2 != NULL) &&
(NodeTypeIsFcb(pFcb2))) {
if (pFcb1->pNetRoot->pSrvCall == pFcb2->pNetRoot->pSrvCall) {
IoStatus->Status = STATUS_SUCCESS;
} else {
IoStatus->Status = STATUS_NOT_SAME_DEVICE;
}
} else {
Status = STATUS_INVALID_PARAMETER;
}
} else {
Status = STATUS_INVALID_PARAMETER;
}
ObDereferenceObject(pFileObject2);
} else {
IoStatus->Status = STATUS_INVALID_PARAMETER;
}
} else {
IoStatus->Status = STATUS_INVALID_PARAMETER;
}
} except( EXCEPTION_EXECUTE_HANDLER ) {
// The I/O request was not handled successfully, abort the I/O request with
// the error status that we get back from the execption code
IoStatus->Status = STATUS_INVALID_PARAMETER;
FastIoSucceeded = TRUE;
}
}
break;
default:
{
FastIoSucceeded = FALSE;
}
}
return FastIoSucceeded;
}
BOOLEAN
RxFastQueryBasicInfo (
IN PFILE_OBJECT FileObject,
IN BOOLEAN Wait,
IN OUT PFILE_BASIC_INFORMATION Buffer,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This routine is for the fast query call for basic file information.
Arguments:
FileObject - Supplies the file object used in this operation
Wait - Indicates if we are allowed to wait for the information
Buffer - Supplies the output buffer to receive the basic information
IoStatus - Receives the final status of the operation
Return Value:
BOOLEAN - TRUE if the operation succeeded and FALSE if the caller
needs to take the long route.
--*/
{
BOOLEAN Results = FALSE;
PFCB Fcb = (PFCB)(FileObject->FsContext);
PFOBX Fobx = (PFOBX)(FileObject->FsContext2);
NODE_TYPE_CODE TypeOfOpen = NodeType(Fcb);
BOOLEAN FcbAcquired = FALSE;
PAGED_CODE();
//
// Determine the type of open for the input file object and only accept
// the user file or directory open
//
if ((TypeOfOpen != RDBSS_NTC_STORAGE_TYPE_FILE) &&
(TypeOfOpen != RDBSS_NTC_STORAGE_TYPE_DIRECTORY)) {
return Results;
}
FsRtlEnterFileSystem();
//
// Get access to the Fcb but only if it is not the paging file
//
if (!FlagOn( Fcb->FcbState, FCB_STATE_PAGING_FILE )) {
if (!ExAcquireResourceSharedLite( Fcb->Header.Resource, Wait )) {
FsRtlExitFileSystem();
return Results;
}
FcbAcquired = TRUE;
}
try {
//
// Set it to indicate that the query is a normal file.
// Later we might overwrite the attribute.
//
Buffer->FileAttributes = FILE_ATTRIBUTE_NORMAL;
//
// If the fcb is not the root dcb then we will fill in the
// buffer otherwise it is all setup for us.
//
if (NodeType(Fcb) != RDBSS_NTC_ROOT_DCB) {
//
// Extract the data and fill in the non zero fields of the output
// buffer
//
Buffer->LastWriteTime = Fcb->LastWriteTime;
Buffer->ChangeTime = Fcb->LastChangeTime;
Buffer->CreationTime = Fcb->CreationTime;
Buffer->LastAccessTime = Fcb->LastAccessTime;
//
// Zero out the field we don't support.
//
Buffer->ChangeTime = RxLargeZero;
if (Fcb->Attributes != 0) {
Buffer->FileAttributes = Fcb->Attributes;
} else {
Buffer->FileAttributes = FILE_ATTRIBUTE_NORMAL;
}
} else {
Buffer->FileAttributes = FILE_ATTRIBUTE_DIRECTORY;
}
//
// If the temporary flag is set, then set it in the buffer.
//
if (FlagOn( Fcb->FcbState, FCB_STATE_TEMPORARY )) {
SetFlag( Buffer->FileAttributes, FILE_ATTRIBUTE_TEMPORARY );
}
IoStatus->Status = STATUS_SUCCESS;
IoStatus->Information = sizeof(FILE_BASIC_INFORMATION);
Results = TRUE;
} finally {
if (FcbAcquired) { ExReleaseResourceLite( Fcb->Header.Resource ); }
FsRtlExitFileSystem();
}
//
// And return to our caller
//
return Results;
}
BOOLEAN
RxFastQueryStdInfo (
IN PFILE_OBJECT FileObject,
IN BOOLEAN Wait,
IN OUT PFILE_STANDARD_INFORMATION Buffer,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This routine is for the fast query call for standard file information.
Arguments:
FileObject - Supplies the file object used in this operation
Wait - Indicates if we are allowed to wait for the information
Buffer - Supplies the output buffer to receive the basic information
IoStatus - Receives the final status of the operation
Return Value:
BOOLEAN - TRUE if the operation succeeded and FALSE if the caller
needs to take the long route.
--*/
{
BOOLEAN Results = FALSE;
PFCB Fcb = (PFCB)(FileObject->FsContext);
PFOBX Fobx = (PFOBX)(FileObject->FsContext2);
NODE_TYPE_CODE TypeOfOpen = NodeType(Fcb);
BOOLEAN FcbAcquired = FALSE;
PAGED_CODE();
//
// Determine the type of open for the input file object and only accept
// the user file or directory open
//
if ((TypeOfOpen != RDBSS_NTC_STORAGE_TYPE_FILE) &&
(TypeOfOpen != RDBSS_NTC_STORAGE_TYPE_DIRECTORY)) {
return Results;
}
//
// Get access to the Fcb but only if it is not the paging file
//
FsRtlEnterFileSystem();
if (!FlagOn( Fcb->FcbState, FCB_STATE_PAGING_FILE )) {
if (!ExAcquireResourceSharedLite( Fcb->Header.Resource, Wait )) {
FsRtlExitFileSystem();
return Results;
}
FcbAcquired = TRUE;
}
try {
Buffer->NumberOfLinks = 1;
Buffer->DeletePending = BooleanFlagOn( Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE );
//
// Case on whether this is a file or a directory, and extract
// the information and fill in the fcb/dcb specific parts
// of the output buffer.
//
if (NodeType(Fcb) == RDBSS_NTC_FCB) {
//
// If we don't alread know the allocation size, we cannot look
// it up in the fast path.
//
if (Fcb->Header.AllocationSize.LowPart == 0xffffffff) {
try_return( Results );
}
Buffer->AllocationSize = Fcb->Header.AllocationSize;
Buffer->EndOfFile = Fcb->Header.FileSize;
Buffer->Directory = FALSE;
} else {
Buffer->AllocationSize = RxLargeZero;
Buffer->EndOfFile = RxLargeZero;
Buffer->Directory = TRUE;
}
IoStatus->Status = STATUS_SUCCESS;
IoStatus->Information = sizeof(FILE_STANDARD_INFORMATION);
Results = TRUE;
try_exit: NOTHING;
} finally {
if (FcbAcquired) { ExReleaseResourceLite( Fcb->Header.Resource ); }
FsRtlExitFileSystem();
}
//
// And return to our caller
//
return Results;
}