Sample Code

Windows Driver Samples/ ClassPnP Storage Class Driver Library/ C++/ src/ power.c/

/*++

Copyright (C) Microsoft Corporation, 1991 - 2010

Module Name:

    power.c

Abstract:

    SCSI class driver routines

Environment:

    kernel mode only

Notes:


Revision History:

--*/

#include "stddef.h"
#include "ntddk.h"
#include "scsi.h"
#include "classp.h"

#include <stdarg.h>

#ifdef DEBUG_USE_WPP
#include "power.tmh"
#endif

#define CLASS_TAG_POWER     'WLcS'

// constants for power transition process. (UNIT: seconds)
#define DEFAULT_POWER_IRP_TIMEOUT_VALUE 10*60
#define TIME_LEFT_FOR_LOWER_DRIVERS     30
#define TIME_LEFT_FOR_UPPER_DRIVERS     5
#define DEFAULT_IO_TIMEOUT_VALUE        10
#define MINIMUM_STOP_UNIT_TIMEOUT_VALUE 2

//
// MINIMAL value is one that has some slack and is the value to use
// if there is a shortened POWER IRP timeout value. If time remaining
// is less than MINIMAL, we will use the MINIMUM value. Both values
// are in the same unit as above (seconds).
//
#define MINIMAL_START_UNIT_TIMEOUT_VALUE 60
#define MINIMUM_START_UNIT_TIMEOUT_VALUE 30

// PoQueryWatchdogTime was introduced in Windows 7.
// Returns TRUE if a watchdog-enabled power IRP is found, otherwise FALSE.
#if (NTDDI_VERSION < NTDDI_WIN7)
#define PoQueryWatchdogTime(A, B) FALSE
#endif

IO_COMPLETION_ROUTINE ClasspPowerDownCompletion;

IO_COMPLETION_ROUTINE ClasspPowerUpCompletion;

IO_COMPLETION_ROUTINE ClasspStartNextPowerIrpCompletion;

NTSTATUS
ClasspPowerHandler(
    IN PDEVICE_OBJECT DeviceObject,
    IN PIRP Irp,
    IN CLASS_POWER_OPTIONS Options
    );

VOID
RetryPowerRequest(
    PDEVICE_OBJECT DeviceObject,
    PIRP Irp,
    PCLASS_POWER_CONTEXT Context
    );

#ifdef ALLOC_PRAGMA
    #pragma alloc_text(PAGE, ClasspPowerSettingCallback)
#endif

/*++////////////////////////////////////////////////////////////////////////////

ClassDispatchPower()

Routine Description:

    This routine acquires the removelock for the irp and then calls the
    appropriate power callback.

Arguments:

    DeviceObject -
    Irp -

Return Value:

--*/
NTSTATUS
ClassDispatchPower(
    IN PDEVICE_OBJECT DeviceObject,
    IN PIRP Irp
    )
{
    PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
    ULONG isRemoved;

    //
    // NOTE: This code may be called at PASSIVE or DISPATCH, depending
    //       upon the device object it is being called for.
    //       don't do anything that would break under either circumstance.
    //

    //
    // If device is added but not yet started, we need to send the Power
    // request down the stack.  If device is started and then stopped,
    // we have enough state to process the power request.
    //

    if (!commonExtension->IsInitialized) {

        PoStartNextPowerIrp(Irp);
        IoSkipCurrentIrpStackLocation(Irp);
        return PoCallDriver(commonExtension->LowerDeviceObject, Irp);
    }

    isRemoved = ClassAcquireRemoveLock(DeviceObject, Irp);

    if (isRemoved) {
        ClassReleaseRemoveLock(DeviceObject, Irp);
        Irp->IoStatus.Status = STATUS_DEVICE_DOES_NOT_EXIST;
        PoStartNextPowerIrp(Irp);
        ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
        return STATUS_DEVICE_DOES_NOT_EXIST;
    }

    return commonExtension->DevInfo->ClassPowerDevice(DeviceObject, Irp);
} // end ClassDispatchPower()

/*++////////////////////////////////////////////////////////////////////////////

ClasspPowerUpCompletion()

Routine Description:

    This routine is used for intermediate completion of a power up request.
    PowerUp requires four requests to be sent to the lower driver in sequence.

        * The queue is "power locked" to ensure that the class driver power-up
          work can be done before request processing resumes.

        * The power irp is sent down the stack for any filter drivers and the
          port driver to return power and resume command processing for the
          device.  Since the queue is locked, no queued irps will be sent
          immediately.

        * A start unit command is issued to the device with appropriate flags
          to override the "power locked" queue.

        * The queue is "power unlocked" to start processing requests again.

    This routine uses the function in the srb which just completed to determine
    which state it is in.

Arguments:

    DeviceObject - the device object being powered up

    Irp - Context->Irp: original power irp; fdoExtension->PrivateFdoData->PowerProcessIrp: power process irp

    Context - Class power context used to perform port/class operations.

Return Value:

    STATUS_MORE_PROCESSING_REQUIRED or
    STATUS_SUCCESS

--*/
NTSTATUS
ClasspPowerUpCompletion(
    IN PDEVICE_OBJECT DeviceObject,
    IN PIRP Irp,
    IN PVOID Context
    )
{
    PCLASS_POWER_CONTEXT PowerContext = (PCLASS_POWER_CONTEXT)Context;
    PCOMMON_DEVICE_EXTENSION commonExtension;
    PFUNCTIONAL_DEVICE_EXTENSION fdoExtension;
    PIRP OriginalIrp;
    PIO_STACK_LOCATION currentStack;
    PIO_STACK_LOCATION nextStack;

    NTSTATUS status = STATUS_MORE_PROCESSING_REQUIRED;
    PSTORAGE_REQUEST_BLOCK_HEADER srbHeader;
    ULONG srbFlags;

    UNREFERENCED_PARAMETER(DeviceObject);

    if (PowerContext == NULL) {
        NT_ASSERT(PowerContext != NULL);
        return STATUS_INVALID_PARAMETER;
    }

    commonExtension = PowerContext->DeviceObject->DeviceExtension;
    fdoExtension = PowerContext->DeviceObject->DeviceExtension;
    OriginalIrp = PowerContext->Irp;

    // currentStack - from original power irp
    // nextStack - from power process irp
    currentStack = IoGetCurrentIrpStackLocation(OriginalIrp);
    nextStack = IoGetNextIrpStackLocation(fdoExtension->PrivateFdoData->PowerProcessIrp);

    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "ClasspPowerUpCompletion: Device Object %p, Irp %p, "
                   "Context %p\n",
                PowerContext->DeviceObject, Irp, Context));

    if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(fdoExtension->PrivateFdoData->PowerSrb.SrbEx);

        //
        // Check if reverted to using legacy SRB.
        //
        if (PowerContext->Srb.Length == sizeof(SCSI_REQUEST_BLOCK)) {
            srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
        }
    } else {
        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
    }

    srbFlags = SrbGetSrbFlags(srbHeader);
    NT_ASSERT(!TEST_FLAG(srbFlags, SRB_FLAGS_FREE_SENSE_BUFFER));
    NT_ASSERT(!TEST_FLAG(srbFlags, SRB_FLAGS_PORT_DRIVER_ALLOCSENSE));
    NT_ASSERT(PowerContext->Options.PowerDown == FALSE);
    NT_ASSERT(PowerContext->Options.HandleSpinUp);

    if ((Irp == OriginalIrp) && (Irp->PendingReturned)) {
        // only for original power irp
        IoMarkIrpPending(Irp);
    }

    PowerContext->PowerChangeState.PowerUp++;

    switch (PowerContext->PowerChangeState.PowerUp) {

        case PowerUpDeviceLocked: {

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously sent power lock\n", Irp));

            //
            // Lock Queue operation has been sent.
            // Now, send the original power irp down to get lower driver and device ready.
            //

            IoCopyCurrentIrpStackLocationToNext(OriginalIrp);

            if ((PowerContext->Options.LockQueue == TRUE) &&
                (!NT_SUCCESS(Irp->IoStatus.Status))) {

                //
                // Lock was not successful:
                // Issue the original power request to the lower driver and next power irp will be started in completion routine.
                //

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIrp status was %lx\n",
                            Irp, Irp->IoStatus.Status));
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tSrb status was %lx\n",
                            Irp, srbHeader->SrbStatus));

                //
                // Set the new power state
                //

                fdoExtension->DevicePowerState = currentStack->Parameters.Power.State.DeviceState;

                OriginalIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;

                IoSetCompletionRoutine(OriginalIrp,
                                       ClasspStartNextPowerIrpCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                //
                // Indicate to Po that we've been successfully powered up so
                // it can do it's notification stuff.
                //

                PoSetPowerState(PowerContext->DeviceObject,
                                currentStack->Parameters.Power.Type,
                                currentStack->Parameters.Power.State);

                ClassReleaseRemoveLock(commonExtension->DeviceObject,
                                       OriginalIrp);

                PoCallDriver(commonExtension->LowerDeviceObject, OriginalIrp);

                return STATUS_MORE_PROCESSING_REQUIRED;

            } else {
                PowerContext->QueueLocked = (UCHAR)PowerContext->Options.LockQueue;
            }

            Irp->IoStatus.Status = STATUS_NOT_SUPPORTED;

            PowerContext->PowerChangeState.PowerUp = PowerUpDeviceLocked;

            IoSetCompletionRoutine(OriginalIrp,
                                   ClasspPowerUpCompletion,
                                   PowerContext,
                                   TRUE,
                                   TRUE,
                                   TRUE);

            status = PoCallDriver(commonExtension->LowerDeviceObject, OriginalIrp);

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIoCallDriver returned %lx\n", OriginalIrp, status));
            break;
        }

        case PowerUpDeviceOn: {

            //
            // Original power irp has been completed by lower driver.
            //

            if (NT_SUCCESS(Irp->IoStatus.Status)) {
                //
                // If power irp succeeded, START UNIT command will be sent.
                //
                PCDB cdb;
                ULONG secondsRemaining = 0;
                ULONG timeoutValue = 0;
                ULONG startUnitTimeout;

                if (PoQueryWatchdogTime(fdoExtension->LowerPdo, &secondsRemaining)) {

                    // do not exceed DEFAULT_POWER_IRP_TIMEOUT_VALUE.
                    secondsRemaining = min(secondsRemaining, DEFAULT_POWER_IRP_TIMEOUT_VALUE);

                    //
                    // It's possible for POWER IRP timeout value to be smaller than default of
                    // START_UNIT_TIMEOUT. If this is the case, use a smaller timeout value.
                    //
                    if (secondsRemaining >= START_UNIT_TIMEOUT) {
                        startUnitTimeout = START_UNIT_TIMEOUT;
                    } else {
                        startUnitTimeout = MINIMAL_START_UNIT_TIMEOUT_VALUE;
                    }

                    // plan to leave (TIME_LEFT_FOR_UPPER_DRIVERS) seconds to upper level drivers
                    // for processing original power irp.
                    if (secondsRemaining >= (TIME_LEFT_FOR_UPPER_DRIVERS + startUnitTimeout)) {
                        fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount =
                            (secondsRemaining - TIME_LEFT_FOR_UPPER_DRIVERS) / startUnitTimeout;

                        // * No 'short' timeouts
                        //
                        //
                        // timeoutValue = (secondsRemaining - TIME_LEFT_FOR_UPPER_DRIVERS) %
                        //                startUnitTimeout;
                        //

                        if (--fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount)
                        {
                            timeoutValue = startUnitTimeout;
                        } else {
                            timeoutValue = secondsRemaining - TIME_LEFT_FOR_UPPER_DRIVERS;
                        }
                    } else {
                        // issue the command with minimum timeout value and do not retry on it.
                        // case of (secondsRemaining < DEFAULT_IO_TIMEOUT_VALUE) is ignored as it should not happen.
                        NT_ASSERT(secondsRemaining >= DEFAULT_IO_TIMEOUT_VALUE);

                        fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = 0;
                        timeoutValue = MINIMUM_START_UNIT_TIMEOUT_VALUE; // use the minimum value for this corner case.
                    }

                } else {
                    // don't know how long left, do not exceed DEFAULT_POWER_IRP_TIMEOUT_VALUE.
                    fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount =
                        DEFAULT_POWER_IRP_TIMEOUT_VALUE / START_UNIT_TIMEOUT - 1;
                    timeoutValue = START_UNIT_TIMEOUT;
                }


                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tSending start unit to device\n", Irp));

                //
                // Issue the start unit command to the device.
                //

                PowerContext->RetryCount = fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount;

                if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
                    status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)srbHeader,
                                                            STORAGE_ADDRESS_TYPE_BTL8,
                                                            CLASS_SRBEX_SCSI_CDB16_BUFFER_SIZE,
                                                            1,
                                                            SrbExDataTypeScsiCdb16);
                    if (NT_SUCCESS(status)) {
                        ((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction = SRB_FUNCTION_EXECUTE_SCSI;

                        //
                        // Set length field in Power Context SRB so we know legacy SRB is not being used.
                        //
                        PowerContext->Srb.Length = 0;

                    } else {
                        //
                        // Should not happen. Revert to legacy SRB.
                        //
                        NT_ASSERT(FALSE);
                        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
                        RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                        srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                        srbHeader->Function = SRB_FUNCTION_EXECUTE_SCSI;
                    }

                } else {
                    RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                    srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                    srbHeader->Function = SRB_FUNCTION_EXECUTE_SCSI;
                }

                SrbSetOriginalRequest(srbHeader, fdoExtension->PrivateFdoData->PowerProcessIrp);
                SrbSetSenseInfoBuffer(srbHeader, commonExtension->PartitionZeroExtension->SenseData);
                SrbSetSenseInfoBufferLength(srbHeader, GET_FDO_EXTENSON_SENSE_DATA_LENGTH(commonExtension->PartitionZeroExtension));

                SrbSetTimeOutValue(srbHeader, timeoutValue);
                SrbAssignSrbFlags(srbHeader,
                                     (SRB_FLAGS_NO_DATA_TRANSFER |
                                      SRB_FLAGS_DISABLE_AUTOSENSE |
                                      SRB_FLAGS_DISABLE_SYNCH_TRANSFER |
                                      SRB_FLAGS_NO_QUEUE_FREEZE));

                if (PowerContext->Options.LockQueue) {
                    SrbSetSrbFlags(srbHeader, SRB_FLAGS_BYPASS_LOCKED_QUEUE);
                }

                SrbSetCdbLength(srbHeader, 6);

                cdb = SrbGetCdb(srbHeader);
                RtlZeroMemory(cdb, sizeof(CDB));

                cdb->START_STOP.OperationCode = SCSIOP_START_STOP_UNIT;
                cdb->START_STOP.Start = 1;

                PowerContext->PowerChangeState.PowerUp = PowerUpDeviceOn;

                IoSetCompletionRoutine(fdoExtension->PrivateFdoData->PowerProcessIrp,
                                       ClasspPowerUpCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                nextStack->Parameters.Scsi.Srb = (PSCSI_REQUEST_BLOCK)srbHeader;
                nextStack->MajorFunction = IRP_MJ_SCSI;

                status = IoCallDriver(commonExtension->LowerDeviceObject, fdoExtension->PrivateFdoData->PowerProcessIrp);

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIoCallDriver returned %lx\n", fdoExtension->PrivateFdoData->PowerProcessIrp, status));

            } else {

                //
                // power irp is failed by lower driver. we're done.
                //

                PowerContext->FinalStatus = Irp->IoStatus.Status;
                goto ClasspPowerUpCompletionFailure;
            }

            break;
        }

        case PowerUpDeviceStarted: { // 3

            //
            // First deal with an error if one occurred.
            //

            if (SRB_STATUS(srbHeader->SrbStatus) != SRB_STATUS_SUCCESS) {

                BOOLEAN retry;
                LONGLONG delta100nsUnits = 0;
                ULONG secondsRemaining = 0;
                ULONG startUnitTimeout = START_UNIT_TIMEOUT;

                TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_POWER, "%p\tError occured when issuing START_UNIT "
                            "command to device. Srb %p, Status %x\n",
                            Irp,
                            srbHeader,
                            srbHeader->SrbStatus));

                NT_ASSERT(!(TEST_FLAG(srbHeader->SrbStatus, SRB_STATUS_QUEUE_FROZEN)));
                NT_ASSERT((srbHeader->Function == SRB_FUNCTION_EXECUTE_SCSI) ||
                          (((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction == SRB_FUNCTION_EXECUTE_SCSI));

                PowerContext->RetryInterval = 0;
                retry = InterpretSenseInfoWithoutHistory(
                            fdoExtension->DeviceObject,
                            Irp,
                            (PSCSI_REQUEST_BLOCK)srbHeader,
                            IRP_MJ_SCSI,
                            IRP_MJ_POWER,
                            fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount - PowerContext->RetryCount,
                            &status,
                            &delta100nsUnits);

                // NOTE: Power context is a public structure, and thus cannot be
                //       updated to use 100ns units.  Therefore, must store the
                //       one-second equivalent.  Round up to ensure minimum delay
                //       requirements have been met.
                delta100nsUnits += (10*1000*1000) - 1;
                delta100nsUnits /= (10*1000*1000);
                // guaranteed not to have high bits set per SAL annotations
                PowerContext->RetryInterval = (ULONG)(delta100nsUnits);


                if ((retry == TRUE) && (PowerContext->RetryCount-- != 0)) {

                    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tRetrying failed request\n", Irp));

                    //
                    // Decrement the state so we come back through here the
                    // next time.
                    //

                    PowerContext->PowerChangeState.PowerUp--;

                    //
                    // Adjust start unit timeout based on remaining time if needed.
                    //
                    if (PoQueryWatchdogTime(fdoExtension->LowerPdo, &secondsRemaining)) {

                        if (secondsRemaining >= TIME_LEFT_FOR_UPPER_DRIVERS) {
                            secondsRemaining -= TIME_LEFT_FOR_UPPER_DRIVERS;
                        }

                        if (secondsRemaining < MINIMAL_START_UNIT_TIMEOUT_VALUE) {
                            startUnitTimeout = MINIMUM_START_UNIT_TIMEOUT_VALUE;
                        } else if (secondsRemaining < START_UNIT_TIMEOUT) {
                            startUnitTimeout = MINIMAL_START_UNIT_TIMEOUT_VALUE;
                        }
                    }

                    SrbSetTimeOutValue(srbHeader, startUnitTimeout);

                    RetryPowerRequest(commonExtension->DeviceObject,
                                      Irp,
                                      PowerContext);

                    break;

                }

                // reset retry count for UNLOCK command.
                fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = MAXIMUM_RETRIES;
                PowerContext->RetryCount = MAXIMUM_RETRIES;
            }

ClasspPowerUpCompletionFailure:

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously spun device up\n", Irp));

            if (PowerContext->QueueLocked) {
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tUnlocking queue\n", Irp));

                if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
                    //
                    // Will reuse SRB for a non-SCSI SRB.
                    //
                    status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)srbHeader,
                                                           STORAGE_ADDRESS_TYPE_BTL8,
                                                           CLASS_SRBEX_NO_SRBEX_DATA_BUFFER_SIZE,
                                                           0);
                    if (NT_SUCCESS(status)) {
                        ((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction = SRB_FUNCTION_UNLOCK_QUEUE;

                        //
                        // Set length field in Power Context SRB so we know legacy SRB is not being used.
                        //
                        PowerContext->Srb.Length = 0;

                    } else {
                        //
                        // Should not occur. Revert to legacy SRB.
                        NT_ASSERT(FALSE);
                        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
                        RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                        srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                        srbHeader->Function = SRB_FUNCTION_UNLOCK_QUEUE;
                    }
                } else {
                    RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                    srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                    srbHeader->Function = SRB_FUNCTION_UNLOCK_QUEUE;
                }
                SrbAssignSrbFlags(srbHeader, SRB_FLAGS_BYPASS_LOCKED_QUEUE);
                SrbSetOriginalRequest(srbHeader, fdoExtension->PrivateFdoData->PowerProcessIrp);

                nextStack->Parameters.Scsi.Srb = (PSCSI_REQUEST_BLOCK)srbHeader;
                nextStack->MajorFunction = IRP_MJ_SCSI;

                PowerContext->PowerChangeState.PowerUp = PowerUpDeviceStarted;

                IoSetCompletionRoutine(fdoExtension->PrivateFdoData->PowerProcessIrp,
                                       ClasspPowerUpCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                status = IoCallDriver(commonExtension->LowerDeviceObject, fdoExtension->PrivateFdoData->PowerProcessIrp);
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIoCallDriver returned %lx\n",
                            fdoExtension->PrivateFdoData->PowerProcessIrp, status));
                break;
            }

            // Fall-through to next case...

        }

        case PowerUpDeviceUnlocked: {

            //
            // This is the end of the dance.
            // We're ignoring possible intermediate error conditions ....
            //

            if (PowerContext->QueueLocked) {
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously unlocked queue\n", OriginalIrp));

                //
                // If the lower device is being removed, the IRP's status may be STATUS_DELETE_PENDING or 
                // STATUS_DEVICE_DOES_NOT_EXIST.
                //
                NT_ASSERT(NT_SUCCESS(Irp->IoStatus.Status) || 
                          Irp->IoStatus.Status == STATUS_DELETE_PENDING || 
                          Irp->IoStatus.Status == STATUS_DEVICE_DOES_NOT_EXIST);

            } else {
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tFall-through (queue not locked)\n", OriginalIrp));
            }

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tFreeing srb and completing\n", OriginalIrp));

            status = PowerContext->FinalStatus;
            OriginalIrp->IoStatus.Status = status;

            //
            // Set the new power state
            //

            if (NT_SUCCESS(status)) {
                fdoExtension->DevicePowerState = currentStack->Parameters.Power.State.DeviceState;
            }

            //
            // Enable tick timer at end of D0 processing if it was previously enabled.
            //
            if (fdoExtension->PrivateFdoData && (fdoExtension->PrivateFdoData->TickTimerEnabled)) {

                //
                // If failure prediction is turned on and we've been powered
                // off longer than the failure prediction query period then
                // force the query on the next timer tick.
                //
                if ((fdoExtension->FailurePredictionInfo != NULL) &&
                    (fdoExtension->FailurePredictionInfo->Method != FailurePredictionNone)) {
                    if (ClasspFailurePredictionPeriodMissed(fdoExtension)) {
                        fdoExtension->FailurePredictionInfo->CountDown = 1;
                    }
                }

                //
                // Finally, enable the timer.
                //
                ClasspEnableTimer(fdoExtension);
            }

            //
            // Indicate to Po that we've been successfully powered up so
            // it can do it's notification stuff.
            //

            PoSetPowerState(PowerContext->DeviceObject,
                            currentStack->Parameters.Power.Type,
                            currentStack->Parameters.Power.State);

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tStarting next power irp\n", OriginalIrp));

            ClassReleaseRemoveLock(PowerContext->DeviceObject, OriginalIrp);

            PowerContext->InUse = FALSE;

            PoStartNextPowerIrp(OriginalIrp);

            // prevent from completing the irp allocated by ourselves
            if ((fdoExtension->PrivateFdoData) && (Irp == fdoExtension->PrivateFdoData->PowerProcessIrp)) {
                // complete original irp if we are processing powerprocess irp,
                // otherwise, by returning status other than STATUS_MORE_PROCESSING_REQUIRED, IO manager will complete it.
                ClassCompleteRequest(commonExtension->DeviceObject, OriginalIrp, IO_NO_INCREMENT);
                status = STATUS_MORE_PROCESSING_REQUIRED;
            }

            return status;
        }
    }

    return STATUS_MORE_PROCESSING_REQUIRED;
} // end ClasspPowerUpCompletion()

/*++////////////////////////////////////////////////////////////////////////////

ClasspPowerDownCompletion()

Routine Description:

    This routine is used for intermediate completion of a power up request.
    PowerUp requires four requests to be sent to the lower driver in sequence.

        * The queue is "power locked" to ensure that the class driver power-up
          work can be done before request processing resumes.

        * The power irp is sent down the stack for any filter drivers and the
          port driver to return power and resume command processing for the
          device.  Since the queue is locked, no queued irps will be sent
          immediately.

        * A start unit command is issued to the device with appropriate flags
          to override the "power locked" queue.

        * The queue is "power unlocked" to start processing requests again.

    This routine uses the function in the srb which just completed to determine
    which state it is in.

Arguments:

    DeviceObject - the device object being powered up

    Irp - the IO_REQUEST_PACKET containing the power request

    Context - the class power context used to perform port/class operations.

Return Value:

    STATUS_MORE_PROCESSING_REQUIRED or
    STATUS_SUCCESS

--*/
NTSTATUS
ClasspPowerDownCompletion(
    IN PDEVICE_OBJECT DeviceObject,
    IN PIRP Irp,
    IN PVOID Context
    )
{
    PCLASS_POWER_CONTEXT PowerContext = (PCLASS_POWER_CONTEXT)Context;
    PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = PowerContext->DeviceObject->DeviceExtension;
    PCOMMON_DEVICE_EXTENSION commonExtension = PowerContext->DeviceObject->DeviceExtension;
    PIRP OriginalIrp = PowerContext->Irp;

    // currentStack is for original power irp
    // nextStack is for power process irp
    PIO_STACK_LOCATION currentStack = IoGetCurrentIrpStackLocation(OriginalIrp);
    PIO_STACK_LOCATION nextStack = IoGetNextIrpStackLocation(fdoExtension->PrivateFdoData->PowerProcessIrp);

    NTSTATUS status = STATUS_MORE_PROCESSING_REQUIRED;
    PSTORAGE_REQUEST_BLOCK_HEADER srbHeader;
    ULONG srbFlags;

    UNREFERENCED_PARAMETER(DeviceObject);

    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "ClasspPowerDownCompletion: Device Object %p, "
                   "Irp %p, Context %p\n",
                PowerContext->DeviceObject, Irp, Context));

    if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(fdoExtension->PrivateFdoData->PowerSrb.SrbEx);

        //
        // Check if reverted to using legacy SRB.
        //
        if (PowerContext->Srb.Length == sizeof(SCSI_REQUEST_BLOCK)) {
            srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
        }
    } else {
        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
    }

    srbFlags = SrbGetSrbFlags(srbHeader);
    NT_ASSERT(!TEST_FLAG(srbFlags, SRB_FLAGS_FREE_SENSE_BUFFER));
    NT_ASSERT(!TEST_FLAG(srbFlags, SRB_FLAGS_PORT_DRIVER_ALLOCSENSE));
    NT_ASSERT(PowerContext->Options.PowerDown == TRUE);
    NT_ASSERT(PowerContext->Options.HandleSpinDown);

    if ((Irp == OriginalIrp) && (Irp->PendingReturned)) {
        // only for original power irp
        IoMarkIrpPending(Irp);
    }

    PowerContext->PowerChangeState.PowerDown3++;

    switch(PowerContext->PowerChangeState.PowerDown3) {

        case PowerDownDeviceLocked3: {

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously sent power lock\n", Irp));

            if ((PowerContext->Options.LockQueue == TRUE) &&
                (!NT_SUCCESS(Irp->IoStatus.Status))) {

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIrp status was %lx\n",
                            Irp,
                            Irp->IoStatus.Status));
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tSrb status was %lx\n",
                            Irp,
                            srbHeader->SrbStatus));

                OriginalIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;

                //
                // Lock was not successful - throw down the power IRP
                // by itself and don't try to spin down the drive or unlock
                // the queue.
                //

                //
                // Set the new power state
                //

                fdoExtension->DevicePowerState =
                    currentStack->Parameters.Power.State.DeviceState;

                //
                // Indicate to Po that we've been successfully powered down
                // so it can do it's notification stuff.
                //

                IoCopyCurrentIrpStackLocationToNext(OriginalIrp);
                IoSetCompletionRoutine(OriginalIrp,
                                       ClasspStartNextPowerIrpCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                PoSetPowerState(PowerContext->DeviceObject,
                                currentStack->Parameters.Power.Type,
                                currentStack->Parameters.Power.State);

                fdoExtension->PowerDownInProgress = FALSE;

                ClassReleaseRemoveLock(commonExtension->DeviceObject,
                                       OriginalIrp);

                PoCallDriver(commonExtension->LowerDeviceObject, OriginalIrp);

                return STATUS_MORE_PROCESSING_REQUIRED;

            } else {
                //
                // Lock the device queue succeeded. Now wait for all outstanding IO to complete.
                // To do this, Srb with SRB_FUNCTION_QUIESCE_DEVICE will be sent down with default timeout value.
                // We need to tolerant failure of this request, no retry will be made.
                //
                PowerContext->QueueLocked = (UCHAR) PowerContext->Options.LockQueue;

                //
                // No retry on device quiescence reqeust
                //
                fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = 0;
                PowerContext->RetryCount = 0;

                if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
                    srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(fdoExtension->PrivateFdoData->PowerSrb.SrbEx);

                    //
                    // Initialize extended SRB for a SRB_FUNCTION_LOCK_QUEUE
                    //
                    status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)srbHeader,
                                                           STORAGE_ADDRESS_TYPE_BTL8,
                                                           CLASS_SRBEX_NO_SRBEX_DATA_BUFFER_SIZE,
                                                           0);
                    if (NT_SUCCESS(status)) {
                        ((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction = SRB_FUNCTION_QUIESCE_DEVICE;
                    } else {
                        //
                        // Should not happen. Revert to legacy SRB.
                        //
                        NT_ASSERT(FALSE);
                        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
                        srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                        srbHeader->Function = SRB_FUNCTION_QUIESCE_DEVICE;
                    }
                } else {
                    srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
                    srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                    srbHeader->Function = SRB_FUNCTION_QUIESCE_DEVICE;
                }

                SrbSetOriginalRequest(srbHeader, fdoExtension->PrivateFdoData->PowerProcessIrp);
                SrbSetTimeOutValue(srbHeader, fdoExtension->TimeOutValue);

                SrbAssignSrbFlags(srbHeader,
                                     (SRB_FLAGS_NO_DATA_TRANSFER |
                                      SRB_FLAGS_DISABLE_AUTOSENSE |
                                      SRB_FLAGS_DISABLE_SYNCH_TRANSFER |
                                      SRB_FLAGS_NO_QUEUE_FREEZE |
                                      SRB_FLAGS_BYPASS_LOCKED_QUEUE |
                                      SRB_FLAGS_D3_PROCESSING));

                IoSetCompletionRoutine(fdoExtension->PrivateFdoData->PowerProcessIrp,
                                       ClasspPowerDownCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                nextStack->Parameters.Scsi.Srb = (PSCSI_REQUEST_BLOCK)srbHeader;
                nextStack->MajorFunction = IRP_MJ_SCSI;

                status = IoCallDriver(commonExtension->LowerDeviceObject, fdoExtension->PrivateFdoData->PowerProcessIrp);

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIoCallDriver returned %lx\n", fdoExtension->PrivateFdoData->PowerProcessIrp, status));
                break;
            }

        }

        case PowerDownDeviceQuiesced3: {

            PCDB cdb;

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously sent device quiesce\n", Irp));

            //
            // don't care the result of device quiesce, we've made the effort.
            // continue on sending other SCSI commands anyway.
            //

            if (!TEST_FLAG(fdoExtension->PrivateFdoData->HackFlags,
                           FDO_HACK_NO_SYNC_CACHE)) {

                //
                // send SCSIOP_SYNCHRONIZE_CACHE
                //

                fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = MAXIMUM_RETRIES;
                PowerContext->RetryCount = MAXIMUM_RETRIES;

                if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
                    status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)srbHeader,
                                                            STORAGE_ADDRESS_TYPE_BTL8,
                                                            CLASS_SRBEX_SCSI_CDB16_BUFFER_SIZE,
                                                            1,
                                                            SrbExDataTypeScsiCdb16);
                    if (NT_SUCCESS(status)) {
                        ((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction = SRB_FUNCTION_EXECUTE_SCSI;

                        //
                        // Set length field in Power Context SRB so we know legacy SRB is not being used.
                        //
                        PowerContext->Srb.Length = 0;

                    } else {
                        //
                        // Should not occur. Revert to legacy SRB.
                        NT_ASSERT(FALSE);
                        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
                        RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                        srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                        srbHeader->Function = SRB_FUNCTION_EXECUTE_SCSI;
                    }

                } else {
                    RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                    srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                    srbHeader->Function = SRB_FUNCTION_EXECUTE_SCSI;
                }


                SrbSetOriginalRequest(srbHeader, fdoExtension->PrivateFdoData->PowerProcessIrp);
                SrbSetSenseInfoBuffer(srbHeader, commonExtension->PartitionZeroExtension->SenseData);
                SrbSetSenseInfoBufferLength(srbHeader, GET_FDO_EXTENSON_SENSE_DATA_LENGTH(commonExtension->PartitionZeroExtension));
                SrbSetTimeOutValue(srbHeader, fdoExtension->TimeOutValue);

                SrbAssignSrbFlags(srbHeader,
                                     (SRB_FLAGS_NO_DATA_TRANSFER |
                                      SRB_FLAGS_DISABLE_AUTOSENSE |
                                      SRB_FLAGS_DISABLE_SYNCH_TRANSFER |
                                      SRB_FLAGS_NO_QUEUE_FREEZE |
                                      SRB_FLAGS_BYPASS_LOCKED_QUEUE |
                                      SRB_FLAGS_D3_PROCESSING));

                SrbSetCdbLength(srbHeader, 10);

                cdb = SrbGetCdb(srbHeader);

                RtlZeroMemory(cdb, sizeof(CDB));
                cdb->SYNCHRONIZE_CACHE10.OperationCode = SCSIOP_SYNCHRONIZE_CACHE;

                IoSetCompletionRoutine(fdoExtension->PrivateFdoData->PowerProcessIrp,
                                       ClasspPowerDownCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                nextStack->Parameters.Scsi.Srb = (PSCSI_REQUEST_BLOCK)srbHeader;
                nextStack->MajorFunction = IRP_MJ_SCSI;

                status = IoCallDriver(commonExtension->LowerDeviceObject, fdoExtension->PrivateFdoData->PowerProcessIrp);

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIoCallDriver returned %lx\n", fdoExtension->PrivateFdoData->PowerProcessIrp, status));
                break;

            } else {

                TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_POWER, "(%p)\tPower Down: not sending SYNCH_CACHE\n",
                            PowerContext->DeviceObject));
                PowerContext->PowerChangeState.PowerDown3++;
                srbHeader->SrbStatus = SRB_STATUS_SUCCESS;
                // and fall through....
            }
            // no break in case the device doesn't like synch_cache commands

        }

        case PowerDownDeviceFlushed3: {

            PCDB cdb;

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously send SCSIOP_SYNCHRONIZE_CACHE\n",
                        Irp));

            //
            // SCSIOP_SYNCHRONIZE_CACHE was sent
            //

            if (SRB_STATUS(srbHeader->SrbStatus) != SRB_STATUS_SUCCESS) {

                BOOLEAN retry;
                LONGLONG delta100nsUnits = 0;

                TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_POWER, "(%p)\tError occured when issuing "
                            "SYNCHRONIZE_CACHE command to device. "
                            "Srb %p, Status %lx\n",
                            Irp,
                            srbHeader,
                            srbHeader->SrbStatus));

                NT_ASSERT(!(TEST_FLAG(srbHeader->SrbStatus, SRB_STATUS_QUEUE_FROZEN)));
                NT_ASSERT((srbHeader->Function == SRB_FUNCTION_EXECUTE_SCSI) ||
                          (((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction == SRB_FUNCTION_EXECUTE_SCSI));

                PowerContext->RetryInterval = 0;
                retry = InterpretSenseInfoWithoutHistory(
                            fdoExtension->DeviceObject,
                            Irp,
                            (PSCSI_REQUEST_BLOCK)srbHeader,
                            IRP_MJ_SCSI,
                            IRP_MJ_POWER,
                            fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount - PowerContext->RetryCount,
                            &status,
                            &delta100nsUnits);

                // NOTE: Power context is a public structure, and thus cannot be
                //       updated to use 100ns units.  Therefore, must store the
                //       one-second equivalent.  Round up to ensure minimum delay
                //       requirements have been met.
                delta100nsUnits += (10*1000*1000) - 1;
                delta100nsUnits /= (10*1000*1000);
                // guaranteed not to have high bits set per SAL annotations
                PowerContext->RetryInterval = (ULONG)(delta100nsUnits);


                if ((retry == TRUE) && (PowerContext->RetryCount-- != 0)) {

                    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tRetrying failed request\n", Irp));

                    //
                    // decrement the state so we come back through here
                    // the next time.
                    //

                    PowerContext->PowerChangeState.PowerDown3--;
                    RetryPowerRequest(commonExtension->DeviceObject,
                                      Irp,
                                      PowerContext);
                    break;
                }

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tSYNCHRONIZE_CACHE not retried\n", Irp));
                fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = MAXIMUM_RETRIES;
                PowerContext->RetryCount = MAXIMUM_RETRIES;
            } // end !SRB_STATUS_SUCCESS

            //
            // note: we are purposefully ignoring any errors.  if the drive
            //       doesn't support a synch_cache, then we're up a creek
            //       anyways.
            //

            if ((currentStack->Parameters.Power.State.DeviceState == PowerDeviceD3) &&
                (currentStack->Parameters.Power.ShutdownType == PowerActionHibernate) &&
                (commonExtension->HibernationPathCount != 0)) {

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPower Down: not sending SPIN DOWN due to hibernation path\n",
                            PowerContext->DeviceObject));

                PowerContext->PowerChangeState.PowerDown3++;
                srbHeader->SrbStatus = SRB_STATUS_SUCCESS;
                status = STATUS_SUCCESS;

                // Fall through to next case...

            } else {
                // Send STOP UNIT command. As "Imme" bit is set to '1', this command should be completed in short time.
                // This command is at low importance, failure of this command has very small impact.

                ULONG secondsRemaining;
                ULONG timeoutValue;

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tSending stop unit to device\n", Irp));

                if (PoQueryWatchdogTime(fdoExtension->LowerPdo, &secondsRemaining)) {
                    // plan to leave some time (TIME_LEFT_FOR_LOWER_DRIVERS) to lower level drivers
                    // for processing the original power irp.
                    if (secondsRemaining >= (TIME_LEFT_FOR_LOWER_DRIVERS + DEFAULT_IO_TIMEOUT_VALUE)) {
                        fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount =
                            (secondsRemaining - TIME_LEFT_FOR_LOWER_DRIVERS) / DEFAULT_IO_TIMEOUT_VALUE;

                        // * No 'short' timeouts
                        //
                        // timeoutValue = (secondsRemaining - TIME_LEFT_FOR_LOWER_DRIVERS) %
                        //                DEFAULT_IO_TIMEOUT_VALUE;
                        // if (timeoutValue < MINIMUM_STOP_UNIT_TIMEOUT_VALUE)
                        // {
                        if (--fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount)
                        {
                            timeoutValue = DEFAULT_IO_TIMEOUT_VALUE;
                        } else {
                            timeoutValue = secondsRemaining - TIME_LEFT_FOR_LOWER_DRIVERS;
                        }
                        // }

                        // Limit to maximum retry count.
                        if (fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount > MAXIMUM_RETRIES) {
                            fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = MAXIMUM_RETRIES;
                        }
                    } else {
                        // issue the command with minimum timeout value and do not retry on it.
                        fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = 0;

                        // minimum as MINIMUM_STOP_UNIT_TIMEOUT_VALUE.
                        if (secondsRemaining > 2 * MINIMUM_STOP_UNIT_TIMEOUT_VALUE) {
                            timeoutValue = secondsRemaining - MINIMUM_STOP_UNIT_TIMEOUT_VALUE;
                        } else {
                            timeoutValue = MINIMUM_STOP_UNIT_TIMEOUT_VALUE;
                        }

                    }

                } else {
                    // do not know how long, use default values.
                    fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = MAXIMUM_RETRIES;
                    timeoutValue = DEFAULT_IO_TIMEOUT_VALUE;
                }

                //
                // Issue STOP UNIT command to the device.
                //

                PowerContext->RetryCount = fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount;

                if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
                    status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)srbHeader,
                                                            STORAGE_ADDRESS_TYPE_BTL8,
                                                            CLASS_SRBEX_SCSI_CDB16_BUFFER_SIZE,
                                                            1,
                                                            SrbExDataTypeScsiCdb16);
                    if (NT_SUCCESS(status)) {
                        ((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction = SRB_FUNCTION_EXECUTE_SCSI;

                        //
                        // Set length field in Power Context SRB so we know legacy SRB is not being used.
                        //
                        PowerContext->Srb.Length = 0;

                    } else {
                        //
                        // Should not occur. Revert to legacy SRB.
                        //
                        NT_ASSERT(FALSE);
                        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
                        RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                        srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                        srbHeader->Function = SRB_FUNCTION_EXECUTE_SCSI;
                    }

                } else {
                    RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                    srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                    srbHeader->Function = SRB_FUNCTION_EXECUTE_SCSI;
                }

                SrbSetOriginalRequest(srbHeader, fdoExtension->PrivateFdoData->PowerProcessIrp);
                SrbSetSenseInfoBuffer(srbHeader, commonExtension->PartitionZeroExtension->SenseData);
                SrbSetSenseInfoBufferLength(srbHeader, GET_FDO_EXTENSON_SENSE_DATA_LENGTH(commonExtension->PartitionZeroExtension));
                SrbSetTimeOutValue(srbHeader, timeoutValue);


                SrbAssignSrbFlags(srbHeader,
                                     (SRB_FLAGS_NO_DATA_TRANSFER |
                                      SRB_FLAGS_DISABLE_AUTOSENSE |
                                      SRB_FLAGS_DISABLE_SYNCH_TRANSFER |
                                      SRB_FLAGS_NO_QUEUE_FREEZE |
                                      SRB_FLAGS_BYPASS_LOCKED_QUEUE |
                                      SRB_FLAGS_D3_PROCESSING));

                SrbSetCdbLength(srbHeader, 6);

                cdb = SrbGetCdb(srbHeader);
                RtlZeroMemory(cdb, sizeof(CDB));

                cdb->START_STOP.OperationCode = SCSIOP_START_STOP_UNIT;
                cdb->START_STOP.Start = 0;
                cdb->START_STOP.Immediate = 1;

                IoSetCompletionRoutine(fdoExtension->PrivateFdoData->PowerProcessIrp,
                                       ClasspPowerDownCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                nextStack->Parameters.Scsi.Srb = (PSCSI_REQUEST_BLOCK)srbHeader;
                nextStack->MajorFunction = IRP_MJ_SCSI;

                status = IoCallDriver(commonExtension->LowerDeviceObject, fdoExtension->PrivateFdoData->PowerProcessIrp);

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIoCallDriver returned %lx\n", fdoExtension->PrivateFdoData->PowerProcessIrp, status));
                break;
            }
        }

        case PowerDownDeviceStopped3: {

            BOOLEAN ignoreError = TRUE;

            //
            // stop was sent
            //

            if (SRB_STATUS(srbHeader->SrbStatus) != SRB_STATUS_SUCCESS) {

                BOOLEAN retry;
                LONGLONG delta100nsUnits = 0;

                TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_POWER, "(%p)\tError occured when issueing STOP_UNIT "
                            "command to device. Srb %p, Status %lx\n",
                            Irp,
                            srbHeader,
                            srbHeader->SrbStatus));

                NT_ASSERT(!(TEST_FLAG(srbHeader->SrbStatus, SRB_STATUS_QUEUE_FROZEN)));
                NT_ASSERT((srbHeader->Function == SRB_FUNCTION_EXECUTE_SCSI) ||
                          (((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction == SRB_FUNCTION_EXECUTE_SCSI));

                PowerContext->RetryInterval = 0;
                retry = InterpretSenseInfoWithoutHistory(
                            fdoExtension->DeviceObject,
                            Irp,
                            (PSCSI_REQUEST_BLOCK)srbHeader,
                            IRP_MJ_SCSI,
                            IRP_MJ_POWER,
                            fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount - PowerContext->RetryCount,
                            &status,
                            &delta100nsUnits);

                // NOTE: Power context is a public structure, and thus cannot be
                //       updated to use 100ns units.  Therefore, must store the
                //       one-second equivalent.  Round up to ensure minimum delay
                //       requirements have been met.
                delta100nsUnits += (10*1000*1000) - 1;
                delta100nsUnits /= (10*1000*1000);
                // guaranteed not to have high bits set per SAL annotations
                PowerContext->RetryInterval = (ULONG)(delta100nsUnits);


                if ((retry == TRUE) && (PowerContext->RetryCount-- != 0)) {

                    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tRetrying failed request\n", Irp));

                    //
                    // decrement the state so we come back through here
                    // the next time.
                    //

                    PowerContext->PowerChangeState.PowerDown3--;

                    SrbSetTimeOutValue(srbHeader, DEFAULT_IO_TIMEOUT_VALUE);

                    RetryPowerRequest(commonExtension->DeviceObject,
                                      Irp,
                                      PowerContext);
                    break;
                }

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tSTOP_UNIT not retried\n", Irp));
                fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = MAXIMUM_RETRIES;
                PowerContext->RetryCount = MAXIMUM_RETRIES;

            } // end !SRB_STATUS_SUCCESS


            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously sent stop unit\n", Irp));

            //
            // some operations, such as a physical format in progress,
            // should not be ignored and should fail the power operation.
            //

            if (!NT_SUCCESS(status)) {

                PVOID senseBuffer = SrbGetSenseInfoBuffer(srbHeader);

                if (TEST_FLAG(srbHeader->SrbStatus, SRB_STATUS_AUTOSENSE_VALID) &&
                    (senseBuffer != NULL)) {

                    BOOLEAN validSense = FALSE;
                    UCHAR senseKey = 0;
                    UCHAR additionalSenseCode = 0;
                    UCHAR additionalSenseCodeQualifier = 0;

                    validSense = ScsiGetSenseKeyAndCodes(senseBuffer,
                                                         SrbGetSenseInfoBufferLength(srbHeader),
                                                         SCSI_SENSE_OPTIONS_FIXED_FORMAT_IF_UNKNOWN_FORMAT_INDICATED,
                                                         &senseKey,
                                                         &additionalSenseCode,
                                                         &additionalSenseCodeQualifier);

                    if (validSense) {
                        if ((senseKey == SCSI_SENSE_NOT_READY) &&
                            (additionalSenseCode == SCSI_ADSENSE_LUN_NOT_READY) &&
                            (additionalSenseCodeQualifier == SCSI_SENSEQ_FORMAT_IN_PROGRESS)) {

                            ignoreError = FALSE;
                            PowerContext->FinalStatus = STATUS_DEVICE_BUSY;
                            status = PowerContext->FinalStatus;
                        }
                    }
                }
            }

            if (NT_SUCCESS(status) || ignoreError) {

                //
                // Issue the original power request to the lower driver.
                //

                OriginalIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;

                IoCopyCurrentIrpStackLocationToNext(OriginalIrp);

                IoSetCompletionRoutine(OriginalIrp,
                                       ClasspPowerDownCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                status = PoCallDriver(commonExtension->LowerDeviceObject, OriginalIrp);

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPoCallDriver returned %lx\n", OriginalIrp, status));
                break;
            }

            // else fall through w/o sending the power irp, since the device
            // is reporting an error that would be "really bad" to power down
            // during.

        }

        case PowerDownDeviceOff3: {

            //
            // SpinDown request completed ... whether it succeeded or not is
            // another matter entirely.
            //

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously sent power irp\n", OriginalIrp));

            if (PowerContext->QueueLocked) {

                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tUnlocking queue\n", OriginalIrp));

                if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
                    //
                    // Will reuse SRB for a non-SCSI SRB.
                    //
                    status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)srbHeader,
                                                           STORAGE_ADDRESS_TYPE_BTL8,
                                                           CLASS_SRBEX_NO_SRBEX_DATA_BUFFER_SIZE,
                                                           0);
                    if (NT_SUCCESS(status)) {
                        ((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction = SRB_FUNCTION_UNLOCK_QUEUE;

                        //
                        // Set length field in Power Context SRB so we know legacy SRB is not being used.
                        //
                        PowerContext->Srb.Length = 0;

                    } else {
                        //
                        // Should not occur. Revert to legacy SRB.
                        //
                        NT_ASSERT(FALSE);
                        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(PowerContext->Srb);
                        RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                        srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                        srbHeader->Function = SRB_FUNCTION_UNLOCK_QUEUE;
                    }
                } else {
                    RtlZeroMemory(srbHeader, sizeof(SCSI_REQUEST_BLOCK));
                    srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
                    srbHeader->Function = SRB_FUNCTION_UNLOCK_QUEUE;
                }

                SrbSetOriginalRequest(srbHeader, fdoExtension->PrivateFdoData->PowerProcessIrp);
                SrbAssignSrbFlags(srbHeader, (SRB_FLAGS_BYPASS_LOCKED_QUEUE |
                                              SRB_FLAGS_D3_PROCESSING));

                nextStack->Parameters.Scsi.Srb = (PSCSI_REQUEST_BLOCK)srbHeader;
                nextStack->MajorFunction = IRP_MJ_SCSI;

                IoSetCompletionRoutine(fdoExtension->PrivateFdoData->PowerProcessIrp,
                                       ClasspPowerDownCompletion,
                                       PowerContext,
                                       TRUE,
                                       TRUE,
                                       TRUE);

                status = IoCallDriver(commonExtension->LowerDeviceObject, fdoExtension->PrivateFdoData->PowerProcessIrp);
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIoCallDriver returned %lx\n",
                            fdoExtension->PrivateFdoData->PowerProcessIrp,
                            status));
                break;
            }

        }

        case PowerDownDeviceUnlocked3: {

            //
            // This is the end of the dance.
            // We're ignoring possible intermediate error conditions ....
            //

            if (PowerContext->QueueLocked == FALSE) {
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tFall through (queue not locked)\n", OriginalIrp));
            } else {
                TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPreviously unlocked queue\n", OriginalIrp));
                NT_ASSERT(NT_SUCCESS(Irp->IoStatus.Status));
                NT_ASSERT(srbHeader->SrbStatus == SRB_STATUS_SUCCESS);

                if (NT_SUCCESS(Irp->IoStatus.Status)) {
                    PowerContext->QueueLocked = FALSE;
                }
            }

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tFreeing srb and completing\n", OriginalIrp));
            status = PowerContext->FinalStatus; // allow failure to propogate

            OriginalIrp->IoStatus.Status = status;
            OriginalIrp->IoStatus.Information = 0;

            if (NT_SUCCESS(status)) {

                //
                // Set the new power state
                //

                fdoExtension->DevicePowerState =
                    currentStack->Parameters.Power.State.DeviceState;

            }

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tStarting next power irp\n", OriginalIrp));

            ClassReleaseRemoveLock(PowerContext->DeviceObject, OriginalIrp);

            PowerContext->InUse = FALSE;

            PoStartNextPowerIrp(OriginalIrp);

            fdoExtension->PowerDownInProgress = FALSE;

            // prevent from completing the irp allocated by ourselves
            if (Irp == fdoExtension->PrivateFdoData->PowerProcessIrp) {
                // complete original irp if we are processing powerprocess irp,
                // otherwise, by returning status other than STATUS_MORE_PROCESSING_REQUIRED, IO manager will complete it.
                ClassCompleteRequest(commonExtension->DeviceObject, OriginalIrp, IO_NO_INCREMENT);
                status = STATUS_MORE_PROCESSING_REQUIRED;
            }

            return status;
        }
    }

    return STATUS_MORE_PROCESSING_REQUIRED;
} // end ClasspPowerDownCompletion()

/*++////////////////////////////////////////////////////////////////////////////

ClasspPowerHandler()

Routine Description:

    This routine reduces the number of useless spinups and spindown requests
    sent to a given device by ignoring transitions to power states we are
    currently in.

    ISSUE-2000/02/20-henrygab - by ignoring spin-up requests, we may be
          allowing the drive

Arguments:

    DeviceObject - the device object which is transitioning power states
    Irp - the power irp
    Options - a set of flags indicating what the device handles

Return Value:

--*/
NTSTATUS
ClasspPowerHandler(
    IN PDEVICE_OBJECT DeviceObject,
    IN PIRP Irp,
    IN CLASS_POWER_OPTIONS Options  // ISSUE-2000/02/20-henrygab - pass pointer, not whole struct
    )
{
    PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
    PDEVICE_OBJECT lowerDevice = commonExtension->LowerDeviceObject;
    PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
    PIO_STACK_LOCATION nextIrpStack;
    PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
    PCLASS_POWER_CONTEXT context;
    PSTORAGE_REQUEST_BLOCK_HEADER srbHeader;
    ULONG srbFlags;
    NTSTATUS status;

    _Analysis_assume_(fdoExtension);
    _Analysis_assume_(fdoExtension->PrivateFdoData);

    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "ClasspPowerHandler: Power irp %p to %s %p\n",
                Irp, (commonExtension->IsFdo ? "fdo" : "pdo"), DeviceObject));

    if (!commonExtension->IsFdo) {

        //
        // certain assumptions are made here,
        // particularly: having the fdoExtension
        //

        TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_POWER, "ClasspPowerHandler: Called for PDO %p???\n",
                    DeviceObject));
        NT_ASSERT(!"PDO using ClasspPowerHandler");

        ClassReleaseRemoveLock(DeviceObject, Irp);
        Irp->IoStatus.Status = STATUS_NOT_SUPPORTED;
        PoStartNextPowerIrp(Irp);
        ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
        return STATUS_NOT_SUPPORTED;
    }

    switch (irpStack->MinorFunction) {

        case IRP_MN_SET_POWER: {
            PCLASS_PRIVATE_FDO_DATA fdoData = fdoExtension->PrivateFdoData;

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIRP_MN_SET_POWER\n", Irp));

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tSetting %s state to %d\n",
                        Irp,
                        (irpStack->Parameters.Power.Type == SystemPowerState ?
                            "System" : "Device"),
                        irpStack->Parameters.Power.State.SystemState));

            switch (irpStack->Parameters.Power.ShutdownType){

                case PowerActionNone:

                    //
                    // Skip if device doesn't need volume verification during idle power
                    // transitions.
                    //
                    if ((fdoExtension->FunctionSupportInfo) &&
                        (fdoExtension->FunctionSupportInfo->IdlePower.NoVerifyDuringIdlePower)) {
                        break;
                    }

                case PowerActionSleep:
                case PowerActionHibernate:
                    if (fdoData->HotplugInfo.MediaRemovable || fdoData->HotplugInfo.MediaHotplug) {
                        /*
                            *  We are suspending device and this drive is either hot-pluggable
                            *  or contains removeable media.
                            *  Set the media dirty bit, since the media may change while
                            *  we are suspended.
                            */
                        SET_FLAG(DeviceObject->Flags, DO_VERIFY_VOLUME);

                        //
                        // Bumping the media  change count  will force the
                        // file system to verify the volume when we resume
                        //

                        InterlockedIncrement((volatile LONG *)&fdoExtension->MediaChangeCount);
                    }

                    break;
                }

            break;
        }

        default: {

            TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tIrp minor code = %#x\n",
                        Irp, irpStack->MinorFunction));
            break;
        }
    }

    if (irpStack->Parameters.Power.Type != DevicePowerState ||
        irpStack->MinorFunction != IRP_MN_SET_POWER) {

        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tSending to lower device\n", Irp));

        goto ClasspPowerHandlerCleanup;

    }

    //
    // already in exact same state, don't work to transition to it.
    //

    if (irpStack->Parameters.Power.State.DeviceState ==
        fdoExtension->DevicePowerState) {

        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tAlready in device state %x\n",
                    Irp, fdoExtension->DevicePowerState));
        goto ClasspPowerHandlerCleanup;

    }

    //
    // or powering down from non-d0 state (device already stopped)
    // NOTE -- we're not sure whether this case can exist or not (the
    // power system may never send this sort of request) but it's trivial
    // to deal with.
    //

    if ((irpStack->Parameters.Power.State.DeviceState != PowerDeviceD0) &&
        (fdoExtension->DevicePowerState != PowerDeviceD0)) {
        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tAlready powered down to %x???\n",
                    Irp, fdoExtension->DevicePowerState));
        fdoExtension->DevicePowerState =
            irpStack->Parameters.Power.State.DeviceState;
        goto ClasspPowerHandlerCleanup;
    }

    //
    // or when not handling powering up and are powering up
    //

    if ((!Options.HandleSpinUp) &&
        (irpStack->Parameters.Power.State.DeviceState == PowerDeviceD0)) {

        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tNot handling spinup to state %x\n",
                    Irp, fdoExtension->DevicePowerState));
        fdoExtension->DevicePowerState =
            irpStack->Parameters.Power.State.DeviceState;
        goto ClasspPowerHandlerCleanup;

    }

    //
    // or when not handling powering down and are powering down
    //

    if ((!Options.HandleSpinDown) &&
        (irpStack->Parameters.Power.State.DeviceState != PowerDeviceD0)) {

        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tNot handling spindown to state %x\n",
                    Irp, fdoExtension->DevicePowerState));
        fdoExtension->DevicePowerState =
            irpStack->Parameters.Power.State.DeviceState;
        goto ClasspPowerHandlerCleanup;

    }

    //
    // validation completed, start the real work.
    //

    IoReuseIrp(fdoExtension->PrivateFdoData->PowerProcessIrp, STATUS_SUCCESS);
    IoSetNextIrpStackLocation(fdoExtension->PrivateFdoData->PowerProcessIrp);
    nextIrpStack = IoGetNextIrpStackLocation(fdoExtension->PrivateFdoData->PowerProcessIrp);

    context = &(fdoExtension->PowerContext);

    NT_ASSERT(context->InUse == FALSE);

    RtlZeroMemory(context, sizeof(CLASS_POWER_CONTEXT));
    context->InUse = TRUE;

    if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(fdoExtension->PrivateFdoData->PowerSrb.SrbEx);

        //
        // Initialize extended SRB for a SRB_FUNCTION_LOCK_QUEUE
        //
        status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)srbHeader,
                                               STORAGE_ADDRESS_TYPE_BTL8,
                                               CLASS_SRBEX_NO_SRBEX_DATA_BUFFER_SIZE,
                                               0);
        if (NT_SUCCESS(status)) {
            ((PSTORAGE_REQUEST_BLOCK)srbHeader)->SrbFunction = SRB_FUNCTION_LOCK_QUEUE;
        } else {
            //
            // Should not happen. Revert to legacy SRB.
            //
            NT_ASSERT(FALSE);
            srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(context->Srb);
            srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
            srbHeader->Function = SRB_FUNCTION_LOCK_QUEUE;
        }
    } else {
        srbHeader = (PSTORAGE_REQUEST_BLOCK_HEADER)&(context->Srb);
        srbHeader->Length = sizeof(SCSI_REQUEST_BLOCK);
        srbHeader->Function = SRB_FUNCTION_LOCK_QUEUE;
    }
    nextIrpStack->Parameters.Scsi.Srb = (PSCSI_REQUEST_BLOCK)srbHeader;
    nextIrpStack->MajorFunction = IRP_MJ_SCSI;

    context->FinalStatus = STATUS_SUCCESS;

    SrbSetOriginalRequest(srbHeader, fdoExtension->PrivateFdoData->PowerProcessIrp);
    SrbSetSrbFlags(srbHeader, (SRB_FLAGS_BYPASS_LOCKED_QUEUE | SRB_FLAGS_NO_QUEUE_FREEZE));

    fdoExtension->PrivateFdoData->MaxPowerOperationRetryCount = MAXIMUM_RETRIES;
    context->RetryCount = MAXIMUM_RETRIES;

    context->Options = Options;
    context->DeviceObject = DeviceObject;
    context->Irp = Irp;

    if (irpStack->Parameters.Power.State.DeviceState == PowerDeviceD0) {

        NT_ASSERT(Options.HandleSpinUp);

        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tpower up - locking queue\n", Irp));

        //
        // We need to issue a queue lock request so that we
        // can spin the drive back up after the power is restored
        // but before any requests are processed.
        //

        context->Options.PowerDown = FALSE;
        context->PowerChangeState.PowerUp = PowerUpDeviceInitial;
        context->CompletionRoutine = ClasspPowerUpCompletion;

    } else {

        NT_ASSERT(Options.HandleSpinDown);

        fdoExtension->PowerDownInProgress = TRUE;

        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tPowering down - locking queue\n", Irp));

        //
        // Disable tick timer at beginning of D3 processing if running.
        //
        if ((fdoExtension->PrivateFdoData->TickTimerEnabled)) {
            ClasspDisableTimer(fdoExtension);
        }

        PoSetPowerState(DeviceObject,
                        irpStack->Parameters.Power.Type,
                        irpStack->Parameters.Power.State);

        context->Options.PowerDown = TRUE;
        context->PowerChangeState.PowerDown3 = PowerDownDeviceInitial3;
        context->CompletionRoutine = ClasspPowerDownCompletion;

    }

    //
    // we are not dealing with port-allocated sense in these routines.
    //

    srbFlags = SrbGetSrbFlags(srbHeader);
    NT_ASSERT(!TEST_FLAG(srbFlags, SRB_FLAGS_FREE_SENSE_BUFFER));
    NT_ASSERT(!TEST_FLAG(srbFlags, SRB_FLAGS_PORT_DRIVER_ALLOCSENSE));

    //
    // Mark the original power irp pending.
    //

    IoMarkIrpPending(Irp);

    if (Options.LockQueue) {

        //
        // Send the lock irp down.
        //

        IoSetCompletionRoutine(fdoExtension->PrivateFdoData->PowerProcessIrp,
                               context->CompletionRoutine,
                               context,
                               TRUE,
                               TRUE,
                               TRUE);

        IoCallDriver(lowerDevice, fdoExtension->PrivateFdoData->PowerProcessIrp);

    } else {

        //
        // Call the completion routine directly.  It won't care what the
        // status of the "lock" was - it will just go and do the next
        // step of the operation.
        //

        context->CompletionRoutine(DeviceObject, fdoExtension->PrivateFdoData->PowerProcessIrp, context);
    }

    return STATUS_PENDING;

ClasspPowerHandlerCleanup:

    //
    // Send the original power irp down, we will start the next power irp in completion routine.
    //
    ClassReleaseRemoveLock(DeviceObject, Irp);

    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tStarting next power irp\n", Irp));
    IoCopyCurrentIrpStackLocationToNext(Irp);
    IoSetCompletionRoutine(Irp,
                           ClasspStartNextPowerIrpCompletion,
                           NULL,
                           TRUE,
                           TRUE,
                           TRUE);
    return PoCallDriver(lowerDevice, Irp);
} // end ClasspPowerHandler()

/*++////////////////////////////////////////////////////////////////////////////

ClassMinimalPowerHandler()

Routine Description:

    This routine is the minimum power handler for a storage driver.  It does
    the least amount of work possible.

--*/
NTSTATUS
ClassMinimalPowerHandler(
    IN PDEVICE_OBJECT DeviceObject,
    IN PIRP Irp
    )
{
    PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
    PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
    NTSTATUS status;

    ClassReleaseRemoveLock(DeviceObject, Irp);
    PoStartNextPowerIrp(Irp);

    switch (irpStack->MinorFunction)
    {
        case IRP_MN_SET_POWER:
        {
            switch (irpStack->Parameters.Power.ShutdownType)
            {
                case PowerActionNone:
                case PowerActionSleep:
                case PowerActionHibernate:
                {
                    if (TEST_FLAG(DeviceObject->Characteristics, FILE_REMOVABLE_MEDIA))
                    {
                        if ((ClassGetVpb(DeviceObject) != NULL) && (ClassGetVpb(DeviceObject)->Flags & VPB_MOUNTED))
                        {
                            //
                            // This flag will cause the filesystem to verify the
                            // volume when coming out of hibernation or standby or runtime power
                            //
                            SET_FLAG(DeviceObject->Flags, DO_VERIFY_VOLUME);
                        }
                    }
                }
                break;
            }
        }

        //
        // Fall through
        //

        case IRP_MN_QUERY_POWER:
        {
            if (!commonExtension->IsFdo)
            {
                Irp->IoStatus.Status = STATUS_SUCCESS;
                Irp->IoStatus.Information = 0;
            }
        }
        break;
    }

    if (commonExtension->IsFdo)
    {
        IoCopyCurrentIrpStackLocationToNext(Irp);
        status = PoCallDriver(commonExtension->LowerDeviceObject, Irp);
    }
    else
    {
        status = Irp->IoStatus.Status;
        ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
    }

    return status;
} // end ClassMinimalPowerHandler()

/*++////////////////////////////////////////////////////////////////////////////

ClassSpinDownPowerHandler()

Routine Description:

    This routine is a callback for disks and other things which require both
    a start and a stop to be sent to the device.  (actually the starts are
    almost always optional, since most device power themselves on to process
    commands, but i digress).

    Determines proper use of spinup, spindown, and queue locking based upon
    ScanForSpecialFlags in the FdoExtension.  This is the most common power
    handler passed into classpnp.sys

Arguments:

    DeviceObject - Supplies the functional device object

    Irp - Supplies the request to be retried.

Return Value:

    None

--*/
__control_entrypoint(DeviceDriver)
NTSTATUS
ClassSpinDownPowerHandler(
    _In_ PDEVICE_OBJECT DeviceObject,
    _In_ PIRP Irp
    )
{
    PFUNCTIONAL_DEVICE_EXTENSION fdoExtension;
    CLASS_POWER_OPTIONS options = {0};

    fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;

    //
    // check the flags to see what options we need to worry about
    //

    if (!TEST_FLAG(fdoExtension->ScanForSpecialFlags,
                  CLASS_SPECIAL_DISABLE_SPIN_DOWN)) {
        options.HandleSpinDown = TRUE;
    }

    if (!TEST_FLAG(fdoExtension->ScanForSpecialFlags,
                  CLASS_SPECIAL_DISABLE_SPIN_UP)) {
        options.HandleSpinUp = TRUE;
    }

    if (!TEST_FLAG(fdoExtension->ScanForSpecialFlags,
                  CLASS_SPECIAL_NO_QUEUE_LOCK)) {
        options.LockQueue = TRUE;
    }

    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "ClasspPowerHandler: Devobj %p\n"
                "\t%shandling spin down\n"
                "\t%shandling spin up\n"
                "\t%slocking queue\n",
                DeviceObject,
                (options.HandleSpinDown ? "" : "not "),
                (options.HandleSpinUp   ? "" : "not "),
                (options.LockQueue      ? "" : "not ")
                ));

    //
    // do all the dirty work
    //

    return ClasspPowerHandler(DeviceObject, Irp, options);
} // end ClassSpinDownPowerHandler()

/*++////////////////////////////////////////////////////////////////////////////

ClassStopUnitPowerHandler()

Routine Description:

    This routine is an outdated call.  To achieve equivalent functionality,
    the driver should set the following flags in ScanForSpecialFlags in the
    FdoExtension:

        CLASS_SPECIAL_DISABLE_SPIN_UP
        CLASS_SPECIAL_NO_QUEUE_LOCK

--*/
NTSTATUS
ClassStopUnitPowerHandler(
    _In_ PDEVICE_OBJECT DeviceObject,
    _In_ PIRP Irp
    )
{
    PFUNCTIONAL_DEVICE_EXTENSION fdoExtension;

    TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_POWER, "ClassStopUnitPowerHandler - Devobj %p using outdated call\n"
                "Drivers should set the following flags in ScanForSpecialFlags "
                " in the FDO extension:\n"
                "\tCLASS_SPECIAL_DISABLE_SPIN_UP\n"
                "\tCLASS_SPECIAL_NO_QUEUE_LOCK\n"
                "This will provide equivalent functionality if the power "
                "routine is then set to ClassSpinDownPowerHandler\n\n",
                DeviceObject));

    fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;

    SET_FLAG(fdoExtension->ScanForSpecialFlags,
             CLASS_SPECIAL_DISABLE_SPIN_UP);
    SET_FLAG(fdoExtension->ScanForSpecialFlags,
             CLASS_SPECIAL_NO_QUEUE_LOCK);

    return ClassSpinDownPowerHandler(DeviceObject, Irp);
} // end ClassStopUnitPowerHandler()

/*++////////////////////////////////////////////////////////////////////////////

RetryPowerRequest()

Routine Description:

    This routine reinitalizes the necessary fields, and sends the request
    to the lower driver.

Arguments:

    DeviceObject - Supplies the device object associated with this request.

    Irp - Supplies the request to be retried.

    Context - Supplies a pointer to the power up context for this request.

Return Value:

    None

--*/
VOID
RetryPowerRequest(
    PDEVICE_OBJECT DeviceObject,
    PIRP Irp,
    PCLASS_POWER_CONTEXT Context
    )
{
    PIO_STACK_LOCATION nextIrpStack = IoGetNextIrpStackLocation(Irp);
    PFUNCTIONAL_DEVICE_EXTENSION fdoExtension =
        (PFUNCTIONAL_DEVICE_EXTENSION)Context->DeviceObject->DeviceExtension;
    PSTORAGE_REQUEST_BLOCK_HEADER srb;
    LONGLONG dueTime;
    ULONG srbFlags;
    ULONG srbFunction;

    TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tDelaying retry by queueing DPC\n", Irp));

    //NT_ASSERT(Context->Irp == Irp);
    if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
        srb = (PSTORAGE_REQUEST_BLOCK_HEADER)&(fdoExtension->PrivateFdoData->PowerSrb.SrbEx);

        //
        // Check if reverted to using legacy SRB.
        //
        if (Context->Srb.Length == sizeof(SCSI_REQUEST_BLOCK)) {
            srb = (PSTORAGE_REQUEST_BLOCK_HEADER)&(Context->Srb);
            srbFunction = srb->Function;
        } else {
            srbFunction = ((PSTORAGE_REQUEST_BLOCK)srb)->SrbFunction;
        }
    } else {
        srb = (PSTORAGE_REQUEST_BLOCK_HEADER)&(Context->Srb);
        srbFunction = srb->Function;
    }

    NT_ASSERT(Context->DeviceObject == DeviceObject);
    srbFlags = SrbGetSrbFlags(srb);
    NT_ASSERT(!TEST_FLAG(srbFlags, SRB_FLAGS_FREE_SENSE_BUFFER));
    NT_ASSERT(!TEST_FLAG(srbFlags, SRB_FLAGS_PORT_DRIVER_ALLOCSENSE));

    if (Context->RetryInterval == 0) {

        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tDelaying minimum time (.2 sec)\n", Irp));
        dueTime = (LONGLONG)1000000 * 2;

    } else {

        TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_POWER, "(%p)\tDelaying %x seconds\n",
                    Irp, Context->RetryInterval));
        dueTime = (LONGLONG)1000000 * 10 * Context->RetryInterval;

    }

    //
    // reset the retry interval
    //

    Context->RetryInterval = 0;

    //
    // Reset byte count of transfer in SRB Extension.
    //

    SrbSetDataTransferLength(srb, 0);

    //
    // Zero SRB statuses.
    //

    srb->SrbStatus = 0;
    if (srbFunction == SRB_FUNCTION_EXECUTE_SCSI) {
        SrbSetScsiStatus(srb, 0);
    }

    //
    // Set up major SCSI function.
    //

    nextIrpStack->MajorFunction = IRP_MJ_SCSI;

    //
    // Save SRB address in next stack for port driver.
    //

    nextIrpStack->Parameters.Scsi.Srb = (PSCSI_REQUEST_BLOCK)srb;

    //
    // Set the completion routine up again.
    //

    IoSetCompletionRoutine(Irp, Context->CompletionRoutine, Context,
                           TRUE, TRUE, TRUE);

    ClassRetryRequest(DeviceObject, Irp, dueTime);

    return;

} // end RetryRequest()

/*++////////////////////////////////////////////////////////////////////////////

ClasspStartNextPowerIrpCompletion()

Routine Description:

    This routine guarantees that the next power irp (power up or down) is not
    sent until the previous one has fully completed.

--*/
NTSTATUS
ClasspStartNextPowerIrpCompletion(
    IN PDEVICE_OBJECT DeviceObject,
    IN PIRP Irp,
    IN PVOID Context
    )
{
    PCLASS_POWER_CONTEXT PowerContext = (PCLASS_POWER_CONTEXT)Context;

    UNREFERENCED_PARAMETER(DeviceObject);

    if (Irp->PendingReturned) {
        IoMarkIrpPending(Irp);
    }

    if (PowerContext != NULL)
    {
        PowerContext->InUse = FALSE;
    }

    PoStartNextPowerIrp(Irp);
    return STATUS_SUCCESS;
} // end ClasspStartNextPowerIrpCompletion()


_IRQL_requires_same_
NTSTATUS
ClasspSendEnableIdlePowerIoctl(
    _In_ PDEVICE_OBJECT DeviceObject
    )
/*++
Description:

    This function is used to send IOCTL_STORAGE_ENABLE_IDLE_POWER to the port
    driver.  It pulls the relevant idle power management properties from the
    FDO's device extension.

Arguments:

    DeviceObject - The class FDO.

Return Value:

    The NTSTATUS code returned from the port driver.  STATUS_SUCCESS indicates
    this device is now enabled for idle (runtime) power management.

--*/
{
    NTSTATUS status;
    STORAGE_IDLE_POWER idlePower = {0};
    IO_STATUS_BLOCK ioStatus = {0};
    PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
    PCOMMON_DEVICE_EXTENSION commonExtension = &(fdoExtension->CommonExtension);

    idlePower.Version = 1;
    idlePower.Size = sizeof(STORAGE_IDLE_POWER);
    idlePower.WakeCapableHint = fdoExtension->FunctionSupportInfo->IdlePower.DeviceWakeable;
    idlePower.D3ColdSupported = fdoExtension->FunctionSupportInfo->IdlePower.D3ColdSupported;
    idlePower.D3IdleTimeout = fdoExtension->FunctionSupportInfo->IdlePower.D3IdleTimeout;

    ClassSendDeviceIoControlSynchronous(
        IOCTL_STORAGE_ENABLE_IDLE_POWER,
        commonExtension->LowerDeviceObject,
        &idlePower,
        sizeof(STORAGE_IDLE_POWER),
        0,
        FALSE,
        &ioStatus
        );

    status = ioStatus.Status;

    TracePrint((TRACE_LEVEL_INFORMATION,
                TRACE_FLAG_POWER,
                "ClasspSendEnableIdlePowerIoctl: Port driver returned status (%x) for FDO (%p)\n"
                "\tWakeCapableHint: %u\n"
                "\tD3ColdSupported: %u\n"
                "\tD3IdleTimeout: %u (ms)",
                status,
                DeviceObject,
                idlePower.WakeCapableHint,
                idlePower.D3ColdSupported,
                idlePower.D3IdleTimeout));

    return status;
}

_Function_class_(POWER_SETTING_CALLBACK)
_IRQL_requires_same_
NTSTATUS
ClasspPowerSettingCallback(
    _In_ LPCGUID SettingGuid,
    _In_reads_bytes_(ValueLength) PVOID Value,
    _In_ ULONG ValueLength,
    _Inout_opt_ PVOID Context
)
/*++
Description:

    This function is the callback for power setting notifications (registered
    when ClasspGetD3IdleTimeout() is called for the first time).

    Currently, this function is used to get the disk idle timeout value from
    the system power settings.

    This function is guaranteed to be called at PASSIVE_LEVEL.

Arguments:

    SettingGuid - The power setting GUID.
    Value - Pointer to the power setting value.
    ValueLength - Size of the Value buffer.
    Context - The FDO's device extension.

Return Value:

    STATUS_SUCCESS

--*/
{    
    PIDLE_POWER_FDO_LIST_ENTRY fdoEntry = NULL;

#pragma warning(suppress:4054) // okay to type cast function pointer to PIRP for this use case
    PIRP removeLockTag = (PIRP)&ClasspPowerSettingCallback;

    UNREFERENCED_PARAMETER(Context);

    PAGED_CODE();

    if (IsEqualGUID(SettingGuid, &GUID_DISK_IDLE_TIMEOUT)) {
        if (ValueLength != sizeof(ULONG) || Value == NULL) {
            return STATUS_INVALID_PARAMETER;
        }

        //
        // The value supplied by this GUID is already in milliseconds.
        //
        DiskIdleTimeoutInMS = *((PULONG)Value);

        //
        // For each FDO on the idle power list, grab the remove lock and send
        // IOCTL_STORAGE_ENABLE_IDLE_POWER to the port driver to update the
        // idle timeout value.
        //
        KeAcquireGuardedMutex(&IdlePowerFDOListMutex);
        fdoEntry = (PIDLE_POWER_FDO_LIST_ENTRY)IdlePowerFDOList.Flink;
        while ((PLIST_ENTRY)fdoEntry != &IdlePowerFDOList) {

            ULONG isRemoved = ClassAcquireRemoveLock(fdoEntry->Fdo, removeLockTag);

            if (!isRemoved) {
                PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)fdoEntry->Fdo->DeviceExtension;

                //
                // Apply the new timeout if the user hasn't overridden it via the registry.
                //
                if (!fdoExtension->FunctionSupportInfo->IdlePower.D3IdleTimeoutOverridden) {
                    fdoExtension->FunctionSupportInfo->IdlePower.D3IdleTimeout = DiskIdleTimeoutInMS;
                    ClasspSendEnableIdlePowerIoctl(fdoEntry->Fdo);
                }
            }

            ClassReleaseRemoveLock(fdoEntry->Fdo, removeLockTag);

            fdoEntry = (PIDLE_POWER_FDO_LIST_ENTRY)fdoEntry->ListEntry.Flink;
        }
        KeReleaseGuardedMutex(&IdlePowerFDOListMutex);

    } else if (IsEqualGUID(SettingGuid, &GUID_CONSOLE_DISPLAY_STATE)) {

        //
        // If monitor is off, change media change requests to not
        // keep device active. This allows removable media devices to
        // go to sleep if there are no other active requests. Otherwise,
        // let media change requests keep the device active.
        //
        if ((ValueLength == sizeof(ULONG)) && (Value != NULL)) {
            if (*((PULONG)Value) == PowerMonitorOff) {
                ClasspScreenOff = TRUE;
            } else {
                ClasspScreenOff = FALSE;
                
                KeAcquireGuardedMutex(&IdlePowerFDOListMutex);
                fdoEntry = (PIDLE_POWER_FDO_LIST_ENTRY)IdlePowerFDOList.Flink;
                while ((PLIST_ENTRY)fdoEntry != &IdlePowerFDOList) {

                    ULONG isRemoved = ClassAcquireRemoveLock(fdoEntry->Fdo, removeLockTag);
                    if (!isRemoved) {
                        PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)fdoEntry->Fdo->DeviceExtension;

                        //
                        // Now that the screen is on, we may need to check for media
                        // for devices that are not in D0 and may have removable media.
                        // This is because the media change polling has been disabled
                        // for devices in D3 and now that the screen is on the user may
                        // have inserted some media that they want to interact with.
                        //
                        if ((fdoExtension->DevicePowerState != PowerDeviceD0) &&                            
                            (fdoExtension->MediaChangeDetectionInfo != NULL) &&
                            (fdoExtension->FunctionSupportInfo->AsynchronousNotificationSupported == FALSE)) {
                            ClassCheckMediaState(fdoExtension);
                        }

                        //
                        // We disabled failure prediction polling during screen-off
                        // so now check to see if we missed a failure prediction
                        // period and if so, force the IOCTL to be sent now.
                        //
                        if ((fdoExtension->FailurePredictionInfo != NULL) &&
                            (fdoExtension->FailurePredictionInfo->Method != FailurePredictionNone)) {
                            if (ClasspFailurePredictionPeriodMissed(fdoExtension)) {
                                fdoExtension->FailurePredictionInfo->CountDown = 1;
                            }
                        }
                    }
                    ClassReleaseRemoveLock(fdoEntry->Fdo, removeLockTag);

                    fdoEntry = (PIDLE_POWER_FDO_LIST_ENTRY)fdoEntry->ListEntry.Flink;
                }
                KeReleaseGuardedMutex(&IdlePowerFDOListMutex);                
            }
        }

    }

    return STATUS_SUCCESS;
}


_IRQL_requires_same_
NTSTATUS
ClasspEnableIdlePower(
    _In_ PDEVICE_OBJECT DeviceObject
    )
/*++
Description:

    This function is used to enable idle (runtime) power management for the
    device.  It will do the work to determine D3Cold support, idle timeout,
    etc. and then notify the port driver that it wants to enable idle power
    management.

    This function may modify some of the idle power fields in the FDO's device
    extension.

Arguments:

    DeviceObject - The class FDO.

Return Value:

    An NTSTATUS code indicating the status of the operation.

--*/
{
    NTSTATUS status = STATUS_SUCCESS;
    ULONG d3ColdDisabledByUser = FALSE;
    PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
    ULONG idleTimeoutOverrideInSeconds = 0;

    //
    // This function should only be called once.
    //
    NT_ASSERT(fdoExtension->FunctionSupportInfo->IdlePower.IdlePowerEnabled == FALSE);

    ClassGetDeviceParameter(fdoExtension,
                        CLASSP_REG_SUBKEY_NAME,
                        CLASSP_REG_DISABLE_D3COLD,
                        &d3ColdDisabledByUser);

    //
    // If the device is hot-pluggable or the user has explicitly
    // disabled D3Cold, do not enable D3Cold for this device.
    //
    if (d3ColdDisabledByUser || fdoExtension->PrivateFdoData->HotplugInfo.DeviceHotplug) {
        fdoExtension->FunctionSupportInfo->IdlePower.D3ColdSupported = 0;
    }

    ClassGetDeviceParameter(fdoExtension,
                            CLASSP_REG_SUBKEY_NAME,
                            CLASSP_REG_IDLE_TIMEOUT_IN_SECONDS,
                            &idleTimeoutOverrideInSeconds);

    //
    // Set the idle timeout.  If the user has not specified an override value,
    // this will either be a default value or will have been updated by the
    // power setting notification callback.
    //
    if (idleTimeoutOverrideInSeconds != 0) {
        fdoExtension->FunctionSupportInfo->IdlePower.D3IdleTimeout = (idleTimeoutOverrideInSeconds * 1000);
        fdoExtension->FunctionSupportInfo->IdlePower.D3IdleTimeoutOverridden = TRUE;
    } else {
        fdoExtension->FunctionSupportInfo->IdlePower.D3IdleTimeout = DiskIdleTimeoutInMS;
    }

    //
    // We don't allow disks to be wakeable.
    //
    fdoExtension->FunctionSupportInfo->IdlePower.DeviceWakeable = FALSE;

    //
    // Send IOCTL_STORAGE_ENABLE_IDLE_POWER to the port driver to enable idle
    // power management by the port driver.
    //
    status = ClasspSendEnableIdlePowerIoctl(DeviceObject);

    if (NT_SUCCESS(status)) {
        PIDLE_POWER_FDO_LIST_ENTRY fdoEntry = NULL;

        //
        // Put this FDO on the list of devices that are idle power managed.
        //
        fdoEntry = ExAllocatePoolWithTag(NonPagedPoolNx, sizeof(IDLE_POWER_FDO_LIST_ENTRY), CLASS_TAG_POWER);
        if (fdoEntry) {

            fdoExtension->FunctionSupportInfo->IdlePower.IdlePowerEnabled = TRUE;

            fdoEntry->Fdo = DeviceObject;

            KeAcquireGuardedMutex(&IdlePowerFDOListMutex);
            InsertHeadList(&IdlePowerFDOList, &(fdoEntry->ListEntry));
            KeReleaseGuardedMutex(&IdlePowerFDOListMutex);

            //
            // If not registered already, register for disk idle timeout power
            // setting notifications.  The power manager will call our power
            // setting callback very soon to set the idle timeout to the actual
            // value.
            //
            if (PowerSettingNotificationHandle == NULL) {
                PoRegisterPowerSettingCallback(DeviceObject,
                                                &GUID_DISK_IDLE_TIMEOUT,
                                                &ClasspPowerSettingCallback,
                                                NULL,
                                                &(PowerSettingNotificationHandle));
            }
        } else {
            fdoExtension->FunctionSupportInfo->IdlePower.IdlePowerEnabled = FALSE;
            status = STATUS_UNSUCCESSFUL;
        }
    }

    return status;
}

Our Services

  • What our customers say about us?

© 2011-2024 All Rights Reserved. Joya Systems. 4425 South Mopac Building II Suite 101 Austin, TX 78735 Tel: 800-DEV-KERNEL

Privacy Policy. Terms of use. Valid XHTML & CSS