1801 lines
47 KiB
C
1801 lines
47 KiB
C
|
/*++
|
||
|
|
||
|
Copyright (c) 1999 Microsoft Corporation
|
||
|
|
||
|
Module Name:
|
||
|
|
||
|
prefparm.c
|
||
|
|
||
|
Abstract:
|
||
|
|
||
|
This module contains the code for prefetcher parameter handling.
|
||
|
|
||
|
Author:
|
||
|
|
||
|
Cenk Ergan (cenke) 15-Mar-2000
|
||
|
|
||
|
Revision History:
|
||
|
|
||
|
--*/
|
||
|
|
||
|
#include "cc.h"
|
||
|
#include "zwapi.h"
|
||
|
#include "prefetch.h"
|
||
|
#include "preftchp.h"
|
||
|
#include "stdio.h"
|
||
|
|
||
|
#ifdef ALLOC_PRAGMA
|
||
|
#pragma alloc_text(INIT, CcPfParametersInitialize)
|
||
|
#pragma alloc_text(INIT, CcPfParametersSetDefaults)
|
||
|
#pragma alloc_text(PAGE, CcPfParametersRead)
|
||
|
#pragma alloc_text(PAGE, CcPfParametersSave)
|
||
|
#pragma alloc_text(PAGE, CcPfParametersVerify)
|
||
|
#pragma alloc_text(PAGE, CcPfParametersWatcher)
|
||
|
#pragma alloc_text(PAGE, CcPfParametersSetChangedEvent)
|
||
|
#pragma alloc_text(PAGE, CcPfGetParameter)
|
||
|
#pragma alloc_text(PAGE, CcPfSetParameter)
|
||
|
#pragma alloc_text(PAGE, CcPfDetermineEnablePrefetcher)
|
||
|
#pragma alloc_text(PAGE, CcPfIsHostingApplication)
|
||
|
#endif // ALLOC_PRAGMA
|
||
|
|
||
|
//
|
||
|
// Globals:
|
||
|
//
|
||
|
|
||
|
extern CCPF_PREFETCHER_GLOBALS CcPfGlobals;
|
||
|
|
||
|
//
|
||
|
// Constants:
|
||
|
//
|
||
|
|
||
|
//
|
||
|
// The following are used as prefixs for the value names for registry
|
||
|
// parameters that are per scenario type.
|
||
|
//
|
||
|
|
||
|
WCHAR *CcPfAppLaunchScenarioTypePrefix = L"AppLaunch";
|
||
|
WCHAR *CcPfBootScenarioTypePrefix = L"Boot";
|
||
|
WCHAR *CcPfInvalidScenarioTypePrefix = L"Invalid";
|
||
|
|
||
|
//
|
||
|
// Routines for prefetcher parameter handling.
|
||
|
//
|
||
|
|
||
|
NTSTATUS
|
||
|
CcPfParametersInitialize (
|
||
|
PCCPF_PREFETCHER_PARAMETERS PrefetcherParameters
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
Initializes specified prefetcher parameters structure.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
PrefetcherParameters - Pointer to structure to initialize.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Status.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
Notes:
|
||
|
|
||
|
The code & local constants for this function gets discarded after system boots.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
||
|
UNICODE_STRING KeyName;
|
||
|
NTSTATUS Status;
|
||
|
|
||
|
//
|
||
|
// Zero out the structure. This initializes:
|
||
|
// ParametersVersion
|
||
|
//
|
||
|
|
||
|
RtlZeroMemory(PrefetcherParameters, sizeof(*PrefetcherParameters));
|
||
|
|
||
|
//
|
||
|
// Initialize the lock protecting the parameters and parameters
|
||
|
// version. Each time parameters are updated, the version is
|
||
|
// bumped.
|
||
|
//
|
||
|
|
||
|
ExInitializeResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
|
||
|
//
|
||
|
// Initialize the workitem used for registry notifications on the
|
||
|
// parameters key.
|
||
|
//
|
||
|
|
||
|
ExInitializeWorkItem(&PrefetcherParameters->RegistryWatchWorkItem,
|
||
|
CcPfParametersWatcher,
|
||
|
PrefetcherParameters);
|
||
|
|
||
|
//
|
||
|
// Set default parameters.
|
||
|
//
|
||
|
|
||
|
CcPfParametersSetDefaults(PrefetcherParameters);
|
||
|
|
||
|
//
|
||
|
// Create / Open the registry key that contains our parameters.
|
||
|
//
|
||
|
|
||
|
RtlInitUnicodeString(&KeyName, CCPF_PARAMETERS_KEY);
|
||
|
|
||
|
InitializeObjectAttributes(&ObjectAttributes,
|
||
|
&KeyName,
|
||
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
||
|
NULL,
|
||
|
NULL);
|
||
|
|
||
|
Status = ZwCreateKey(&PrefetcherParameters->ParametersKey,
|
||
|
KEY_ALL_ACCESS,
|
||
|
&ObjectAttributes,
|
||
|
0,
|
||
|
NULL,
|
||
|
REG_OPTION_NON_VOLATILE,
|
||
|
0);
|
||
|
|
||
|
if (NT_SUCCESS(Status)) {
|
||
|
|
||
|
//
|
||
|
// Update the default parameters with those in the registry.
|
||
|
//
|
||
|
|
||
|
Status = CcPfParametersRead(PrefetcherParameters);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
DBGPR((CCPFID,PFERR,"CCPF: Init-FailedReadParams=%x\n",Status));
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Request notification when something changes in the
|
||
|
// prefetcher parameters key.
|
||
|
//
|
||
|
|
||
|
Status = ZwNotifyChangeKey(PrefetcherParameters->ParametersKey,
|
||
|
NULL,
|
||
|
(PIO_APC_ROUTINE)&PrefetcherParameters->RegistryWatchWorkItem,
|
||
|
(PVOID)(UINT_PTR)(unsigned int)DelayedWorkQueue,
|
||
|
&PrefetcherParameters->RegistryWatchIosb,
|
||
|
REG_LEGAL_CHANGE_FILTER,
|
||
|
FALSE,
|
||
|
&PrefetcherParameters->RegistryWatchBuffer,
|
||
|
sizeof(PrefetcherParameters->RegistryWatchBuffer),
|
||
|
TRUE);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
|
||
|
//
|
||
|
// Although we could not register a notification, this
|
||
|
// is not a fatal error.
|
||
|
//
|
||
|
|
||
|
DBGPR((CCPFID,PFERR,"CCPF: Init-FailedSetParamNotify=%x\n",Status));
|
||
|
}
|
||
|
|
||
|
} else {
|
||
|
|
||
|
DBGPR((CCPFID,PFERR,"CCPF: Init-FailedCreateParamKey=%x\n",Status));
|
||
|
|
||
|
//
|
||
|
// Make sure parameters key handle is invalid.
|
||
|
//
|
||
|
|
||
|
PrefetcherParameters->ParametersKey = NULL;
|
||
|
}
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
CcPfParametersSetDefaults (
|
||
|
PCCPF_PREFETCHER_PARAMETERS PrefetcherParameters
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
Initializes specified parameters structure to default values.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
Parameters - Pointer to structure to initialize.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
Notes:
|
||
|
|
||
|
The code & local constants for this function gets discarded after system boots.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
PPF_SYSTEM_PREFETCH_PARAMETERS Parameters;
|
||
|
PPF_TRACE_LIMITS TraceLimits;
|
||
|
PF_SCENARIO_TYPE ScenarioType;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
Parameters = &PrefetcherParameters->Parameters;
|
||
|
|
||
|
for (ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
|
||
|
|
||
|
//
|
||
|
// PfSvNotSpecified is currently treated as disabled.
|
||
|
//
|
||
|
|
||
|
Parameters->EnableStatus[ScenarioType] = PfSvNotSpecified;
|
||
|
|
||
|
//
|
||
|
// Trace limits are determined based on scenario type.
|
||
|
//
|
||
|
|
||
|
TraceLimits = &Parameters->TraceLimits[ScenarioType];
|
||
|
|
||
|
switch(ScenarioType) {
|
||
|
|
||
|
case PfApplicationLaunchScenarioType:
|
||
|
|
||
|
TraceLimits->MaxNumPages = 4000;
|
||
|
TraceLimits->MaxNumSections = 170;
|
||
|
TraceLimits->TimerPeriod = (-1 * 1000 * 1000 * 10);
|
||
|
|
||
|
PrefetcherParameters->ScenarioTypePrefixes[ScenarioType] =
|
||
|
CcPfAppLaunchScenarioTypePrefix;
|
||
|
|
||
|
break;
|
||
|
|
||
|
case PfSystemBootScenarioType:
|
||
|
|
||
|
TraceLimits->MaxNumPages = 128000;
|
||
|
TraceLimits->MaxNumSections = 4080;
|
||
|
TraceLimits->TimerPeriod = (-1 * 12000 * 1000 * 10);
|
||
|
|
||
|
PrefetcherParameters->ScenarioTypePrefixes[ScenarioType] =
|
||
|
CcPfBootScenarioTypePrefix;
|
||
|
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
|
||
|
//
|
||
|
// We should be handling all scenario types above.
|
||
|
//
|
||
|
|
||
|
CCPF_ASSERT(FALSE);
|
||
|
|
||
|
TraceLimits->MaxNumPages = PF_MAXIMUM_PAGES;
|
||
|
TraceLimits->MaxNumSections = PF_MAXIMUM_SECTIONS;
|
||
|
TraceLimits->TimerPeriod = (-1 * 1000 * 1000 * 10);
|
||
|
|
||
|
PrefetcherParameters->ScenarioTypePrefixes[ScenarioType] =
|
||
|
CcPfInvalidScenarioTypePrefix;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// These limits ensure that we don't monopolize system resources
|
||
|
// for prefetching.
|
||
|
//
|
||
|
|
||
|
Parameters->MaxNumActiveTraces = 8;
|
||
|
Parameters->MaxNumSavedTraces = 8;
|
||
|
|
||
|
//
|
||
|
// This is the default directory under SystemRoot where we
|
||
|
// find prefetch instructions for scenarios. During upgrades
|
||
|
// we remove the contents of this directory, so "Prefetch" is
|
||
|
// hardcoded in txtsetup.inx.
|
||
|
//
|
||
|
|
||
|
wcsncpy(Parameters->RootDirPath,
|
||
|
L"Prefetch",
|
||
|
PF_MAX_PREFETCH_ROOT_PATH);
|
||
|
|
||
|
Parameters->RootDirPath[PF_MAX_PREFETCH_ROOT_PATH - 1] = 0;
|
||
|
|
||
|
//
|
||
|
// This is the default list of known hosting applications.
|
||
|
//
|
||
|
|
||
|
wcsncpy(Parameters->HostingApplicationList,
|
||
|
L"DLLHOST.EXE,MMC.EXE,RUNDLL32.EXE",
|
||
|
PF_HOSTING_APP_LIST_MAX_CHARS);
|
||
|
|
||
|
Parameters->HostingApplicationList[PF_HOSTING_APP_LIST_MAX_CHARS - 1] = 0;
|
||
|
|
||
|
//
|
||
|
// Make sure the default parameters make sense.
|
||
|
//
|
||
|
|
||
|
CCPF_ASSERT(NT_SUCCESS(CcPfParametersVerify(Parameters)));
|
||
|
|
||
|
}
|
||
|
|
||
|
NTSTATUS
|
||
|
CcPfParametersRead (
|
||
|
PCCPF_PREFETCHER_PARAMETERS PrefetcherParameters
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine updates the parameters structure with the
|
||
|
parameters in the registry.
|
||
|
|
||
|
Keep the value names that are used in sync with the function to
|
||
|
save the parameters.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
PrefetcherParameters - Pointer to parameters.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Status.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
NTSTATUS Status;
|
||
|
PF_SYSTEM_PREFETCH_PARAMETERS Parameters;
|
||
|
PPF_TRACE_LIMITS TraceLimits;
|
||
|
PF_SCENARIO_TYPE ScenarioType;
|
||
|
WCHAR ValueName[CCPF_MAX_PARAMETER_NAME_LENGTH];
|
||
|
WCHAR *ValueNamePrefix;
|
||
|
HANDLE ParametersKey;
|
||
|
BOOLEAN EnableStatusSpecified;
|
||
|
ULONG EnablePrefetcher;
|
||
|
BOOLEAN AcquiredParametersLock;
|
||
|
ULONG Length;
|
||
|
LONG CurrentVersion;
|
||
|
ULONG RetryCount;
|
||
|
PKTHREAD CurrentThread;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
CurrentThread = KeGetCurrentThread ();
|
||
|
AcquiredParametersLock = FALSE;
|
||
|
RetryCount = 0;
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersRead()\n"));
|
||
|
|
||
|
do {
|
||
|
|
||
|
//
|
||
|
// Get the parameters lock shared.
|
||
|
//
|
||
|
|
||
|
KeEnterCriticalRegionThread(CurrentThread);
|
||
|
ExAcquireResourceSharedLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
AcquiredParametersLock = TRUE;
|
||
|
|
||
|
//
|
||
|
// If we could not initialize the parameters key, we would fail
|
||
|
// all the following ops miserably.
|
||
|
//
|
||
|
|
||
|
if (!PrefetcherParameters->ParametersKey) {
|
||
|
Status = STATUS_REINITIALIZATION_NEEDED;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
ParametersKey = PrefetcherParameters->ParametersKey;
|
||
|
|
||
|
//
|
||
|
// Save current version of parameters. Each time parameters gets
|
||
|
// updated, the version is bumped.
|
||
|
//
|
||
|
|
||
|
CurrentVersion = PrefetcherParameters->ParametersVersion;
|
||
|
|
||
|
//
|
||
|
// Copy over existing parameters to the parameters structure we
|
||
|
// are building. This way, if we cannot get a value from the
|
||
|
// registry we'll keep the value we already have.
|
||
|
//
|
||
|
|
||
|
Parameters = PrefetcherParameters->Parameters;
|
||
|
|
||
|
//
|
||
|
// Read the prefetcher enable value. Depending on whether it is
|
||
|
// specified and if so its value we will set enable status for
|
||
|
// prefetch scenario types.
|
||
|
//
|
||
|
|
||
|
Length = sizeof(EnablePrefetcher);
|
||
|
Status = CcPfGetParameter(ParametersKey,
|
||
|
L"EnablePrefetcher",
|
||
|
REG_DWORD,
|
||
|
&EnablePrefetcher,
|
||
|
&Length);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
|
||
|
//
|
||
|
// Enable status is not specified or we cannot access it.
|
||
|
//
|
||
|
|
||
|
EnableStatusSpecified = FALSE;
|
||
|
|
||
|
} else {
|
||
|
|
||
|
EnableStatusSpecified = TRUE;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Get per scenario parameters.
|
||
|
//
|
||
|
|
||
|
for (ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
|
||
|
|
||
|
ValueNamePrefix = PrefetcherParameters->ScenarioTypePrefixes[ScenarioType];
|
||
|
|
||
|
//
|
||
|
// Determine enable status. If EnableStatusSpecified, whether
|
||
|
// prefeching for this scenario type is on or off is
|
||
|
// determined by the ScenarioType'th bit in EnablePrefetcher.
|
||
|
//
|
||
|
|
||
|
if (EnableStatusSpecified) {
|
||
|
if (EnablePrefetcher & (1 << ScenarioType)) {
|
||
|
Parameters.EnableStatus[ScenarioType] = PfSvEnabled;
|
||
|
} else {
|
||
|
Parameters.EnableStatus[ScenarioType] = PfSvDisabled;
|
||
|
}
|
||
|
} else {
|
||
|
Parameters.EnableStatus[ScenarioType] = PfSvNotSpecified;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update trace limits for this scenario type. Ignore return
|
||
|
// value from GetParameter since the value may not be
|
||
|
// specified in the registry. If so the current value is kept
|
||
|
// intact.
|
||
|
//
|
||
|
|
||
|
TraceLimits = &Parameters.TraceLimits[ScenarioType];
|
||
|
|
||
|
wcscpy(ValueName, ValueNamePrefix);
|
||
|
wcscat(ValueName, L"MaxNumPages");
|
||
|
Length = sizeof(TraceLimits->MaxNumPages);
|
||
|
CcPfGetParameter(ParametersKey,
|
||
|
ValueName,
|
||
|
REG_DWORD,
|
||
|
&TraceLimits->MaxNumPages,
|
||
|
&Length);
|
||
|
|
||
|
wcscpy(ValueName, ValueNamePrefix);
|
||
|
wcscat(ValueName, L"MaxNumSections");
|
||
|
Length = sizeof(TraceLimits->MaxNumSections);
|
||
|
CcPfGetParameter(ParametersKey,
|
||
|
ValueName,
|
||
|
REG_DWORD,
|
||
|
&TraceLimits->MaxNumSections,
|
||
|
&Length);
|
||
|
|
||
|
wcscpy(ValueName, ValueNamePrefix);
|
||
|
wcscat(ValueName, L"TimerPeriod");
|
||
|
Length = sizeof(TraceLimits->TimerPeriod);
|
||
|
CcPfGetParameter(ParametersKey,
|
||
|
ValueName,
|
||
|
REG_BINARY,
|
||
|
&TraceLimits->TimerPeriod,
|
||
|
&Length);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update maximum number of active traces.
|
||
|
//
|
||
|
|
||
|
Length = sizeof(Parameters.MaxNumActiveTraces);
|
||
|
CcPfGetParameter(ParametersKey,
|
||
|
L"MaxNumActiveTraces",
|
||
|
REG_DWORD,
|
||
|
&Parameters.MaxNumActiveTraces,
|
||
|
&Length);
|
||
|
|
||
|
//
|
||
|
// Update maximum number of saved traces.
|
||
|
//
|
||
|
|
||
|
Length = sizeof(Parameters.MaxNumSavedTraces);
|
||
|
CcPfGetParameter(ParametersKey,
|
||
|
L"MaxNumSavedTraces",
|
||
|
REG_DWORD,
|
||
|
&Parameters.MaxNumSavedTraces,
|
||
|
&Length);
|
||
|
|
||
|
//
|
||
|
// Update the root directory path.
|
||
|
//
|
||
|
|
||
|
Length = sizeof(Parameters.RootDirPath);
|
||
|
CcPfGetParameter(ParametersKey,
|
||
|
L"RootDirPath",
|
||
|
REG_SZ,
|
||
|
Parameters.RootDirPath,
|
||
|
&Length);
|
||
|
|
||
|
Parameters.RootDirPath[PF_MAX_PREFETCH_ROOT_PATH - 1] = 0;
|
||
|
|
||
|
//
|
||
|
// Update list of known hosting applications.
|
||
|
//
|
||
|
|
||
|
Length = sizeof(Parameters.HostingApplicationList);
|
||
|
CcPfGetParameter(ParametersKey,
|
||
|
L"HostingAppList",
|
||
|
REG_SZ,
|
||
|
Parameters.HostingApplicationList,
|
||
|
&Length);
|
||
|
|
||
|
Parameters.HostingApplicationList[PF_HOSTING_APP_LIST_MAX_CHARS - 1] = 0;
|
||
|
_wcsupr(Parameters.HostingApplicationList);
|
||
|
|
||
|
//
|
||
|
// Verify the parameters updated from the registry.
|
||
|
//
|
||
|
|
||
|
Status = CcPfParametersVerify(&Parameters);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Release the shared lock and acquire it exclusive.
|
||
|
//
|
||
|
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
|
||
|
KeEnterCriticalRegionThread(CurrentThread);
|
||
|
ExAcquireResourceExclusiveLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
|
||
|
//
|
||
|
// Check if somebody already updated the parameters before us.
|
||
|
//
|
||
|
|
||
|
if (CurrentVersion != PrefetcherParameters->ParametersVersion) {
|
||
|
|
||
|
//
|
||
|
// Bummer. Somebody updated parameters when we released
|
||
|
// our shared lock to acquire it exclusive. We have to try
|
||
|
// again. The default values we used for parameters that
|
||
|
// were not in the registry may have been changed.
|
||
|
//
|
||
|
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
AcquiredParametersLock = FALSE;
|
||
|
|
||
|
RetryCount++;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// We are updating the parameters, bump the version.
|
||
|
//
|
||
|
|
||
|
PrefetcherParameters->ParametersVersion++;
|
||
|
|
||
|
PrefetcherParameters->Parameters = Parameters;
|
||
|
|
||
|
//
|
||
|
// Release the exclusive lock and break out.
|
||
|
//
|
||
|
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
AcquiredParametersLock = FALSE;
|
||
|
|
||
|
break;
|
||
|
|
||
|
} while (RetryCount < 10);
|
||
|
|
||
|
//
|
||
|
// See if we looped too many times and could not achive updating
|
||
|
// the parameters.
|
||
|
//
|
||
|
|
||
|
if (RetryCount >= 10) {
|
||
|
Status = STATUS_RETRY;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Otherwise we were successful.
|
||
|
//
|
||
|
|
||
|
Status = STATUS_SUCCESS;
|
||
|
|
||
|
cleanup:
|
||
|
|
||
|
if (AcquiredParametersLock) {
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
}
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersRead()=%x\n", Status));
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
NTSTATUS
|
||
|
CcPfParametersSave (
|
||
|
PCCPF_PREFETCHER_PARAMETERS PrefetcherParameters
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine updates the registry with the specified prefetch
|
||
|
parameters.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
PrefetcherParameters - Pointer to parameters structure.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Status.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
NTSTATUS Status;
|
||
|
PPF_TRACE_LIMITS TraceLimits;
|
||
|
PF_SCENARIO_TYPE ScenarioType;
|
||
|
WCHAR ValueName[CCPF_MAX_PARAMETER_NAME_LENGTH];
|
||
|
WCHAR *ValueNamePrefix;
|
||
|
HANDLE ParametersKey;
|
||
|
BOOLEAN EnableStatusSpecified;
|
||
|
ULONG EnablePrefetcher;
|
||
|
BOOLEAN AcquiredParametersLock;
|
||
|
ULONG Length;
|
||
|
PPF_SYSTEM_PREFETCH_PARAMETERS Parameters;
|
||
|
PKTHREAD CurrentThread;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
CurrentThread = KeGetCurrentThread ();
|
||
|
Parameters = &PrefetcherParameters->Parameters;
|
||
|
AcquiredParametersLock = FALSE;
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersSave()\n"));
|
||
|
|
||
|
//
|
||
|
// Get the parameters lock shared.
|
||
|
//
|
||
|
|
||
|
KeEnterCriticalRegionThread(CurrentThread);
|
||
|
ExAcquireResourceSharedLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
AcquiredParametersLock = TRUE;
|
||
|
|
||
|
//
|
||
|
// If we could not initialize the parameters key, we would fail
|
||
|
// all the following ops miserably.
|
||
|
//
|
||
|
|
||
|
if (!PrefetcherParameters->ParametersKey) {
|
||
|
Status = STATUS_REINITIALIZATION_NEEDED;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
ParametersKey = PrefetcherParameters->ParametersKey;
|
||
|
|
||
|
//
|
||
|
// Build up the prefetcher enable value.
|
||
|
//
|
||
|
|
||
|
EnableStatusSpecified = FALSE;
|
||
|
EnablePrefetcher = 0;
|
||
|
|
||
|
for (ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
|
||
|
|
||
|
//
|
||
|
// By default prefetching for all scenario types will be
|
||
|
// disabled, except it is explicitly enabled.
|
||
|
//
|
||
|
|
||
|
if (Parameters->EnableStatus[ScenarioType] == PfSvEnabled) {
|
||
|
EnablePrefetcher |= (1 << ScenarioType);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Even if enable status for one scenario type is specified,
|
||
|
// we have to save the enable prefetcher key.
|
||
|
//
|
||
|
|
||
|
if (Parameters->EnableStatus[ScenarioType] != PfSvNotSpecified) {
|
||
|
EnableStatusSpecified = TRUE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (EnableStatusSpecified) {
|
||
|
|
||
|
//
|
||
|
// Save the prefetcher enable key.
|
||
|
//
|
||
|
|
||
|
Length = sizeof(EnablePrefetcher);
|
||
|
|
||
|
Status = CcPfSetParameter(ParametersKey,
|
||
|
L"EnablePrefetcher",
|
||
|
REG_DWORD,
|
||
|
&EnablePrefetcher,
|
||
|
Length);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Save per scenario parameters.
|
||
|
//
|
||
|
|
||
|
for (ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
|
||
|
|
||
|
ValueNamePrefix = PrefetcherParameters->ScenarioTypePrefixes[ScenarioType];
|
||
|
|
||
|
//
|
||
|
// Update trace limits for this scenario type.
|
||
|
//
|
||
|
|
||
|
TraceLimits = &Parameters->TraceLimits[ScenarioType];
|
||
|
|
||
|
wcscpy(ValueName, ValueNamePrefix);
|
||
|
wcscat(ValueName, L"MaxNumPages");
|
||
|
Length = sizeof(TraceLimits->MaxNumPages);
|
||
|
Status = CcPfSetParameter(ParametersKey,
|
||
|
ValueName,
|
||
|
REG_DWORD,
|
||
|
&TraceLimits->MaxNumPages,
|
||
|
Length);
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
wcscpy(ValueName, ValueNamePrefix);
|
||
|
wcscat(ValueName, L"MaxNumSections");
|
||
|
Length = sizeof(TraceLimits->MaxNumSections);
|
||
|
Status = CcPfSetParameter(ParametersKey,
|
||
|
ValueName,
|
||
|
REG_DWORD,
|
||
|
&TraceLimits->MaxNumSections,
|
||
|
Length);
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
wcscpy(ValueName, ValueNamePrefix);
|
||
|
wcscat(ValueName, L"TimerPeriod");
|
||
|
Length = sizeof(TraceLimits->TimerPeriod);
|
||
|
Status = CcPfSetParameter(ParametersKey,
|
||
|
ValueName,
|
||
|
REG_BINARY,
|
||
|
&TraceLimits->TimerPeriod,
|
||
|
Length);
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update maximum number of active traces.
|
||
|
//
|
||
|
|
||
|
Length = sizeof(Parameters->MaxNumActiveTraces);
|
||
|
Status = CcPfSetParameter(ParametersKey,
|
||
|
L"MaxNumActiveTraces",
|
||
|
REG_DWORD,
|
||
|
&Parameters->MaxNumActiveTraces,
|
||
|
Length);
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update maximum number of saved traces.
|
||
|
//
|
||
|
|
||
|
Length = sizeof(Parameters->MaxNumSavedTraces);
|
||
|
Status = CcPfSetParameter(ParametersKey,
|
||
|
L"MaxNumSavedTraces",
|
||
|
REG_DWORD,
|
||
|
&Parameters->MaxNumSavedTraces,
|
||
|
Length);
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update the root directory path.
|
||
|
//
|
||
|
|
||
|
Length = (wcslen(Parameters->RootDirPath) + 1) * sizeof(WCHAR);
|
||
|
Status = CcPfSetParameter(ParametersKey,
|
||
|
L"RootDirPath",
|
||
|
REG_SZ,
|
||
|
Parameters->RootDirPath,
|
||
|
Length);
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update the hosting application list path.
|
||
|
//
|
||
|
|
||
|
Length = (wcslen(Parameters->HostingApplicationList) + 1) * sizeof(WCHAR);
|
||
|
Status = CcPfSetParameter(ParametersKey,
|
||
|
L"HostingAppList",
|
||
|
REG_SZ,
|
||
|
Parameters->HostingApplicationList,
|
||
|
Length);
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
Status = STATUS_SUCCESS;
|
||
|
|
||
|
cleanup:
|
||
|
|
||
|
if (AcquiredParametersLock) {
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
}
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersSave()=%x\n", Status));
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
NTSTATUS
|
||
|
CcPfParametersVerify (
|
||
|
PPF_SYSTEM_PREFETCH_PARAMETERS Parameters
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine verifies that the specified parameters structure is
|
||
|
valid and within sanity limits.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
Parameters - Pointer to parameters structure.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Status.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
NTSTATUS Status;
|
||
|
ULONG FailedCheckId;
|
||
|
ULONG CharIdx;
|
||
|
BOOLEAN FoundNUL;
|
||
|
PF_SCENARIO_TYPE ScenarioType;
|
||
|
PPF_TRACE_LIMITS TraceLimits;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
Status = STATUS_INVALID_PARAMETER;
|
||
|
FailedCheckId = 0;
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersVerify\n"));
|
||
|
|
||
|
//
|
||
|
// Make sure RootDirPath is NUL terminated.
|
||
|
//
|
||
|
|
||
|
FoundNUL = FALSE;
|
||
|
|
||
|
for (CharIdx = 0; CharIdx < PF_MAX_PREFETCH_ROOT_PATH; CharIdx++) {
|
||
|
if (Parameters->RootDirPath[CharIdx] == 0) {
|
||
|
FoundNUL = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (FoundNUL == FALSE) {
|
||
|
FailedCheckId = 10;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Make sure HostingApplicationList is NUL terminated.
|
||
|
//
|
||
|
|
||
|
FoundNUL = FALSE;
|
||
|
|
||
|
for (CharIdx = 0; CharIdx < PF_HOSTING_APP_LIST_MAX_CHARS; CharIdx++) {
|
||
|
if (Parameters->HostingApplicationList[CharIdx] == 0) {
|
||
|
FoundNUL = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Make sure the list is upper case.
|
||
|
//
|
||
|
|
||
|
if (towupper(Parameters->HostingApplicationList[CharIdx]) !=
|
||
|
Parameters->HostingApplicationList[CharIdx]) {
|
||
|
|
||
|
FailedCheckId = 13;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (FoundNUL == FALSE) {
|
||
|
FailedCheckId = 15;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Make sure all per scenario type parameters types are within
|
||
|
// sanity limits.
|
||
|
//
|
||
|
|
||
|
for (ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
|
||
|
|
||
|
if (Parameters->EnableStatus[ScenarioType] < 0 ||
|
||
|
Parameters->EnableStatus[ScenarioType] >= PfSvMaxEnableStatus) {
|
||
|
FailedCheckId = 20;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Check trace limits.
|
||
|
//
|
||
|
|
||
|
TraceLimits = &Parameters->TraceLimits[ScenarioType];
|
||
|
|
||
|
if (TraceLimits->MaxNumPages > PF_MAXIMUM_PAGES) {
|
||
|
FailedCheckId = 30;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
if (TraceLimits->MaxNumSections > PF_MAXIMUM_SECTIONS) {
|
||
|
FailedCheckId = 40;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
if ((TraceLimits->TimerPeriod < PF_MAXIMUM_TIMER_PERIOD) ||
|
||
|
(TraceLimits->TimerPeriod >= 0)) {
|
||
|
FailedCheckId = 50;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Check limits on active/saved traces.
|
||
|
//
|
||
|
|
||
|
if (Parameters->MaxNumActiveTraces > PF_MAXIMUM_ACTIVE_TRACES) {
|
||
|
FailedCheckId = 60;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
if (Parameters->MaxNumSavedTraces > PF_MAXIMUM_SAVED_TRACES) {
|
||
|
FailedCheckId = 70;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// We passed all the checks.
|
||
|
//
|
||
|
|
||
|
Status = STATUS_SUCCESS;
|
||
|
|
||
|
cleanup:
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersVerify()=%x,%d\n", Status, FailedCheckId));
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
CcPfParametersWatcher(
|
||
|
IN PCCPF_PREFETCHER_PARAMETERS PrefetcherParameters
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine gets called when our parameters in the registry change.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
PrefetcherParameters - Pointer to parameters structure.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
NTSTATUS Status;
|
||
|
UNICODE_STRING KeyName;
|
||
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
||
|
HANDLE ParametersKey;
|
||
|
PKTHREAD CurrentThread;
|
||
|
HANDLE TempHandle;
|
||
|
BOOLEAN HoldingParametersLock;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
HoldingParametersLock = FALSE;
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersWatcher()\n"));
|
||
|
|
||
|
//
|
||
|
// Our change notify triggered. Request further notification. But
|
||
|
// first wait until we can get the parameters lock exclusive, so
|
||
|
// while we are saving parameters to the registry we don't kick
|
||
|
// off a notification for each key.
|
||
|
//
|
||
|
|
||
|
CurrentThread = KeGetCurrentThread ();
|
||
|
KeEnterCriticalRegionThread(CurrentThread);
|
||
|
ExAcquireResourceExclusiveLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
|
||
|
//
|
||
|
// Hold the parameters lock shared since we are using ParametersKey.
|
||
|
//
|
||
|
|
||
|
KeEnterCriticalRegionThread(CurrentThread);
|
||
|
ExAcquireResourceSharedLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
HoldingParametersLock = TRUE;
|
||
|
|
||
|
//
|
||
|
// Make sure we still have a parameters key.
|
||
|
//
|
||
|
|
||
|
if (!PrefetcherParameters->ParametersKey) {
|
||
|
|
||
|
//
|
||
|
// In order to have setup a registry watch, we should have
|
||
|
// initialized the parameters key successfully.
|
||
|
//
|
||
|
|
||
|
CCPF_ASSERT(PrefetcherParameters->ParametersKey);
|
||
|
Status = STATUS_UNSUCCESSFUL;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
Status = ZwNotifyChangeKey(PrefetcherParameters->ParametersKey,
|
||
|
NULL,
|
||
|
(PIO_APC_ROUTINE)&PrefetcherParameters->RegistryWatchWorkItem,
|
||
|
(PVOID)(UINT_PTR)(unsigned int)DelayedWorkQueue,
|
||
|
&PrefetcherParameters->RegistryWatchIosb,
|
||
|
REG_LEGAL_CHANGE_FILTER,
|
||
|
FALSE,
|
||
|
&PrefetcherParameters->RegistryWatchBuffer,
|
||
|
sizeof(PrefetcherParameters->RegistryWatchBuffer),
|
||
|
TRUE);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
|
||
|
//
|
||
|
// Somebody may have deleted the key. We have to recreate it then.
|
||
|
//
|
||
|
|
||
|
if (Status == STATUS_KEY_DELETED) {
|
||
|
|
||
|
RtlInitUnicodeString(&KeyName, CCPF_PARAMETERS_KEY);
|
||
|
|
||
|
InitializeObjectAttributes(&ObjectAttributes,
|
||
|
&KeyName,
|
||
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
||
|
NULL,
|
||
|
NULL);
|
||
|
|
||
|
Status = ZwCreateKey(&ParametersKey,
|
||
|
KEY_ALL_ACCESS,
|
||
|
&ObjectAttributes,
|
||
|
0,
|
||
|
NULL,
|
||
|
REG_OPTION_NON_VOLATILE,
|
||
|
0);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
DBGPR((CCPFID,PFERR,"CCPF: ParametersWatcher-FailedRecreate=%x\n",Status));
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update global key handle. To do this release the shared lock
|
||
|
// and get it exclusive.
|
||
|
//
|
||
|
|
||
|
CCPF_ASSERT(HoldingParametersLock);
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
ExAcquireResourceExclusiveLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
|
||
|
TempHandle = PrefetcherParameters->ParametersKey;
|
||
|
PrefetcherParameters->ParametersKey = ParametersKey;
|
||
|
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
ExAcquireResourceSharedLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
|
||
|
//
|
||
|
// Close the old handle.
|
||
|
//
|
||
|
|
||
|
if (TempHandle) {
|
||
|
ZwClose(TempHandle);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Did someone steal the parameters key from beneath us?
|
||
|
//
|
||
|
|
||
|
if (!PrefetcherParameters->ParametersKey) {
|
||
|
Status = STATUS_UNSUCCESSFUL;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Retry setting a notification again.
|
||
|
//
|
||
|
|
||
|
Status = ZwNotifyChangeKey(PrefetcherParameters->ParametersKey,
|
||
|
NULL,
|
||
|
(PIO_APC_ROUTINE)&PrefetcherParameters->RegistryWatchWorkItem,
|
||
|
(PVOID)(UINT_PTR)(unsigned int)DelayedWorkQueue,
|
||
|
&PrefetcherParameters->RegistryWatchIosb,
|
||
|
REG_LEGAL_CHANGE_FILTER,
|
||
|
FALSE,
|
||
|
&PrefetcherParameters->RegistryWatchBuffer,
|
||
|
sizeof(PrefetcherParameters->RegistryWatchBuffer),
|
||
|
TRUE);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
DBGPR((CCPFID,PFERR,"CCPF: ParametersWatcher-FailedReSetNotify=%x\n",Status));
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
} else {
|
||
|
DBGPR((CCPFID,PFERR,"CCPF: ParametersWatcher-FailedSetNotify=%x\n",Status));
|
||
|
goto cleanup;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Release the parameters lock as we'll need it when re-reading the
|
||
|
// parameters.
|
||
|
//
|
||
|
|
||
|
if (HoldingParametersLock) {
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
HoldingParametersLock = FALSE;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update the global parameters.
|
||
|
//
|
||
|
|
||
|
Status = CcPfParametersRead(PrefetcherParameters);
|
||
|
|
||
|
if (NT_SUCCESS(Status)) {
|
||
|
|
||
|
//
|
||
|
// Determine if prefetching is enabled.
|
||
|
//
|
||
|
|
||
|
CcPfDetermineEnablePrefetcher();
|
||
|
|
||
|
//
|
||
|
// Set the event so the service queries for the latest parameters.
|
||
|
//
|
||
|
|
||
|
CcPfParametersSetChangedEvent(PrefetcherParameters);
|
||
|
}
|
||
|
|
||
|
cleanup:
|
||
|
|
||
|
if (HoldingParametersLock) {
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
HoldingParametersLock = FALSE;
|
||
|
}
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
NTSTATUS
|
||
|
CcPfParametersSetChangedEvent(
|
||
|
PCCPF_PREFETCHER_PARAMETERS PrefetcherParameters
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine tries to open and set the event that tells the
|
||
|
service system prefetch parameters have changed.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Status.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
NTSTATUS Status;
|
||
|
UNICODE_STRING EventName;
|
||
|
OBJECT_ATTRIBUTES EventObjAttr;
|
||
|
HANDLE EventHandle;
|
||
|
PKTHREAD CurrentThread;
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersSetChangedEvent()\n"));
|
||
|
|
||
|
//
|
||
|
// If we have already opened the event, just signal it.
|
||
|
//
|
||
|
|
||
|
if (PrefetcherParameters->ParametersChangedEvent) {
|
||
|
|
||
|
ZwSetEvent(PrefetcherParameters->ParametersChangedEvent, NULL);
|
||
|
|
||
|
Status = STATUS_SUCCESS;
|
||
|
|
||
|
} else {
|
||
|
|
||
|
//
|
||
|
// Try to open the event. We don't open this at initialization
|
||
|
// because our service may not have started to create this
|
||
|
// event yet. If csrss.exe has not initialized, we may not
|
||
|
// even have the BaseNamedObjects object directory created, in
|
||
|
// which Win32 events reside.
|
||
|
//
|
||
|
|
||
|
RtlInitUnicodeString(&EventName, PF_PARAMETERS_CHANGED_EVENT_NAME);
|
||
|
|
||
|
InitializeObjectAttributes(&EventObjAttr,
|
||
|
&EventName,
|
||
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
||
|
NULL,
|
||
|
NULL);
|
||
|
|
||
|
Status = ZwOpenEvent(&EventHandle,
|
||
|
EVENT_ALL_ACCESS,
|
||
|
&EventObjAttr);
|
||
|
|
||
|
if (NT_SUCCESS(Status)) {
|
||
|
|
||
|
//
|
||
|
// Acquire the lock and set the global handle.
|
||
|
//
|
||
|
CurrentThread = KeGetCurrentThread ();
|
||
|
|
||
|
KeEnterCriticalRegionThread(CurrentThread);
|
||
|
ExAcquireResourceExclusiveLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
|
||
|
if (!PrefetcherParameters->ParametersChangedEvent) {
|
||
|
|
||
|
//
|
||
|
// Set the global handle.
|
||
|
//
|
||
|
|
||
|
PrefetcherParameters->ParametersChangedEvent = EventHandle;
|
||
|
CCPF_ASSERT(EventHandle);
|
||
|
|
||
|
EventHandle = NULL;
|
||
|
}
|
||
|
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
|
||
|
if (EventHandle != NULL) {
|
||
|
//
|
||
|
// Somebody already initialized the global handle
|
||
|
// before us. Close our handle and use the one they
|
||
|
// initialized.
|
||
|
//
|
||
|
|
||
|
ZwClose(EventHandle);
|
||
|
}
|
||
|
|
||
|
|
||
|
//
|
||
|
// We have an event now. Signal it.
|
||
|
//
|
||
|
|
||
|
ZwSetEvent(PrefetcherParameters->ParametersChangedEvent, NULL);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: ParametersSetChangedEvent()=%x\n", Status));
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
NTSTATUS
|
||
|
CcPfGetParameter (
|
||
|
HANDLE ParametersKey,
|
||
|
WCHAR *ValueNameBuffer,
|
||
|
ULONG ValueType,
|
||
|
PVOID Value,
|
||
|
ULONG *ValueSize
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine queries a value under the specified registry into the
|
||
|
specified buffer. Contents of Value and ValueSize are not changed
|
||
|
if returning failure.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
ParametersKey - Handle to key to query value under.
|
||
|
|
||
|
ValueNameBuffer - Name of the value.
|
||
|
|
||
|
ValueType - What the type of that value should be. (e.g. REG_DWORD).
|
||
|
|
||
|
Value - Queried value data gets put here.
|
||
|
|
||
|
ValueSize - Size of Value buffer in bytes. On successful return
|
||
|
this is set to number of bytes copied into Value.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Status.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
UNICODE_STRING ValueName;
|
||
|
CHAR Buffer[CCPF_MAX_PARAMETER_VALUE_BUFFER];
|
||
|
PKEY_VALUE_PARTIAL_INFORMATION ValueBuffer;
|
||
|
ULONG Length;
|
||
|
NTSTATUS Status;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
ValueBuffer = (PKEY_VALUE_PARTIAL_INFORMATION) Buffer;
|
||
|
Length = CCPF_MAX_PARAMETER_VALUE_BUFFER;
|
||
|
RtlInitUnicodeString(&ValueName, ValueNameBuffer);
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: GetParameter(%ws,%x)\n", ValueNameBuffer, ValueType));
|
||
|
|
||
|
//
|
||
|
// Verify parameters.
|
||
|
//
|
||
|
|
||
|
if (!ParametersKey) {
|
||
|
return STATUS_INVALID_PARAMETER;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Query value.
|
||
|
//
|
||
|
|
||
|
Status = ZwQueryValueKey(ParametersKey,
|
||
|
&ValueName,
|
||
|
KeyValuePartialInformation,
|
||
|
ValueBuffer,
|
||
|
Length,
|
||
|
&Length);
|
||
|
|
||
|
if (!NT_SUCCESS(Status)) {
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Make sure ZwQueryValue returns valid information.
|
||
|
//
|
||
|
|
||
|
if (Length < sizeof(KEY_VALUE_PARTIAL_INFORMATION)) {
|
||
|
CCPF_ASSERT(Length >= sizeof(KEY_VALUE_PARTIAL_INFORMATION));
|
||
|
Status = STATUS_UNSUCCESSFUL;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Check value type.
|
||
|
//
|
||
|
|
||
|
if (ValueBuffer->Type != ValueType) {
|
||
|
Status = STATUS_OBJECT_TYPE_MISMATCH;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Check if data will fit into the buffer caller passed in.
|
||
|
//
|
||
|
|
||
|
if (ValueBuffer->DataLength > *ValueSize) {
|
||
|
Status = STATUS_BUFFER_TOO_SMALL;
|
||
|
goto cleanup;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Copy data into user's buffer.
|
||
|
//
|
||
|
|
||
|
RtlCopyMemory(Value, ValueBuffer->Data, ValueBuffer->DataLength);
|
||
|
|
||
|
//
|
||
|
// Set copied number of bytes.
|
||
|
//
|
||
|
|
||
|
*ValueSize = ValueBuffer->DataLength;
|
||
|
|
||
|
Status = STATUS_SUCCESS;
|
||
|
|
||
|
cleanup:
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: GetParameter(%ws)=%x\n", ValueNameBuffer, Status));
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
NTSTATUS
|
||
|
CcPfSetParameter (
|
||
|
HANDLE ParametersKey,
|
||
|
WCHAR *ValueNameBuffer,
|
||
|
ULONG ValueType,
|
||
|
PVOID Value,
|
||
|
ULONG ValueSize
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine sets a parameter under the specified registry.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
ParametersKey - Handle to key to query value under.
|
||
|
|
||
|
ValueNameBuffer - Name of the value.
|
||
|
|
||
|
ValueType - What the type of that value should be. (e.g. REG_DWORD).
|
||
|
|
||
|
Value - Data to save.
|
||
|
|
||
|
ValueSize - Size of Value buffer in bytes.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Status.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
UNICODE_STRING ValueName;
|
||
|
NTSTATUS Status;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
RtlInitUnicodeString(&ValueName, ValueNameBuffer);
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: SetParameter(%ws,%x)\n", ValueNameBuffer, ValueType));
|
||
|
|
||
|
//
|
||
|
// Verify parameters.
|
||
|
//
|
||
|
|
||
|
if (!ParametersKey) {
|
||
|
return STATUS_INVALID_PARAMETER;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Save the value.
|
||
|
//
|
||
|
|
||
|
Status = ZwSetValueKey(ParametersKey,
|
||
|
&ValueName,
|
||
|
0,
|
||
|
ValueType,
|
||
|
Value,
|
||
|
ValueSize);
|
||
|
|
||
|
//
|
||
|
// Return the status.
|
||
|
//
|
||
|
|
||
|
DBGPR((CCPFID,PFTRC,"CCPF: SetParameter(%ws)=%x\n", ValueNameBuffer, Status));
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
LOGICAL
|
||
|
CcPfDetermineEnablePrefetcher(
|
||
|
VOID
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine sets the global CcPfEnablePrefetcher based on the
|
||
|
EnableStatus'es for all scenario types in global parameters as
|
||
|
well as other factors, such as whether we have booted safe mode.
|
||
|
|
||
|
Note: Acquires Parameters lock exclusive.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
New value of CcPfEnablePrefetcher.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
PF_SCENARIO_TYPE ScenarioType;
|
||
|
LOGICAL EnablePrefetcher;
|
||
|
PKTHREAD CurrentThread;
|
||
|
BOOLEAN IgnoreBootScenarioType;
|
||
|
PCCPF_PREFETCHER_PARAMETERS PrefetcherParameters;
|
||
|
|
||
|
extern PF_BOOT_PHASE_ID CcPfBootPhase;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
EnablePrefetcher = FALSE;
|
||
|
PrefetcherParameters = &CcPfGlobals.Parameters;
|
||
|
CurrentThread = KeGetCurrentThread ();
|
||
|
|
||
|
//
|
||
|
// Ignore whether prefetching is enabled for boot, if we've
|
||
|
// already past the point in boot where this matters.
|
||
|
//
|
||
|
|
||
|
IgnoreBootScenarioType = (CcPfBootPhase >= PfSessionManagerInitPhase) ? TRUE : FALSE;
|
||
|
|
||
|
KeEnterCriticalRegionThread(CurrentThread);
|
||
|
ExAcquireResourceExclusiveLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
|
||
|
//
|
||
|
// If we have booted to safe mode, the prefetcher will be disabled.
|
||
|
//
|
||
|
|
||
|
if (InitSafeBootMode) {
|
||
|
|
||
|
EnablePrefetcher = FALSE;
|
||
|
|
||
|
} else {
|
||
|
|
||
|
//
|
||
|
// By default prefetching is disabled. If prefetching is
|
||
|
// enabled for any scenario type, then the prefetcher is
|
||
|
// enabled.
|
||
|
//
|
||
|
|
||
|
for (ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
|
||
|
|
||
|
//
|
||
|
// Skip enable status for the boot scenario if requested.
|
||
|
//
|
||
|
|
||
|
if (IgnoreBootScenarioType) {
|
||
|
if (ScenarioType == PfSystemBootScenarioType) {
|
||
|
continue;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (PrefetcherParameters->Parameters.EnableStatus[ScenarioType] == PfSvEnabled) {
|
||
|
EnablePrefetcher = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Update global enable status.
|
||
|
//
|
||
|
|
||
|
CcPfEnablePrefetcher = EnablePrefetcher;
|
||
|
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
|
||
|
return CcPfEnablePrefetcher;
|
||
|
}
|
||
|
|
||
|
BOOLEAN
|
||
|
CcPfIsHostingApplication(
|
||
|
IN PWCHAR ExecutableName
|
||
|
)
|
||
|
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine determines whether the specified executable is in the
|
||
|
list of known hosting applications, e.g. rundll32, dllhost etc.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
ExecutableName - NUL terminated UPCASED executable name, e.g. "MMC.EXE"
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
TRUE - Executable is for a known hosting application.
|
||
|
|
||
|
FALSE - It is not.
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
Kernel mode. IRQL == PASSIVE_LEVEL.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
PCCPF_PREFETCHER_PARAMETERS PrefetcherParameters;
|
||
|
PKTHREAD CurrentThread;
|
||
|
PWCHAR CurrentPosition;
|
||
|
PWCHAR ListStart;
|
||
|
PWCHAR ListEnd;
|
||
|
ULONG ExecutableNameLength;
|
||
|
BOOLEAN FoundInList;
|
||
|
|
||
|
//
|
||
|
// Initialize locals.
|
||
|
//
|
||
|
|
||
|
PrefetcherParameters = &CcPfGlobals.Parameters;
|
||
|
CurrentThread = KeGetCurrentThread();
|
||
|
ExecutableNameLength = wcslen(ExecutableName);
|
||
|
FoundInList = FALSE;
|
||
|
|
||
|
//
|
||
|
// Get the parameters lock for read.
|
||
|
//
|
||
|
|
||
|
KeEnterCriticalRegionThread(CurrentThread);
|
||
|
ExAcquireResourceSharedLite(&PrefetcherParameters->ParametersLock, TRUE);
|
||
|
|
||
|
//
|
||
|
// Search for executable in hosting application list.
|
||
|
//
|
||
|
|
||
|
ListStart = PrefetcherParameters->Parameters.HostingApplicationList;
|
||
|
ListEnd = ListStart + wcslen(PrefetcherParameters->Parameters.HostingApplicationList);
|
||
|
|
||
|
for (CurrentPosition = wcsstr(ListStart, ExecutableName);
|
||
|
CurrentPosition != NULL;
|
||
|
CurrentPosition = wcsstr(CurrentPosition + 1, ExecutableName)) {
|
||
|
|
||
|
//
|
||
|
// We should not go beyond the limits.
|
||
|
//
|
||
|
|
||
|
if (CurrentPosition < ListStart || CurrentPosition >= ListEnd) {
|
||
|
CCPF_ASSERT(CurrentPosition >= ListStart);
|
||
|
CCPF_ASSERT(CurrentPosition < ListEnd);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// It should be the first item in the list or be preceded by a comma.
|
||
|
//
|
||
|
|
||
|
if (CurrentPosition != ListStart && *(CurrentPosition - 1) != L',') {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// It should be the last item in the list or be followed by a comma.
|
||
|
//
|
||
|
|
||
|
if (CurrentPosition + ExecutableNameLength != ListEnd &&
|
||
|
CurrentPosition[ExecutableNameLength] != L',') {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// We found it in the list.
|
||
|
//
|
||
|
|
||
|
FoundInList = TRUE;
|
||
|
break;
|
||
|
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Release the parameters lock.
|
||
|
//
|
||
|
|
||
|
ExReleaseResourceLite(&PrefetcherParameters->ParametersLock);
|
||
|
KeLeaveCriticalRegionThread(CurrentThread);
|
||
|
|
||
|
//
|
||
|
// Return whether the executable was found in the list.
|
||
|
//
|
||
|
|
||
|
return FoundInList;
|
||
|
}
|
||
|
|