2020-09-30 16:53:55 +02:00

344 lines
9.5 KiB
C

/*++
Copyright (c) 1990 Microsoft Corporation
Module Name:
idle.c
Abstract:
This module implements the power management idle timing code for
device objects
Author:
Bryan Willman (bryanwi) 7-Nov-96
Revision History:
--*/
#include "pop.h"
NTKERNELAPI
PULONG
PoRegisterDeviceForIdleDetection (
IN PDEVICE_OBJECT DeviceObject,
IN ULONG ConservationIdleTime,
IN ULONG PerformanceIdleTime,
IN DEVICE_POWER_STATE State
)
/*++
Routine Description:
A device driver calls this routine to either:
a. Create and initialize a new idle detection block
b. Reset values in an existing idle detection block
If the device object has an idle detection block, it is
filled in with new values.
Otherwise, an idle detect block is created and linked to the device
object.
Arguments:
DeviceObject - Device object which wants idle detection, set_power
IRPs will be sent here
ConservationIdleTime - timeout for system in "conserve mode"
PerformanceIdleTime - timeout for system in "performance mode"
Type - Type of set_power sent (for set_power irp)
State - what state to go to (for set_power irp)
Return Value:
NULL - if an attempt to create a new idle block failed
non-NULL - if an idle block was created, or if an existing one was reset
--*/
{
PDEVICE_OBJECT_POWER_EXTENSION pdope;
KIRQL OldIrql;
ULONG DeviceType, OldDeviceType;
ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
ASSERT(DeviceObject != NULL);
//
// deal with the case where idle detection is being turned off
//
if ((ConservationIdleTime == 0) && (PerformanceIdleTime == 0)) {
PopLockDopeGlobal(&OldIrql);
pdope = DeviceObject->DeviceObjectExtension->Dope;
if (pdope == NULL) {
//
// cannot be linked into the chain, so must already be off,
// so we're done
//
} else {
//
// there is a pdope, so we may be on the idle list
//
if ((pdope->IdleList.Flink == &(pdope->IdleList)) &&
(pdope->IdleList.Blink == &(pdope->IdleList)))
{
//
// we're off the queue already, so we're done
//
} else {
//
// a dope vector exists and is on the idle scan list,
// so we must delist ourselves
//
RemoveEntryList(&(pdope->IdleList));
OldDeviceType = pdope->DeviceType | ES_CONTINUOUS;
pdope->DeviceType = 0;
PopApplyAttributeState (ES_CONTINUOUS, OldDeviceType);
pdope->ConservationIdleTime = 0L;
pdope->PerformanceIdleTime = 0L;
pdope->State = PowerDeviceUnspecified;
pdope->IdleCount = 0;
InitializeListHead(&(pdope->IdleList));
}
}
PopUnlockDopeGlobal(OldIrql);
return NULL;
}
//
// Set DeviceType if this is an idle registration by type
//
DeviceType = 0;
if (ConservationIdleTime == (ULONG) -1 &&
PerformanceIdleTime == (ULONG) -1) {
switch (DeviceObject->DeviceType) {
case FILE_DEVICE_DISK:
case FILE_DEVICE_MASS_STORAGE:
//
// Or-in ES_CONTINUOUS which will go up
// in the high bits. We'll ignore this bit
// when we typecast this into a UCHAR as we
// assign this to pdope->DeviceType, however, we
// need this high bit set for when we call into
// PopApplyAttributeState. That's because we
// overload the flags we send into this function
// with both the device type and the flags to
// apply to that device type.
//
DeviceType = POP_DISK_SPINDOWN | ES_CONTINUOUS;
break;
default:
//
// Unsupported type
//
return NULL;
}
}
//
// now, the case where it's being turned on
//
pdope = PopGetDope(DeviceObject);
if (pdope == NULL) {
//
// we didn't have a DOPE structure and couldn't allocate one, fail
//
return NULL;
}
//
// May be a newly allocated Dope, or an existing one.
// In either case, update values.
// Enqueue if not already in queue
//
PopLockDopeGlobal(&OldIrql);
OldDeviceType = pdope->DeviceType | ES_CONTINUOUS;
pdope->ConservationIdleTime = ConservationIdleTime;
pdope->PerformanceIdleTime = PerformanceIdleTime;
pdope->State = State;
pdope->IdleCount = 0;
//
// type cast this so we ignore any high bits set which
// may contain attributes. All we care about is the
// device type, which is in the low byte.
//
pdope->DeviceType = (UCHAR) DeviceType;
if ((pdope->IdleList.Flink == &(pdope->IdleList)) &&
(pdope->IdleList.Blink == &(pdope->IdleList)))
{
//
// we're off the queue, and must be enqueued
//
InsertTailList(&PopIdleDetectList, &(pdope->IdleList));
}
PopUnlockDopeGlobal(OldIrql);
PopApplyAttributeState(DeviceType, OldDeviceType);
PopCheckForWork(TRUE);
return (PULONG) &(pdope->IdleCount); // success
}
VOID
PopScanIdleList(
IN PKDPC Dpc,
IN PVOID DeferredContext,
IN PVOID SystemArgument1,
IN PVOID SystemArgument2
)
/*++
Routine Description:
Called by the PopIdleScanTimer at PopIdleScanTimeInseconds interval,
this routine runs the list of Idle Blocks, finding any that meet the
trip conditions, and sends commands to the appropriate device objects
to change state.
The timer that calls this DPC is setup in poinit.c.
Arguments:
Standard DPC arguments, all are ignored.
Return Value:
None.
--*/
{
KIRQL OldIrql;
PLIST_ENTRY link;
ULONG idlelimit;
PDEVICE_OBJECT_POWER_EXTENSION pblock;
POWER_STATE PowerState;
PLONG pIdleCount;
ULONG oldCount;
UNREFERENCED_PARAMETER (Dpc);
UNREFERENCED_PARAMETER (DeferredContext);
UNREFERENCED_PARAMETER (SystemArgument1);
UNREFERENCED_PARAMETER (SystemArgument2);
PopLockDopeGlobal(&OldIrql);
link = PopIdleDetectList.Flink;
while (link != &PopIdleDetectList) {
pblock = CONTAINING_RECORD(link, DEVICE_OBJECT_POWER_EXTENSION, IdleList);
pIdleCount = &(pblock->IdleCount);
oldCount = InterlockedIncrement(pIdleCount);
switch (pblock->DeviceType) {
case 0:
idlelimit = pblock->PerformanceIdleTime;
if (PopIdleDetectionMode == PO_IDLE_CONSERVATION) {
idlelimit = pblock->ConservationIdleTime;
}
break;
case POP_DISK_SPINDOWN:
idlelimit = PopPolicy->SpindownTimeout;
break;
default:
idlelimit = 0;
PopInternalAddToDumpFile( NULL, 0, pblock->DeviceObject, NULL, NULL, pblock );
KeBugCheckEx( INTERNAL_POWER_ERROR,
0x200,
POP_IDLE,
(ULONG_PTR)pblock->DeviceObject,
(ULONG_PTR)pblock );
}
if ((idlelimit > 0) && ((oldCount+1) == idlelimit)) {
PowerState.DeviceState = pblock->State;
PoRequestPowerIrp (
pblock->DeviceObject,
IRP_MN_SET_POWER,
PowerState,
NULL,
NULL,
NULL
);
}
link = link->Flink;
}
PopUnlockDopeGlobal(OldIrql);
return;
}
PDEVICE_OBJECT_POWER_EXTENSION
PopGetDope (
PDEVICE_OBJECT DeviceObject
)
{
PDEVOBJ_EXTENSION Doe;
PDEVICE_OBJECT_POWER_EXTENSION Dope;
KIRQL OldIrql;
Doe = (PDEVOBJ_EXTENSION) DeviceObject->DeviceObjectExtension;
if (!Doe->Dope) {
PopLockDopeGlobal(&OldIrql);
if (!Doe->Dope) {
Dope = (PDEVICE_OBJECT_POWER_EXTENSION)
ExAllocatePoolWithTag(
NonPagedPool,
sizeof(DEVICE_OBJECT_POWER_EXTENSION),
POP_DOPE_TAG
);
if (Dope) {
RtlZeroMemory (Dope, sizeof(DEVICE_OBJECT_POWER_EXTENSION));
Dope->DeviceObject = DeviceObject;
Dope->State = PowerDeviceUnspecified;
InitializeListHead(&(Dope->IdleList));
InitializeListHead(&(Dope->NotifySourceList));
InitializeListHead(&(Dope->NotifyTargetList));
// force the signature to 0 so buildpowerchannel gets called
Dope->PowerChannelSummary.Signature = (ULONG)0;
InitializeListHead(&(Dope->PowerChannelSummary.NotifyList));
Doe->Dope = Dope;
}
}
PopUnlockDopeGlobal(OldIrql);
}
return Doe->Dope;
}