|
|
/*++
Copyright (c) 2001 Microsoft Corporation
Module Name:
PpPathPath.c
Abstract:
The file implements support for managing devices on the paging path.
Author:
Adrian J. Oney (AdriaO) February 3rd, 2001
Revision History:
Originally taken from ChuckL's implementation in mm\modwrite.c.
--*/
#include "pnpmgrp.h"
#include "pipagepath.h"
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, PpPagePathAssign)
#pragma alloc_text(PAGE, PpPagePathRelease)
#pragma alloc_text(PAGE, PiPagePathSetState)
#endif
NTSTATUS PpPagePathAssign( IN PFILE_OBJECT FileObject ) /*++
Routine Description:
This routine informs driver stacks that they are now on the paging path. Drivers need to take appropriate actions when on the path, such as failing IRP_MN_QUERY_STOP and IRP_MN_QUERY_REMOVE, locking their code and clearing the DO_POWER_PAGABLE bit, etc.
Arguments:
FileObject - File object for the paging file itself.
Return Value:
NTSTATUS.
--*/ { PAGED_CODE();
return PiPagePathSetState(FileObject, TRUE); }
NTSTATUS PpPagePathRelease( IN PFILE_OBJECT FileObject ) /*++
Routine Description:
This routine informs driver stacks that the passed in file is no longer a paging file. Each driver stack notified may still be on the paging path however if their hardware supports a different paging file on another drive.
Arguments:
FileObject - File object for the paging file itself.
Return Value:
NTSTATUS.
--*/ { PAGED_CODE();
return PiPagePathSetState(FileObject, FALSE); }
NTSTATUS PiPagePathSetState( IN PFILE_OBJECT FileObject, IN BOOLEAN InPath ) /*++
Routine Description:
This routine notifies driver stacks when a paging file is shut down on their device, or if a paging file is being started up on their device. If a paging file is being started up, this request is also a query as the stack may not be able to support a pagefile.
Arguments:
FileObject - File object for the paging file itself.
InPath - Whether the page file is being started or shut down.
Return Value:
NTSTATUS.
--*/ { PIRP irp; NTSTATUS status; PDEVICE_OBJECT deviceObject; KEVENT event; PIO_STACK_LOCATION irpSp; IO_STATUS_BLOCK localIoStatus;
PAGED_CODE();
//
// Reference the file object here so that no special checks need be made
// in I/O completion to determine whether or not to dereference the file
// object.
//
ObReferenceObject(FileObject);
//
// Initialize the local event.
//
KeInitializeEvent(&event, NotificationEvent, FALSE);
//
// Get the address of the target device object.
//
deviceObject = IoGetRelatedDeviceObject(FileObject);
//
// Allocate and initialize the irp for this operation.
//
irp = IoAllocateIrp(deviceObject->StackSize, FALSE);
if (irp == NULL) {
//
// Don't dereference the file object, our caller will take care of that.
//
return STATUS_NO_MEMORY; }
irp->Tail.Overlay.OriginalFileObject = FileObject; irp->Tail.Overlay.Thread = PsGetCurrentThread(); irp->RequestorMode = KernelMode;
//
// Fill in the service independent parameters in the IRP.
//
irp->UserEvent = &event; irp->Flags = IRP_SYNCHRONOUS_API; irp->UserIosb = &localIoStatus; irp->Overlay.AsynchronousParameters.UserApcRoutine = (PIO_APC_ROUTINE) NULL;
//
// Get a pointer to the stack location for the first driver. This will be
// used to pass the original function codes and parameters.
//
irpSp = IoGetNextIrpStackLocation(irp); irpSp->MajorFunction = IRP_MJ_PNP; irpSp->MinorFunction = IRP_MN_DEVICE_USAGE_NOTIFICATION; irpSp->FileObject = FileObject; irp->IoStatus.Status = STATUS_NOT_SUPPORTED; irp->AssociatedIrp.SystemBuffer = NULL; // irp->Flags = 0;
irpSp->Parameters.UsageNotification.InPath = InPath; irpSp->Parameters.UsageNotification.Type = DeviceUsageTypePaging;
//
// Insert the packet at the head of the IRP list for the thread.
//
IoQueueThreadIrp(irp);
//
// Acquire the engine lock to ensure no rebalances, removes, or power
// operations are in progress during this notification.
//
PpDevNodeLockTree(PPL_TREEOP_ALLOW_READS);
//
// Now simply invoke the driver at its dispatch entry with the IRP.
//
status = IoCallDriver(deviceObject, irp);
//
// Wait for the local event and copy the final status information
// back to the caller.
//
if (status == STATUS_PENDING) {
(VOID) KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL);
status = localIoStatus.Status; }
//
// Unlock the tree.
//
PpDevNodeUnlockTree(PPL_TREEOP_ALLOW_READS);
return status; }
|