Windows2000/private/ntos/io/parse.c

1572 lines
70 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*++
Copyright (c) 1989-1993 Microsoft Corporation
Module Name:
parse.c
Abstract:
This module contains the code to implement the device object parse routine.
Author:
Darryl E. Havens (darrylh) 15-May-1988
Environment:
Kernel mode
Revision History:
--*/
#include "iop.h"
// Define macro to round up the size of a name for buffer optimization.
#define RoundNameSize( Length ) ( \
(Length < 64 - 8) ? 64 - 8 : \
(Length < 128 - 8) ? 128 - 8 :\
(Length < 256 - 8) ? 256 - 8 : Length )
#define IO_MAX_REMOUNT_REPARSE_ATTEMPTS 32
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, IopParseFile)
#pragma alloc_text(PAGE, IopParseDevice)
#pragma alloc_text(PAGE, IopQueryName)
#pragma alloc_text(PAGE, IopCheckBackupRestorePrivilege)
#endif
NTSTATUS IopCheckDeviceAndDriver(POPEN_PACKET op, PDEVICE_OBJECT parseDeviceObject)
{
NTSTATUS status;
KIRQL irql;
// Make sure that the device and its driver are really there and they are
// going to stay there. The object itself cannot go away just yet because
// the object management system has performed a reference which bumps the
// count of the number of reasons why the object must stick around.
// However, the driver could be attempting to unload itself, so perform
// this check. If the driver is being unloaded, then set the final status
// of the operation to "No such device" and return with a NULL file object
// pointer.
// Note that it is possible to "open" an exclusive device more than once
// provided that the caller is performing a relative open. This feature
// is how users "allocate" a device, and then use it to perform operations.
ExAcquireFastLock( &IopDatabaseLock, &irql );
if (parseDeviceObject->DeviceObjectExtension->ExtensionFlags &
(DOE_UNLOAD_PENDING | DOE_DELETE_PENDING | DOE_REMOVE_PENDING | DOE_REMOVE_PROCESSED | DOE_START_PENDING) ||
parseDeviceObject->Flags & DO_DEVICE_INITIALIZING) {
status = STATUS_NO_SUCH_DEVICE;
} else if (parseDeviceObject->Flags & DO_EXCLUSIVE && parseDeviceObject->ReferenceCount != 0 && op->RelatedFileObject == NULL && !(op->Options & IO_ATTACH_DEVICE)) {
status = STATUS_ACCESS_DENIED;
} else {
parseDeviceObject->ReferenceCount++;
status = STATUS_SUCCESS;
}
ExReleaseFastLock( &IopDatabaseLock, irql );
return status;
}
PVPB
IopCheckVpbMounted(
IN POPEN_PACKET op,
IN PDEVICE_OBJECT parseDeviceObject,
IN OUT PUNICODE_STRING RemainingName,
OUT PNTSTATUS status
)
{
PVPB vpb;
KIRQL irql;
BOOLEAN alertable;
// Loop here until the VPB_MOUNTED test can be passed while holding the
// VPB spinlock. After the mount succeeds, it is still necessary to acquire
// the spinlock to check that the VPB (which may be different from the one
// before the mount) is still mounted. If it is, then its reference count
// is incremented before releasing the spinlock.
ExAcquireFastLock( &IopVpbSpinLock, &irql );
alertable = (op->CreateOptions & FILE_SYNCHRONOUS_IO_ALERT) ? TRUE : FALSE;
while (!(parseDeviceObject->Vpb->Flags & VPB_MOUNTED)) {
ExReleaseFastLock( &IopVpbSpinLock, irql );
// Try to mount the volume, allowing only RAW to perform the mount if this is a DASD open.
*status = IopMountVolume( parseDeviceObject, (BOOLEAN) (!RemainingName->Length && !op->RelatedFileObject), FALSE, alertable );
// If the mount operation was unsuccessful, adjust the reference
// count for the device and return now.
if (!NT_SUCCESS( *status ) || *status == STATUS_USER_APC || *status == STATUS_ALERTED) {
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
if (!NT_SUCCESS( *status )) {
return NULL;
} else {
*status = STATUS_WRONG_VOLUME;
return NULL;
}
}
ExAcquireFastLock( &IopVpbSpinLock, &irql );
}
// Synchronize here with the file system to make sure that volumes do not
// go away while en route to the FS.
vpb = parseDeviceObject->Vpb;
// Check here that the VPB is not locked.
if (vpb->Flags & VPB_LOCKED) {
*status = STATUS_ACCESS_DENIED;
vpb = NULL;
} else {
vpb->ReferenceCount += 1;
}
ExReleaseFastLock( &IopVpbSpinLock, irql );
return vpb;
}
VOID IopDereferenceVpbAndFree(IN PVPB Vpb)
{
KIRQL irql;
PVPB vpb = (PVPB) NULL;
ExAcquireFastLock( &IopVpbSpinLock, &irql );
Vpb->ReferenceCount--;
if ((Vpb->ReferenceCount == 0) && (Vpb->RealDevice->Vpb != Vpb) && !(Vpb->Flags & VPB_PERSISTENT)) {
vpb = Vpb;
}
ExReleaseFastLock( &IopVpbSpinLock, irql );
if (vpb) {
ExFreePool( vpb );
}
}
NTSTATUS
IopParseDevice(
IN PVOID ParseObject,
IN PVOID ObjectType,
IN PACCESS_STATE AccessState,
IN KPROCESSOR_MODE AccessMode,
IN ULONG Attributes,
IN OUT PUNICODE_STRING CompleteName,
IN OUT PUNICODE_STRING RemainingName,
IN OUT PVOID Context OPTIONAL,
IN PSECURITY_QUALITY_OF_SERVICE SecurityQos OPTIONAL,
OUT PVOID *Object
)
/*++
Routine Description:
This routine interfaces to the NT Object Manager. It is invoked when
the object system is given the name of an entity to create or open and the
name translates to a device object. This routine is specified as the parse
routine for all device objects.
In the normal case of an NtCreateFile, the user specifies either the name
of a device or of a file. In the former situation, this routine is invoked
with a pointer to the device and a null ("") string. For this case, the
routine simply allocates an IRP, fills it in, and passes it to the driver
for the device. The driver will then perform whatever rudimentary functions
are necessary and will return a status code indicating whether an error was
incurred. This status code is remembered in the Open Packet (OP).
In the latter situation, the name string to be opened/created is non-null.
That is, it contains the remainder of the pathname to the file that is to
be opened or created. For this case, the routine allocates an IRP, fills
it in, and passes it to the driver for the device. The driver may then
need to take further action or it may complete the request immediately. If
it needs to perform some work asynchronously, then it can queue the request
and return a status of STATUS_PENDING. This allows this routine and its
caller to return to the user so that he can continue. Otherwise, the open/
create is basically finished.
If the driver supports symbolic links, then it is also possible for the
driver to return a new name. This name will be returned to the Object
Manager as a new name to look up. The parsing will then begin again from
the start.
It is also the responsibility of this routine to create a file object for
the file, if the name specifies a file. The file object's address is
returned to the NtCreateFile service through the OP.
Arguments:
ParseObject - Pointer to the device object the name translated into.
ObjectType - Type of the object being opened.
AccessState - Running security access state information for operation.
AccessMode - Access mode of the original caller.
Attributes - Attributes to be applied to the object.
CompleteName - Complete name of the object.
RemainingName - Remaining name of the object.
Context - Pointer to an Open Packet (OP) from NtCreateFile service.
SecurityQos - Optional security quality of service indicator.
Object - The address of a variable to receive the created file object, if any.
Return Value:
The function return value is one of the following:
a) Success - This indicates that the function succeeded and the object parameter contains the address of the created file object.
b) Error - This indicates that the file was not found or created and no file object was created.
c) Reparse - This indicates that the remaining name string has been replaced by a new name that is to be parsed.
--*/
{
#define COPY_ATTRIBUTES( n, b, s ) { \
(n)->CreationTime.QuadPart = (b)->CreationTime.QuadPart; \
(n)->LastAccessTime.QuadPart = (b)->LastAccessTime.QuadPart; \
(n)->LastWriteTime.QuadPart = (b)->LastWriteTime.QuadPart; \
(n)->ChangeTime.QuadPart = (b)->ChangeTime.QuadPart; \
(n)->AllocationSize.QuadPart = (s)->AllocationSize.QuadPart; \
(n)->EndOfFile.QuadPart = (s)->EndOfFile.QuadPart; \
(n)->FileAttributes = (b)->FileAttributes; }
PIRP irp;
PIO_STACK_LOCATION irpSp;
POPEN_PACKET op;
PFILE_OBJECT fileObject;
NTSTATUS status;
IO_STATUS_BLOCK ioStatus;
IO_SECURITY_CONTEXT securityContext;
PDEVICE_OBJECT deviceObject;
PDEVICE_OBJECT parseDeviceObject;
BOOLEAN directDeviceOpen;
PVPB vpb;
ACCESS_MASK desiredAccess;
PDUMMY_FILE_OBJECT localFileObject;
BOOLEAN realFileObjectRequired;
KPROCESSOR_MODE modeForPrivilegeCheck;
ULONG retryCount = 0;
BOOLEAN relativeVolumeOpen = FALSE; // True if opening a filesystem volume
PAGED_CODE();
reparse_loop:
// Assume failure by setting the returned object pointer to NULL.
*Object = (PVOID) NULL;
// Get the address of the Open Packet (OP).
op = Context;
// Ensure that this routine is actually being invoked because someone is
// attempting to open a device or a file through NtCreateFile. This code
// must be invoked from there (as opposed to some other random object
// create or open routine).
if (op == NULL || op->Type != IO_TYPE_OPEN_PACKET || op->Size != sizeof( OPEN_PACKET )) {
return STATUS_OBJECT_TYPE_MISMATCH;
}
// Obtain a pointer to the parse object as a device object, which is the actual type of the object anyway.
parseDeviceObject = (PDEVICE_OBJECT) ParseObject;
// If this is a relative open, then get the device on which the file
// is really being opened from the related file object and use that for
// the remainder of this function and for all operations performed on the file object that is about to be created.
if (op->RelatedFileObject) {
parseDeviceObject = op->RelatedFileObject->DeviceObject;
}
// Make sure that the device and its driver are really there and they are
// going to stay there. The object itself cannot go away just yet because
// the object management system has performed a reference which bumps the
// count of the number of reasons why the object must stick around.
// However, the driver could be attempting to unload itself, so perform
// this check. If the driver is being unloaded, then set the final status
// of the operation to "No such device" and return with a NULL file object pointer.
// Note that it is possible to "open" an exclusive device more than once
// provided that the caller is performing a relative open. This feature
// is how users "allocate" a device, and then use it to perform operations.
status = IopCheckDeviceAndDriver( op, parseDeviceObject );
if (!NT_SUCCESS(status)) {
return op->FinalStatus = status;
}
// Since ObOpenObjectByName is called without being passed
// any object type information, we need to map the generic
// bits in the DesiredAccess mask here. We also need to save
// the object's generic mapping in the access state structure
// here, because this is the earliest opportunity we have to do so.
RtlMapGenericMask( &AccessState->RemainingDesiredAccess, &IoFileObjectType->TypeInfo.GenericMapping );
RtlMapGenericMask( &AccessState->OriginalDesiredAccess, &IoFileObjectType->TypeInfo.GenericMapping );
SeSetAccessStateGenericMapping( AccessState, &IoFileObjectType->TypeInfo.GenericMapping );
desiredAccess = AccessState->RemainingDesiredAccess;
// Compute the previous mode to be passed in to the privilege check
if (AccessMode != KernelMode || op->Options & IO_FORCE_ACCESS_CHECK) {
modeForPrivilegeCheck = UserMode;
} else {
modeForPrivilegeCheck = KernelMode;
}
IopCheckBackupRestorePrivilege( AccessState, &op->CreateOptions, modeForPrivilegeCheck, op->Disposition);
// If this is not the first time through here for this object, and the
// object itself is being opened, then the desired access must also
// include the previously granted access from the last pass. Likewise,
// if the privileges have been checked already, then this is another
// pass through for a file, so add in the previously granted access.
if ((op->Override && !RemainingName->Length) || AccessState->Flags & SE_BACKUP_PRIVILEGES_CHECKED) {
desiredAccess |= AccessState->PreviouslyGrantedAccess;
}
// If its a filesystem volume open and we are doing a relative open to it
// then do the access check. Note that relative opens can be nested and we propagate
// the fact that the relative open is for a volume using the FO_VOLUME_OPEN flag.
if (op->RelatedFileObject) {
if ((op->RelatedFileObject->Flags & FO_VOLUME_OPEN) && RemainingName->Length == 0) {
relativeVolumeOpen = TRUE;
}
}
// Now determine what type of security check should be made. This is
// based on whether the remaining name string is null. If it is null,
// then the device itself is being opened, so a full security check is
// performed. Otherwise, only a check to ensure that the caller can
// traverse the device object is made. Note that these checks are only
// made if the caller's mode is user, or if access checking is being
// forced. Note also that if an access check was already made on the
// device itself, and this code is being executed again because of a
// reparse, then the access check need not be made the second time
// around.
if ((AccessMode != KernelMode || op->Options & IO_FORCE_ACCESS_CHECK) && (!op->RelatedFileObject || relativeVolumeOpen) && !op->Override) {
BOOLEAN subjectContextLocked = FALSE;
BOOLEAN accessGranted;
ACCESS_MASK grantedAccess;
// The caller's mode is either user or access checking is being
// forced. Perform the appropriate access check on the device object.
if (!RemainingName->Length) {
UNICODE_STRING nameString;
PPRIVILEGE_SET privileges = NULL;
// The device itself is being opened. Make a full security check to ensure that the caller has the appropriate access.
KeEnterCriticalRegion( );
ExAcquireResourceShared( &IopSecurityResource, TRUE );
SeLockSubjectContext( &AccessState->SubjectSecurityContext );
subjectContextLocked = TRUE;
accessGranted = SeAccessCheck( parseDeviceObject->SecurityDescriptor,
&AccessState->SubjectSecurityContext,
subjectContextLocked,
desiredAccess,
0,
&privileges,
&IoFileObjectType->TypeInfo.GenericMapping,
UserMode,
&grantedAccess,
&status );
if (privileges) {
(VOID) SeAppendPrivileges( AccessState, privileges );
SeFreePrivileges( privileges );
}
if (accessGranted) {
AccessState->PreviouslyGrantedAccess |= grantedAccess;
AccessState->RemainingDesiredAccess &= ~( grantedAccess | MAXIMUM_ALLOWED );
op->Override = TRUE;
}
nameString.Length = 8;
nameString.MaximumLength = 8;
nameString.Buffer = L"File";
SeOpenObjectAuditAlarm( &nameString,
parseDeviceObject,
CompleteName,
parseDeviceObject->SecurityDescriptor,
AccessState,
FALSE,
accessGranted,
UserMode,
&AccessState->GenerateOnClose );
ExReleaseResource( &IopSecurityResource );
KeLeaveCriticalRegion();
} else {
// The device is not being opened, rather, a file on the device
// is being opened or created. Therefore, only perform a check here for traverse access to the device.
// First determine if we have to perform traverse checking at all.
// Traverse checking only needs to be done if the device being
// traversed is a disk, or if the caller does not already have
// traverse checking privilege. Note that the former case is so
// that an administrator can turn off access to the "system
// partition", or someone would be able to install a trojan horse
// into the system by simply replacing one of the files there with something of their own.
if (!(AccessState->Flags & TOKEN_HAS_TRAVERSE_PRIVILEGE) ||
parseDeviceObject->DeviceType == FILE_DEVICE_DISK ||
parseDeviceObject->DeviceType == FILE_DEVICE_CD_ROM ) {
KeEnterCriticalRegion( );
ExAcquireResourceShared( &IopSecurityResource, TRUE );
// If the token is restricted we need to do the full access check.
if ((AccessState->Flags & TOKEN_IS_RESTRICTED) == 0) {
accessGranted = SeFastTraverseCheck( parseDeviceObject->SecurityDescriptor, FILE_TRAVERSE, UserMode );
} else {
accessGranted = FALSE;
}
if (!accessGranted) {
PPRIVILEGE_SET privileges = NULL;
// The caller was not granted traverse access through the
// normal fast path lookup. Perform a full-blown access
// check to determine whether some other ACE allows traverse access.
SeLockSubjectContext( &AccessState->SubjectSecurityContext );
subjectContextLocked = TRUE;
accessGranted = SeAccessCheck( parseDeviceObject->SecurityDescriptor,
&AccessState->SubjectSecurityContext,
subjectContextLocked,
FILE_TRAVERSE,
0,
&privileges,
&IoFileObjectType->TypeInfo.GenericMapping,
UserMode,
&grantedAccess,
&status );
if (privileges) {
(VOID) SeAppendPrivileges( AccessState, privileges );
SeFreePrivileges( privileges );
}
}
// Perform the traverse audit alarm if necessary.
SeTraverseAuditAlarm( &AccessState->OperationID,
parseDeviceObject,
parseDeviceObject->SecurityDescriptor,
&AccessState->SubjectSecurityContext,
subjectContextLocked,
FILE_TRAVERSE,
(PPRIVILEGE_SET) NULL,
accessGranted,
UserMode );
ExReleaseResource( &IopSecurityResource );
KeLeaveCriticalRegion();
} else {
accessGranted = TRUE;
}
}
// Unlock the subject's security context so that it can be changed,
// if it was locked.
if (subjectContextLocked) {
SeUnlockSubjectContext( &AccessState->SubjectSecurityContext );
}
// Finally, determine whether or not access was granted to the device.
// If not, clean everything up and get out now without even invoking the device driver.
if (!accessGranted) {
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
return STATUS_ACCESS_DENIED;
}
}
realFileObjectRequired = !(op->QueryOnly || op->DeleteOnly);
if (RemainingName->Length == 0 &&
op->RelatedFileObject == NULL &&
((desiredAccess & ~(SYNCHRONIZE |
FILE_READ_ATTRIBUTES |
READ_CONTROL |
ACCESS_SYSTEM_SECURITY |
WRITE_OWNER |
WRITE_DAC)) == 0) &&
realFileObjectRequired) {
// If the name of the object being opened is just the name of the
// device itself, and there is no related file object, and the caller
// is opening the device for only read attributes access, then this
// device will not be mounted. This allows applications to obtain
// attributes about the device without actually mounting it.
// Note that if this *is* a direct device open, then the normal path
// through the I/O system and drivers may never be used, even if
// the device appears to be mounted. This is because the user may
// remove the media from the drive (even though it is mounted), and
// now attempting to determine what type of drive it is will still
// fail, this time very hard, because a whole mount process is now
// required, thus defeating this feature.
directDeviceOpen = TRUE;
} else {
// Otherwise, this is a normal open of a file, directory, device, or volume.
directDeviceOpen = FALSE;
}
// There are now five different cases. These are as follows:
// 1) This is a relative open, in which case we want to send the
// request to then same device that opened the relative file object.
// 2) The VPB pointer in the device object is NULL. This means that
// this device does not support a file system. This includes devices such as terminals, etc.
// 3) The VPB pointer in the device object is not NULL and:
// a) The VPB is "blank". That is, the VPB has never been filled in, which means that the device has never been mounted.
// b) The VPB is non-blank, but the verify flag on the device is
// set, indicating that the door to the drive may have been opened and the media may therefore have been changed.
// c) The VPB is non-blank and the verify flag is not set.
// Both of the latter are not explicitly checked for, as #c is
// the normal case, and #b is the responsibility of the file system to check.
// If this is a file system that supports volumes, vpbRefCount will
// be filled in to point to the reference count in the Vpb. Error
// exits paths later on key off this value to see if they should
// decrement the ref count. Note that a direct device open does not
// make it to the file system, so no increment is needed, and no
// decrement will be performed in objsup.c IopDeleteFile().
vpb = NULL;
// If the related open was a direct device open then we should go through the full mount
// path for this open as this may not be a direct device open.
if (op->RelatedFileObject && (!(op->RelatedFileObject->Flags & FO_DIRECT_DEVICE_OPEN))) {
deviceObject = (PDEVICE_OBJECT)ParseObject;
if (op->RelatedFileObject->Vpb) {
vpb = op->RelatedFileObject->Vpb;
// Synchronize here with the file system to make sure that
// volumes don't go away while en route to the FS.
ExInterlockedAddUlong( &vpb->ReferenceCount, 1, &IopVpbSpinLock );
}
} else {
deviceObject = parseDeviceObject;
if (parseDeviceObject->Vpb && !directDeviceOpen) {
vpb = IopCheckVpbMounted( op, parseDeviceObject, RemainingName, &status );
if ( !vpb ) {
return status;
}
// Set the address of the device object associated with the VPB.
deviceObject = vpb->DeviceObject;
}
// Walk the attached device list.
if (deviceObject->AttachedDevice) {
deviceObject = IoGetAttachedDevice( deviceObject );
}
}
// If the driver says that the IO manager should do the access checks, lets do it here.
// We do the check against the parse device object as that device object has a name and we can set an ACL against it.
// We only worry about related opens of devices as the other case is taken care of in the filesystem.
if ((deviceObject->Characteristics & FILE_DEVICE_SECURE_OPEN) && (op->RelatedFileObject || RemainingName->Length) && (!relativeVolumeOpen)) {
BOOLEAN subjectContextLocked = FALSE;
BOOLEAN accessGranted;
ACCESS_MASK grantedAccess;
UNICODE_STRING nameString;
PPRIVILEGE_SET privileges = NULL;
// If the device wants to ensure secure opens then lets check the two
// cases which were skipped earlier. These cases are if its a relative open or if there are trailing names.
KeEnterCriticalRegion( );
ExAcquireResourceShared( &IopSecurityResource, TRUE );
SeLockSubjectContext( &AccessState->SubjectSecurityContext );
subjectContextLocked = TRUE;
accessGranted = SeAccessCheck( parseDeviceObject->SecurityDescriptor,
&AccessState->SubjectSecurityContext,
subjectContextLocked,
desiredAccess,
0,
&privileges,
&IoFileObjectType->TypeInfo.GenericMapping,
UserMode,
&grantedAccess,
&status );
if (privileges) {
(VOID) SeAppendPrivileges( AccessState, privileges );
SeFreePrivileges( privileges );
}
if (accessGranted) {
AccessState->PreviouslyGrantedAccess |= grantedAccess;
AccessState->RemainingDesiredAccess &= ~( grantedAccess | MAXIMUM_ALLOWED );
}
nameString.Length = 8;
nameString.MaximumLength = 8;
nameString.Buffer = L"File";
SeOpenObjectAuditAlarm( &nameString,
deviceObject,
CompleteName,
parseDeviceObject->SecurityDescriptor,
AccessState,
FALSE,
accessGranted,
UserMode,
&AccessState->GenerateOnClose );
SeUnlockSubjectContext( &AccessState->SubjectSecurityContext );
ExReleaseResource( &IopSecurityResource );
KeLeaveCriticalRegion();
if (!accessGranted) {
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
if (vpb) {
IopDereferenceVpbAndFree(vpb);
}
return STATUS_ACCESS_DENIED;
}
}
// Allocate and fill in the I/O Request Packet (IRP) to use in interfacing
// to the driver. The allocation is done using an exception handler in
// case the caller does not have enough quota to allocate the packet.
irp = IopAllocateIrp( deviceObject->StackSize, TRUE );
if (!irp) {
// An IRP could not be allocated. Cleanup and return an appropriate error status code.
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
if (vpb) {
IopDereferenceVpbAndFree(vpb);
}
return STATUS_INSUFFICIENT_RESOURCES;
}
irp->Tail.Overlay.Thread = PsGetCurrentThread();
irp->RequestorMode = AccessMode;
irp->Flags = IRP_CREATE_OPERATION | IRP_SYNCHRONOUS_API | IRP_DEFER_IO_COMPLETION;
securityContext.SecurityQos = SecurityQos;
securityContext.AccessState = AccessState;
securityContext.DesiredAccess = desiredAccess;
securityContext.FullCreateOptions = op->CreateOptions;
// Get a pointer to the stack location for the first driver. This is where the original function codes and parameters are passed.
irpSp = IoGetNextIrpStackLocation( irp );
irpSp->Control = 0;
if (op->CreateFileType == CreateFileTypeNone) {
// This is a normal file open or create function.
irpSp->MajorFunction = IRP_MJ_CREATE;
irpSp->Parameters.Create.EaLength = op->EaLength;
irpSp->Flags = (UCHAR) op->Options;
if (!(Attributes & OBJ_CASE_INSENSITIVE)) {
irpSp->Flags |= SL_CASE_SENSITIVE;
}
} else if (op->CreateFileType == CreateFileTypeNamedPipe) {
// A named pipe is being created.
irpSp->MajorFunction = IRP_MJ_CREATE_NAMED_PIPE;
irpSp->Parameters.CreatePipe.Parameters = op->ExtraCreateParameters;
} else {
// A mailslot is being created.
irpSp->MajorFunction = IRP_MJ_CREATE_MAILSLOT;
irpSp->Parameters.CreateMailslot.Parameters = op->ExtraCreateParameters;
}
// Also fill in the NtCreateFile service's caller's parameters.
irp->Overlay.AllocationSize = op->AllocationSize;
irp->AssociatedIrp.SystemBuffer = op->EaBuffer;
irpSp->Parameters.Create.Options = (op->Disposition << 24) | (op->CreateOptions & 0x00ffffff);
irpSp->Parameters.Create.FileAttributes = op->FileAttributes;
irpSp->Parameters.Create.ShareAccess = op->ShareAccess;
irpSp->Parameters.Create.SecurityContext = &securityContext;
// Fill in local parameters so this routine can determine when the I/O is
// finished, and the normal I/O completion code will not get any errors.
irp->UserIosb = &ioStatus;
irp->MdlAddress = (PMDL) NULL;
irp->PendingReturned = FALSE;
irp->Cancel = FALSE;
irp->UserEvent = (PKEVENT) NULL;
irp->CancelRoutine = (PDRIVER_CANCEL) NULL;
irp->Tail.Overlay.AuxiliaryBuffer = (PVOID) NULL;
// Allocate and initialize the file object that will be used in dealing
// with the device for the remainder of this session with the user. How
// the file object is allocated is based on whether or not a real file
// object is actually required. It is not required for the query and delete only operations.
if (realFileObjectRequired) {
OBJECT_ATTRIBUTES objectAttributes;
// A real, full-blown file object is actually required.
InitializeObjectAttributes( &objectAttributes,
(PUNICODE_STRING) NULL,
Attributes,
(HANDLE) NULL,
(PSECURITY_DESCRIPTOR) NULL
);
status = ObCreateObject( KernelMode,
IoFileObjectType,
&objectAttributes,
AccessMode,
(PVOID) NULL,
(ULONG) sizeof( FILE_OBJECT ),
0,
0,
(PVOID *) &fileObject );
if (!NT_SUCCESS( status )) {
IoFreeIrp( irp );
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
if (vpb) {
IopDereferenceVpbAndFree(vpb);
}
return op->FinalStatus = status;
}
RtlZeroMemory( fileObject, sizeof( FILE_OBJECT ) );
fileObject->Type = IO_TYPE_FILE;
fileObject->Size = sizeof( FILE_OBJECT );
fileObject->RelatedFileObject = op->RelatedFileObject;
if (op->CreateOptions & (FILE_SYNCHRONOUS_IO_ALERT | FILE_SYNCHRONOUS_IO_NONALERT)) {
fileObject->Flags = FO_SYNCHRONOUS_IO;
if (op->CreateOptions & FILE_SYNCHRONOUS_IO_ALERT) {
fileObject->Flags |= FO_ALERTABLE_IO;
}
}
// Now fill in the file object as best is possible at this point and set
// a pointer to it in the IRP so everyone else can find it.
if (fileObject->Flags & FO_SYNCHRONOUS_IO) {
KeInitializeEvent( &fileObject->Lock, SynchronizationEvent, FALSE );
fileObject->Waiters = 0;
fileObject->CurrentByteOffset.QuadPart = 0;
}
if (op->CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING) {
fileObject->Flags |= FO_NO_INTERMEDIATE_BUFFERING;
}
if (op->CreateOptions & FILE_WRITE_THROUGH) {
fileObject->Flags |= FO_WRITE_THROUGH;
}
if (op->CreateOptions & FILE_SEQUENTIAL_ONLY) {
fileObject->Flags |= FO_SEQUENTIAL_ONLY;
}
if (op->CreateOptions & FILE_RANDOM_ACCESS) {
fileObject->Flags |= FO_RANDOM_ACCESS;
}
} else {
// This is either a quick delete or query operation. For these cases,
// it is possible to optimize the Object Manager out of the picture by
// simply putting together something that "looks" like a file object, and then operating on it.
localFileObject = op->LocalFileObject;
RtlZeroMemory( localFileObject, sizeof( DUMMY_FILE_OBJECT ) );
fileObject = (PFILE_OBJECT) &localFileObject->ObjectHeader.Body;
localFileObject->ObjectHeader.Type = IoFileObjectType;
localFileObject->ObjectHeader.PointerCount = 1;
}
if (directDeviceOpen) {
fileObject->Flags |= FO_DIRECT_DEVICE_OPEN;
}
if (!(Attributes & OBJ_CASE_INSENSITIVE)) {
fileObject->Flags |= FO_OPENED_CASE_SENSITIVE;
}
fileObject->Type = IO_TYPE_FILE;
fileObject->Size = sizeof( FILE_OBJECT );
fileObject->RelatedFileObject = op->RelatedFileObject;
fileObject->DeviceObject = parseDeviceObject;
irp->Tail.Overlay.OriginalFileObject = fileObject;
irpSp->FileObject = fileObject;
// Allocate a file name string buffer which is large enough to contain
// the entire remaining name string and initialize the maximum length.
if (RemainingName->Length) {
fileObject->FileName.MaximumLength = RoundNameSize( RemainingName->Length );
fileObject->FileName.Buffer = ExAllocatePoolWithTag( PagedPool, fileObject->FileName.MaximumLength, 'mNoI' );
if (!fileObject->FileName.Buffer) {
IoFreeIrp( irp );
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
if (vpb) {
IopDereferenceVpbAndFree(vpb);
}
fileObject->DeviceObject = (PDEVICE_OBJECT) NULL;
if (realFileObjectRequired) {
ObDereferenceObject( fileObject );
}
return STATUS_INSUFFICIENT_RESOURCES;
}
}
// Now copy the name string into the file object from the remaining name
// that is being reparsed. If the driver decides to reparse, then it must replace this name.
RtlCopyUnicodeString( &fileObject->FileName, RemainingName );
// Before invoking the driver's open routine, check to see whether or not
// this is a fast network attributes query and, if so, and the driver implements the function, attempt to call it here.
if (op->QueryOnly) {
PFAST_IO_DISPATCH fastIoDispatch = deviceObject->DriverObject->FastIoDispatch;
BOOLEAN result;
if (fastIoDispatch &&
fastIoDispatch->SizeOfFastIoDispatch > FIELD_OFFSET( FAST_IO_DISPATCH, FastIoQueryOpen ) &&
fastIoDispatch->FastIoQueryOpen) {
IoSetNextIrpStackLocation( irp );
irpSp->DeviceObject = deviceObject;
result = (fastIoDispatch->FastIoQueryOpen)( irp, op->NetworkInformation, deviceObject );
if (result) {
op->FinalStatus = irp->IoStatus.Status;
op->Information = irp->IoStatus.Information;
// The operation worked, so simply dereference and free the resources acquired up to this point.
if ((op->FinalStatus == STATUS_REPARSE) && irp->Tail.Overlay.AuxiliaryBuffer) {
ASSERT( op->Information > IO_REPARSE_TAG_RESERVED_ONE );
ExFreePool( irp->Tail.Overlay.AuxiliaryBuffer );
irp->Tail.Overlay.AuxiliaryBuffer = NULL;
op->RelatedFileObject = (PFILE_OBJECT) NULL;
}
if (fileObject->FileName.Length) {
ExFreePool( fileObject->FileName.Buffer );
}
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
if (vpb) {
IopDereferenceVpbAndFree(vpb);
}
#if DBG
irp->CurrentLocation = irp->StackCount + 2;
#endif // DBG
IoFreeIrp( irp );
// Finally, indicate that the parse routine was actually
// invoked and that the information returned herein can be used.
op->ParseCheck = OPEN_PACKET_PATTERN;
status = STATUS_SUCCESS;
if (!op->FullAttributes) {
try {
op->BasicInformation->FileAttributes = op->NetworkInformation->FileAttributes;
} except(EXCEPTION_EXECUTE_HANDLER) {
status = GetExceptionCode();
}
}
return status;
} else {
// The fast I/O operation did not work, so take the longer route.
irp->Tail.Overlay.CurrentStackLocation++;
irp->CurrentLocation++;
}
}
}
// Finally, initialize the file object's event to the Not Signaled state and remember that a file object was created.
KeInitializeEvent( &fileObject->Event, NotificationEvent, FALSE );
op->FileObject = fileObject;
// Insert the packet at the head of the IRP list for the thread.
IopQueueThreadIrp( irp );
// Now invoke the driver itself to open the file.
status = IoCallDriver( deviceObject, irp );
// One of four things may have happened when the driver was invoked:
// 1. The I/O operation is pending (Status == STATUS_PENDING). This can
// occur on devices which need to perform some sort of device
// manipulation (such as opening a file for a file system).
// 2. The driver returned an error (Status < 0). This occurs when either
// a supplied parameter was in error, or the device or file system
// incurred or discovered an error.
// 3. The operation ended in a reparse (Status == STATUS_REPARSE). This
// occurs when a file system opens the file, only to discover that it
// represents a symbolic link.
// 4. The operation is complete and was successful (Status ==
// STATUS_SUCCESS). Note that for this case the only action is to
// return a pointer to the file object.
if (status == STATUS_PENDING) {
(VOID) KeWaitForSingleObject( &fileObject->Event, Executive, KernelMode, FALSE, (PLARGE_INTEGER) NULL );
status = ioStatus.Status;
} else {
// The I/O operation was completed without returning a status of
// pending. This means that at this point, the IRP has not been
// fully completed. Complete it now.
PKNORMAL_ROUTINE normalRoutine;
PVOID normalContext;
KIRQL irql;
ASSERT( !irp->PendingReturned );
ASSERT( !irp->MdlAddress );
// In the case of name junctions do the transmogrify work.
if (irp->IoStatus.Status == STATUS_REPARSE && irp->IoStatus.Information == IO_REPARSE_TAG_MOUNT_POINT ) {
PREPARSE_DATA_BUFFER reparseBuffer = NULL;
ASSERT ( irp->Tail.Overlay.AuxiliaryBuffer != NULL );
reparseBuffer = (PREPARSE_DATA_BUFFER) irp->Tail.Overlay.AuxiliaryBuffer;
ASSERT( reparseBuffer->ReparseTag == IO_REPARSE_TAG_MOUNT_POINT );
ASSERT( reparseBuffer->ReparseDataLength < MAXIMUM_REPARSE_DATA_BUFFER_SIZE );
ASSERT( reparseBuffer->Reserved < MAXIMUM_REPARSE_DATA_BUFFER_SIZE );
IopDoNameTransmogrify( irp, fileObject, reparseBuffer );
}
// Now finish up the request.
KeRaiseIrql( APC_LEVEL, &irql );
// Note that normally the system would simply call IopCompleteRequest
// here to complete the packet. However, because this is a create
// operation, several assumptions can be made that make it much faster
// to perform the couple of operations that completing the request
// would perform. These include: copying the I/O status block,
// dequeueing the IRP and freeing it, and setting the file object's
// event to the signalled state. The latter is done here by hand,
// since it is known that it is not possible for any thread to be waiting on the event.
ioStatus = irp->IoStatus;
status = ioStatus.Status;
fileObject->Event.Header.SignalState = 1;
IopDequeueThreadIrp( irp );
// The SystemBuffer is in some cases used by the driver, and needs to be freed if present.
if ((irp->Flags & IRP_BUFFERED_IO) && (irp->Flags & IRP_DEALLOCATE_BUFFER)) {
ExFreePool(irp->AssociatedIrp.SystemBuffer);
}
IoFreeIrp( irp );
KeLowerIrql( irql );
}
// Copy the information field of the I/O status block back to the original caller in case it is required.
op->Information = ioStatus.Information;
if (!NT_SUCCESS( status )) {
int openCancelled;
// The operation ended in an error. Kill the file object, dereference
// the device object, and return a null pointer.
if (fileObject->FileName.Length) {
ExFreePool( fileObject->FileName.Buffer );
fileObject->FileName.Length = 0;
}
fileObject->DeviceObject = (PDEVICE_OBJECT) NULL;
openCancelled = (fileObject->Flags & FO_FILE_OPEN_CANCELLED);
if (realFileObjectRequired) {
ObDereferenceObject( fileObject );
}
op->FileObject = (PFILE_OBJECT) NULL;
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
if ((!openCancelled) && (vpb )) {
IopDereferenceVpbAndFree(vpb);
}
return op->FinalStatus = status;
} else if (status == STATUS_REPARSE) {
// The operation resulted in a reparse. This means that the file
// name in the file object is the new name to be looked up. Replace
// the complete name string with the new name and return STATUS_REPARSE
// so the object manager knows to start over again. Note, however,
// that the file name buffer in the file object itself is kept intact
// so that it can be reused when coming back here again.
// A reparse status may also have been returned from the file system if
// the volume that was in a drive needed to have been verified, but
// the verification failed, and a new volume was mounted. In this case, everything starts over again using the new volume.
ASSERT( IO_REPARSE == IO_REPARSE_TAG_RESERVED_ZERO );
if ((ioStatus.Information == IO_REPARSE) || (ioStatus.Information == IO_REPARSE_TAG_MOUNT_POINT)) {
// If the complete name buffer isn't large enough, reallocate it.
if (CompleteName->MaximumLength < fileObject->FileName.Length) {
PVOID buffer;
buffer = ExAllocatePoolWithTag( PagedPool, fileObject->FileName.Length, 'cFoI' );
if (!buffer) {
return op->FinalStatus = STATUS_INSUFFICIENT_RESOURCES;
} else {
if (CompleteName->Buffer) {
ExFreePool( CompleteName->Buffer );
}
CompleteName->Buffer = buffer;
CompleteName->MaximumLength = fileObject->FileName.Length;
}
}
RtlCopyUnicodeString( CompleteName, &fileObject->FileName );
// For NTFS directory junction points we NULL the RelatedFileObject.
// If the prior call was a relative open, the subsequent one will not be.
if (ioStatus.Information == IO_REPARSE_TAG_MOUNT_POINT) {
op->RelatedFileObject = (PFILE_OBJECT) NULL;
}
}
// Kill the file object, dereference the device object, and return a null pointer.
if (fileObject->FileName.Length) {
ExFreePool( fileObject->FileName.Buffer );
fileObject->FileName.Length = 0;
}
fileObject->DeviceObject = (PDEVICE_OBJECT) NULL;
if (realFileObjectRequired) {
ObDereferenceObject( fileObject );
}
op->FileObject = (PFILE_OBJECT) NULL;
IopDecrementDeviceObjectRef( parseDeviceObject, FALSE );
if (vpb) {
IopDereferenceVpbAndFree(vpb);
}
ASSERT( IO_REMOUNT == IO_REPARSE_TAG_RESERVED_ONE );
if (ioStatus.Information == IO_REPARSE_TAG_RESERVED_ONE) {
// If we are reparsing to verify a volume, restart the reparse
// by attempting to parse the device once again. Note that it
// would be best to simply recurse, but it's not possible since
// there is a limited amount of stack available to kernel mode
// and a limit needs to be enforced for the number of times that
// verify reparse can occur.
if (++retryCount > IO_MAX_REMOUNT_REPARSE_ATTEMPTS) {
return STATUS_UNSUCCESSFUL;
}
goto reparse_loop;
} else {
// Really reparsing a symbolic link, so go back to the object
// manager so it can begin the parse from the top.
op->RelatedFileObject = (PFILE_OBJECT) NULL;
return STATUS_REPARSE;
}
} else {
// The operation was successful. The first thing to do is to see if
// the device that processed the open also opened the file. If
// not, we need to adjust the vpb reference counts. Then, if this is
// not a query or a delete, but rather a normal open/create, return
// the address of the FileObject to the caller and set the
// information returned in the original requestor's I/O status block.
// Also set the value of the parse check field in the open packet to
// a value which will let the caller know that this routine was
// successful in creating the file object. Finally, return the status
// of the operation to the caller.
PDEVICE_OBJECT deviceObjectThatOpenedFile;
deviceObjectThatOpenedFile = IoGetRelatedDeviceObject(fileObject);
if (deviceObject != deviceObjectThatOpenedFile) {
// The device that opened the related file is not the one
// that opened this file. So, readjust the vpb reference counts.
if (vpb) {
IopDereferenceVpbAndFree(vpb);
}
vpb = fileObject->Vpb;
if (vpb) {
ExInterlockedAddUlong(
&vpb->ReferenceCount, 1, &IopVpbSpinLock );
}
}
if (realFileObjectRequired) {
*Object = fileObject;
op->ParseCheck = OPEN_PACKET_PATTERN;
// Add a reference so the file object cannot go away before
// the create routine gets chance to flag the object for handle create.
ObReferenceObject( fileObject );
// If the filename length is zero and its not a relative open or
// its a relative open to a volume open then set the volume open flag.
// Also set it only for filesystem device object volume.
if ((!fileObject->RelatedFileObject || fileObject->RelatedFileObject->Flags & FO_VOLUME_OPEN) &&
(!fileObject->FileName.Length)) {
switch (deviceObjectThatOpenedFile->DeviceType) {
case FILE_DEVICE_DISK_FILE_SYSTEM:
case FILE_DEVICE_CD_ROM_FILE_SYSTEM:
case FILE_DEVICE_TAPE_FILE_SYSTEM:
case FILE_DEVICE_FILE_SYSTEM:
fileObject->Flags |= FO_VOLUME_OPEN;
break;
default:
break;
}
}
return op->FinalStatus = ioStatus.Status;
} else {
// This is either a quick query or delete operation. Determine which it is and quickly perform the operation.
if (op->QueryOnly) {
PFAST_IO_DISPATCH fastIoDispatch;
BOOLEAN queryResult = FALSE;
fastIoDispatch = deviceObjectThatOpenedFile->DriverObject->FastIoDispatch;
if (!op->FullAttributes) {
PFILE_BASIC_INFORMATION basicInfo = NULL;
// This is a simple FAT file attribute query. Attempt to obtain the basic information about the file.
try {
if (fastIoDispatch && fastIoDispatch->FastIoQueryBasicInfo) {
queryResult = fastIoDispatch->FastIoQueryBasicInfo(fileObject, TRUE, op->BasicInformation, &ioStatus, deviceObjectThatOpenedFile);
}
if (!queryResult) {
ULONG returnedLength;
basicInfo = ExAllocatePool( NonPagedPool, sizeof( FILE_BASIC_INFORMATION ) );
if (basicInfo) {
status = IoQueryFileInformation(fileObject, FileBasicInformation, sizeof( FILE_BASIC_INFORMATION ), basicInfo, &returnedLength);
if (NT_SUCCESS( status )) {
RtlCopyMemory( op->BasicInformation, basicInfo, returnedLength );
}
ExFreePool( basicInfo );
} else {
status = STATUS_INSUFFICIENT_RESOURCES;
}
} else {
status = ioStatus.Status;
}
} except(EXCEPTION_EXECUTE_HANDLER) {
if (basicInfo) {
ExFreePool( basicInfo );
}
status = GetExceptionCode();
}
} else {
// This is a full attribute query. Attempt to obtain the
// full network attributes for the file. This includes
// both the basic and standard information about the file. Try the fast path first, if it exists.
if (fastIoDispatch &&
fastIoDispatch->SizeOfFastIoDispatch > FIELD_OFFSET( FAST_IO_DISPATCH, FastIoQueryNetworkOpenInfo ) &&
fastIoDispatch->FastIoQueryNetworkOpenInfo) {
queryResult = fastIoDispatch->FastIoQueryNetworkOpenInfo(fileObject, TRUE, op->NetworkInformation, &ioStatus, deviceObjectThatOpenedFile);
}
if (!queryResult) {
ULONG returnedLength;
// Either the fast dispatch routine did not exist, or
// it simply wasn't callable at this time. Attempt to
// obtain all of the information at once via an IRP-based call.
status = IoQueryFileInformation(
fileObject,
FileNetworkOpenInformation,
sizeof( FILE_NETWORK_OPEN_INFORMATION ),
op->NetworkInformation,
&returnedLength
);
if (!NT_SUCCESS( status )) {
if (status == STATUS_INVALID_PARAMETER || status == STATUS_NOT_IMPLEMENTED) {
FILE_BASIC_INFORMATION basicInfo;
FILE_STANDARD_INFORMATION stdInfo;
// The IRP-based call did not work either, so
// simply try to obtain the information by
// doing IRP-based queries for the basic and
// standard information and piecing together
// the results into the caller's buffer. Note
// that it might be possible to perform fast
// I/O operations to get the data, but it
// might also fail because of the above. So
// simply query the information the long way.
status = IoQueryFileInformation(fileObject, FileBasicInformation, sizeof( FILE_BASIC_INFORMATION ), &basicInfo, &returnedLength);
if (NT_SUCCESS( status )) {
status = IoQueryFileInformation(
fileObject,
FileStandardInformation,
sizeof( FILE_STANDARD_INFORMATION ),
&stdInfo,
&returnedLength);
if (NT_SUCCESS( status )) {
COPY_ATTRIBUTES( op->NetworkInformation, &basicInfo, &stdInfo );
}
}
}
}
}
}
} else {
// There is nothing to do for a quick delete since the caller
// set the FILE_DELETE_ON_CLOSE CreateOption so it is already
// set in the file system.
NOTHING;
}
op->ParseCheck = OPEN_PACKET_PATTERN;
if (realFileObjectRequired) {
ObDereferenceObject( fileObject );
} else {
IopDeleteFile( fileObject );
}
op->FileObject = (PFILE_OBJECT) NULL;
op->FinalStatus = status;
return status;
}
}
}
NTSTATUS
IopParseFile(
IN PVOID ParseObject,
IN PVOID ObjectType,
IN PACCESS_STATE AccessState,
IN KPROCESSOR_MODE AccessMode,
IN ULONG Attributes,
IN OUT PUNICODE_STRING CompleteName,
IN OUT PUNICODE_STRING RemainingName,
IN OUT PVOID Context OPTIONAL,
IN PSECURITY_QUALITY_OF_SERVICE SecurityQos OPTIONAL,
OUT PVOID *Object
)
/*++
Routine Description:
This routine interfaces to the NT Object Manager. It is invoked when
the object system is given the name of an entity to create or open and is
also given a handle to a directory file object that the operation is to be
performed relative to. This routine is specified as the parse routine for all file objects.
This routine simply invokes the parse routine for the appropriate device
that is associated with the file object. It is the responsibility of that routine to perform the operation.
Arguments:
ParseObject - Pointer to the file object that the name is to be opened or created relative to.
ObjectType - Type of the object being opened.
AccessState - Running security access state information for operation.
AccessMode - Access mode of the original caller.
Attributes - Attributes to be applied to the object.
CompleteName - Complete name of the object.
RemainingName - Remaining name of the object.
Context - Pointer to an Open Packet (OP) from NtCreateFile service.
SecurityQos - Supplies a pointer to the captured QOS information if available.
Object - The address of a variable to receive the created file object, if any.
Return Value:
The function return value is one of the following:
a) Success - This indicates that the function succeeded and the object
parameter contains the address of the created file object.
b) Error - This indicates that the file was not found or created and
no file object was created.
c) Reparse - This indicates that the remaining name string has been
replaced by a new name that is to be parsed.
--*/
{
PDEVICE_OBJECT deviceObject;
POPEN_PACKET op;
PAGED_CODE();
// Get the address of the Open Packet (OP).
op = (POPEN_PACKET) Context;
// Ensure that this routine is actually being invoked because someone is
// attempting to open a device or a file through NtCreateFile. This code
// must be invoked from there (as opposed to some other random object
// create or open routine).
if (op == NULL || op->Type != IO_TYPE_OPEN_PACKET || op->Size != sizeof( OPEN_PACKET )) {
return STATUS_OBJECT_TYPE_MISMATCH;
}
// Get a pointer to the device object for this file.
deviceObject = IoGetRelatedDeviceObject( (PFILE_OBJECT) ParseObject );
// Pass the related file object to the device object parse routine.
op->RelatedFileObject = (PFILE_OBJECT) ParseObject;
// Open or create the specified file.
return IopParseDevice( deviceObject, ObjectType, AccessState, AccessMode, Attributes, CompleteName, RemainingName, Context, SecurityQos, Object );
}
NTSTATUS IopQueryName(IN PVOID Object, IN BOOLEAN HasObjectName, OUT POBJECT_NAME_INFORMATION ObjectNameInfo, IN ULONG Length, OUT PULONG ReturnLength)
/*++
Routine Description:
This function implements the query name procedure for the Object Manager for querying the names of file objects.
Arguments:
Object - Pointer to the file object whose name is to be retrieved.
HasObjectName - Indicates whether or not the object has a name.
ObjectNameInfo - Buffer in which to return the name.
Length - Specifies the length of the output buffer, in bytes.
ReturnLength - Specifies the number of bytes actually returned in the output buffer.
Return Value:
The function return value is the final status of the query operation.
--*/
{
NTSTATUS status;
ULONG lengthNeeded;
PFILE_OBJECT fileObject;
PUCHAR buffer;
PWSTR p;
POBJECT_NAME_INFORMATION deviceNameInfo;
PFILE_NAME_INFORMATION fileNameInfo;
ULONG length;
UNREFERENCED_PARAMETER( HasObjectName );
PAGED_CODE();
ASSERT( FIELD_OFFSET( FILE_NAME_INFORMATION, FileName ) < sizeof( OBJECT_NAME_INFORMATION ) );
// Ensure that the size of the output buffer is at least the minimum
// size required to include the basic object name information structure.
if (Length < sizeof( OBJECT_NAME_INFORMATION )) {
return STATUS_INFO_LENGTH_MISMATCH;
}
// Begin by allocating a buffer in which to build the name of the file.
buffer = ExAllocatePoolWithTag( PagedPool, Length, ' oI' );
if (!buffer) {
return STATUS_INSUFFICIENT_RESOURCES;
}
try {
// Query the name of the device on which the file is open.
fileObject = (PFILE_OBJECT) Object;
deviceNameInfo = (POBJECT_NAME_INFORMATION) buffer;
status = ObQueryNameString( (PVOID) fileObject->DeviceObject, deviceNameInfo, Length, &lengthNeeded );
if (!NT_SUCCESS( status )) {
return status;
}
// Ensure that there is enough room in the output buffer to return the name and copy it.
RtlCopyMemory( ObjectNameInfo, deviceNameInfo, lengthNeeded > Length ? Length : lengthNeeded );
p = (PWSTR) (ObjectNameInfo + 1);
ObjectNameInfo->Name.Buffer = p;
p = (PWSTR) ((PCHAR) p + deviceNameInfo->Name.Length);
// If the buffer is already full, then return.
if (lengthNeeded > Length) {
return STATUS_BUFFER_OVERFLOW;
}
// Reset the state for the buffer to obtain the filename portion of the
// name and calculate the remaining length of the caller's buffer. Note
// that in the following calculations, there are two assumptions and
// and dependencies:
// 1) The above query of the device name's returned length needed
// include a NULL character which will be included at the end
// of the entire name. This is included in the calculations
// although it does not appear to be included.
// 2) The sizeof the object name information buffer is assumed
// (and guaranteed because it can never change) to be larger
// than the filename offset in a file name information buffer.
// Therefore it is known that the new length of the "buffer"
// variable can be set to the remaining length plus at least 4.
fileNameInfo = (PFILE_NAME_INFORMATION) buffer;
length = Length - lengthNeeded;
length += FIELD_OFFSET( FILE_NAME_INFORMATION, FileName );
if (KeGetPreviousMode() == UserMode || !(fileObject->Flags & FO_SYNCHRONOUS_IO)) {
// Query the name of the file based using an intermediary buffer.
status = IoQueryFileInformation( fileObject, FileNameInformation, length, (PVOID) fileNameInfo, &lengthNeeded );
}
else {
// This is a kernel mode request for a file that was opened for
// synchronous I/O. A special function that does not obtain the
// file object lock is required, otherwise the request may deadlock
// since the lock is probably already owned.
status = IopGetFileName( fileObject, length, fileNameInfo, &lengthNeeded );
}
// If an error occurred attempting to obtain the filename return now. Note that buffer overflow is a warning, not an error.
if (NT_ERROR( status )) {
if (status == STATUS_INVALID_PARAMETER ||
status == STATUS_INVALID_DEVICE_REQUEST ||
status == STATUS_NOT_IMPLEMENTED ||
status == STATUS_INVALID_INFO_CLASS) {
lengthNeeded = FIELD_OFFSET( FILE_NAME_INFORMATION, FileName );
fileNameInfo->FileNameLength = 0;
fileNameInfo->FileName[0] = OBJ_NAME_PATH_SEPARATOR;
status = STATUS_SUCCESS;
}
else {
return status;
}
}
// Set the remaining length of the caller's buffer as well as the total
// length needed to contain the entire name of the file.
length = lengthNeeded - FIELD_OFFSET( FILE_NAME_INFORMATION, FileName );
lengthNeeded = (ULONG)((PUCHAR) p - (PUCHAR) ObjectNameInfo) + fileNameInfo->FileNameLength;
// Attempt to copy the name of the file into the output buffer. Note
// that if the file name does not begin w/a '\', then it is not volume
// relative, so the name of the file cannot be expressed as the
// concatenation of the name of the device and the file. Therefore an error is returned.
// The only example of this situation known at this time is when one
// opens a directory by file ID, and then opens a file relative to that
// directory. When attempting to query the path, if the caller did not
// have traverse access to open the directory, then the only name that
// can be returned is the path name to the file from the directory, but
// the volume-relative name cannot be returned. Therefore, the file
// system returns only the name of the directory and the path to the
// file, but this is not volume-relative so the only recourse is to return an error.
// Note that if the caller were to call NtQueryInformationFile and
// request FileNameInformation, then the name above named will be successfully returned from the file system.
if (fileNameInfo->FileName[0] != OBJ_NAME_PATH_SEPARATOR) {
return STATUS_OBJECT_PATH_INVALID;
}
RtlMoveMemory( p, fileNameInfo->FileName, length );
p = (PWSTR) ((PCH) p + length);
*p = '\0';
lengthNeeded += sizeof( WCHAR );
*ReturnLength = lengthNeeded;
length = (ULONG)((PUCHAR) p - (PUCHAR) ObjectNameInfo);
ObjectNameInfo->Name.Length = (USHORT) (length - sizeof( *ObjectNameInfo ));
ObjectNameInfo->Name.MaximumLength = (USHORT) ((length - sizeof( *ObjectNameInfo )) + sizeof( WCHAR ));
}
finally {
// Finally, free the temporary buffer.
ExFreePool( buffer );
}
return status;
}
VOID IopCheckBackupRestorePrivilege(IN PACCESS_STATE AccessState, IN OUT PULONG CreateOptions, IN KPROCESSOR_MODE PreviousMode, IN ULONG Disposition)
/*++
Routine Description:
This funcion will determine if the caller is asking for any accesses
that may be satisfied by Backup or Restore privileges, and if so,
perform the privilge checks. If the privilege checks succeed, then
the appropriate bits will be moved out of the RemainingDesiredAccess
field in the AccessState structure and placed into the PreviouslyGrantedAccess field.
Note that access is not denied if the caller does not have either or
both of the privileges, since he may be granted the desired access via the security descriptor on the object.
This routine will also set a flag in the AccessState structure so that
it will not perform these privilege checks again in case we come through this way again due to a reparse.
Arguments:
AccessState - The AccessState containing the current state of this access attempt.
CreateOptions - The CreateOptions field from the OPEN_PACKET structure for this open attempt.
PreviousMode - The processor mode to be used in checking parameters.
Disposition - The create disposition for this request.
Return Value:
None.
--*/
{
ACCESS_MASK desiredAccess;
ACCESS_MASK readAccess;
ACCESS_MASK writeAccess;
PRIVILEGE_SET requiredPrivileges;
BOOLEAN accessGranted;
BOOLEAN keepBackupIntent = FALSE;
BOOLEAN ForceRestoreCheck = FALSE;
PAGED_CODE();
// Check to determine whether or not this check has already been made.
// If so, simply return back to the caller.
if (AccessState->Flags & SE_BACKUP_PRIVILEGES_CHECKED) {
return;
}
if (*CreateOptions & FILE_OPEN_FOR_BACKUP_INTENT) {
AccessState->Flags |= SE_BACKUP_PRIVILEGES_CHECKED;
readAccess = READ_CONTROL | ACCESS_SYSTEM_SECURITY | FILE_GENERIC_READ | FILE_TRAVERSE;
writeAccess = WRITE_DAC | WRITE_OWNER | ACCESS_SYSTEM_SECURITY | FILE_GENERIC_WRITE | FILE_ADD_FILE | FILE_ADD_SUBDIRECTORY | DELETE;
desiredAccess = AccessState->RemainingDesiredAccess;
// If the caller has requested MAXIMUM_ALLOWED, then make it appear as
// if the request was for everything permitted by Backup and Restore,
// and then grant everything that can actually be granted.
if (desiredAccess & MAXIMUM_ALLOWED) {
desiredAccess |= ( readAccess | writeAccess );
}
// If the disposition says that we're opening the file, check for both backup
// and restore privilege, depending on what's in the desired access.
// If the disposition says that we're creating or trying to overwrite the file,
// then all we need to do is to check for restore privilege, and if it's there,
// grant every possible access.
if ( Disposition & FILE_OPEN ) {
// If the request was for any of the bits in the read access mask, then
// assume that this is a backup operation, and check for the Backup
// privielege. If the caller has it, then grant the intersection of
// the desired access and read access masks.
if (readAccess & desiredAccess) {
requiredPrivileges.PrivilegeCount = 1;
requiredPrivileges.Control = PRIVILEGE_SET_ALL_NECESSARY;
requiredPrivileges.Privilege[0].Luid = SeBackupPrivilege;
requiredPrivileges.Privilege[0].Attributes = 0;
accessGranted = SePrivilegeCheck( &requiredPrivileges, &AccessState->SubjectSecurityContext, PreviousMode );
if (accessGranted) {
// The caller has Backup privilege, so grant the appropriate accesses.
keepBackupIntent = TRUE;
(VOID) SeAppendPrivileges( AccessState, &requiredPrivileges );
AccessState->PreviouslyGrantedAccess |= ( desiredAccess & readAccess );
AccessState->RemainingDesiredAccess &= ~readAccess;
desiredAccess &= ~readAccess;
AccessState->Flags |= TOKEN_HAS_BACKUP_PRIVILEGE;
}
}
} else {
ForceRestoreCheck = TRUE;
}
// If the request was for any of the bits in the write access mask, then
// assume that this is a restore operation, so check for the Restore
// privilege. If the caller has it, then grant the intersection of
// the desired access and write access masks.
if ((writeAccess & desiredAccess) || ForceRestoreCheck) {
requiredPrivileges.PrivilegeCount = 1;
requiredPrivileges.Control = PRIVILEGE_SET_ALL_NECESSARY;
requiredPrivileges.Privilege[0].Luid = SeRestorePrivilege;
requiredPrivileges.Privilege[0].Attributes = 0;
accessGranted = SePrivilegeCheck( &requiredPrivileges, &AccessState->SubjectSecurityContext, PreviousMode );
if (accessGranted) {
// The caller has Restore privilege, so grant the appropriate accesses.
keepBackupIntent = TRUE;
(VOID) SeAppendPrivileges( AccessState, &requiredPrivileges );
AccessState->PreviouslyGrantedAccess |= (desiredAccess & writeAccess);
AccessState->RemainingDesiredAccess &= ~writeAccess;
AccessState->Flags |= TOKEN_HAS_RESTORE_PRIVILEGE;
}
}
// If either of the access types was granted because the caller had
// backup or restore privilege, then the backup intent flag is kept.
// Otherwise, it is cleared so that it is not passed onto the driver
// so that it is not incorrectly propogated anywhere else, since this
// caller does not actually have the privilege enabled.
if (!keepBackupIntent) {
*CreateOptions &= ~FILE_OPEN_FOR_BACKUP_INTENT;
}
}
}