Windows2003-3790/admin/services/sched/service/pfsvc.c
2020-09-30 16:53:55 +02:00

17906 lines
438 KiB
C

/*++
Copyright (c) 2000 Microsoft Corporation
Module Name:
pfsvc.c
Abstract:
This module contains the main rountines for the prefetcher service
responsible for maintaining prefetch scenario files.
Author:
Stuart Sechrest (stuartse)
Cenk Ergan (cenke)
Chuck Leinzmeier (chuckl)
Environment:
User Mode
--*/
#include <nt.h>
#include <ntrtl.h>
#include <nturtl.h>
#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <limits.h>
#include <aclapi.h>
#include <dbghelp.h>
#include <idletask.h>
#include <prefetch.h>
#include <shdcom.h>
#include <tchar.h>
#include "pfsvc.h"
//
// Routine called to register for notifications when processing of all idle
// tasks is requested from the idle task server.
//
typedef VOID (*PIT_PROCESS_IDLE_TASKS_NOTIFY_ROUTINE)(VOID);
BOOL
ItSpSetProcessIdleTasksNotifyRoutine (
PIT_PROCESS_IDLE_TASKS_NOTIFY_ROUTINE NotifyRoutine
);
//
// Globals.
//
PFSVC_GLOBALS PfSvcGlobals = {0};
//
// Exposed routines:
//
DWORD
WINAPI
PfSvcMainThread(
VOID *Param
)
/*++
Routine Description:
This is the main routine for the prefetcher service. It sets up
file notification on the input files directory and waits for work
or the signaling of the termination event.
Arguments:
Param - Pointer to handle to the event that will signal our
termination.
Return Value:
Win32 error code.
--*/
{
HANDLE hStopEvent;
HANDLE hTracesReadyEvent;
HANDLE hParametersChangedEvent;
HANDLE hEvents[4];
ULONG NumEvents;
DWORD ErrorCode;
BOOLEAN bExitMainLoop;
DWORD dwWait;
NTSTATUS Status;
PF_SCENARIO_TYPE ScenarioType;
BOOLEAN UpdatedParameters;
BOOLEAN PrefetchingEnabled;
HANDLE PrefetcherThreads[1];
ULONG NumPrefetcherThreads;
ULONG ThreadIdx;
//
// Initialize locals.
//
// FUTURE-2002/03/29-ScottMa -- NumEvents should be set where the events
// array is initialized, and asserted against the maximum value.
NumEvents = sizeof(hEvents) / sizeof(HANDLE);
hStopEvent = *((HANDLE *) Param);
hTracesReadyEvent = NULL;
hParametersChangedEvent = NULL;
NumPrefetcherThreads = 0;
DBGPR((PFID,PFTRC,"PFSVC: MainThread()\n"));
//
// Initialize globals.
//
ErrorCode = PfSvInitializeGlobals();
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedInitGlobals\n"));
goto cleanup;
}
//
// Save service start time, prefetcher version etc.
//
PfSvSaveStartInfo(PfSvcGlobals.ServiceDataKey);
//
// Get necessary permissions for this thread to perform prefetch
// service tasks.
//
ErrorCode = PfSvGetPrefetchServiceThreadPrivileges();
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedGetPrivileges\n"));
goto cleanup;
}
//
// Set permissions on the event that can be set to override
// waiting for system to be idle before processing traces, so it
// can be set by administrators.
//
ErrorCode = PfSvSetAdminOnlyPermissions(PFSVC_OVERRIDE_IDLE_EVENT_NAME,
PfSvcGlobals.OverrideIdleProcessingEvent,
SE_KERNEL_OBJECT);
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedSetPermissions1\n"));
goto cleanup;
}
ErrorCode = PfSvSetAdminOnlyPermissions(PFSVC_PROCESSING_COMPLETE_EVENT_NAME,
PfSvcGlobals.ProcessingCompleteEvent,
SE_KERNEL_OBJECT);
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedSetPermissions2\n"));
goto cleanup;
}
//
// Get system prefetch parameters.
//
ErrorCode = PfSvQueryPrefetchParameters(&PfSvcGlobals.Parameters);
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedQueryParameters\n"));
goto cleanup;
}
//
// Depending on system type, if various types of prefetching is
// not specified in the registry (i.e. not specifically disabled),
// enable it.
//
UpdatedParameters = FALSE;
if (PfSvcGlobals.OsVersion.wProductType == VER_NT_WORKSTATION) {
//
// Enable all prefetching types if they are not disabled.
//
for(ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
if (PfSvcGlobals.Parameters.EnableStatus[ScenarioType] == PfSvNotSpecified) {
PfSvcGlobals.Parameters.EnableStatus[ScenarioType] = PfSvEnabled;
UpdatedParameters = TRUE;
}
}
} else if (PfSvcGlobals.OsVersion.wProductType == VER_NT_SERVER ||
PfSvcGlobals.OsVersion.wProductType == VER_NT_DOMAIN_CONTROLLER) {
//
// Enable only boot prefetching.
//
if (PfSvcGlobals.Parameters.EnableStatus[PfSystemBootScenarioType] == PfSvNotSpecified) {
PfSvcGlobals.Parameters.EnableStatus[PfSystemBootScenarioType] = PfSvEnabled;
UpdatedParameters = TRUE;
}
}
//
// If we enabled prefetching for a scenario type, call the kernel
// to update the parameters.
//
if (UpdatedParameters) {
ErrorCode = PfSvSetPrefetchParameters(&PfSvcGlobals.Parameters);
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedSetParameters\n"));
goto cleanup;
}
}
//
// Continue only if prefetching for a scenario type is enabled.
//
PrefetchingEnabled = FALSE;
for(ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
if (PfSvcGlobals.Parameters.EnableStatus[ScenarioType] == PfSvEnabled) {
PrefetchingEnabled = TRUE;
break;
}
}
if (PrefetchingEnabled == FALSE) {
ErrorCode = ERROR_NOT_SUPPORTED;
DBGPR((PFID,PFERR,"PFSVC: MainThread()-PrefetchingNotEnabled\n"));
goto cleanup;
}
//
// Initialize the directory that contains prefetch instructions.
//
ErrorCode = PfSvInitializePrefetchDirectory(PfSvcGlobals.Parameters.RootDirPath);
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedInitPrefetchDir\n"));
goto cleanup;
}
//
// Create the event that the kernel will set when raw traces are
// available. Then set the event so that the first time into the loop we
// will immediately process whatever raw traces are already waiting.
//
// The event is an autoclearing event, so it resets to the not-signaled
// state when our wait is satisfied. This allows proper synchronization
// with the kernel prefetcher.
//
hTracesReadyEvent = CreateEvent(NULL,
FALSE,
FALSE,
PF_COMPLETED_TRACES_EVENT_WIN32_NAME);
if (hTracesReadyEvent == NULL) {
ErrorCode = GetLastError();
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedTracesReadyEvent\n"));
goto cleanup;
}
SetEvent(hTracesReadyEvent);
//
// Create the event that the kernel will set when system prefetch
// parameters change.
//
hParametersChangedEvent = CreateEvent(NULL,
FALSE,
FALSE,
PF_PARAMETERS_CHANGED_EVENT_WIN32_NAME);
if (hParametersChangedEvent == NULL) {
ErrorCode = GetLastError();
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedParamsChangedEvent\n"));
goto cleanup;
}
//
// Set permissions on the events kernel uses to communicate with us.
//
ErrorCode = PfSvSetAdminOnlyPermissions(PF_COMPLETED_TRACES_EVENT_WIN32_NAME,
hTracesReadyEvent,
SE_KERNEL_OBJECT);
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedSetPermissions3\n"));
goto cleanup;
}
ErrorCode = PfSvSetAdminOnlyPermissions(PF_PARAMETERS_CHANGED_EVENT_WIN32_NAME,
hParametersChangedEvent,
SE_KERNEL_OBJECT);
if (ErrorCode != ERROR_SUCCESS) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedSetPermissions3\n"));
goto cleanup;
}
//
// Queue a work item to wait for the shell ready event and notify
// the kernel.
//
QueueUserWorkItem(PfSvPollShellReadyWorker, NULL, WT_EXECUTELONGFUNCTION);
//
// Create a thread to process traces retrieved from the kernel.
//
PrefetcherThreads[NumPrefetcherThreads] = CreateThread(NULL,
0,
PfSvProcessTraceThread,
NULL,
0,
NULL);
if (PrefetcherThreads[NumPrefetcherThreads]) {
NumPrefetcherThreads++;
}
//
// Register a notification routine with the idle task server.
//
ItSpSetProcessIdleTasksNotifyRoutine(PfSvProcessIdleTasksCallback);
//
// Set up handles we are going to wait on.
//
hEvents[0] = hStopEvent;
hEvents[1] = hTracesReadyEvent;
hEvents[2] = hParametersChangedEvent;
hEvents[3] = PfSvcGlobals.CheckForMissedTracesEvent;
//
// This is the main loop. Wait on the events for work or for exit
// signal.
//
bExitMainLoop = FALSE;
do {
DBGPR((PFID,PFWAIT,"PFSVC: MainThread()-WaitForWork\n"));
dwWait = WaitForMultipleObjects(NumEvents, hEvents, FALSE, INFINITE);
DBGPR((PFID,PFWAIT,"PFSVC: MainThread()-EndWaitForWork=%x\n",dwWait));
switch(dwWait) {
case WAIT_OBJECT_0:
//
// Service exit event:
//
//
// Break out, cleanup and exit.
//
ErrorCode = ERROR_SUCCESS;
bExitMainLoop = TRUE;
break;
case WAIT_OBJECT_0 + 3:
//
// The event that is set when we had max number of queued
// traces and we processed one. We should check for traces
// we could not pick up because the queue had maxed.
//
//
// Fall through to retrieve traces from the kernel.
//
case WAIT_OBJECT_0 + 1:
//
// New traces are available event set by the kernel:
//
PfSvGetRawTraces();
break;
case WAIT_OBJECT_0 + 2:
//
// Prefetch parameters changed event:
//
//
// Get new system prefetch parameters.
//
ErrorCode = PfSvQueryPrefetchParameters(&PfSvcGlobals.Parameters);
//
// If we were not successful, we should not continue.
//
if (ErrorCode != ERROR_SUCCESS) {
bExitMainLoop = TRUE;
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedQueryParameters2\n"));
break;
}
//
// Update the path to the prefetch instructions directory.
//
ErrorCode = PfSvInitializePrefetchDirectory(PfSvcGlobals.Parameters.RootDirPath);
if (ErrorCode != ERROR_SUCCESS) {
bExitMainLoop = TRUE;
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedReinitPrefetchDir\n"));
break;
}
break;
default:
//
// Something gone wrong. Break out, cleanup and exit.
//
DBGPR((PFID,PFERR,"PFSVC: MainThread()-WaitForWorkFailed\n"));
ErrorCode = ERROR_INVALID_HANDLE;
bExitMainLoop = TRUE;
break;
}
} while (!bExitMainLoop);
cleanup:
//
// If we could create this event, make sure it is signaled when the
// service is exiting, so no one gets stuck on it.
//
if (PfSvcGlobals.ProcessingCompleteEvent) {
SetEvent(PfSvcGlobals.ProcessingCompleteEvent);
}
//
// Save exit information.
//
if (PfSvcGlobals.ServiceDataKey) {
PfSvSaveExitInfo(PfSvcGlobals.ServiceDataKey, ErrorCode);
}
//
// Make sure the terminate event is set and wait for all our
// threads to exit.
//
if (NumPrefetcherThreads) {
//
// We could not have created worker threads without having
// initialized the globals successfully.
//
PFSVC_ASSERT(PfSvcGlobals.TerminateServiceEvent);
SetEvent(PfSvcGlobals.TerminateServiceEvent);
for (ThreadIdx = 0; ThreadIdx < NumPrefetcherThreads; ThreadIdx++) {
PFSVC_ASSERT(PrefetcherThreads[ThreadIdx]);
DBGPR((PFID,PFWAIT,"PFSVC: MainThread()-WaitForThreadIdx(%d)\n", ThreadIdx));
WaitForSingleObject(PrefetcherThreads[ThreadIdx], INFINITE);
DBGPR((PFID,PFWAIT,"PFSVC: MainThread()-EndWaitForThreadIdx(%d)\n", ThreadIdx));
CloseHandle(PrefetcherThreads[ThreadIdx]);
}
}
if (hTracesReadyEvent != NULL) {
CloseHandle(hTracesReadyEvent);
}
if (hParametersChangedEvent != NULL) {
CloseHandle(hParametersChangedEvent);
}
//
// Cleanup all globals.
//
PfSvCleanupGlobals();
DBGPR((PFID,PFTRC,"PFSVC: MainThread()=%x\n", ErrorCode));
return ErrorCode;
}
//
// Internal service routines:
//
//
// Thread routines:
//
DWORD
WINAPI
PfSvProcessTraceThread(
VOID *Param
)
/*++
Routine Description:
This is the routine for the thread that processes traces and
updates scenarios.
Arguments:
Param - Ignored.
Return Value:
Win32 error code.
--*/
{
PFSVC_IDLE_TASK LayoutTask;
PFSVC_IDLE_TASK DirectoryCleanupTask;
PPFSVC_TRACE_BUFFER TraceBuffer;
PLIST_ENTRY HeadEntry;
HANDLE CheckForQueuedTracesEvents[3];
HANDLE BootTraceEvents[2];
DWORD ErrorCode;
ULONG NumCheckForQueuedTracesEvents;
ULONG OrgNumQueuedTraces;
ULONG WaitResult;
ULONG NumEvents;
ULONG NumFailedTraces;
BOOLEAN AcquiredTracesLock;
//
// Intialize locals.
//
TraceBuffer = NULL;
AcquiredTracesLock = FALSE;
PfSvInitializeTask(&LayoutTask);
PfSvInitializeTask(&DirectoryCleanupTask);
NumFailedTraces = 0;
//
// These are the events we wait on before picking up traces to
// process.
//
CheckForQueuedTracesEvents[0] = PfSvcGlobals.TerminateServiceEvent;
CheckForQueuedTracesEvents[1] = PfSvcGlobals.NewTracesToProcessEvent;
CheckForQueuedTracesEvents[2] = PfSvcGlobals.OverrideIdleProcessingEvent;
NumCheckForQueuedTracesEvents = sizeof(CheckForQueuedTracesEvents) / sizeof(HANDLE);
DBGPR((PFID,PFTRC,"PFSVC: ProcessTraceThread()\n"));
//
// Get necessary permissions for this thread to perform prefetch
// service tasks.
//
ErrorCode = PfSvGetPrefetchServiceThreadPrivileges();
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// If we are allowed to run the defragger...
// We'll update the layout file even if the registry setting prevents us
// from launching the defragger.
//
if (PfSvAllowedToRunDefragger(FALSE)) {
//
// Queue an idle task to check & update the optimal disk layout if
// necessary. Ignore failure to do so.
//
ErrorCode = PfSvRegisterTask(&LayoutTask,
ItOptimalDiskLayoutTaskId,
PfSvCommonTaskCallback,
PfSvUpdateOptimalLayout);
}
//
// Loop forever waiting for traces to process and processing them.
//
while(TRUE) {
//
// Grab queued traces lock to check for queued traces.
//
PFSVC_ASSERT(!AcquiredTracesLock);
PFSVC_ACQUIRE_LOCK(PfSvcGlobals.TracesLock);
AcquiredTracesLock = TRUE;
if (!IsListEmpty(&PfSvcGlobals.Traces)) {
//
// Dequeue and process the first entry in the list.
//
HeadEntry = RemoveHeadList(&PfSvcGlobals.Traces);
TraceBuffer = CONTAINING_RECORD(HeadEntry,
PFSVC_TRACE_BUFFER,
TracesLink);
PFSVC_ASSERT(PfSvcGlobals.NumTraces);
OrgNumQueuedTraces = PfSvcGlobals.NumTraces;
PfSvcGlobals.NumTraces--;
//
// Release the lock.
//
PFSVC_RELEASE_LOCK(PfSvcGlobals.TracesLock);
AcquiredTracesLock = FALSE;
//
// If we had maxed the queue, note to check for traces
// that we may have failed to pick up because the queue
// was full.
//
if (OrgNumQueuedTraces == PFSVC_MAX_NUM_QUEUED_TRACES) {
SetEvent(PfSvcGlobals.CheckForMissedTracesEvent);
//
// Let the thread that queries the kernel for traces
// wake up and run.
//
Sleep(0);
}
//
// Clear the event that says we don't have traces to
// process.
//
ResetEvent(PfSvcGlobals.ProcessingCompleteEvent);
//
// If this is a boot trace, wait for a little while for
// boot to be really over before processing it.
//
if (TraceBuffer->Trace.ScenarioType == PfSystemBootScenarioType) {
BootTraceEvents[0] = PfSvcGlobals.TerminateServiceEvent;
BootTraceEvents[1] = PfSvcGlobals.OverrideIdleProcessingEvent;
NumEvents = 2;
PFSVC_ASSERT(NumEvents <= (sizeof(BootTraceEvents) / sizeof(HANDLE)));
WaitResult = WaitForMultipleObjects(NumEvents,
BootTraceEvents,
FALSE,
45000); // 45 seconds.
if (WaitResult == WAIT_OBJECT_0) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
}
ErrorCode = PfSvProcessTrace(&TraceBuffer->Trace);
//
// Update statistics.
//
PfSvcGlobals.NumTracesProcessed++;
if (ErrorCode != ERROR_SUCCESS) {
PfSvcGlobals.LastTraceFailure = ErrorCode;
} else {
PfSvcGlobals.NumTracesSuccessful++;
}
//
// Free the trace buffer.
//
VirtualFree(TraceBuffer, 0, MEM_RELEASE);
TraceBuffer = NULL;
//
// Did we just create too many scenario files in the prefetch directory?
// Queue an idle task to clean up.
//
if (PfSvcGlobals.NumPrefetchFiles >= PFSVC_MAX_PREFETCH_FILES) {
if (!DirectoryCleanupTask.Registered) {
//
// Make sure we've cleaned up after a possible previous run.
//
PfSvCleanupTask(&DirectoryCleanupTask);
PfSvInitializeTask(&DirectoryCleanupTask);
ErrorCode = PfSvRegisterTask(&DirectoryCleanupTask,
ItPrefetchDirectoryCleanupTaskId,
PfSvCommonTaskCallback,
PfSvCleanupPrefetchDirectory);
}
}
//
// Every so many scenario launches it is good to see if we should update
// disk layout.
//
if (((PfSvcGlobals.NumTracesSuccessful + 1) % 32) == 0) {
//
// Even if we can't launch the defragger because of the
// registry setting, we'll update layout.ini, so if the user
// manually launches the defragger it will optimize layout
// as well.
//
if (PfSvAllowedToRunDefragger(FALSE)) {
if (!LayoutTask.Registered) {
//
// Make sure we've cleaned up after a possible previous run.
//
PfSvCleanupTask(&LayoutTask);
PfSvInitializeTask(&LayoutTask);
ErrorCode = PfSvRegisterTask(&LayoutTask,
ItOptimalDiskLayoutTaskId,
PfSvCommonTaskCallback,
PfSvUpdateOptimalLayout);
}
}
}
} else {
//
// The list is empty. Signal that we are done with all the
// queued traces if we don't have idle tasks to complete.
//
if (!LayoutTask.Registered &&
!DirectoryCleanupTask.Registered) {
SetEvent(PfSvcGlobals.ProcessingCompleteEvent);
}
//
// Release the lock.
//
PFSVC_RELEASE_LOCK(PfSvcGlobals.TracesLock);
AcquiredTracesLock = FALSE;
//
// Update the statistics if there were new failed traces.
//
// FUTURE-2002/03/29-ScottMa -- The statistics are not updated
// unless there were new failed traces, therefore the logged
// values for number of traces processed is stale.
if (NumFailedTraces != (PfSvcGlobals.NumTracesProcessed -
PfSvcGlobals.NumTracesSuccessful)) {
NumFailedTraces = PfSvcGlobals.NumTracesProcessed -
PfSvcGlobals.NumTracesSuccessful;
PfSvSaveTraceProcessingStatistics(PfSvcGlobals.ServiceDataKey);
}
//
// Wait until new traces are queued.
//
DBGPR((PFID,PFWAIT,"PFSVC: ProcessTraceThread()-WaitForTrace\n"));
NumEvents = NumCheckForQueuedTracesEvents;
WaitResult = WaitForMultipleObjects(NumEvents,
CheckForQueuedTracesEvents,
FALSE,
INFINITE);
DBGPR((PFID,PFWAIT,"PFSVC: ProcessTraceThread()-EndWaitForTrace=%x\n", WaitResult));
switch(WaitResult) {
case WAIT_OBJECT_0:
//
// Service exit event:
//
ErrorCode = ERROR_SUCCESS;
goto cleanup;
break;
case WAIT_OBJECT_0 + 1:
//
// New traces queued for processing event:
//
break;
case WAIT_OBJECT_0 + 2:
//
// Idle detection was overriden. If we had registered tasks
// to be run, we will unregister them and run them manually.
//
PfSvSaveTraceProcessingStatistics(PfSvcGlobals.ServiceDataKey);
if (LayoutTask.Registered) {
PfSvUnregisterTask(&LayoutTask, FALSE);
PfSvCleanupTask(&LayoutTask);
PfSvInitializeTask(&LayoutTask);
PfSvUpdateOptimalLayout(NULL);
}
if (DirectoryCleanupTask.Registered) {
PfSvUnregisterTask(&DirectoryCleanupTask, FALSE);
PfSvCleanupTask(&DirectoryCleanupTask);
PfSvInitializeTask(&DirectoryCleanupTask);
PfSvCleanupPrefetchDirectory(NULL);
}
//
// We will drop out of this block, check & process queued traces
// and then set the processing complete event.
//
break;
default:
//
// Something went wrong...
//
ErrorCode = ERROR_INVALID_HANDLE;
goto cleanup;
}
}
//
// Loop to check if there are new traces.
//
}
//
// We should not break out of the loop.
//
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_INVALID_FUNCTION;
cleanup:
if (AcquiredTracesLock) {
PFSVC_RELEASE_LOCK(PfSvcGlobals.TracesLock);
}
if (TraceBuffer) {
VirtualFree(TraceBuffer, 0, MEM_RELEASE);
}
PfSvUnregisterTask(&LayoutTask, FALSE);
PfSvCleanupTask(&LayoutTask);
PfSvUnregisterTask(&DirectoryCleanupTask, FALSE);
PfSvCleanupTask(&DirectoryCleanupTask);
DBGPR((PFID,PFTRC,"PFSVC: ProcessTraceThread()=%x\n", ErrorCode));
return ErrorCode;
}
DWORD
WINAPI
PfSvPollShellReadyWorker(
VOID *Param
)
/*++
Routine Description:
This is the routine for the thread that is spawned to poll the
ShellReadyEvent.
Arguments:
Param - Ignored.
Return Value:
Win32 error code.
--*/
{
HANDLE ShellReadyEvent;
HANDLE Events[2];
ULONG NumEvents;
ULONG PollPeriod;
ULONG TotalPollPeriod;
DWORD WaitResult;
DWORD ErrorCode;
NTSTATUS Status;
PREFETCHER_INFORMATION PrefetcherInformation;
PF_BOOT_PHASE_ID PhaseId;
//
// Initialize locals.
//
ShellReadyEvent = NULL;
Events[0] = PfSvcGlobals.TerminateServiceEvent;
NumEvents = 1;
DBGPR((PFID,PFTRC,"PFSVC: PollShellReadyThread()\n"));
//
// Get necessary permissions for this thread to perform prefetch
// service tasks.
//
ErrorCode = PfSvGetPrefetchServiceThreadPrivileges();
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Until we can open the shell ready event, wait on the service
// termination event and retry every PollPeriod milliseconds.
//
PollPeriod = 1000;
TotalPollPeriod = 0;
do {
//
// Try to open the shell ready event.
//
// FUTURE-2002/03/29-ScottMa -- The EVENT_ALL_ACCESS flag only needs
// to be SYNCHRONIZE, following the principle of least privelege.
ShellReadyEvent = OpenEvent(EVENT_ALL_ACCESS,FALSE,L"ShellReadyEvent");
if (ShellReadyEvent) {
break;
}
//
// Wait for a while.
//
DBGPR((PFID,PFWAIT,"PFSVC: PollShellReadyThread()-WaitForOpen\n"));
WaitResult = WaitForMultipleObjects(NumEvents,
Events,
FALSE,
PollPeriod);
DBGPR((PFID,PFWAIT,"PFSVC: PollShellReadyThread()-EndWaitForOpen=%d\n", WaitResult));
switch(WaitResult) {
case WAIT_OBJECT_0:
//
// Service exit event:
//
ErrorCode = ERROR_PROCESS_ABORTED;
goto cleanup;
break;
case WAIT_TIMEOUT:
//
// Fall through and try opening the shell ready event again.
//
break;
default:
//
// Something gone wrong. Break out, cleanup and exit.
//
ErrorCode = ERROR_INVALID_HANDLE;
goto cleanup;
}
TotalPollPeriod += PollPeriod;
} while (TotalPollPeriod < 180000);
//
// If we could not get the ShellReadyEvent, we timed out.
//
if (ShellReadyEvent == NULL) {
ErrorCode = ERROR_TIMEOUT;
goto cleanup;
}
//
// Wait on the ShellReadyEvent to be signaled.
//
Events[NumEvents] = ShellReadyEvent;
NumEvents++;
DBGPR((PFID,PFWAIT,"PFSVC: PollShellReadyThread()-WaitForShell\n"));
WaitResult = WaitForMultipleObjects(NumEvents,
Events,
FALSE,
60000);
DBGPR((PFID,PFWAIT,"PFSVC: PollShellReadyThread()-EndWaitForShell=%d\n",WaitResult));
switch (WaitResult) {
case WAIT_OBJECT_0:
//
// Service exit event:
//
ErrorCode = ERROR_PROCESS_ABORTED;
goto cleanup;
break;
case WAIT_OBJECT_0 + 1:
//
// Shell ready event got signaled. Let the kernel mode
// prefetcher know.
//
PhaseId = PfUserShellReadyPhase;
PrefetcherInformation.Magic = PF_SYSINFO_MAGIC_NUMBER;
PrefetcherInformation.Version = PF_CURRENT_VERSION;
PrefetcherInformation.PrefetcherInformationClass = PrefetcherBootPhase;
PrefetcherInformation.PrefetcherInformation = &PhaseId;
PrefetcherInformation.PrefetcherInformationLength = sizeof(PhaseId);
Status = NtSetSystemInformation(SystemPrefetcherInformation,
&PrefetcherInformation,
sizeof(PrefetcherInformation));
//
// Fall through with the status.
//
ErrorCode = RtlNtStatusToDosError(Status);
break;
case WAIT_TIMEOUT:
//
// Shell ready event was created but not signaled...
//
ErrorCode = ERROR_TIMEOUT;
break;
default:
//
// Something gone wrong.
//
ErrorCode = GetLastError();
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = ERROR_INVALID_FUNCTION;
}
}
//
// Fall through with status from the switch statement.
//
cleanup:
if (ShellReadyEvent) {
CloseHandle(ShellReadyEvent);
}
DBGPR((PFID,PFTRC,"PFSVC: PollShellReadyThread()=%x\n", ErrorCode));
return ErrorCode;
}
//
// Routines called by the main prefetcher thread.
//
DWORD
PfSvGetRawTraces(
VOID
)
/*++
Routine Description:
This routine checks for new traces prepared by the kernel. The new
traces are downloaded and queued so they can be processed.
Arguments:
None.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
NTSTATUS Status;
PPFSVC_TRACE_BUFFER TraceBuffer;
ULONG TraceBufferMaximumLength;
ULONG TraceBufferLength;
PREFETCHER_INFORMATION PrefetcherInformation;
ULONG NumTracesRetrieved;
ULONG FailedCheck;
//
// Initialize locals.
//
TraceBuffer = NULL;
TraceBufferMaximumLength = 0;
NumTracesRetrieved = 0;
DBGPR((PFID,PFTRC,"PFSVC: GetRawTraces()\n"));
//
// Clear the event that asks us to check for more traces.
//
ResetEvent(PfSvcGlobals.CheckForMissedTracesEvent);
//
// While we do not already have too many traces to process, get
// traces from the kernel.
//
while (PfSvcGlobals.NumTraces < PFSVC_MAX_NUM_QUEUED_TRACES) {
//
// Retrieve a trace from the kernel.
//
PrefetcherInformation.Version = PF_CURRENT_VERSION;
PrefetcherInformation.Magic = PF_SYSINFO_MAGIC_NUMBER;
PrefetcherInformation.PrefetcherInformationClass = PrefetcherRetrieveTrace;
PrefetcherInformation.PrefetcherInformation = &TraceBuffer->Trace;
if (TraceBufferMaximumLength <= FIELD_OFFSET(PFSVC_TRACE_BUFFER, Trace)) {
PrefetcherInformation.PrefetcherInformationLength = 0;
} else {
PrefetcherInformation.PrefetcherInformationLength =
TraceBufferMaximumLength - FIELD_OFFSET(PFSVC_TRACE_BUFFER, Trace);
}
Status = NtQuerySystemInformation(SystemPrefetcherInformation,
&PrefetcherInformation,
sizeof(PrefetcherInformation),
&TraceBufferLength);
if (!NT_SUCCESS(Status)) {
if (Status == STATUS_BUFFER_TOO_SMALL) {
if (TraceBuffer != NULL) {
VirtualFree(TraceBuffer, 0, MEM_RELEASE);
}
//
// Add room for the header we wrap over it.
//
TraceBufferLength += sizeof(PFSVC_TRACE_BUFFER) - sizeof(PF_TRACE_HEADER);
TraceBufferMaximumLength = ROUND_TRACE_BUFFER_SIZE(TraceBufferLength);
TraceBuffer = VirtualAlloc(NULL,
TraceBufferMaximumLength,
MEM_COMMIT,
PAGE_READWRITE);
if (TraceBuffer == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
continue;
} else if (Status == STATUS_NO_MORE_ENTRIES) {
break;
}
ErrorCode = RtlNtStatusToDosError(Status);
goto cleanup;
}
#ifdef PFSVC_DBG
//
// Write out the trace to a file:
//
if (PfSvcDbgMaxNumSavedTraces) {
WCHAR TraceFilePath[MAX_PATH + 1];
LONG NumChars;
//
// Build up a file name.
//
InterlockedIncrement(&PfSvcDbgTraceNumber);
PFSVC_ACQUIRE_LOCK(PfSvcGlobals.PrefetchRootLock);
NumChars = _snwprintf(TraceFilePath,
MAX_PATH,
L"%ws\\%ws%d.trc",
PfSvcGlobals.PrefetchRoot,
PfSvcDbgTraceBaseName,
PfSvcDbgTraceNumber % PfSvcDbgMaxNumSavedTraces);
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
if (NumChars > 0 && NumChars < MAX_PATH) {
//
// Make sure the path is terminated.
//
TraceFilePath[MAX_PATH - 1] = 0;
//
// Write out the trace.
//
PfSvWriteBuffer(TraceFilePath,
&TraceBuffer->Trace,
TraceBuffer->Trace.Size);
}
}
#endif // PFSVC_DBG
//
// Verify integrity of the trace.
//
if (!PfVerifyTraceBuffer(&TraceBuffer->Trace,
TraceBuffer->Trace.Size,
&FailedCheck)) {
DBGPR((PFID,PFWARN,"PFSVC: IGNORING TRACE\n"));
continue;
}
//
// Put it on the list of traces to process.
//
PFSVC_ACQUIRE_LOCK(PfSvcGlobals.TracesLock);
InsertTailList(&PfSvcGlobals.Traces, &TraceBuffer->TracesLink);
PfSvcGlobals.NumTraces++;
PFSVC_RELEASE_LOCK(PfSvcGlobals.TracesLock);
//
// Notify that there are new traces to process.
//
SetEvent(PfSvcGlobals.NewTracesToProcessEvent);
//
// Clean out the loop variables.
//
TraceBuffer = NULL;
TraceBufferMaximumLength = 0;
TraceBufferLength = 0;
NumTracesRetrieved++;
}
//
// We should never go above the limit of queued traces.
//
PFSVC_ASSERT(PfSvcGlobals.NumTraces <= PFSVC_MAX_NUM_QUEUED_TRACES);
ErrorCode = ERROR_SUCCESS;
cleanup:
if (TraceBuffer != NULL) {
VirtualFree(TraceBuffer, 0, MEM_RELEASE);
}
DBGPR((PFID,PFTRC,"PFSVC: GetRawTraces()=%x,%d\n", ErrorCode, NumTracesRetrieved));
return ErrorCode;
}
DWORD
PfSvInitializeGlobals(
VOID
)
/*++
Routine Description:
This routine initializes the global variables / tables etc.
Arguments:
None.
Return Value:
Win32 error code.
--*/
{
NTSTATUS Status;
DWORD ErrorCode;
ULONG FileIdx;
WCHAR *CSCRootPath;
ULONG CSCRootPathMaxChars;
//
// These are the path suffices to recognize files we don't want to
// prefetch for boot. Keep these sorted lexically going from
// LAST CHARACTER TO FIRST and UPCASE.
//
static WCHAR *FilesToIgnoreForBoot[] = {
L"SYSTEM32\\CONFIG\\SOFTWARE",
L"\\WMI\\TRACE.LOG",
L"SYSTEM32\\CONFIG\\SOFTWARE.LOG",
L"SYSTEM32\\CONFIG\\SAM.LOG",
L"SYSTEM32\\CONFIG\\SYSTEM.LOG",
L"SYSTEM32\\CONFIG\\DEFAULT.LOG",
L"SYSTEM32\\CONFIG\\SECURITY.LOG",
L"\\PERF.ETL",
L"SYSTEM32\\CONFIG\\SAM",
L"SYSTEM32\\CONFIG\\SYSTEM",
L"SYSTEM32\\CONFIG\\SYSTEM.ALT",
L"SYSTEM32\\CONFIG\\DEFAULT",
L"SYSTEM32\\CONFIG\\SECURITY",
};
DBGPR((PFID,PFTRC,"PFSVC: InitializeGlobals()\n"));
//
// Initialize locals.
//
CSCRootPath = NULL;
//
// Zero out the globals structure so we know what to cleanup if
// the initialization fails in the middle.
//
RtlZeroMemory(&PfSvcGlobals, sizeof(PfSvcGlobals));
//
// Initialize the list of traces to be processed.
//
InitializeListHead(&PfSvcGlobals.Traces);
PfSvcGlobals.NumTraces = 0;
//
// We have not launched the defragger for anything yet.
//
PfSvcGlobals.DefraggerErrorCode = ERROR_SUCCESS;
//
// Initialize table for registry files that we don't want to
// prefetch for boot.
//
PfSvcGlobals.FilesToIgnoreForBoot = FilesToIgnoreForBoot;
PfSvcGlobals.NumFilesToIgnoreForBoot =
sizeof(FilesToIgnoreForBoot) / sizeof(WCHAR *);
//
// Get OS version information.
//
RtlZeroMemory(&PfSvcGlobals.OsVersion, sizeof(PfSvcGlobals.OsVersion));
PfSvcGlobals.OsVersion.dwOSVersionInfoSize = sizeof(PfSvcGlobals.OsVersion);
Status = RtlGetVersion((PRTL_OSVERSIONINFOW)&PfSvcGlobals.OsVersion);
if (!NT_SUCCESS(Status)) {
DBGPR((PFID,PFERR,"PFSVC: MainThread()-FailedGetOSVersion\n"));
ErrorCode = RtlNtStatusToDosError(Status);
goto cleanup;
}
//
// Initialize the table of ignored files' suffix lengths.
//
PfSvcGlobals.FileSuffixLengths =
PFSVC_ALLOC(PfSvcGlobals.NumFilesToIgnoreForBoot * sizeof(ULONG));
if (!PfSvcGlobals.FileSuffixLengths) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
for (FileIdx = 0;
FileIdx < PfSvcGlobals.NumFilesToIgnoreForBoot;
FileIdx++) {
PfSvcGlobals.FileSuffixLengths[FileIdx] =
wcslen(PfSvcGlobals.FilesToIgnoreForBoot[FileIdx]);
}
//
// Create an event that will get signaled when the service is
// exiting.
//
PfSvcGlobals.TerminateServiceEvent = CreateEvent(NULL,
TRUE,
FALSE,
NULL);
if (PfSvcGlobals.TerminateServiceEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Initialize the lock for the list of traces to be processed.
//
PfSvcGlobals.TracesLock = CreateMutex(NULL, FALSE, NULL);
if (PfSvcGlobals.TracesLock == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Initialize the events that are used to communicate between the
// acquirer and processor of the traces.
//
PfSvcGlobals.NewTracesToProcessEvent = CreateEvent(NULL,
FALSE,
FALSE,
NULL);
if (PfSvcGlobals.NewTracesToProcessEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
PfSvcGlobals.CheckForMissedTracesEvent = CreateEvent(NULL,
FALSE,
FALSE,
NULL);
if (PfSvcGlobals.CheckForMissedTracesEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// This named manual-reset event can be set to force all traces to
// be processed as soon as they become available rather than
// waiting for the system to become idle first.
//
// NOTICE-2002/03/29-ScottMa -- Have we considered the impact of squatting
// on all the named events used in the code?
PfSvcGlobals.OverrideIdleProcessingEvent = CreateEvent(NULL,
TRUE,
FALSE,
PFSVC_OVERRIDE_IDLE_EVENT_NAME);
if (PfSvcGlobals.OverrideIdleProcessingEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// This named manual-reset event is created signaled. When this
// event is signaled, it means there are no traces we have to
// process now.
//
PfSvcGlobals.ProcessingCompleteEvent = CreateEvent(NULL,
TRUE,
TRUE,
PFSVC_PROCESSING_COMPLETE_EVENT_NAME);
if (PfSvcGlobals.ProcessingCompleteEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Initialize prefetch root path and the lock to protect it. The
// real root path will be initialized after parameters are queried
// from the kernel.
//
PfSvcGlobals.PrefetchRoot[0] = 0;
PfSvcGlobals.PrefetchRootLock = CreateMutex(NULL, FALSE, NULL);
if (PfSvcGlobals.PrefetchRootLock == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
PfSvcGlobals.NumPrefetchFiles = 0;
//
// Open the service data registry key, creating it if necessary.
//
ErrorCode = RegCreateKey(HKEY_LOCAL_MACHINE,
PFSVC_SERVICE_DATA_KEY,
&PfSvcGlobals.ServiceDataKey);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Check the registry to see if the user does not want us to run
// the defragger.
//
ErrorCode = PfSvGetDontRunDefragger(&PfSvcGlobals.DontRunDefragger);
if (ErrorCode != ERROR_SUCCESS) {
//
// By default we will run the defragger.
//
PfSvcGlobals.DontRunDefragger = FALSE;
}
//
// Determine CSC root path. It won't be used if we can't allocate or
// determine it, so don't worry about the error code.
//
CSCRootPathMaxChars = MAX_PATH + 1;
CSCRootPath = PFSVC_ALLOC(CSCRootPathMaxChars * sizeof(CSCRootPath[0]));
if (CSCRootPath) {
ErrorCode = PfSvGetCSCRootPath(CSCRootPath, CSCRootPathMaxChars);
if (ErrorCode == ERROR_SUCCESS) {
PfSvcGlobals.CSCRootPath = CSCRootPath;
CSCRootPath = NULL;
}
}
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: InitializeGlobals()=%x\n", ErrorCode));
if (CSCRootPath) {
PFSVC_FREE(CSCRootPath);
}
return ErrorCode;
}
VOID
PfSvCleanupGlobals(
VOID
)
/*++
Routine Description:
This routine uninitializes the global variables / tables etc.
Arguments:
None.
Return Value:
VOID
--*/
{
PPFSVC_TRACE_BUFFER TraceBuffer;
PLIST_ENTRY ListHead;
DBGPR((PFID,PFTRC,"PFSVC: CleanupGlobals()\n"));
//
// Free allocated table.
//
if (PfSvcGlobals.FileSuffixLengths) {
PFSVC_FREE(PfSvcGlobals.FileSuffixLengths);
}
//
// Free queued traces.
//
while (!IsListEmpty(&PfSvcGlobals.Traces)) {
ListHead = RemoveHeadList(&PfSvcGlobals.Traces);
PFSVC_ASSERT(PfSvcGlobals.NumTraces);
PfSvcGlobals.NumTraces--;
TraceBuffer = CONTAINING_RECORD(ListHead,
PFSVC_TRACE_BUFFER,
TracesLink);
VirtualFree(TraceBuffer, 0, MEM_RELEASE);
}
//
// Close handles to opened events/mutexes.
//
if (PfSvcGlobals.TerminateServiceEvent) {
CloseHandle(PfSvcGlobals.TerminateServiceEvent);
}
if (PfSvcGlobals.TracesLock) {
CloseHandle(PfSvcGlobals.TracesLock);
}
if (PfSvcGlobals.NewTracesToProcessEvent) {
CloseHandle(PfSvcGlobals.NewTracesToProcessEvent);
}
if (PfSvcGlobals.CheckForMissedTracesEvent) {
CloseHandle(PfSvcGlobals.CheckForMissedTracesEvent);
}
if (PfSvcGlobals.OverrideIdleProcessingEvent) {
CloseHandle(PfSvcGlobals.OverrideIdleProcessingEvent);
}
if (PfSvcGlobals.ProcessingCompleteEvent) {
CloseHandle(PfSvcGlobals.ProcessingCompleteEvent);
}
if (PfSvcGlobals.PrefetchRootLock) {
CloseHandle(PfSvcGlobals.PrefetchRootLock);
}
//
// Close service data key handle.
//
if (PfSvcGlobals.ServiceDataKey) {
RegCloseKey(PfSvcGlobals.ServiceDataKey);
}
//
// Free CSC root path.
//
if (PfSvcGlobals.CSCRootPath) {
PFSVC_FREE(PfSvcGlobals.CSCRootPath);
}
}
DWORD
PfSvGetCSCRootPath (
WCHAR *CSCRootPath,
ULONG CSCRootPathMaxChars
)
/*++
Routine Description:
This routine determines the root path for CSC (client side caching) files.
Arguments:
CSCRootPath - If successful, a NUL terminated string is copied into this buffer.
CSCRootPathMaxChars - Maximum characters we can copy into CSCRootPath
buffer including the terminating NUL.
Return Value:
Win32 error code.
--*/
{
WCHAR CSCDirName[] = L"CSC";
HKEY CSCKeyHandle;
ULONG WindowsDirectoryLength;
ULONG CSCRootPathLength;
ULONG RequiredNumChars;
DWORD ErrorCode;
DWORD BufferSize;
DWORD ValueType;
//
// Initialize locals.
//
CSCKeyHandle = NULL;
//
// Open CSC parameters key.
//
ErrorCode = RegOpenKey(HKEY_LOCAL_MACHINE,
TEXT(REG_STRING_NETCACHE_KEY_A),
&CSCKeyHandle);
if (ErrorCode == ERROR_SUCCESS) {
//
// Query system setting for the CSC root path.
//
BufferSize = CSCRootPathMaxChars * sizeof(CSCRootPath[0]);
ErrorCode = RegQueryValueEx(CSCKeyHandle,
TEXT(REG_STRING_DATABASE_LOCATION_A),
NULL,
&ValueType,
(PVOID)CSCRootPath,
&BufferSize);
if (ErrorCode == ERROR_SUCCESS) {
//
// Sanity check the length.
//
if ((BufferSize / sizeof(CSCRootPath[0])) < MAX_PATH) {
//
// Make sure the buffer is NUL terminated.
//
CSCRootPath[CSCRootPathMaxChars-1] = 0;
//
// We got what we wanted. Make sure it has room for and is terminated
// by a slash.
//
CSCRootPathLength = wcslen(CSCRootPath);
if (CSCRootPathLength < CSCRootPathMaxChars - 1) {
if (CSCRootPath[CSCRootPathLength - 1] != L'\\') {
CSCRootPath[CSCRootPathLength] = L'\\';
CSCRootPathLength++;
CSCRootPath[CSCRootPathLength] = L'\0';
}
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
}
}
}
//
// If we come here, we have to use the default CSC path i.e. %windir%\CSC
//
WindowsDirectoryLength = GetWindowsDirectory(CSCRootPath,
CSCRootPathMaxChars - 1);
if (WindowsDirectoryLength == 0) {
//
// There was an error.
//
ErrorCode = GetLastError();
PFSVC_ASSERT(ErrorCode != ERROR_SUCCESS);
goto cleanup;
}
//
// See if we have room to add \CSC\ and a terminating NUL.
//
RequiredNumChars = WindowsDirectoryLength;
RequiredNumChars ++; // leading backslash.
RequiredNumChars += wcslen(CSCDirName); // CSC.
RequiredNumChars ++; // ending backslash.
RequiredNumChars ++; // terminating NUL.
if (CSCRootPathMaxChars < RequiredNumChars) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Build up the path:
//
CSCRootPathLength = WindowsDirectoryLength;
if (CSCRootPath[CSCRootPathLength - 1] != L'\\') {
CSCRootPath[CSCRootPathLength] = L'\\';
CSCRootPathLength++;
}
wcscpy(CSCRootPath + CSCRootPathLength, CSCDirName);
CSCRootPathLength += wcslen(CSCDirName);
CSCRootPath[CSCRootPathLength] = L'\\';
CSCRootPathLength++;
//
// Terminate the string.
//
CSCRootPath[CSCRootPathLength] = L'\0';
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
if (CSCKeyHandle) {
RegCloseKey(CSCKeyHandle);
}
if (ErrorCode == ERROR_SUCCESS) {
//
// We have the path in CSCRootPath. It should be in the X:\path\
// format. It should also be somewhat long, otherwise we will mismatch
// to too many files that we will not prefetch. It should also be
// terminated by a \ and NUL.
//
PFSVC_ASSERT(CSCRootPathLength < CSCRootPathMaxChars);
if ((CSCRootPathLength > 6) &&
(CSCRootPath[1] == L':') &&
(CSCRootPath[2] == L'\\') &&
(CSCRootPath[CSCRootPathLength - 1] == L'\\') &&
(CSCRootPath[CSCRootPathLength] == L'\0')) {
//
// Remove the X: from the beginning of the path so we can match
// it to NT paths like \Device\HarddiskVolume1. Note that we have
// to move the terminating NUL too.
//
MoveMemory(CSCRootPath,
CSCRootPath + 2,
(CSCRootPathLength - 1) * sizeof(CSCRootPath[0]));
CSCRootPathLength -= 2;
//
// Upcase the path so we don't have to do expensive case insensitive
// comparisons.
//
_wcsupr(CSCRootPath);
} else {
ErrorCode = ERROR_BAD_FORMAT;
}
}
return ErrorCode;
}
DWORD
PfSvSetPrefetchParameters(
PPF_SYSTEM_PREFETCH_PARAMETERS Parameters
)
/*++
Routine Description:
This routine updates the system prefetch parameters in the kernel.
Arguments:
Parameters - Pointer to parameters structure.
Return Value:
Win32 error code.
--*/
{
PREFETCHER_INFORMATION PrefetcherInformation;
NTSTATUS Status;
DWORD ErrorCode;
PrefetcherInformation.Magic = PF_SYSINFO_MAGIC_NUMBER;
PrefetcherInformation.Version = PF_CURRENT_VERSION;
PrefetcherInformation.PrefetcherInformationClass = PrefetcherSystemParameters;
PrefetcherInformation.PrefetcherInformation = Parameters;
PrefetcherInformation.PrefetcherInformationLength = sizeof(*Parameters);
Status = NtSetSystemInformation(SystemPrefetcherInformation,
&PrefetcherInformation,
sizeof(PrefetcherInformation));
if (!NT_SUCCESS(Status)) {
ErrorCode = RtlNtStatusToDosError(Status);
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvQueryPrefetchParameters(
PPF_SYSTEM_PREFETCH_PARAMETERS Parameters
)
/*++
Routine Description:
This routine queries the system prefetch parameters from the kernel.
The calling thread must have called PfSvGetPrefetchServiceThreadPrivileges.
Arguments:
Parameters - Pointer to structure to update.
Return Value:
Win32 error code.
--*/
{
PREFETCHER_INFORMATION PrefetcherInformation;
NTSTATUS Status;
DWORD ErrorCode;
ULONG Length;
PrefetcherInformation.Magic = PF_SYSINFO_MAGIC_NUMBER;
PrefetcherInformation.Version = PF_CURRENT_VERSION;
PrefetcherInformation.PrefetcherInformationClass = PrefetcherSystemParameters;
PrefetcherInformation.PrefetcherInformation = Parameters;
PrefetcherInformation.PrefetcherInformationLength = sizeof(*Parameters);
Status = NtQuerySystemInformation(SystemPrefetcherInformation,
&PrefetcherInformation,
sizeof(PrefetcherInformation),
&Length);
if (!NT_SUCCESS(Status)) {
ErrorCode = RtlNtStatusToDosError(Status);
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvInitializePrefetchDirectory(
WCHAR *PathFromSystemRoot
)
/*++
Routine Description:
This routine builds up full path for the prefetch instructions
directory given PathFromSystemRoot, makes sure this directory
exists, and sets the security information on it. Finally, the
global PrefetchRoot path is updated with path to the new
directory.
Global NumPrefetchFiles is also updated.
The calling thread must have the SE_TAKE_OWNERSHIP_NAME privilege.
Arguments:
PathFromSystemRoot - Path to the prefetch directory from SystemRoot.
Return Value:
Win32 error code.
--*/
{
ULONG PathLength;
ULONG NumFiles;
HANDLE DirHandle;
DWORD ErrorCode;
DWORD FileAttributes;
WCHAR FullDirPathBuffer[MAX_PATH + 1];
//
// Initialize locals.
//
DirHandle = INVALID_HANDLE_VALUE;
DBGPR((PFID,PFTRC,"PFSVC: InitPrefetchDir(%ws)\n",PathFromSystemRoot));
//
// Build path name to the prefetch files directory.
// ExpandEnvironmentStrings return length includes space for
// the terminating NUL character.
//
PathLength = ExpandEnvironmentStrings(L"%SystemRoot%\\",
FullDirPathBuffer,
MAX_PATH);
PathLength += wcslen(PathFromSystemRoot);
if (PathLength > MAX_PATH) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Copy the path from system root.
//
wcscat(FullDirPathBuffer, PathFromSystemRoot);
//
// Create the directory if it does not already exist.
//
if (!CreateDirectory(FullDirPathBuffer, NULL)) {
ErrorCode = GetLastError();
if (ErrorCode == ERROR_ALREADY_EXISTS) {
//
// The directory, or a file with that name may already
// exist. Make sure it is the former.
//
FileAttributes = GetFileAttributes(FullDirPathBuffer);
if (FileAttributes == INVALID_FILE_ATTRIBUTES) {
ErrorCode = GetLastError();
goto cleanup;
}
if (!(FileAttributes & FILE_ATTRIBUTE_DIRECTORY)) {
ErrorCode = ERROR_CANNOT_MAKE;
goto cleanup;
}
} else {
goto cleanup;
}
}
//
// Disable indexing of the prefetch directory.
//
FileAttributes = GetFileAttributes(FullDirPathBuffer);
if (FileAttributes == INVALID_FILE_ATTRIBUTES) {
ErrorCode = GetLastError();
goto cleanup;
}
if (!SetFileAttributes(FullDirPathBuffer,
FileAttributes | FILE_ATTRIBUTE_NOT_CONTENT_INDEXED)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Set permissions.
//
ErrorCode = PfSvSetAdminOnlyPermissions(FullDirPathBuffer, NULL, SE_FILE_OBJECT);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Count the scenario files in the directory.
//
ErrorCode = PfSvCountFilesInDirectory(FullDirPathBuffer,
L"*." PF_PREFETCH_FILE_EXTENSION,
&NumFiles);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Update the global prefetch root directory path.
//
PFSVC_ACQUIRE_LOCK(PfSvcGlobals.PrefetchRootLock);
wcscpy(PfSvcGlobals.PrefetchRoot, FullDirPathBuffer);
PfSvcGlobals.NumPrefetchFiles = NumFiles;
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: InitPrefetchDir(%ws)=%x\n",PathFromSystemRoot,ErrorCode));
return ErrorCode;
}
DWORD
PfSvCountFilesInDirectory(
WCHAR *DirectoryPath,
WCHAR *MatchExpression,
PULONG NumFiles
)
/*++
Routine Description:
This is routine returns the number of files in the specified
directory whose names match the specified expression.
Arguments:
DirectoryPath - NULL terminated path to the directory.
MatchExpression - Something like "*.pf" Don't go nuts with DOS
type expressions, this function won't try to transmogrify them.
NumFiles - Number of files are returned here. Bogus if returned error.
Return Value:
Win32 error code.
--*/
{
IO_STATUS_BLOCK IoStatusBlock;
OBJECT_ATTRIBUTES ObjectAttributes;
UNICODE_STRING DirectoryPathU;
UNICODE_STRING MatchExpressionU;
HANDLE DirectoryHandle;
PVOID QueryBuffer;
PFILE_NAMES_INFORMATION FileInfo;
ULONG QueryBufferSize;
ULONG FileCount;
NTSTATUS Status;
DWORD ErrorCode;
BOOLEAN Success;
BOOLEAN AllocatedDirectoryPathU;
BOOLEAN OpenedDirectory;
BOOLEAN RestartScan;
//
// Initialize locals.
//
AllocatedDirectoryPathU = FALSE;
OpenedDirectory = FALSE;
QueryBuffer = NULL;
QueryBufferSize = 0;
RtlInitUnicodeString(&MatchExpressionU, MatchExpression);
DBGPR((PFID,PFTRC,"PFSVC: CountFilesInDirectory(%ws,%ws)\n", DirectoryPath, MatchExpression));
//
// Convert the path to NT path.
//
Success = RtlDosPathNameToNtPathName_U(DirectoryPath,
&DirectoryPathU,
NULL,
NULL);
if (!Success) {
ErrorCode = ERROR_PATH_NOT_FOUND;
goto cleanup;
}
AllocatedDirectoryPathU = TRUE;
//
// Open the directory.
//
InitializeObjectAttributes(&ObjectAttributes,
&DirectoryPathU,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
Status = NtOpenFile(&DirectoryHandle,
FILE_LIST_DIRECTORY | SYNCHRONIZE,
&ObjectAttributes,
&IoStatusBlock,
FILE_SHARE_READ | FILE_SHARE_WRITE,
FILE_DIRECTORY_FILE |
FILE_SYNCHRONOUS_IO_NONALERT |
FILE_OPEN_FOR_BACKUP_INTENT);
if (!NT_SUCCESS(Status)) {
ErrorCode = RtlNtStatusToDosError(Status);
goto cleanup;
}
OpenedDirectory = TRUE;
//
// Allocate a decent sized query buffer.
//
QueryBufferSize = sizeof(FILE_NAMES_INFORMATION) + MAX_PATH * sizeof(WCHAR);
QueryBufferSize *= 16;
QueryBuffer = PFSVC_ALLOC(QueryBufferSize);
if (!QueryBuffer) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Loop querying file data. We query FileNamesInformation so
// we don't have to access file metadata.
//
RestartScan = TRUE;
FileCount = 0;
while (TRUE) {
Status = NtQueryDirectoryFile(DirectoryHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
QueryBuffer,
QueryBufferSize,
FileNamesInformation,
FALSE,
&MatchExpressionU,
RestartScan);
RestartScan = FALSE;
//
// If there are no files that match the format, then we'll get
// STATUS_NO_SUCH_FILE.
//
if (Status == STATUS_NO_SUCH_FILE && (FileCount == 0)) {
//
// We'll return the fact that there are no such files in the
// directory.
//
break;
}
if (Status == STATUS_NO_MORE_FILES) {
//
// We are done.
//
break;
}
if (NT_ERROR(Status)) {
ErrorCode = RtlNtStatusToDosError(Status);
goto cleanup;
}
//
// Go through the files returned in the buffer.
//
for (FileInfo = QueryBuffer;
((PUCHAR) FileInfo < ((PUCHAR) QueryBuffer + QueryBufferSize));
FileInfo = (PVOID) (((PUCHAR) FileInfo) + FileInfo->NextEntryOffset)) {
FileCount++;
if (!FileInfo->NextEntryOffset) {
break;
}
}
}
*NumFiles = FileCount;
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: CountFilesInDirectory(%ws)=%d,%x\n", DirectoryPath, *NumFiles, ErrorCode));
if (AllocatedDirectoryPathU) {
RtlFreeHeap(RtlProcessHeap(), 0, DirectoryPathU.Buffer);
}
if (OpenedDirectory) {
NtClose(DirectoryHandle);
}
if (QueryBuffer) {
PFSVC_FREE(QueryBuffer);
}
return ErrorCode;
}
//
// Routines to process acquired traces:
//
DWORD
PfSvProcessTrace(
PPF_TRACE_HEADER Trace
)
/*++
Routine Description:
This routine is called to process a trace and update the the
scenario file.
Arguments:
Trace - Pointer to trace.
Return Value:
Win32 error code.
--*/
{
PPF_SCENARIO_HEADER Scenario;
PFSVC_SCENARIO_INFO ScenarioInfo;
WCHAR ScenarioFilePath[MAX_PATH];
ULONG ScenarioFilePathMaxChars;
DWORD ErrorCode;
//
// Initialize locals.
//
PfSvInitializeScenarioInfo(&ScenarioInfo,
&Trace->ScenarioId,
Trace->ScenarioType);
ScenarioFilePathMaxChars = sizeof(ScenarioFilePath) /
sizeof(ScenarioFilePath[0]);
Scenario = NULL;
DBGPR((PFID,PFTRC,"PFSVC: ProcessTrace(%p)\n", Trace));
//
// Build file path to existing information for this scenario.
//
ErrorCode = PfSvScenarioGetFilePath(ScenarioFilePath,
ScenarioFilePathMaxChars,
&Trace->ScenarioId);
if (ErrorCode != ERROR_SUCCESS) {
//
// The buffer we specified should have been big enough. This call
// should not fail.
//
PFSVC_ASSERT(ErrorCode == ERROR_SUCCESS);
goto cleanup;
}
//
// Map and verify scenario file if it exists. If we cannot open it,
// NULL Scenario should be returned.
//
ErrorCode = PfSvScenarioOpen(ScenarioFilePath,
&Trace->ScenarioId,
Trace->ScenarioType,
&Scenario);
PFSVC_ASSERT(Scenario || ErrorCode);
//
// Allocate memory upfront for trace & scenario processing.
//
ErrorCode = PfSvScenarioInfoPreallocate(&ScenarioInfo,
Scenario,
Trace);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Incorporate information from any existing scenario file.
//
if (Scenario) {
ErrorCode = PfSvAddExistingScenarioInfo(&ScenarioInfo, Scenario);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Unmap the scenario so we can write over it when done.
//
UnmapViewOfFile(Scenario);
Scenario = NULL;
}
//
// If this is the first launch of this scenario, it is likely that we
// will create a new scenario file for it.
//
if (ScenarioInfo.ScenHeader.NumLaunches == 1) {
//
// Do we already have too many scenario files in the prefetch directory?
//
if (PfSvcGlobals.NumPrefetchFiles > PFSVC_MAX_PREFETCH_FILES) {
//
// If this is not the boot scenario, we'll ignore it. We don't
// create new scenario files until we clean up the old ones.
//
if (ScenarioInfo.ScenHeader.ScenarioType != PfSystemBootScenarioType) {
#ifndef PFSVC_DBG
ErrorCode = ERROR_TOO_MANY_OPEN_FILES;
goto cleanup;
#endif // !PFSVC_DBG
}
}
PfSvcGlobals.NumPrefetchFiles++;
}
//
// Verify that volume magics from existing scenario match those in
// the new trace. If volumes change beneath us we'd need to fix
// file paths in the existing scenario. But that is too much work,
// so for now we just start new.
//
if (!PfSvVerifyVolumeMagics(&ScenarioInfo, Trace)) {
PfSvCleanupScenarioInfo(&ScenarioInfo);
PfSvInitializeScenarioInfo(&ScenarioInfo,
&Trace->ScenarioId,
Trace->ScenarioType);
ErrorCode = PfSvScenarioInfoPreallocate(&ScenarioInfo,
NULL,
Trace);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Also delete the existing scenario instruction in case we
// fail to update them since they are invalid now.
//
PfSvcGlobals.NumPrefetchFiles--;
DeleteFile(ScenarioFilePath);
}
//
// Merge information from new trace.
//
ErrorCode = PfSvAddTraceInfo(&ScenarioInfo, Trace);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Decide which pages to actually prefetch next time, and
// eliminate uninteresting sections and pages.
//
ErrorCode = PfSvApplyPrefetchPolicy(&ScenarioInfo);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// If no pages/sections are left in the scenario after applying
// the policy, we'll delete the scenario file.
//
if (ScenarioInfo.ScenHeader.NumSections == 0 ||
ScenarioInfo.ScenHeader.NumPages == 0) {
//
// We cannot have sections without pages or vice versa.
//
PFSVC_ASSERT(ScenarioInfo.ScenHeader.NumSections == 0);
PFSVC_ASSERT(ScenarioInfo.ScenHeader.NumPages == 0);
//
// Remove the scenario file.
//
DeleteFile(ScenarioFilePath);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Sort remaining sections by first access.
//
ErrorCode = PfSvSortSectionNodesByFirstAccess(&ScenarioInfo.SectionList);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Write out new scenario file.
//
ErrorCode = PfSvWriteScenario(&ScenarioInfo, ScenarioFilePath);
//
// Fall through with status.
//
cleanup:
PfSvCleanupScenarioInfo(&ScenarioInfo);
if (Scenario) {
UnmapViewOfFile(Scenario);
}
DBGPR((PFID,PFTRC,"PFSVC: ProcessTrace(%p)=%x\n", Trace, ErrorCode));
return ErrorCode;
}
VOID
PfSvInitializeScenarioInfo (
PPFSVC_SCENARIO_INFO ScenarioInfo,
PPF_SCENARIO_ID ScenarioId,
PF_SCENARIO_TYPE ScenarioType
)
/*++
Routine Description:
This routine initializes the specified new scenario structure. It
sets the fields of the embedded scenario header as if no previous
scenario information is available.
Arguments:
ScenarioInfo - Pointer to structure to initialize.
ScenarioId & ScenarioType - Identifiers for the scenario.
Return Value:
None.
--*/
{
//
// Initialize ScenarioInfo so we know what to cleanup. Zeroing the structure
// takes care of the following fields:
// OneBigAllocation
// NewPages
// HitPages
// MissedOpportunityPages
// IgnoredPages
// PrefetchedPages
//
RtlZeroMemory(ScenarioInfo, sizeof(PFSVC_SCENARIO_INFO));
InitializeListHead(&ScenarioInfo->SectionList);
InitializeListHead(&ScenarioInfo->VolumeList);
PfSvChunkAllocatorInitialize(&ScenarioInfo->SectionNodeAllocator);
PfSvChunkAllocatorInitialize(&ScenarioInfo->PageNodeAllocator);
PfSvChunkAllocatorInitialize(&ScenarioInfo->VolumeNodeAllocator);
PfSvStringAllocatorInitialize(&ScenarioInfo->PathAllocator);
//
// Initialize the embedded scenario header.
//
ScenarioInfo->ScenHeader.Version = PF_CURRENT_VERSION;
ScenarioInfo->ScenHeader.MagicNumber = PF_SCENARIO_MAGIC_NUMBER;
ScenarioInfo->ScenHeader.ServiceVersion = PFSVC_SERVICE_VERSION;
ScenarioInfo->ScenHeader.Size = 0;
ScenarioInfo->ScenHeader.ScenarioId = *ScenarioId;
ScenarioInfo->ScenHeader.ScenarioType = ScenarioType;
ScenarioInfo->ScenHeader.NumSections = 0;
ScenarioInfo->ScenHeader.NumPages = 0;
ScenarioInfo->ScenHeader.FileNameInfoSize = 0;
ScenarioInfo->ScenHeader.NumLaunches = 1;
ScenarioInfo->ScenHeader.Sensitivity = PF_MIN_SENSITIVITY;
//
// These fields help us not prefetch if a scenario is getting
// launched too frequently. RePrefetchTime and ReTraceTime's get
// set to default values after the scenario is launched a number
// of times. This allows training scenarios run after clearing the
// prefetch cache to be traced correctly.
//
ScenarioInfo->ScenHeader.LastLaunchTime.QuadPart = 0;
ScenarioInfo->ScenHeader.MinRePrefetchTime.QuadPart = 0;
ScenarioInfo->ScenHeader.MinReTraceTime.QuadPart = 0;
return;
}
VOID
PfSvCleanupScenarioInfo(
PPFSVC_SCENARIO_INFO ScenarioInfo
)
/*++
Routine Description:
This function cleans up a scenario info structure. It does not
free the structure itself. The structure should have been
initialized by PfSvInitializeScenarioInfo.
Arguments:
ScenarioInfo - Pointer to structure.
Return Value:
None.
--*/
{
PPFSVC_SECTION_NODE SectionNode;
PLIST_ENTRY SectListEntry;
PPFSVC_VOLUME_NODE VolumeNode;
PLIST_ENTRY VolumeListEntry;
//
// Walk through the volume nodes and free them. Do this before
// freeing section nodes, so when we are trying to cleanup a
// section node, it is not on a volume node's list.
//
while (!IsListEmpty(&ScenarioInfo->VolumeList)) {
VolumeListEntry = RemoveHeadList(&ScenarioInfo->VolumeList);
VolumeNode = CONTAINING_RECORD(VolumeListEntry,
PFSVC_VOLUME_NODE,
VolumeLink);
//
// Cleanup the volume node.
//
PfSvCleanupVolumeNode(ScenarioInfo, VolumeNode);
//
// Free the volume node.
//
PfSvChunkAllocatorFree(&ScenarioInfo->VolumeNodeAllocator, VolumeNode);
}
//
// Walk through the section nodes and free them.
//
while (!IsListEmpty(&ScenarioInfo->SectionList)) {
SectListEntry = RemoveHeadList(&ScenarioInfo->SectionList);
SectionNode = CONTAINING_RECORD(SectListEntry,
PFSVC_SECTION_NODE,
SectionLink);
//
// Cleanup the section node.
//
PfSvCleanupSectionNode(ScenarioInfo, SectionNode);
//
// Free the section node.
//
PfSvChunkAllocatorFree(&ScenarioInfo->SectionNodeAllocator, SectionNode);
}
//
// Cleanup allocators.
//
PfSvChunkAllocatorCleanup(&ScenarioInfo->SectionNodeAllocator);
PfSvChunkAllocatorCleanup(&ScenarioInfo->PageNodeAllocator);
PfSvChunkAllocatorCleanup(&ScenarioInfo->VolumeNodeAllocator);
PfSvStringAllocatorCleanup(&ScenarioInfo->PathAllocator);
//
// Free the one big allocation we made.
//
if (ScenarioInfo->OneBigAllocation) {
PFSVC_FREE(ScenarioInfo->OneBigAllocation);
}
return;
}
DWORD
PfSvScenarioGetFilePath(
OUT PWCHAR FilePath,
IN ULONG FilePathMaxChars,
IN PPF_SCENARIO_ID ScenarioId
)
/*++
Routine Description:
This routine builds the file path for the specified scenario.
Arguments:
FilePath - Output buffer.
FilePathMaxChars - Size of FilePath buffer in characters including NUL.
ScenarioId - Scenario identifier.
Return Value:
Win32 error code.
--*/
{
ULONG NumChars;
DWORD ErrorCode;
WCHAR ScenarioFileName[PF_MAX_SCENARIO_FILE_NAME];
BOOLEAN AcquiredPrefetchRootLock;
//
// Get the lock so the path to prefetch folder does not change
// beneath our feet.
//
PFSVC_ACQUIRE_LOCK(PfSvcGlobals.PrefetchRootLock);
AcquiredPrefetchRootLock = TRUE;
//
// Calculate how big an input buffer we will need.
//
NumChars = wcslen(PfSvcGlobals.PrefetchRoot);
NumChars += wcslen(L"\\");
NumChars += PF_MAX_SCENARIO_FILE_NAME;
if (NumChars >= FilePathMaxChars) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Build the scenario file name from scenario identifier.
//
swprintf(ScenarioFileName,
PF_SCEN_FILE_NAME_FORMAT,
ScenarioId->ScenName,
ScenarioId->HashId,
PF_PREFETCH_FILE_EXTENSION);
//
// Build file path from prefetch directory path and file name.
//
swprintf(FilePath,
L"%ws\\%ws",
PfSvcGlobals.PrefetchRoot,
ScenarioFileName);
PFSVC_ASSERT(wcslen(FilePath) < FilePathMaxChars);
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
AcquiredPrefetchRootLock = FALSE;
ErrorCode = ERROR_SUCCESS;
cleanup:
if (AcquiredPrefetchRootLock) {
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
}
return ErrorCode;
}
DWORD
PfSvScenarioOpen (
IN PWCHAR FilePath,
IN PPF_SCENARIO_ID ScenarioId,
IN PF_SCENARIO_TYPE ScenarioType,
OUT PPF_SCENARIO_HEADER *Scenario
)
/*++
Routine Description:
This routine maps & verifies the scenario instructions at FilePath.
If a Scenario is returned, caller has to call UnmapViewOfFile to cleanup.
Arguments:
FilePath - Path to scenario instructions.
Scenario - Pointer to base of mapping of scenario instructions or NULL
if the function returns an error.
Return Value:
Win32 error code.
--*/
{
PPF_SCENARIO_HEADER OpenedScenario;
DWORD FailedCheck;
DWORD ErrorCode;
DWORD FileSize;
//
// Initialize locals.
//
OpenedScenario = NULL;
//
// Initialize output parameters.
//
*Scenario = NULL;
//
// Try to map the scenario file.
//
ErrorCode = PfSvGetViewOfFile(FilePath,
&OpenedScenario,
&FileSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Verify the scenario file.
//
FailedCheck = 0;
if (!PfSvVerifyScenarioBuffer(OpenedScenario, FileSize, &FailedCheck) ||
(OpenedScenario->ScenarioType != ScenarioType) ||
OpenedScenario->ServiceVersion != PFSVC_SERVICE_VERSION) {
//
// This is a bogus / wrong / outdated scenario file. Remove
// it.
//
UnmapViewOfFile(OpenedScenario);
OpenedScenario = NULL;
DeleteFile(FilePath);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
*Scenario = OpenedScenario;
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ErrorCode != ERROR_SUCCESS) {
if (OpenedScenario) {
UnmapViewOfFile(OpenedScenario);
}
*Scenario = NULL;
} else {
//
// If we are returning success we should be returning a valid Scenario.
//
PFSVC_ASSERT(*Scenario);
}
return ErrorCode;
}
DWORD
PfSvScenarioInfoPreallocate(
IN PPFSVC_SCENARIO_INFO ScenarioInfo,
OPTIONAL IN PPF_SCENARIO_HEADER Scenario,
OPTIONAL IN PPF_TRACE_HEADER Trace
)
/*++
Routine Description:
This routine preallocates a heap to be divided up and used by the
various allocators when processing a prefetch trace. The default allocation
size is determined from the Trace and Scenario size.
Arguments:
ScenarioInfo - Pointer to scenario containing allocators to initialize.
Scenario - Pointer to scenario instructions.
Trace - Pointer to prefetch trace.
Return Value:
Win32 error code.
--*/
{
PUCHAR Allocation;
PUCHAR ChunkStart;
DWORD ErrorCode;
ULONG AllocationSize;
ULONG NumSections;
ULONG NumPages;
ULONG NumVolumes;
ULONG PathSize;
//
// Initialize locals.
//
Allocation = NULL;
NumSections = 0;
NumPages = 0;
NumVolumes = 0;
PathSize = 0;
//
// Estimate how much to preallocate. Over-estimate rather than under-
// estimate because we will have to go to the heap for individual allocations
// if we underestimate. If we overestimate, as long as we don't touch the extra
// pages allocated we don't get a hit.
//
if (Trace) {
NumSections += Trace->NumSections;
}
if (Scenario) {
NumSections += Scenario->NumSections;
}
if (Trace) {
NumPages += Trace->NumEntries;
}
if (Scenario) {
NumPages += Scenario->NumPages;
}
if (Trace) {
NumVolumes += Trace->NumVolumes;
}
if (Scenario) {
NumVolumes += Scenario->NumMetadataRecords;
//
// It is very likely that we will at least share the volume containing the
// main executables between the trace and existing scenario instructions.
// So if we have both Trace and Scenario take one volume node off the estimate.
//
if (Trace) {
PFSVC_ASSERT(NumVolumes);
NumVolumes--;
}
}
//
// It is hard to estimate how much we will allocate for various paths
// e.g. file paths & each level of parent directory paths etc. It should be less
// than the size of the total trace, although it probably makes up most of it.
//
if (Trace) {
PathSize += Trace->Size;
}
if (Scenario) {
PathSize += Scenario->FileNameInfoSize;
PathSize += Scenario->MetadataInfoSize;
}
//
// Add it all up.
//
AllocationSize = 0;
AllocationSize += _alignof(PFSVC_VOLUME_NODE);
AllocationSize += NumVolumes * sizeof(PFSVC_VOLUME_NODE);
AllocationSize += _alignof(PFSVC_SECTION_NODE);
AllocationSize += NumSections * sizeof(PFSVC_SECTION_NODE);
AllocationSize += _alignof(PFSVC_PAGE_NODE);
AllocationSize += NumPages * sizeof(PFSVC_PAGE_NODE);
AllocationSize += PathSize;
//
// Make one big allocation.
//
Allocation = PFSVC_ALLOC(AllocationSize);
if (!Allocation) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Divide up the big allocation. Since we are providing the buffers,
// allocators should not fail.
//
ChunkStart = Allocation;
//
// Volume nodes.
//
ErrorCode = PfSvChunkAllocatorStart(&ScenarioInfo->VolumeNodeAllocator,
ChunkStart,
sizeof(PFSVC_VOLUME_NODE),
NumVolumes);
if (ErrorCode != ERROR_SUCCESS) {
PFSVC_ASSERT(ErrorCode == ERROR_SUCCESS);
goto cleanup;
}
ChunkStart += (ULONG_PTR) NumVolumes * sizeof(PFSVC_VOLUME_NODE);
//
// Section nodes.
//
ChunkStart = PF_ALIGN_UP(ChunkStart, _alignof(PFSVC_SECTION_NODE));
ErrorCode = PfSvChunkAllocatorStart(&ScenarioInfo->SectionNodeAllocator,
ChunkStart,
sizeof(PFSVC_SECTION_NODE),
NumSections);
if (ErrorCode != ERROR_SUCCESS) {
PFSVC_ASSERT(ErrorCode == ERROR_SUCCESS);
goto cleanup;
}
ChunkStart += (ULONG_PTR) NumSections * sizeof(PFSVC_SECTION_NODE);
//
// Page nodes.
//
ChunkStart = PF_ALIGN_UP(ChunkStart, _alignof(PFSVC_PAGE_NODE));
ErrorCode = PfSvChunkAllocatorStart(&ScenarioInfo->PageNodeAllocator,
ChunkStart,
sizeof(PFSVC_PAGE_NODE),
NumPages);
if (ErrorCode != ERROR_SUCCESS) {
PFSVC_ASSERT(ErrorCode == ERROR_SUCCESS);
goto cleanup;
}
ChunkStart += (ULONG_PTR) NumPages * sizeof(PFSVC_PAGE_NODE);
//
// Path names.
//
ErrorCode = PfSvStringAllocatorStart(&ScenarioInfo->PathAllocator,
ChunkStart,
PathSize);
if (ErrorCode != ERROR_SUCCESS) {
PFSVC_ASSERT(ErrorCode == ERROR_SUCCESS);
goto cleanup;
}
ChunkStart += (ULONG_PTR) PathSize;
//
// We should not have passed beyond what we allocated.
//
PFSVC_ASSERT(ChunkStart > (PUCHAR) Allocation);
PFSVC_ASSERT(ChunkStart < (PUCHAR) Allocation + (ULONG_PTR) AllocationSize);
ScenarioInfo->OneBigAllocation = Allocation;
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ErrorCode != ERROR_SUCCESS) {
if (Allocation) {
PFSVC_FREE(Allocation);
}
}
return ErrorCode;
}
DWORD
PfSvAddExistingScenarioInfo(
PPFSVC_SCENARIO_INFO ScenarioInfo,
PPF_SCENARIO_HEADER Scenario
)
/*++
Routine Description:
This function gets existing scenario information for the specified
scenario and updates ScenarioInfo.
Arguments:
ScenarioInfo - Initialized scenario info structure.
Scenario - Pointer to mapped scenario instructions.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
PPFSVC_SECTION_NODE SectionNode;
PPFSVC_PAGE_NODE PageNode;
PPF_SECTION_RECORD Sections;
PPF_SECTION_RECORD SectionRecord;
PPF_PAGE_RECORD Pages;
PCHAR FileNameInfo;
WCHAR *FileName;
LONG PageIdx;
ULONG SectionIdx;
ULONG NumPages;
PCHAR MetadataInfoBase;
PPF_METADATA_RECORD MetadataRecordTable;
PPF_METADATA_RECORD MetadataRecord;
ULONG MetadataRecordIdx;
PWCHAR VolumePath;
//
// Copy over the existing scenario header.
//
ScenarioInfo->ScenHeader = *Scenario;
//
// Update number of launches.
//
ScenarioInfo->ScenHeader.NumLaunches++;
//
// Convert the scenario data into intermediate data structures
// we can manipulate easier:
//
//
// Create volume nodes from metadata records.
//
MetadataInfoBase = (PCHAR)Scenario + Scenario->MetadataInfoOffset;
MetadataRecordTable = (PPF_METADATA_RECORD) MetadataInfoBase;
for (MetadataRecordIdx = 0;
MetadataRecordIdx < Scenario->NumMetadataRecords;
MetadataRecordIdx++) {
MetadataRecord = &MetadataRecordTable[MetadataRecordIdx];
VolumePath = (PWCHAR)(MetadataInfoBase + MetadataRecord->VolumeNameOffset);
ErrorCode = PfSvCreateVolumeNode(ScenarioInfo,
VolumePath,
MetadataRecord->VolumeNameLength,
&MetadataRecord->CreationTime,
MetadataRecord->SerialNumber);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
//
// Convert page and section nodes.
//
Sections = (PPF_SECTION_RECORD) ((PCHAR)Scenario + Scenario->SectionInfoOffset);
Pages = (PPF_PAGE_RECORD) ((PCHAR)Scenario + Scenario->PageInfoOffset);
FileNameInfo = (PCHAR)Scenario + Scenario->FileNameInfoOffset;
for (SectionIdx = 0; SectionIdx < Scenario->NumSections; SectionIdx++) {
//
// Build a section node from this section record in the
// scenario file. PfSvGetSectionRecord will insert it into
// the new scenario by the section record's name.
//
SectionRecord = &Sections[SectionIdx];
FileName = (PWSTR) (FileNameInfo + SectionRecord->FileNameOffset);
SectionNode = PfSvGetSectionRecord (ScenarioInfo,
FileName,
SectionRecord->FileNameLength);
if (!SectionNode) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// There should not be duplicate sections in the
// scenario. The section node we got should have an empty
// section record.
//
PFSVC_ASSERT(SectionNode->SectionRecord.FirstPageIdx == 0);
PFSVC_ASSERT(SectionNode->SectionRecord.NumPages == 0);
PFSVC_ASSERT(SectionNode->OrgSectionIndex == ULONG_MAX);
//
// Update the index of this section in the scenario file.
//
SectionNode->OrgSectionIndex = SectionIdx;
//
// Update the section record in the section node.
//
SectionNode->SectionRecord = *SectionRecord;
//
// Put page records for the section into the list.
//
PageIdx = SectionRecord->FirstPageIdx;
NumPages = 0;
while (PageIdx != PF_INVALID_PAGE_IDX) {
if (NumPages >= SectionRecord->NumPages) {
//
// There should not be more pages on the list than
// what the section record says there is.
//
PFSVC_ASSERT(FALSE);
break;
}
PageNode = PfSvChunkAllocatorAllocate(&ScenarioInfo->PageNodeAllocator);
if (!PageNode) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Copy over the page record.
//
PageNode->PageRecord = Pages[PageIdx];
//
// Insert it into the section's page list. Note that
// the page records in the section should be sorted by
// offset. By inserting to the tail, we maintain that.
//
InsertTailList(&SectionNode->PageList, &PageNode->PageLink);
//
// Shift the usage history for this page record making
// room for whether this page was used in this launch.
//
PageNode->PageRecord.UsageHistory <<= 1;
//
// Shift the prefetch history for this page record and
// note whether we had asked this page to be
// prefetched in this launch.
//
PageNode->PageRecord.PrefetchHistory <<= 1;
if (!PageNode->PageRecord.IsIgnore) {
PageNode->PageRecord.PrefetchHistory |= 0x1;
}
//
// Keep the count of pages we had asked to be
// prefetched, so we can calculate hit rate and adjust
// the sensitivity.
//
if(!PageNode->PageRecord.IsIgnore) {
if (PageNode->PageRecord.IsImage) {
ScenarioInfo->PrefetchedPages++;
}
if (PageNode->PageRecord.IsData) {
ScenarioInfo->PrefetchedPages++;
}
} else {
ScenarioInfo->IgnoredPages++;
}
//
// Update next page idx.
//
PageIdx = Pages[PageIdx].NextPageIdx;
//
// Update number of pages we've copied.
//
NumPages++;
}
//
// We should have copied as many pages as the section said
// there were.
//
PFSVC_ASSERT(NumPages == SectionRecord->NumPages);
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvVerifyVolumeMagics(
PPFSVC_SCENARIO_INFO ScenarioInfo,
PPF_TRACE_HEADER Trace
)
/*++
Routine Description:
Walk through the volumes in the trace and make sure their magics
match the ones in ScenarioInfo.
Arguments:
ScenarioInfo - Pointer to scenario info structure.
Trace - Pointer to trace.
Return Value:
Win32 error code.
--*/
{
PPFSVC_VOLUME_NODE VolumeNode;
PPF_VOLUME_INFO VolumeInfo;
ULONG VolumeInfoSize;
ULONG VolumeIdx;
BOOLEAN VolumeMagicsMatch;
//
// Walk the volumes in the trace.
//
VolumeInfo = (PPF_VOLUME_INFO) ((PCHAR)Trace + Trace->VolumeInfoOffset);
for (VolumeIdx = 0; VolumeIdx < Trace->NumVolumes; VolumeIdx++) {
//
// Get the scenario info's volume node for this volume.
//
VolumeNode = PfSvGetVolumeNode(ScenarioInfo,
VolumeInfo->VolumePath,
VolumeInfo->VolumePathLength);
if (VolumeNode) {
//
// Make sure the magics match.
//
if (VolumeNode->SerialNumber != VolumeInfo->SerialNumber ||
VolumeNode->CreationTime.QuadPart != VolumeInfo->CreationTime.QuadPart) {
VolumeMagicsMatch = FALSE;
goto cleanup;
}
}
//
// Get the next volume.
//
VolumeInfoSize = sizeof(PF_VOLUME_INFO);
VolumeInfoSize += VolumeInfo->VolumePathLength * sizeof(WCHAR);
VolumeInfo = (PPF_VOLUME_INFO) ((PCHAR) VolumeInfo + VolumeInfoSize);
//
// Make sure VolumeInfo is aligned.
//
VolumeInfo = PF_ALIGN_UP(VolumeInfo, _alignof(PF_VOLUME_INFO));
}
//
// Volume magics for volumes that appear both in the trace and the
// scenario info matched.
//
VolumeMagicsMatch = TRUE;
cleanup:
return VolumeMagicsMatch;
}
DWORD
PfSvAddTraceInfo(
PPFSVC_SCENARIO_INFO ScenarioInfo,
PPF_TRACE_HEADER Trace
)
/*++
Routine Description:
Add information in a raw trace to the specified scenario info
structure.
Arguments:
ScenarioInfo - Pointer to scenario info structure.
Trace - Pointer to trace.
Return Value:
Win32 error code.
--*/
{
PPF_SECTION_INFO Section;
PPF_LOG_ENTRY LogEntries;
PCHAR pFileName;
PPFSVC_SECTION_NODE *SectionTable;
PPFSVC_SECTION_NODE SectionNode;
PPFSVC_VOLUME_NODE VolumeNode;
ULONG TraceEndIdx;
ULONG SectionIdx;
ULONG EntryIdx;
DWORD ErrorCode;
ULONG SectionLength;
ULONG NextSectionIndex;
PPF_VOLUME_INFO VolumeInfo;
ULONG VolumeInfoSize;
ULONG VolumeIdx;
ULONG SectionTableSize;
//
// Initialize locals so we know what to clean up.
//
SectionTable = NULL;
DBGPR((PFID,PFTRC,"PFSVC: AddTraceInfo()\n"));
//
// Update last launch time.
//
ScenarioInfo->ScenHeader.LastLaunchTime = Trace->LaunchTime;
//
// If this scenario has been launched a number of times, we update
// the min reprefetch and retrace times. See comment for
// PFSVC_MIN_LAUNCHES_FOR_LAUNCH_FREQ_CHECK.
//
if (ScenarioInfo->ScenHeader.NumLaunches >= PFSVC_MIN_LAUNCHES_FOR_LAUNCH_FREQ_CHECK) {
ScenarioInfo->ScenHeader.MinRePrefetchTime.QuadPart = PFSVC_DEFAULT_MIN_REPREFETCH_TIME;
ScenarioInfo->ScenHeader.MinReTraceTime.QuadPart = PFSVC_DEFAULT_MIN_RETRACE_TIME;
}
#ifdef PFSVC_DBG
//
// On checked build, always set these to 0, so we do prefetch every
// scenario launch.
//
ScenarioInfo->ScenHeader.MinRePrefetchTime.QuadPart = 0;
ScenarioInfo->ScenHeader.MinReTraceTime.QuadPart = 0;
#endif // PFSVC_DBG
//
// Walk through the volumes in the trace and create volume nodes
// for them.
//
VolumeInfo = (PPF_VOLUME_INFO) ((PCHAR)Trace + Trace->VolumeInfoOffset);
for (VolumeIdx = 0; VolumeIdx < Trace->NumVolumes; VolumeIdx++) {
//
// Upcase the path so we don't have to do expensive case
// insensitive comparisons.
//
_wcsupr(VolumeInfo->VolumePath);
ErrorCode = PfSvCreateVolumeNode(ScenarioInfo,
VolumeInfo->VolumePath,
VolumeInfo->VolumePathLength,
&VolumeInfo->CreationTime,
VolumeInfo->SerialNumber);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Get the next volume.
//
VolumeInfoSize = sizeof(PF_VOLUME_INFO);
VolumeInfoSize += VolumeInfo->VolumePathLength * sizeof(WCHAR);
VolumeInfo = (PPF_VOLUME_INFO) ((PCHAR) VolumeInfo + VolumeInfoSize);
//
// Make sure VolumeInfo is aligned.
//
VolumeInfo = PF_ALIGN_UP(VolumeInfo, _alignof(PF_VOLUME_INFO));
}
//
// Allocate section node table so we know where to put the logged
// page faults.
//
SectionTableSize = sizeof(PPFSVC_SECTION_NODE) * Trace->NumSections;
SectionTable = PFSVC_ALLOC(SectionTableSize);
if (!SectionTable) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
RtlZeroMemory(SectionTable, SectionTableSize);
//
// Walk through the sections in the trace, and either find
// existing section records in the new scenario or create new
// ones.
//
Section = (PPF_SECTION_INFO) ((PCHAR)Trace + Trace->SectionInfoOffset);
for (SectionIdx = 0; SectionIdx < Trace->NumSections; SectionIdx++) {
//
// Upcase the path so we don't have to do expensive case
// insensitive comparisons.
//
_wcsupr(Section->FileName);
//
// If the section is for metafile, simply add it as a directory
// to be prefetched. We don't keep track of individual faults,
// since we can't prefetch only parts of directories.
//
if (Section->Metafile) {
VolumeNode = PfSvGetVolumeNode(ScenarioInfo,
Section->FileName,
Section->FileNameLength);
PFSVC_ASSERT(VolumeNode);
if (VolumeNode) {
PfSvAddParentDirectoriesToList(&VolumeNode->DirectoryList,
VolumeNode->VolumePathLength,
Section->FileName,
Section->FileNameLength);
}
goto NextSection;
}
//
// Find or create a section record for this section.
//
SectionTable[SectionIdx] = PfSvGetSectionRecord(ScenarioInfo,
Section->FileName,
Section->FileNameLength);
//
// If we could not get a record, it is because we had to
// create one and we did not have enough memory.
//
if (!SectionTable[SectionIdx]) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
NextSection:
//
// Get the next section record in the trace.
//
SectionLength = sizeof(PF_SECTION_INFO) +
(Section->FileNameLength) * sizeof(WCHAR);
Section = (PPF_SECTION_INFO) ((PUCHAR) Section + SectionLength);
}
//
// Determine after which log entry the trace ends.
//
TraceEndIdx = PfSvGetTraceEndIdx(Trace);
//
// If the determined trace end is zero (as is the case for most
// applications running under stress that don't get any pagefaults
// traced for the first few seconds), bail out.
//
if (TraceEndIdx == 0) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Add logged pagefault information up to the determined trace end
// to the new scenario info.
//
LogEntries = (PPF_LOG_ENTRY) ((PCHAR)Trace + Trace->TraceBufferOffset);
//
// Keep track of NextSectionIdx so we can order the sections by
// the first access [i.e. first page fault in the trace]
//
NextSectionIndex = 0;
for (EntryIdx = 0; EntryIdx < TraceEndIdx; EntryIdx++) {
SectionNode = SectionTable[LogEntries[EntryIdx].SectionId];
//
// For metafile sections we don't create section nodes.
//
if (!SectionNode) {
continue;
}
//
// NewSectionIndex fields of all section nodes are initialized
// to ULONG_MAX. If we have not already seen this section in
// the trace note its order and update NextSectionIdx.
//
if (SectionNode->NewSectionIndex == ULONG_MAX) {
SectionNode->NewSectionIndex = NextSectionIndex;
NextSectionIndex++;
}
//
// Add fault information to our section record.
//
ErrorCode = PfSvAddFaultInfoToSection(ScenarioInfo,
&LogEntries[EntryIdx],
SectionNode);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
ErrorCode = ERROR_SUCCESS;
cleanup:
if (SectionTable) {
PFSVC_FREE(SectionTable);
}
DBGPR((PFID,PFTRC,"PFSVC: AddTraceInfo()=%x\n", ErrorCode));
return ErrorCode;
}
// FUTURE-2002/03/29-ScottMa -- The function below should not be named
// PfSvGetSectionRecord, since a PF_SECTION_RECORD is not returned.
// Consider renaming it to PfSvGetSectionNode.
PPFSVC_SECTION_NODE
PfSvGetSectionRecord(
PPFSVC_SCENARIO_INFO ScenarioInfo,
WCHAR *FilePath,
ULONG FilePathLength
)
/*++
Routine Description:
Find or create a section node in the scenario info for the
specified file path.
Arguments:
ScenarioInfo - Pointer to scenario info structure.
FilePath - NUL terminated NT path to file.
FilePathLength - Length of FilePath in chars excluding NUL.
Return Value:
Pointer to created or found section node or NULL if there was a
problem.
--*/
{
PPFSVC_SECTION_NODE SectionNode;
PLIST_ENTRY HeadEntry;
PLIST_ENTRY NextEntry;
LONG ComparisonResult;
ULONG FilePathSize;
PPFSVC_SECTION_NODE ReturnNode;
//
// Initialize locals.
//
ReturnNode = NULL;
//
// Walk through the existing sections records looking for a file
// name match. Section records are on a lexically sorted list.
//
HeadEntry = &ScenarioInfo->SectionList;
NextEntry = HeadEntry->Flink;
while (HeadEntry != NextEntry) {
SectionNode = CONTAINING_RECORD(NextEntry,
PFSVC_SECTION_NODE,
SectionLink);
ComparisonResult = wcscmp(SectionNode->FilePath, FilePath);
if (ComparisonResult == 0) {
//
// We found a match. Return this section record.
//
ReturnNode = SectionNode;
goto cleanup;
} else if (ComparisonResult > 0) {
//
// We won't find the name in our list. We have to create a
// new section record.
//
break;
}
NextEntry = NextEntry->Flink;
}
//
// We have to create a new section record. NextEntry points to
// where we have to insert it in the list.
//
SectionNode = PfSvChunkAllocatorAllocate(&ScenarioInfo->SectionNodeAllocator);
if (!SectionNode) {
ReturnNode = NULL;
goto cleanup;
}
//
// Initialize the section node.
//
SectionNode->FilePath = NULL;
InitializeListHead(&SectionNode->PageList);
InitializeListHead(&SectionNode->SectionVolumeLink);
SectionNode->NewSectionIndex = ULONG_MAX;
SectionNode->OrgSectionIndex = ULONG_MAX;
SectionNode->FileIndexNumber.QuadPart = -1i64;
//
// Initialize the section record.
//
RtlZeroMemory(&SectionNode->SectionRecord, sizeof(PF_SECTION_RECORD));
//
// Allocate and copy over the file name.
//
FilePathSize = (FilePathLength + 1) * sizeof(WCHAR);
SectionNode->FilePath = PfSvStringAllocatorAllocate(&ScenarioInfo->PathAllocator,
FilePathSize);
if (!SectionNode->FilePath) {
PfSvCleanupSectionNode(ScenarioInfo, SectionNode);
PfSvChunkAllocatorFree(&ScenarioInfo->SectionNodeAllocator, SectionNode);
ReturnNode = NULL;
goto cleanup;
}
RtlCopyMemory(SectionNode->FilePath, FilePath, FilePathSize);
//
// Update the file name length on the section record.
//
SectionNode->SectionRecord.FileNameLength = FilePathLength;
//
// Insert the section into the right spot on the scenario's list.
//
InsertTailList(NextEntry, &SectionNode->SectionLink);
//
// Return the newly setup section record.
//
ReturnNode = SectionNode;
cleanup:
return ReturnNode;
}
DWORD
PfSvAddFaultInfoToSection(
PPFSVC_SCENARIO_INFO ScenarioInfo,
PPF_LOG_ENTRY LogEntry,
PPFSVC_SECTION_NODE SectionNode
)
/*++
Routine Description:
Add fault information from a trace log entry to proper section
record in the new scenario.
Arguments:
ScenarioInfo - Pointer to scenario info structure.
LogEntry - Pointer to trace log entry.
SectionNode - Pointer to the section node the log entry belongs to.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
PPFSVC_PAGE_NODE PageNode;
PLIST_ENTRY HeadEntry;
PLIST_ENTRY NextEntry;
//
// Walk through the page records for this section.
//
HeadEntry = &SectionNode->PageList;
NextEntry = HeadEntry->Flink;
while (HeadEntry != NextEntry) {
PageNode = CONTAINING_RECORD(NextEntry,
PFSVC_PAGE_NODE,
PageLink);
if (PageNode->PageRecord.FileOffset > LogEntry->FileOffset) {
//
// We won't find this fault in this sorted list.
//
break;
} else if (PageNode->PageRecord.FileOffset == LogEntry->FileOffset) {
//
// We found the page, update the page record and section
// record with the info in log entry.
//
if (LogEntry->IsImage) {
PageNode->PageRecord.IsImage = 1;
} else {
PageNode->PageRecord.IsData = 1;
}
//
// Note the this page was used in this launch.
//
PageNode->PageRecord.UsageHistory |= 0x1;
//
// See if this page was prefetched for this launch and
// update appropriate stats.
//
if(PageNode->PageRecord.IsIgnore) {
ScenarioInfo->MissedOpportunityPages++;
} else {
ScenarioInfo->HitPages++;
}
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
NextEntry = NextEntry->Flink;
}
//
// We have to add a new page record before NextEntry in the list.
//
PageNode = PfSvChunkAllocatorAllocate(&ScenarioInfo->PageNodeAllocator);
if (!PageNode) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Set up new page record. First initialize fields.
//
PageNode->PageRecord.IsImage = 0;
PageNode->PageRecord.IsData = 0;
PageNode->PageRecord.IsIgnore = 0;
PageNode->PageRecord.FileOffset = LogEntry->FileOffset;
if (LogEntry->IsImage) {
PageNode->PageRecord.IsImage = 1;
} else {
PageNode->PageRecord.IsData = 1;
}
//
// Initialize usage history for this new page record noting that
// it was used in this launch.
//
PageNode->PageRecord.UsageHistory = 0x1;
//
// Initialize prefetch history for this new page record.
//
PageNode->PageRecord.PrefetchHistory = 0;
//
// Insert it into the sections pages list.
//
InsertTailList(NextEntry, &PageNode->PageLink);
//
// Update stats on the new scenario.
//
ScenarioInfo->NewPages++;
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvApplyPrefetchPolicy(
PPFSVC_SCENARIO_INFO ScenarioInfo
)
/*++
Routine Description:
Go through all the information in ScenarioInfo and determine which
pages/sections to prefetch for the next launch of the scenario.
Arguments:
ScenarioInfo - Pointer to scenario info structure.
Return Value:
Win32 error code.
--*/
{
ULONG Sensitivity;
PPFSVC_SECTION_NODE SectionNode;
PPFSVC_PAGE_NODE PageNode;
ULONG SectNumPagesToPrefetch;
ULONG HitPages;
ULONG MissedOpportunityPages;
ULONG PrefetchedPages;
ULONG IgnoredPages;
PLIST_ENTRY SectHead;
PLIST_ENTRY SectNext;
PLIST_ENTRY PageHead;
PLIST_ENTRY PageNext;
ULONG NumUsed;
PPF_SCENARIO_HEADER Scenario;
ULONG FileNameSize;
ULONG IgnoredFileIdx;
BOOLEAN bSkipSection;
PFSV_SUFFIX_COMPARISON_RESULT ComparisonResult;
DWORD ErrorCode;
PPFSVC_VOLUME_NODE VolumeNode;
PWCHAR MFTSuffix;
PWCHAR PathSuffix;
FILE_BASIC_INFORMATION FileInformation;
ULONG MFTSuffixLength;
//
// Initialize locals.
//
Scenario = &ScenarioInfo->ScenHeader;
MFTSuffix = L"\\$MFT";
MFTSuffixLength = wcslen(MFTSuffix);
DBGPR((PFID,PFTRC,"PFSVC: ApplyPrefetchPolicy()\n"));
//
// Initialize fields of the scenario header we will set up.
//
Scenario->NumSections = 0;
Scenario->NumPages = 0;
Scenario->FileNameInfoSize = 0;
//
// Determine sensitivity based on usage of the pages we prefetched
// and we ignored.
//
HitPages = ScenarioInfo->HitPages;
MissedOpportunityPages = ScenarioInfo->MissedOpportunityPages;
PrefetchedPages = ScenarioInfo->PrefetchedPages;
IgnoredPages = ScenarioInfo->IgnoredPages;
//
// Check what percent of the pages we brought were used.
//
if (PrefetchedPages &&
(((HitPages * 100) / PrefetchedPages) < PFSVC_MIN_HIT_PERCENTAGE)) {
//
// Our hit rate is low. Increase sensitivity of the
// scenario, so for us to prefetch a page, it has to be
// used in more of the last launches.
//
if (ScenarioInfo->ScenHeader.Sensitivity < PF_MAX_SENSITIVITY) {
ScenarioInfo->ScenHeader.Sensitivity ++;
}
} else if (IgnoredPages &&
(((MissedOpportunityPages * 100) / IgnoredPages) > PFSVC_MAX_IGNORED_PERCENTAGE)) {
//
// If we are using most of what we prefetched (or we are not
// prefetching anything!), but we ignored some pages we could
// have prefetched, and they were used too, time to decrease
// sensitivity so we ignore less pages.
//
if (ScenarioInfo->ScenHeader.Sensitivity > PF_MIN_SENSITIVITY) {
ScenarioInfo->ScenHeader.Sensitivity --;
}
}
//
// Don't let the boot scenario's sensitivity to fall below 2.
// This makes sure we don't pick up all the application setup &
// configuration updates that happen during boot once.
//
if (ScenarioInfo->ScenHeader.ScenarioType == PfSystemBootScenarioType) {
PFSVC_ASSERT(PF_MIN_SENSITIVITY <= 2);
if (ScenarioInfo->ScenHeader.Sensitivity < 2) {
ScenarioInfo->ScenHeader.Sensitivity = 2;
}
}
Sensitivity = ScenarioInfo->ScenHeader.Sensitivity;
//
// If number of times this scenario was launched is less
// than sensitivity, adjust sensitivity. Otherwise we
// won't end up prefetching anything.
//
if (Sensitivity > ScenarioInfo->ScenHeader.NumLaunches) {
Sensitivity = ScenarioInfo->ScenHeader.NumLaunches;
}
//
// Walk through pages for every section and determine if they
// should be prefetched or not based on scenario sensitivity and
// their usage history in the last launches.
//
SectHead = &ScenarioInfo->SectionList;
SectNext = SectHead->Flink;
while (SectHead != SectNext) {
SectionNode = CONTAINING_RECORD(SectNext,
PFSVC_SECTION_NODE,
SectionLink);
SectNext = SectNext->Flink;
//
// Initialize section records fields.
//
SectionNode->SectionRecord.IsImage = 0;
SectionNode->SectionRecord.IsData = 0;
SectionNode->SectionRecord.NumPages = 0;
//
// If we are nearing the limits for number of sections and
// pages, ignore the rest of the sections.
//
if (Scenario->NumSections >= PF_MAXIMUM_SECTIONS ||
Scenario->NumPages + PF_MAXIMUM_SECTION_PAGES >= PF_MAXIMUM_PAGES) {
//
// Remove this section node from our list.
//
PfSvCleanupSectionNode(ScenarioInfo, SectionNode);
RemoveEntryList(&SectionNode->SectionLink);
PfSvChunkAllocatorFree(&ScenarioInfo->SectionNodeAllocator, SectionNode);
continue;
}
//
// If this is the boot scenario, check to see if this is one
// of the sections we ignore.
//
if (Scenario->ScenarioType == PfSystemBootScenarioType) {
bSkipSection = FALSE;
for (IgnoredFileIdx = 0;
IgnoredFileIdx < PfSvcGlobals.NumFilesToIgnoreForBoot;
IgnoredFileIdx++) {
ComparisonResult = PfSvCompareSuffix(SectionNode->FilePath,
SectionNode->SectionRecord.FileNameLength,
PfSvcGlobals.FilesToIgnoreForBoot[IgnoredFileIdx],
PfSvcGlobals.FileSuffixLengths[IgnoredFileIdx],
TRUE);
if (ComparisonResult == PfSvSuffixIdentical) {
//
// The suffix matched.
//
bSkipSection = TRUE;
break;
} else if (ComparisonResult == PfSvSuffixGreaterThan) {
//
// Since the ignore-suffices are lexically sorted,
// this file name's suffix won't match others
// either.
//
bSkipSection = FALSE;
break;
}
}
if (bSkipSection) {
//
// Remove this section node from our list.
//
PfSvCleanupSectionNode(ScenarioInfo, SectionNode);
RemoveEntryList(&SectionNode->SectionLink);
PfSvChunkAllocatorFree(&ScenarioInfo->SectionNodeAllocator, SectionNode);
continue;
}
}
//
// Keep track of num pages to prefetch for this section.
//
SectNumPagesToPrefetch = 0;
PageHead = &SectionNode->PageList;
PageNext = PageHead->Flink;
while (PageHead != PageNext) {
PageNode = CONTAINING_RECORD(PageNext,
PFSVC_PAGE_NODE,
PageLink);
PageNext = PageNext->Flink;
//
// Get number of times this page was used in the launches
// in usage history.
//
NumUsed = PfSvGetNumTimesUsed(PageNode->PageRecord.UsageHistory,
PF_PAGE_HISTORY_SIZE);
//
// If it was not used at all in the history we've kept
// track of, remove it.
//
if (NumUsed == 0) {
RemoveEntryList(&PageNode->PageLink);
PfSvChunkAllocatorFree(&ScenarioInfo->PageNodeAllocator, PageNode);
continue;
}
//
// Update the number of pages for this section.
//
SectionNode->SectionRecord.NumPages++;
//
// Check if this page qualifies to be prefetched next time.
//
if (NumUsed >= Sensitivity) {
PageNode->PageRecord.IsIgnore = 0;
//
// Update the number of pages we are prefetching for
// this section.
//
SectNumPagesToPrefetch++;
//
// Update whether we are going to prefetch this
// section as image, data [or both].
//
SectionNode->SectionRecord.IsImage |= PageNode->PageRecord.IsImage;
SectionNode->SectionRecord.IsData |= PageNode->PageRecord.IsData;
} else {
PageNode->PageRecord.IsIgnore = 1;
}
}
//
// Check if we want to keep this section in the scenario:
//
bSkipSection = FALSE;
if (SectionNode->SectionRecord.NumPages == 0) {
//
// If we don't have any pages left for this section, remove
// it.
//
bSkipSection = TRUE;
} else if (SectionNode->SectionRecord.NumPages >= PF_MAXIMUM_SECTION_PAGES) {
//
// If we ended up with too many pages for this section, remove
// it.
//
bSkipSection = TRUE;
} else if (PfSvcGlobals.CSCRootPath &&
wcsstr(SectionNode->FilePath, PfSvcGlobals.CSCRootPath)) {
//
// Skip client side cache (CSC) files. These files may get encrypted as
// LocalSystem, and when the AppData folder is redirected, we may take
// minutes trying to open them when prefetching for shell launch.
//
bSkipSection = TRUE;
} else {
//
// Encrypted files may result in several network accesses during open,
// even if they are local. This is especially so if the AppData folder is
// redirected to a server. We cannot afford these network delays when
// blocking the scenario for prefetching.
//
ErrorCode = PfSvGetFileBasicInformation(SectionNode->FilePath,
&FileInformation);
if ((ErrorCode == ERROR_SUCCESS) &&
(FileInformation.FileAttributes & FILE_ATTRIBUTE_ENCRYPTED)) {
bSkipSection = TRUE;
}
}
if (bSkipSection) {
PfSvCleanupSectionNode(ScenarioInfo, SectionNode);
RemoveEntryList(&SectionNode->SectionLink);
PfSvChunkAllocatorFree(&ScenarioInfo->SectionNodeAllocator, SectionNode);
continue;
}
//
// Get the volume node for the volume this section is
// on. The volume node should have been added when the
// existing scenario information or the new trace
// information was added to the scenario info.
//
VolumeNode = PfSvGetVolumeNode(ScenarioInfo,
SectionNode->FilePath,
SectionNode->SectionRecord.FileNameLength);
PFSVC_ASSERT(VolumeNode);
if (VolumeNode) {
VolumeNode->NumAllSections++;
}
//
// If we are not prefetching any pages from this section for
// the next launch, mark it ignore.
//
if (SectNumPagesToPrefetch == 0) {
SectionNode->SectionRecord.IsIgnore = 1;
} else {
SectionNode->SectionRecord.IsIgnore = 0;
//
// If this is MFT section for this volume, save it on the volume
// node. We will add the pages referenced from MFT to the list of
// files to prefetch metadata for.
//
if ((VolumeNode && VolumeNode->MFTSectionNode == NULL) &&
(VolumeNode->VolumePathLength == (SectionNode->SectionRecord.FileNameLength - MFTSuffixLength))) {
PathSuffix = SectionNode->FilePath + SectionNode->SectionRecord.FileNameLength;
PathSuffix -= MFTSuffixLength;
if (wcscmp(PathSuffix, MFTSuffix) == 0) {
//
// This is the MFT section node for this volume.
//
VolumeNode->MFTSectionNode = SectionNode;
//
// Mark the MFT section node as "ignore" so kernel does
// not attempt to prefetch it directly.
//
VolumeNode->MFTSectionNode->SectionRecord.IsIgnore = 1;
//
// Save how many pages we'll prefetch from MFT on the section
// node. We save this instead of FileIndexNumber field, since
// there won't be one for MFT. We won't try to get one either
// since we are marking this section node ignore.
//
VolumeNode->MFTSectionNode->MFTNumPagesToPrefetch = SectNumPagesToPrefetch;
}
}
}
//
// If we are not ignoring this section, update its file system
// index number so its metadata can be prefetched.
//
if (SectionNode->SectionRecord.IsIgnore == 0) {
ErrorCode = PfSvGetFileIndexNumber(SectionNode->FilePath,
&SectionNode->FileIndexNumber);
if (ErrorCode == ERROR_SUCCESS) {
if (VolumeNode) {
//
// Insert this section node into the section list of
// the volume it is on.
//
InsertTailList(&VolumeNode->SectionList,
&SectionNode->SectionVolumeLink);
VolumeNode->NumSections++;
//
// Update volume node's directory list with parent
// directories of this file.
//
PfSvAddParentDirectoriesToList(&VolumeNode->DirectoryList,
VolumeNode->VolumePathLength,
SectionNode->FilePath,
SectionNode->SectionRecord.FileNameLength);
}
}
}
//
// Update number of sections, number of pages and file name
// info length on the scenario.
//
Scenario->NumSections++;
Scenario->NumPages += SectionNode->SectionRecord.NumPages;
FileNameSize = sizeof(WCHAR) *
(SectionNode->SectionRecord.FileNameLength + 1);
Scenario->FileNameInfoSize += FileNameSize;
}
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
DBGPR((PFID,PFTRC,"PFSVC: ApplyPrefetchPolicy()=%x\n", ErrorCode));
return ErrorCode;
}
ULONG
PfSvGetNumTimesUsed(
ULONG UsageHistory,
ULONG UsageHistorySize
)
/*++
Routine Description:
Calculate how many times a page seems to be used according to
UsageHistory.
Arguments:
UsageHistory - Bitmap. 1's correspond to "was used", 0 = "not used".
UsageHistorySize - Size of UsageHistory in bits from the least
significant bit.
Return Value:
How many times the page seems to be used.
--*/
{
ULONG NumUsed;
ULONG BitIdx;
//
// Initialize locals.
//
NumUsed = 0;
//
// Walk through the bits in usage history starting from the least
// significant and count how many bits are on. We can probably do
// this more efficiently.
//
// FUTURE-2002/03/29-ScottMa -- You're right... We CAN do this more
// efficiently!
for (BitIdx = 0; BitIdx < UsageHistorySize; BitIdx++) {
if (UsageHistory & (1 << BitIdx)) {
NumUsed++;
}
}
return NumUsed;
}
ULONG
PfSvGetTraceEndIdx(
PPF_TRACE_HEADER Trace
)
/*++
Routine Description:
Determines the index of the last page logged in the trace.
Arguments:
Trace - Pointer to trace.
Return Value:
Index of the last page logged.
--*/
{
ULONG TotalFaults;
ULONG PeriodIdx;
ULONG *Id;
DBGPR((PFID,PFSTRC,"PFSVC: GetTraceEndIdx(%p)\n", Trace));
TotalFaults = Trace->FaultsPerPeriod[0];
for (PeriodIdx = 1; PeriodIdx < PF_MAX_NUM_TRACE_PERIODS; PeriodIdx++) {
if(Trace->FaultsPerPeriod[PeriodIdx] < PFSVC_MIN_FAULT_THRESHOLD) {
//
// If this is not the boot scenario, determine that
// scenario has ended when logged pagefaults for a time
// slice falls below minimum.
//
if (Trace->ScenarioType != PfSystemBootScenarioType) {
break;
}
}
TotalFaults += Trace->FaultsPerPeriod[PeriodIdx];
}
//
// Sum of entries per period should not be greater than all
// entries logged.
//
PFSVC_ASSERT(TotalFaults <= Trace->NumEntries);
DBGPR((PFID,PFSTRC,"PFSVC: GetTraceEndIdx(%p)=%d\n", Trace, TotalFaults));
return TotalFaults;
}
//
// Routines to write updated scenario instructions to the scenario
// file.
//
DWORD
PfSvWriteScenario(
PPFSVC_SCENARIO_INFO ScenarioInfo,
PWCHAR ScenarioFilePath
)
/*++
Routine Description:
Prepare scenario instructions structure from the scenarion info
and write it to the specified file.
Arguments:
ScenarioInfo - Pointer to scenario info structure.
ScenarioFilePath - Path to scenarion file to update.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
PPF_SCENARIO_HEADER Scenario;
ULONG FailedCheck;
//
// Initialize locals.
//
Scenario = NULL;
DBGPR((PFID,PFTRC,"PFSVC: WriteScenario(%ws)\n", ScenarioFilePath));
//
// Build scenario dump from information we gathered.
//
ErrorCode = PfSvPrepareScenarioDump(ScenarioInfo, &Scenario);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Make sure the scenario we built passes the checks.
//
if (!PfSvVerifyScenarioBuffer(Scenario, Scenario->Size, &FailedCheck) ||
Scenario->ServiceVersion != PFSVC_SERVICE_VERSION) {
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Write out the buffer.
//
ErrorCode = PfSvWriteBuffer(ScenarioFilePath, Scenario, Scenario->Size);
//
// Fall through with ErrorCode.
//
cleanup:
if (Scenario) {
PFSVC_FREE(Scenario);
}
DBGPR((PFID,PFTRC,"PFSVC: WriteScenario(%ws)=%x\n", ScenarioFilePath, ErrorCode));
return ErrorCode;
}
DWORD
PfSvPrepareScenarioDump(
IN PPFSVC_SCENARIO_INFO ScenarioInfo,
OUT PPF_SCENARIO_HEADER *ScenarioPtr
)
/*++
Routine Description:
Allocate a contiguous scenario buffer and fill it in with
information in ScenarioInfo. ScenarioInfo is not modified.
Arguments:
ScenarioInfo - Pointer to scenario information built from an
existing scenario file and a scenario trace.
ScenarioPtr - If successful, pointer to allocated and built
scenario is put here. The caller should free this buffer when
done with it.
Return Value:
Win32 error code.
--*/
{
PPF_SCENARIO_HEADER Scenario;
ULONG Size;
DWORD ErrorCode;
PPF_SECTION_RECORD Sections;
PPF_SECTION_RECORD Section;
PPFSVC_SECTION_NODE SectionNode;
ULONG CurSectionIdx;
PPF_PAGE_RECORD Pages;
PPF_PAGE_RECORD Page;
PPF_PAGE_RECORD PreviousPage;
PPFSVC_PAGE_NODE PageNode;
ULONG CurPageIdx;
PCHAR FileNames;
ULONG CurFileInfoOffset;
PCHAR DestPtr;
PLIST_ENTRY SectHead;
PLIST_ENTRY SectNext;
PLIST_ENTRY PageHead;
PLIST_ENTRY PageNext;
ULONG FileNameSize;
PPFSVC_VOLUME_NODE VolumeNode;
PLIST_ENTRY HeadVolume;
PLIST_ENTRY NextVolume;
PCHAR MetadataInfoBase;
PPF_METADATA_RECORD MetadataRecordTable;
PPF_METADATA_RECORD MetadataRecord;
ULONG MetadataInfoSize;
ULONG NumMetadataRecords;
ULONG CurMetadataRecordIdx;
ULONG CopySize;
ULONG CurFilePrefetchIdx;
ULONG FilePrefetchInfoSize;
ULONG FilePrefetchCount;
PFILE_PREFETCH FilePrefetchInfo;
WCHAR *DirectoryPath;
ULONG DirectoryPathLength;
PPFSVC_PATH PathEntry;
LARGE_INTEGER IndexNumber;
ULONG DirectoryPathInfoSize;
ULONG DirectoryPathSize;
PPF_COUNTED_STRING DirectoryPathCS;
//
// Initialize locals.
//
Scenario = NULL;
DBGPR((PFID,PFTRC,"PFSVC: PrepareScenarioDump()\n"));
//
// Calculate how big the scenario is going to be.
//
Size = sizeof(PF_SCENARIO_HEADER);
Size += ScenarioInfo->ScenHeader.NumSections * sizeof(PF_SECTION_RECORD);
Size += ScenarioInfo->ScenHeader.NumPages * sizeof(PF_PAGE_RECORD);
Size += ScenarioInfo->ScenHeader.FileNameInfoSize;
//
// Add space for the metadata prefetch information.
//
//
// Make some space for aligning the metadata records table.
//
MetadataInfoSize = _alignof(PF_METADATA_RECORD);
HeadVolume = &ScenarioInfo->VolumeList;
NextVolume = HeadVolume->Flink;
NumMetadataRecords = 0;
while (NextVolume != HeadVolume) {
VolumeNode = CONTAINING_RECORD(NextVolume,
PFSVC_VOLUME_NODE,
VolumeLink);
NextVolume = NextVolume->Flink;
//
// If there are no sections at all on this volume, skip it.
//
if (VolumeNode->NumAllSections == 0) {
continue;
}
NumMetadataRecords++;
//
// Metadata record:
//
MetadataInfoSize += sizeof(PF_METADATA_RECORD);
//
// Volume Path:
//
MetadataInfoSize += (VolumeNode->VolumePathLength + 1) * sizeof(WCHAR);
//
// FilePrefetchInfo buffer: This has to be ULONGLONG
// aligned. Add extra space for that in case.
//
MetadataInfoSize += _alignof(FILE_PREFETCH);
MetadataInfoSize += sizeof(FILE_PREFETCH);
if (VolumeNode->NumSections) {
MetadataInfoSize += (VolumeNode->NumSections - 1) * sizeof(ULONGLONG);
}
MetadataInfoSize += VolumeNode->DirectoryList.NumPaths * sizeof(ULONGLONG);
if (VolumeNode->MFTSectionNode) {
MetadataInfoSize += VolumeNode->MFTSectionNode->MFTNumPagesToPrefetch * sizeof(ULONGLONG);
}
//
// Add space for the directory paths on this volume.
//
MetadataInfoSize += VolumeNode->DirectoryList.NumPaths * sizeof(PF_COUNTED_STRING);
MetadataInfoSize += VolumeNode->DirectoryList.TotalLength * sizeof(WCHAR);
//
// Note that PF_COUNTED_STRING contains space for one
// character. DirectoryList's total length excludes NUL's at
// the end of each path.
//
}
Size += MetadataInfoSize;
//
// Allocate scenario buffer.
//
Scenario = PFSVC_ALLOC(Size);
if (!Scenario) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Copy the header and set the size.
//
*Scenario = ScenarioInfo->ScenHeader;
Scenario->Size = Size;
DestPtr = (PCHAR) Scenario + sizeof(*Scenario);
//
// Initialize where our data is going.
//
Sections = (PPF_SECTION_RECORD) DestPtr;
Scenario->SectionInfoOffset = (ULONG) (DestPtr - (PCHAR) Scenario);
CurSectionIdx = 0;
DestPtr += Scenario->NumSections * sizeof(PF_SECTION_RECORD);
Pages = (PPF_PAGE_RECORD) DestPtr;
Scenario->PageInfoOffset = (ULONG) (DestPtr - (PCHAR) Scenario);
CurPageIdx = 0;
DestPtr += Scenario->NumPages * sizeof(PF_PAGE_RECORD);
FileNames = DestPtr;
Scenario->FileNameInfoOffset = (ULONG) (DestPtr - (PCHAR) Scenario);
CurFileInfoOffset = 0;
DestPtr += Scenario->FileNameInfoSize;
//
// Extra space for this alignment was allocated upfront.
//
PFSVC_ASSERT(PF_IS_POWER_OF_TWO(_alignof(PF_METADATA_RECORD)));
MetadataInfoBase = PF_ALIGN_UP(DestPtr, _alignof(PF_METADATA_RECORD));
DestPtr += MetadataInfoSize;
MetadataRecordTable = (PPF_METADATA_RECORD) MetadataInfoBase;
Scenario->MetadataInfoOffset = (ULONG) (MetadataInfoBase - (PCHAR) Scenario);
Scenario->MetadataInfoSize = (ULONG) (DestPtr - MetadataInfoBase);
Scenario->NumMetadataRecords = NumMetadataRecords;
//
// Destination pointer should be at the end of the allocated
// buffer now.
//
if (DestPtr != (PCHAR) Scenario + Scenario->Size) {
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Walk through the sections on the new scenario info and copy
// them.
//
SectHead = &ScenarioInfo->SectionList;
SectNext = SectHead->Flink;
while (SectHead != SectNext) {
SectionNode = CONTAINING_RECORD(SectNext,
PFSVC_SECTION_NODE,
SectionLink);
//
// The target section record.
//
Section = &Sections[CurSectionIdx];
//
// Copy section record info.
//
*Section = SectionNode->SectionRecord;
//
// Copy pages for the section.
//
Section->FirstPageIdx = PF_INVALID_PAGE_IDX;
PreviousPage = NULL;
PageHead = &SectionNode->PageList;
PageNext = PageHead->Flink;
while (PageNext != PageHead) {
PageNode = CONTAINING_RECORD(PageNext,
PFSVC_PAGE_NODE,
PageLink);
Page = &Pages[CurPageIdx];
//
// If this is the first page in the section, update first
// page index on the section record.
//
if (Section->FirstPageIdx == PF_INVALID_PAGE_IDX) {
Section->FirstPageIdx = CurPageIdx;
}
//
// Copy page record.
//
*Page = PageNode->PageRecord;
//
// Update NextPageIdx on the previous page if there is
// one.
//
if (PreviousPage) {
PreviousPage->NextPageIdx = CurPageIdx;
}
//
// Update previous page.
//
PreviousPage = Page;
//
// Set next link to list termination now. If there is a
// next page it is going to update this.
//
Page->NextPageIdx = PF_INVALID_PAGE_IDX;
//
// Update position in the page record table.
//
CurPageIdx++;
PFSVC_ASSERT(CurPageIdx <= Scenario->NumPages);
PageNext = PageNext->Flink;
}
//
// Copy over file name.
//
FileNameSize = (Section->FileNameLength + 1) * sizeof(WCHAR);
RtlCopyMemory(FileNames + CurFileInfoOffset,
SectionNode->FilePath,
FileNameSize);
//
// Update section record's file name offset.
//
Section->FileNameOffset = CurFileInfoOffset;
//
// Update current index into file name info.
//
CurFileInfoOffset += FileNameSize;
PFSVC_ASSERT(CurFileInfoOffset <= Scenario->FileNameInfoSize);
//
// Update our position in the section table.
//
CurSectionIdx++;
PFSVC_ASSERT(CurSectionIdx <= Scenario->NumSections);
SectNext = SectNext->Flink;
}
//
// Make sure we filled up the tables.
//
if (CurSectionIdx != Scenario->NumSections ||
CurPageIdx != Scenario->NumPages ||
CurFileInfoOffset != Scenario->FileNameInfoSize) {
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Build and copy the metadata prefetch information.
//
//
// Set our target to right after the metadata records table.
//
DestPtr = MetadataInfoBase + sizeof(PF_METADATA_RECORD) * NumMetadataRecords;
CurMetadataRecordIdx = 0;
HeadVolume = &ScenarioInfo->VolumeList;
NextVolume = HeadVolume->Flink;
while (NextVolume != HeadVolume) {
VolumeNode = CONTAINING_RECORD(NextVolume,
PFSVC_VOLUME_NODE,
VolumeLink);
NextVolume = NextVolume->Flink;
//
// If there are no sections at all on this volume, skip it.
//
if (VolumeNode->NumAllSections == 0) {
continue;
}
//
// Make sure we are within bounds.
//
if (CurMetadataRecordIdx >= NumMetadataRecords) {
PFSVC_ASSERT(CurMetadataRecordIdx < NumMetadataRecords);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
MetadataRecord = &MetadataRecordTable[CurMetadataRecordIdx];
CurMetadataRecordIdx++;
//
// Copy volume identifiers.
//
MetadataRecord->SerialNumber = VolumeNode->SerialNumber;
MetadataRecord->CreationTime = VolumeNode->CreationTime;
//
// Copy volume name.
//
MetadataRecord->VolumeNameOffset = (ULONG) (DestPtr - MetadataInfoBase);
MetadataRecord->VolumeNameLength = VolumeNode->VolumePathLength;
CopySize = (VolumeNode->VolumePathLength + 1) * sizeof(WCHAR);
if (DestPtr + CopySize > (PCHAR) Scenario + Scenario->Size) {
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
RtlCopyMemory(DestPtr, VolumeNode->VolumePath, CopySize);
DestPtr += CopySize;
//
// Align and update DestPtr for the FILE_PREFETCH structure.
//
PFSVC_ASSERT(PF_IS_POWER_OF_TWO(_alignof(FILE_PREFETCH)));
DestPtr = PF_ALIGN_UP(DestPtr, _alignof(FILE_PREFETCH));
FilePrefetchInfo = (PFILE_PREFETCH) DestPtr;
MetadataRecord->FilePrefetchInfoOffset = (ULONG) (DestPtr - MetadataInfoBase);
//
// Calculate size of the file prefetch information structure.
//
FilePrefetchCount = VolumeNode->NumSections;
if (VolumeNode->MFTSectionNode) {
FilePrefetchCount += VolumeNode->MFTSectionNode->MFTNumPagesToPrefetch;
}
FilePrefetchCount += VolumeNode->DirectoryList.NumPaths;
FilePrefetchInfoSize = sizeof(FILE_PREFETCH);
if (FilePrefetchCount) {
//
// Note that there is room for one entry in the FILE_PREFETCH
// structure.
//
FilePrefetchInfoSize += (FilePrefetchCount - 1) * sizeof(ULONGLONG);
}
MetadataRecord->FilePrefetchInfoSize = FilePrefetchInfoSize;
if (DestPtr + FilePrefetchInfoSize > (PCHAR) Scenario + Scenario->Size) {
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Update destination pointer.
//
DestPtr += FilePrefetchInfoSize;
//
// Initialize file prefetch information structure.
//
FilePrefetchInfo->Type = FILE_PREFETCH_TYPE_FOR_CREATE;
FilePrefetchInfo->Count = FilePrefetchCount;
//
// Build list of file indexes to prefetch:
//
CurFilePrefetchIdx = 0;
//
// Add file system index numbers for sections.
//
SectHead = &VolumeNode->SectionList;
SectNext = SectHead->Flink;
while(SectNext != SectHead) {
SectionNode = CONTAINING_RECORD(SectNext,
PFSVC_SECTION_NODE,
SectionVolumeLink);
SectNext = SectNext->Flink;
if (CurFilePrefetchIdx >= VolumeNode->NumSections) {
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Add the filesystem index number for this section to the list.
//
FilePrefetchInfo->Prefetch[CurFilePrefetchIdx] =
SectionNode->FileIndexNumber.QuadPart;
CurFilePrefetchIdx++;
}
//
// Add file system index numbers for directories.
//
PathEntry = NULL;
while (PathEntry = PfSvGetNextPathSorted(&VolumeNode->DirectoryList,
PathEntry)) {
DirectoryPath = PathEntry->Path;
//
// Get the file index number for this directory and add it
// to the list we'll ask the filesystem to prefetch.
//
ErrorCode = PfSvGetFileIndexNumber(DirectoryPath, &IndexNumber);
if (ErrorCode == ERROR_SUCCESS) {
FilePrefetchInfo->Prefetch[CurFilePrefetchIdx] = IndexNumber.QuadPart;
} else {
FilePrefetchInfo->Prefetch[CurFilePrefetchIdx] = 0;
}
CurFilePrefetchIdx++;
}
//
// Add file system index numbers that we drive from direct MFT access.
//
if (VolumeNode->MFTSectionNode) {
SectionNode = VolumeNode->MFTSectionNode;
for (PageNext = SectionNode->PageList.Flink;
PageNext != &SectionNode->PageList;
PageNext = PageNext->Flink) {
PageNode = CONTAINING_RECORD(PageNext,
PFSVC_PAGE_NODE,
PageLink);
if (!PageNode->PageRecord.IsIgnore) {
//
// We know the file offset in MFT. Every file record is
// 1KB == 2^10 bytes. To convert fileoffset in MFT to a
// file record number we just shift it by 10.
//
FilePrefetchInfo->Prefetch[CurFilePrefetchIdx] =
PageNode->PageRecord.FileOffset >> 10;
CurFilePrefetchIdx++;
}
}
}
//
// We should have specified all the file index numbers.
//
PFSVC_ASSERT(CurFilePrefetchIdx == FilePrefetchInfo->Count);
//
// Add paths for directories accessed on this volume.
//
MetadataRecord->NumDirectories = VolumeNode->DirectoryList.NumPaths;
MetadataRecord->DirectoryPathsOffset = (ULONG)(DestPtr - MetadataInfoBase);
PathEntry = NULL;
while (PathEntry = PfSvGetNextPathSorted(&VolumeNode->DirectoryList,
PathEntry)) {
DirectoryPath = PathEntry->Path;
DirectoryPathLength = PathEntry->Length;
//
// Calculate how big the entry for this path is going to
// be and make sure it will be within bounds.
//
DirectoryPathSize = sizeof(PF_COUNTED_STRING);
DirectoryPathSize += DirectoryPathLength * sizeof(WCHAR);
if (DestPtr + DirectoryPathSize > (PCHAR) Scenario + Scenario->Size) {
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Copy over the directory path.
//
DirectoryPathCS = (PPF_COUNTED_STRING) DestPtr;
DirectoryPathCS->Length = (USHORT) DirectoryPathLength;
RtlCopyMemory(DirectoryPathCS->String,
DirectoryPath,
(DirectoryPathLength + 1) * sizeof(WCHAR));
DestPtr += DirectoryPathSize;
}
}
//
// Make sure we are not past the end of the buffer.
//
if (DestPtr > (PCHAR) Scenario + Scenario->Size) {
PFSVC_ASSERT(FALSE);
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Set up return pointer.
//
*ScenarioPtr = Scenario;
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ErrorCode != ERROR_SUCCESS) {
if (Scenario != NULL) {
PFSVC_FREE(Scenario);
}
}
DBGPR((PFID,PFTRC,"PFSVC: PrepareScenarioDump()=%x\n", ErrorCode));
return ErrorCode;
}
//
// Routines to maintain the optimal disk layout file and update disk
// layout.
//
DWORD
PfSvUpdateOptimalLayout(
PPFSVC_IDLE_TASK Task
)
/*++
Routine Description:
This routine will determine if the optimal disk layout has to be
updated and if so it will write out a new layout file and launch
the defragger.
Arguments:
Task - If specified the function will check Task every once in a
while to see if it should exit with ERROR_RETRY.
Return Value:
Win32 error code.
--*/
{
ULARGE_INTEGER CurrentTimeLI;
ULARGE_INTEGER LastDiskLayoutTimeLI;
ULARGE_INTEGER MinTimeBeforeRelayoutLI;
PFSVC_PATH_LIST OptimalLayout;
PFSVC_PATH_LIST CurrentLayout;
FILETIME LastDiskLayoutTime;
FILETIME LayoutFileTime;
FILETIME CurrentTime;
PPFSVC_PATH_LIST NewLayout;
PWCHAR LayoutFilePath;
ULONG LayoutFilePathBufferSize;
DWORD ErrorCode;
DWORD BootScenarioProcessed;
DWORD BootFilesWereOptimized;
DWORD MinHoursBeforeRelayout;
DWORD Size;
DWORD RegValueType;
BOOLEAN LayoutChanged;
BOOLEAN MissingOriginalLayoutFile;
BOOLEAN CheckForLayoutFrequencyLimit;
//
// Initialize locals.
//
LayoutFilePath = NULL;
LayoutFilePathBufferSize = 0;
PfSvInitializePathList(&OptimalLayout, NULL, FALSE);
PfSvInitializePathList(&CurrentLayout, NULL, FALSE);
DBGPR((PFID,PFTRC,"PFSVC: UpdateOptimalLayout(%p)\n", Task));
//
// Determine when we updated the disk layout from the layout file.
//
ErrorCode = PfSvGetLastDiskLayoutTime(&LastDiskLayoutTime);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Query whether boot files have been optimized.
//
Size = sizeof(BootFilesWereOptimized);
ErrorCode = RegQueryValueEx(PfSvcGlobals.ServiceDataKey,
PFSVC_BOOT_FILES_OPTIMIZED_VALUE_NAME,
NULL,
&RegValueType,
(PVOID) &BootFilesWereOptimized,
&Size);
if (ErrorCode != ERROR_SUCCESS) {
BootFilesWereOptimized = FALSE;
}
//
// Get optimal layout file path.
//
ErrorCode = PfSvGetLayoutFilePath(&LayoutFilePath,
&LayoutFilePathBufferSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Determine when the file was last modified.
//
ErrorCode = PfSvGetLastWriteTime(LayoutFilePath, &LayoutFileTime);
if (ErrorCode == ERROR_SUCCESS) {
MissingOriginalLayoutFile = FALSE;
//
// If the file was modified after we laid out the files on the disk
// its contents are not interesting. Otherwise, if the new optimal
// layout is similar to layout specified in the file, we may not
// have to re-layout the files.
//
if (CompareFileTime(&LayoutFileTime, &LastDiskLayoutTime) <= 0) {
//
// Read the current layout.
//
ErrorCode = PfSvReadLayout(LayoutFilePath,
&CurrentLayout,
&LayoutFileTime);
if (ErrorCode != ERROR_SUCCESS) {
//
// The layout file seems to be bad / inaccesible.
// Cleanup the path list, so a brand new one gets
// built.
//
PfSvCleanupPathList(&CurrentLayout);
PfSvInitializePathList(&CurrentLayout, NULL, FALSE);
}
}
} else {
//
// We could not get the timestamp on the original layout file.
// It might have been deleted.
//
MissingOriginalLayoutFile = TRUE;
}
//
// Determine what the current optimal layout should be from
// scenario files.
//
ErrorCode = PfSvDetermineOptimalLayout(Task,
&OptimalLayout,
&BootScenarioProcessed);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Update current layout based on what optimal layout should be.
// If the two are similar we don't need to launch the defragger.
//
ErrorCode = PfSvUpdateLayout(&CurrentLayout,
&OptimalLayout,
&LayoutChanged);
if (ErrorCode == ERROR_SUCCESS) {
if (!LayoutChanged) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// We'll use the updated layout.
//
NewLayout = &CurrentLayout;
} else {
//
// We'll run with the optimal layout.
//
NewLayout = &OptimalLayout;
}
//
// Optimal way to layout files has changed. Write out the new layout.
//
ErrorCode = PfSvSaveLayout(LayoutFilePath,
NewLayout,
&LayoutFileTime);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// If enough time has not passed since last disk layout don't run the
// defragger again unless...
//
CheckForLayoutFrequencyLimit = TRUE;
//
// - We've been explicitly asked to update the layout (i.e. no idle
// task context.)
//
if (!Task) {
CheckForLayoutFrequencyLimit = FALSE;
}
//
// - Someone seems to have deleted the layout file and we recreated it.
//
if (MissingOriginalLayoutFile) {
CheckForLayoutFrequencyLimit = FALSE;
}
//
// - Boot prefetching is enabled but boot files have not been optimized
// yet and we processed the list of files from the boot this time.
//
if (PfSvcGlobals.Parameters.EnableStatus[PfSystemBootScenarioType] == PfSvEnabled) {
if (!BootFilesWereOptimized && BootScenarioProcessed) {
CheckForLayoutFrequencyLimit = FALSE;
}
}
if (CheckForLayoutFrequencyLimit) {
//
// We will check to see if enough time has passed by getting current
// time and comparing it to last disk layout time.
//
LastDiskLayoutTimeLI.LowPart = LastDiskLayoutTime.dwLowDateTime;
LastDiskLayoutTimeLI.HighPart = LastDiskLayoutTime.dwHighDateTime;
//
// Get current time as file time.
//
GetSystemTimeAsFileTime(&CurrentTime);
CurrentTimeLI.LowPart = CurrentTime.dwLowDateTime;
CurrentTimeLI.HighPart = CurrentTime.dwHighDateTime;
//
// Check to make sure that current time is after last disk layout time
// (in case the user has played with time.)
//
if (CurrentTimeLI.QuadPart > LastDiskLayoutTimeLI.QuadPart) {
//
// Query how long has to pass before we re-layout the files on
// disk.
//
Size = sizeof(MinHoursBeforeRelayout);
ErrorCode = RegQueryValueEx(PfSvcGlobals.ServiceDataKey,
PFSVC_MIN_RELAYOUT_HOURS_VALUE_NAME,
NULL,
&RegValueType,
(PVOID) &MinHoursBeforeRelayout,
&Size);
if (ErrorCode == ERROR_SUCCESS) {
MinTimeBeforeRelayoutLI.QuadPart = PFSVC_NUM_100NS_IN_AN_HOUR * MinHoursBeforeRelayout;
} else {
MinTimeBeforeRelayoutLI.QuadPart = PFSVC_MIN_TIME_BEFORE_DISK_RELAYOUT;
}
if (CurrentTimeLI.QuadPart < LastDiskLayoutTimeLI.QuadPart +
MinTimeBeforeRelayoutLI.QuadPart) {
//
// Not enough time has passed before last disk layout.
//
ErrorCode = ERROR_INVALID_TIME;
goto cleanup;
}
}
}
//
// Launch the defragger for layout optimization.
//
ErrorCode = PfSvLaunchDefragger(Task, TRUE, NULL);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Save whether boot files were optimized.
//
ErrorCode = RegSetValueEx(PfSvcGlobals.ServiceDataKey,
PFSVC_BOOT_FILES_OPTIMIZED_VALUE_NAME,
0,
REG_DWORD,
(PVOID) &BootScenarioProcessed,
sizeof(BootScenarioProcessed));
//
// Save the last time we updated disk layout to the registry.
//
ErrorCode = PfSvSetLastDiskLayoutTime(&LayoutFileTime);
//
// Fall through with error code.
//
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: UpdateOptimalLayout(%p)=%x\n", Task, ErrorCode));
PfSvCleanupPathList(&OptimalLayout);
PfSvCleanupPathList(&CurrentLayout);
if (LayoutFilePath) {
PFSVC_FREE(LayoutFilePath);
}
return ErrorCode;
}
DWORD
PfSvUpdateLayout (
PPFSVC_PATH_LIST CurrentLayout,
PPFSVC_PATH_LIST OptimalLayout,
PBOOLEAN LayoutChanged
)
/*++
Routine Description:
This routine updates the specified layout based on the new optimal
layout. If the two layouts are similar, CurrentLayout is not updated.
An error may be returned while CurrentLayout is being updated. It is the
caller's responsibility to revert CurrentLayout to its original in that case.
Arguments:
CurrentLayout - Current file layout.
OptimalLayout - Newly determined optimal file layout.
LayoutChanged - Whether Layout was changed.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
PPFSVC_PATH PathEntry;
ULONG NumOptimalLayoutFiles;
ULONG NumMissingFiles;
ULONG NumCommonFiles;
ULONG NumCurrentLayoutOnlyFiles;
//
// Initialize locals.
//
NumOptimalLayoutFiles = 0;
NumMissingFiles = 0;
//
// Go through the paths in the new layout counting the differences with
// the current layout.
//
PathEntry = NULL;
while (PathEntry = PfSvGetNextPathInOrder(OptimalLayout, PathEntry)) {
NumOptimalLayoutFiles++;
if (!PfSvIsInPathList(CurrentLayout, PathEntry->Path, PathEntry->Length)) {
NumMissingFiles++;
}
}
//
// Make some sanity checks about the statistics gathered.
//
PFSVC_ASSERT(NumOptimalLayoutFiles == OptimalLayout->NumPaths);
PFSVC_ASSERT(NumOptimalLayoutFiles >= NumMissingFiles);
NumCommonFiles = NumOptimalLayoutFiles - NumMissingFiles;
PFSVC_ASSERT(CurrentLayout->NumPaths >= NumCommonFiles);
NumCurrentLayoutOnlyFiles = CurrentLayout->NumPaths - NumCommonFiles;
//
// If there are not that many new files: no need to update the layout.
//
if (NumMissingFiles <= 20) {
*LayoutChanged = FALSE;
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// We will be updating the current layout.
//
*LayoutChanged = TRUE;
//
// If there are too many files in the current layout that don't need to be
// there anymore, rebuild the list.
//
if (NumCurrentLayoutOnlyFiles >= CurrentLayout->NumPaths / 4) {
PfSvCleanupPathList(CurrentLayout);
PfSvInitializePathList(CurrentLayout, NULL, FALSE);
}
//
// Add files from the optimal layout to the end of current layout.
//
while (PathEntry = PfSvGetNextPathInOrder(OptimalLayout, PathEntry)) {
ErrorCode = PfSvAddToPathList(CurrentLayout, PathEntry->Path, PathEntry->Length);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: UpdateLayout(%p,%p)=%d,%x\n",CurrentLayout,OptimalLayout,*LayoutChanged,ErrorCode));
return ErrorCode;
}
DWORD
PfSvDetermineOptimalLayout (
PPFSVC_IDLE_TASK Task,
PPFSVC_PATH_LIST OptimalLayout,
BOOL *BootScenarioProcessed
)
/*++
Routine Description:
This routine will determine if the optimal disk layout has to be
updated by looking at the existing scenario files.
Arguments:
Task - If specified the function will check Task every once in a
while to see if it should exit with ERROR_RETRY.
OptimalLayout - Initialized empty path list that will be built.
BootScenarioProcessed - Whether we got the list of boot files from
the boot scenario.
Return Value:
Win32 error code.
--*/
{
PFSVC_SCENARIO_FILE_CURSOR FileCursor;
FILETIME LayoutFileTime;
PNTPATH_TRANSLATION_LIST TranslationList;
PWCHAR DosPathBuffer;
ULONG DosPathBufferSize;
DWORD ErrorCode;
BOOLEAN AcquiredLock;
WCHAR BootScenarioFileName[PF_MAX_SCENARIO_FILE_NAME];
WCHAR BootScenarioFilePath[MAX_PATH + 1];
//
// Initialize locals.
//
PfSvInitializeScenarioFileCursor(&FileCursor);
TranslationList = NULL;
AcquiredLock = FALSE;
DosPathBuffer = NULL;
DosPathBufferSize = 0;
DBGPR((PFID,PFTRC,"PFSVC: DetermineOptimalLayout(%p,%p)\n",Task,OptimalLayout));
//
// Initialize output variables.
//
*BootScenarioProcessed = FALSE;
//
// Acquire the prefetch root directory lock and initialize some locals.
//
PFSVC_ACQUIRE_LOCK(PfSvcGlobals.PrefetchRootLock);
AcquiredLock = TRUE;
//
// Start the file cursor.
//
ErrorCode = PfSvStartScenarioFileCursor(&FileCursor, PfSvcGlobals.PrefetchRoot);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Build the boot scenario file path.
//
swprintf(BootScenarioFileName,
PF_SCEN_FILE_NAME_FORMAT,
PF_BOOT_SCENARIO_NAME,
PF_BOOT_SCENARIO_HASHID,
PF_PREFETCH_FILE_EXTENSION);
_snwprintf(BootScenarioFilePath,
MAX_PATH,
L"%ws\\%ws",
PfSvcGlobals.PrefetchRoot,
BootScenarioFileName);
BootScenarioFilePath[MAX_PATH - 1] = 0;
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
AcquiredLock = FALSE;
//
// Get translation list so we can convert NT paths in the trace to
// Dos paths that the defragger understands.
//
ErrorCode = PfSvBuildNtPathTranslationList(&TranslationList);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Should we continue to run?
//
ErrorCode = PfSvContinueRunningTask(Task);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Add boot loader files to optimal layout.
//
ErrorCode = PfSvBuildBootLoaderFilesList(OptimalLayout);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Add files from the boot scenario.
//
ErrorCode = PfSvUpdateLayoutForScenario(OptimalLayout,
BootScenarioFilePath,
TranslationList,
&DosPathBuffer,
&DosPathBufferSize);
if (ErrorCode == ERROR_SUCCESS) {
*BootScenarioProcessed = TRUE;
}
//
// Go through all the other scenario files.
//
while (TRUE) {
//
// Should we continue to run?
//
ErrorCode = PfSvContinueRunningTask(Task);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Get file info for the next scenario file.
//
ErrorCode = PfSvGetNextScenarioFileInfo(&FileCursor);
if (ErrorCode == ERROR_NO_MORE_FILES) {
break;
}
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
PfSvUpdateLayoutForScenario(OptimalLayout,
FileCursor.FilePath,
TranslationList,
&DosPathBuffer,
&DosPathBufferSize);
}
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: DetermineOptimalLayout(%p,%p)=%x\n",Task,OptimalLayout,ErrorCode));
if (AcquiredLock) {
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
}
PfSvCleanupScenarioFileCursor(&FileCursor);
if (TranslationList) {
PfSvFreeNtPathTranslationList(TranslationList);
}
if (DosPathBuffer) {
PFSVC_ASSERT(DosPathBufferSize);
PFSVC_FREE(DosPathBuffer);
}
return ErrorCode;
}
DWORD
PfSvUpdateLayoutForScenario (
PPFSVC_PATH_LIST OptimalLayout,
WCHAR *ScenarioFilePath,
PNTPATH_TRANSLATION_LIST TranslationList,
PWCHAR *DosPathBuffer,
PULONG DosPathBufferSize
)
/*++
Routine Description:
This routine will add the directories and files referenced in a
scenario in the order they appear to the specified optimal layout
path list .
Arguments:
OptimalLayout - Pointer to path list.
ScenarioFilePath - Scenario file.
TranslationList, DosPathBuffer, DosPathBufferSize - These are used
to translate NT path names in the scenario file to Dos path names
that should be in the layout file.
Return Value:
Win32 error code.
--*/
{
PPF_SCENARIO_HEADER Scenario;
PCHAR MetadataInfoBase;
PPF_METADATA_RECORD MetadataRecordTable;
PPF_METADATA_RECORD MetadataRecord;
PPF_COUNTED_STRING DirectoryPath;
PPF_SECTION_RECORD Sections;
PCHAR FilePathInfo;
PWCHAR FilePath;
ULONG FilePathLength;
ULONG SectionIdx;
ULONG MetadataRecordIdx;
ULONG DirectoryIdx;
DWORD ErrorCode;
DWORD FileSize;
DWORD FailedCheck;
//
// Initialize locals.
//
Scenario = NULL;
//
// Map the scenario file.
//
ErrorCode = PfSvGetViewOfFile(ScenarioFilePath,
&Scenario,
&FileSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Verify scenario file.
//
if (!PfSvVerifyScenarioBuffer(Scenario, FileSize, &FailedCheck) ||
Scenario->ServiceVersion != PFSVC_SERVICE_VERSION) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// First add the directories that need to be accessed.
//
MetadataInfoBase = (PCHAR)Scenario + Scenario->MetadataInfoOffset;
MetadataRecordTable = (PPF_METADATA_RECORD) MetadataInfoBase;
for (MetadataRecordIdx = 0;
MetadataRecordIdx < Scenario->NumMetadataRecords;
MetadataRecordIdx++) {
MetadataRecord = &MetadataRecordTable[MetadataRecordIdx];
DirectoryPath = (PPF_COUNTED_STRING)
(MetadataInfoBase + MetadataRecord->DirectoryPathsOffset);
for (DirectoryIdx = 0;
DirectoryIdx < MetadataRecord->NumDirectories;
DirectoryIdx++,
DirectoryPath = (PPF_COUNTED_STRING) (&DirectoryPath->String[DirectoryPath->Length + 1])) {
ErrorCode = PfSvTranslateNtPath(TranslationList,
DirectoryPath->String,
DirectoryPath->Length,
DosPathBuffer,
DosPathBufferSize);
//
// We may not be able to translate all NT paths to Dos paths.
//
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvAddToPathList(OptimalLayout,
*DosPathBuffer,
wcslen(*DosPathBuffer));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
}
}
//
// Now add the file paths.
//
Sections = (PPF_SECTION_RECORD) ((PCHAR)Scenario + Scenario->SectionInfoOffset);
FilePathInfo = (PCHAR)Scenario + Scenario->FileNameInfoOffset;
for (SectionIdx = 0; SectionIdx < Scenario->NumSections; SectionIdx++) {
FilePath = (PWSTR) (FilePathInfo + Sections[SectionIdx].FileNameOffset);
FilePathLength = Sections[SectionIdx].FileNameLength;
ErrorCode = PfSvTranslateNtPath(TranslationList,
FilePath,
FilePathLength,
DosPathBuffer,
DosPathBufferSize);
//
// We may not be able to translate all NT paths to Dos paths.
//
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvAddToPathList(OptimalLayout,
*DosPathBuffer,
wcslen(*DosPathBuffer));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
}
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
if (Scenario) {
UnmapViewOfFile(Scenario);
}
DBGPR((PFID,PFTRC,"PFSVC: UpdateLayoutForScenario(%p,%ws)=%x\n",OptimalLayout,ScenarioFilePath,ErrorCode));
return ErrorCode;
}
DWORD
PfSvReadLayout(
IN WCHAR *FilePath,
OUT PPFSVC_PATH_LIST Layout,
OUT FILETIME *LastWriteTime
)
/*++
Routine Description:
This function adds contents of the optimal layout file to the
specified path list. Note that failure may be returned after
adding several files to the list.
Arguments:
FilePath - NUL terminated path to optimal layout file.
Layout - Pointer to initialized path list.
LastWriteTime - Last write time of the read file.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
FILE *LayoutFile;
WCHAR *LineBuffer;
ULONG LineBufferMaxChars;
ULONG LineLength;
//
// Initialize locals.
//
LayoutFile = NULL;
LineBuffer = NULL;
LineBufferMaxChars = 0;
//
// Open the layout file.
//
LayoutFile = _wfopen(FilePath, L"rb");
if (!LayoutFile) {
ErrorCode = ERROR_FILE_NOT_FOUND;
goto cleanup;
}
//
// Read and verify header.
//
ErrorCode = PfSvReadLine(LayoutFile,
&LineBuffer,
&LineBufferMaxChars,
&LineLength);
if (ErrorCode != ERROR_SUCCESS || !LineLength) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
PfSvRemoveEndOfLineChars(LineBuffer, &LineLength);
if (wcscmp(LineBuffer, L"[OptimalLayoutFile]")) {
//
// Notepad puts a weird first character in the UNICODE text files.
// Skip the first character and compare again.
//
// FUTURE-2002/03/29-ScottMa -- Strictly speaking, that first character
// is the UNICODE marker character, which must be 0xFFFE if the file
// was written on the same -endian machine as it is being read, while
// 0xFEFF would indicate that the file needs to be converted.
if ((LineLength < 1) ||
wcscmp(&LineBuffer[1], L"[OptimalLayoutFile]")) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
}
//
// Read and verify version.
//
ErrorCode = PfSvReadLine(LayoutFile,
&LineBuffer,
&LineBufferMaxChars,
&LineLength);
if (ErrorCode != ERROR_SUCCESS || !LineLength) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
PfSvRemoveEndOfLineChars(LineBuffer, &LineLength);
if (wcscmp(LineBuffer, L"Version=1")) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Read in file names.
//
do {
ErrorCode = PfSvReadLine(LayoutFile,
&LineBuffer,
&LineBufferMaxChars,
&LineLength);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
if (!LineLength) {
//
// We hit end of file.
//
break;
}
PfSvRemoveEndOfLineChars(LineBuffer, &LineLength);
//
// Add it to the list.
//
ErrorCode = PfSvAddToPathList(Layout,
LineBuffer,
LineLength);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
} while (TRUE);
//
// Get the last write time on the file.
//
ErrorCode = PfSvGetLastWriteTime(FilePath, LastWriteTime);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
if (LayoutFile) {
fclose(LayoutFile);
}
if (LineBuffer) {
PFSVC_ASSERT(LineBufferMaxChars);
PFSVC_FREE(LineBuffer);
}
return ErrorCode;
}
DWORD
PfSvSaveLayout(
IN WCHAR *FilePath,
IN PPFSVC_PATH_LIST Layout,
OUT FILETIME *LastWriteTime
)
/*++
Routine Description:
This routine saves the specified file layout list in order to the
specified file in the right format.
Arguments:
FilePath - Path to output layout file.
Layout - Pointer to layout.
LastWriteTime - Last write time on the file after we are done
saving the layout.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
HANDLE LayoutFile;
WCHAR *FileHeader;
ULONG BufferSize;
ULONG NumBytesWritten;
PPFSVC_PATH PathEntry;
WCHAR *NewLine;
ULONG SizeOfNewLine;
//
// Initialize locals.
//
LayoutFile = INVALID_HANDLE_VALUE;
NewLine = L"\r\n";
SizeOfNewLine = wcslen(NewLine) * sizeof(WCHAR);
//
// Open & truncate the layout file. We are also opening with read
// permissions so we can query the last write time when we are
// done.
//
LayoutFile = CreateFile(FilePath,
GENERIC_READ | GENERIC_WRITE,
0,
0,
CREATE_ALWAYS,
0,
NULL);
if (LayoutFile == INVALID_HANDLE_VALUE) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Write out the header.
//
FileHeader = L"[OptimalLayoutFile]\r\nVersion=1\r\n";
BufferSize = wcslen(FileHeader) * sizeof(WCHAR);
if (!WriteFile(LayoutFile,
FileHeader,
BufferSize,
&NumBytesWritten,
NULL)) {
ErrorCode = GetLastError();
goto cleanup;
}
PathEntry = NULL;
while (PathEntry = PfSvGetNextPathInOrder(Layout, PathEntry)) {
//
// Write the path.
//
BufferSize = PathEntry->Length * sizeof(WCHAR);
if (!WriteFile(LayoutFile,
PathEntry->Path,
BufferSize,
&NumBytesWritten,
NULL)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Write the newline.
//
if (!WriteFile(LayoutFile,
NewLine,
SizeOfNewLine,
&NumBytesWritten,
NULL)) {
ErrorCode = GetLastError();
goto cleanup;
}
}
//
// Make sure everything is written to the file so our
// LastWriteTime will be accurate.
//
if (!FlushFileBuffers(LayoutFile)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Get the last write time.
//
if (!GetFileTime(LayoutFile, NULL, NULL, LastWriteTime)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
if (LayoutFile != INVALID_HANDLE_VALUE) {
CloseHandle(LayoutFile);
}
return ErrorCode;
}
DWORD
PfSvGetLayoutFilePath(
PWCHAR *FilePathBuffer,
PULONG FilePathBufferSize
)
/*++
Routine Description:
This function tries to query the layout file path into the
specified buffer. If the buffer is too small, or NULL, it is
reallocated. If not NULL, the buffer should have been allocated by
PFSVC_ALLOC. It is the callers responsibility to free the returned
buffer using PFSVC_FREE.
In order to avoid having somebody cause us to overwrite any file
in the system always the default layout file path is saved in the
registry and returned.
Arguments:
FilePathBuffer - Layout file path will be put into this buffer
after it is reallocated if it is NULL or not big enough.
FilePathBufferSize - Maximum size of *FilePathBuffer in bytes.
Return Value:
Win32 error code.
--*/
{
ULONG DefaultPathSize;
ULONG DefaultPathLength;
HKEY DefragParametersKey;
DWORD ErrorCode;
BOOLEAN AcquiredPrefetchRootLock;
BOOLEAN OpenedParametersKey;
//
// Initialize locals.
//
AcquiredPrefetchRootLock = FALSE;
OpenedParametersKey = FALSE;
//
// Verify parameters.
//
if (*FilePathBufferSize) {
PFSVC_ASSERT(*FilePathBuffer);
}
PFSVC_ACQUIRE_LOCK(PfSvcGlobals.PrefetchRootLock);
AcquiredPrefetchRootLock = TRUE;
DefaultPathLength = wcslen(PfSvcGlobals.PrefetchRoot);
DefaultPathLength += 1; // for '\\'
DefaultPathLength += wcslen(PFSVC_OPTIMAL_LAYOUT_FILE_DEFAULT_NAME);
DefaultPathSize = (DefaultPathLength + 1) * sizeof(WCHAR);
//
// Check if we have to allocate/reallocate the buffer.
//
if ((*FilePathBufferSize) < DefaultPathSize) {
if (*FilePathBuffer) {
PFSVC_ASSERT(*FilePathBufferSize);
PFSVC_FREE(*FilePathBuffer);
}
(*FilePathBufferSize) = 0;
(*FilePathBuffer) = PFSVC_ALLOC(DefaultPathSize);
if (!(*FilePathBuffer)) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
(*FilePathBufferSize) = DefaultPathSize;
}
//
// Build the path in the FilePathBuffer
//
wcscpy((*FilePathBuffer), PfSvcGlobals.PrefetchRoot);
wcscat((*FilePathBuffer), L"\\");
wcscat((*FilePathBuffer), PFSVC_OPTIMAL_LAYOUT_FILE_DEFAULT_NAME);
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
AcquiredPrefetchRootLock = FALSE;
//
// Save the default path in the registry so it is used by the
// defragger:
//
//
// Open the parameters key, creating it if necessary.
//
ErrorCode = RegCreateKey(HKEY_LOCAL_MACHINE,
PFSVC_OPTIMAL_LAYOUT_REG_KEY_PATH,
&DefragParametersKey);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
OpenedParametersKey = TRUE;
ErrorCode = RegSetValueEx(DefragParametersKey,
PFSVC_OPTIMAL_LAYOUT_REG_VALUE_NAME,
0,
REG_SZ,
(PVOID) (*FilePathBuffer),
(*FilePathBufferSize));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
cleanup:
if (AcquiredPrefetchRootLock) {
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
}
if (OpenedParametersKey) {
CloseHandle(DefragParametersKey);
}
return ErrorCode;
}
//
// Routines to defrag the disks once after setup when the system is idle.
//
DWORD
PfSvLaunchDefragger(
PPFSVC_IDLE_TASK Task,
BOOLEAN ForLayoutOptimization,
PWCHAR TargetDrive
)
/*++
Routine Description:
This routine will launch the defragger. It will create an event that
will be passed to the defragger so the defragger can be stopped if
the service is stopping or the Task (if one is specified) is being
unregistered, etc.
Arguments:
Task - If specified the function will check Task every once in a
while to see if it should exit with ERROR_RETRY.
ForLayoutOptimization - Whether we are launching the defragger
only for layout optimization.
TargetDrive - If we are not launching for layout optimization, the
drive that we want to defrag.
Return Value:
Win32 error code.
--*/
{
PROCESS_INFORMATION ProcessInfo;
STARTUPINFO StartupInfo;
WCHAR *CommandLine;
WCHAR *DefragCommand;
WCHAR *DoLayoutParameter;
WCHAR *DriveToDefrag;
HANDLE StopDefraggerEvent;
HANDLE ProcessHandle;
HANDLE Events[4];
ULONG NumEvents;
ULONG MaxEvents;
DWORD ErrorCode;
DWORD ExitCode;
DWORD WaitResult;
DWORD ProcessId;
ULONG SystemDirLength;
ULONG CommandLineLength;
ULONG RetryCount;
BOOL DefraggerExitOnItsOwn;
WCHAR SystemDrive[3];
WCHAR ProcessIdString[35];
WCHAR StopEventString[35];
WCHAR SystemDir[MAX_PATH + 1];
//
// Initialize locals.
//
StopDefraggerEvent = NULL;
DefragCommand = L"\\defrag.exe\" ";
DoLayoutParameter = L"-b ";
CommandLine = NULL;
RtlZeroMemory(&ProcessInfo, sizeof(PROCESS_INFORMATION));
RtlZeroMemory(&StartupInfo, sizeof(STARTUPINFO));
StartupInfo.cb = sizeof(STARTUPINFO);
ProcessHandle = NULL;
MaxEvents = sizeof(Events) / sizeof(HANDLE);
DBGPR((PFID,PFTRC,"PFSVC: LaunchDefragger(%p,%d,%ws)\n",Task,(DWORD)ForLayoutOptimization,TargetDrive));
//
// If we are not allowed to run the defragger, don't.
//
if (!PfSvAllowedToRunDefragger(TRUE)) {
ErrorCode = ERROR_ACCESS_DENIED;
goto cleanup;
}
//
// Get current process ID as string.
//
ProcessId = GetCurrentProcessId();
swprintf(ProcessIdString, L"-p %x ", ProcessId);
//
// Create a stop event and convert handle value to string.
//
StopDefraggerEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
if (!StopDefraggerEvent) {
ErrorCode = GetLastError();
goto cleanup;
}
swprintf(StopEventString, L"-s %p ", StopDefraggerEvent);
//
// Get path to system32 directory.
//
SystemDirLength = GetSystemDirectory(SystemDir, MAX_PATH);
if (!SystemDirLength) {
ErrorCode = GetLastError();
goto cleanup;
}
if (SystemDirLength >= MAX_PATH) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
SystemDir[MAX_PATH - 1] = 0;
//
// Determine which drive we will be defragmenting.
//
if (ForLayoutOptimization) {
//
// Get system drive from system directory path.
//
SystemDrive[0] = SystemDir[0];
SystemDrive[1] = SystemDir[1];
SystemDrive[2] = 0;
DriveToDefrag = SystemDrive;
} else {
DriveToDefrag = TargetDrive;
}
//
// Build the command line to launch the process. All strings we put
// together include a trailing space.
//
CommandLineLength = 0;
CommandLineLength += wcslen(L"\""); // protect against spaces in SystemDir.
CommandLineLength += wcslen(SystemDir);
CommandLineLength += wcslen(DefragCommand);
CommandLineLength += wcslen(ProcessIdString);
CommandLineLength += wcslen(StopEventString);
if (ForLayoutOptimization) {
CommandLineLength += wcslen(DoLayoutParameter);
}
CommandLineLength += wcslen(DriveToDefrag);
CommandLine = PFSVC_ALLOC((CommandLineLength + 1) * sizeof(WCHAR));
if (!CommandLine) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
wcscpy(CommandLine, L"\"");
wcscat(CommandLine, SystemDir);
wcscat(CommandLine, DefragCommand);
wcscat(CommandLine, ProcessIdString);
wcscat(CommandLine, StopEventString);
if (ForLayoutOptimization) {
wcscat(CommandLine, DoLayoutParameter);
}
wcscat(CommandLine, DriveToDefrag);
//
// We may have to launch the defragger multiple times for it to make
// or determine the space in which to layout files etc.
//
for (RetryCount = 0; RetryCount < 20; RetryCount++) {
PFSVC_ASSERT(!ProcessHandle);
//
// Create the process.
//
// FUTURE-2002/03/29-ScottMa -- CreateProcess is safer if you supply
// the first parameter. Since the full command-line was built up
// by this function, the first parameter is readily available.
if (!CreateProcess (NULL,
CommandLine,
NULL,
NULL,
FALSE,
CREATE_NO_WINDOW,
NULL,
NULL,
&StartupInfo,
&ProcessInfo)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Close handle to the thread, save the process handle.
//
CloseHandle(ProcessInfo.hThread);
ProcessHandle = ProcessInfo.hProcess;
//
// Setup the events we will wait on.
//
NumEvents = 0;
Events[NumEvents] = ProcessHandle;
NumEvents++;
Events[NumEvents] = PfSvcGlobals.TerminateServiceEvent;
NumEvents ++;
if (Task) {
Events[NumEvents] = Task->StartedUnregisteringEvent;
NumEvents++;
Events[NumEvents] = Task->StopEvent;
NumEvents++;
}
PFSVC_ASSERT(NumEvents <= MaxEvents);
DefraggerExitOnItsOwn = FALSE;
WaitResult = WaitForMultipleObjects(NumEvents,
Events,
FALSE,
INFINITE);
switch(WaitResult) {
case WAIT_OBJECT_0:
//
// The defragger process exit.
//
DefraggerExitOnItsOwn = TRUE;
break;
case WAIT_OBJECT_0 + 1:
//
// The service is exiting, Signal the defragger to exit, but don't
// wait for it.
//
SetEvent(StopDefraggerEvent);
ErrorCode = ERROR_SHUTDOWN_IN_PROGRESS;
goto cleanup;
break;
case WAIT_OBJECT_0 + 2:
case WAIT_OBJECT_0 + 3:
//
// We would have specified these wait events only if a Task was
// specified.
//
PFSVC_ASSERT(Task);
//
// Signal the defragger process to exit and wait for it to exit.
//
SetEvent(StopDefraggerEvent);
NumEvents = 0;
Events[NumEvents] = ProcessHandle;
NumEvents++;
Events[NumEvents] = PfSvcGlobals.TerminateServiceEvent;
NumEvents++;
WaitResult = WaitForMultipleObjects(NumEvents,
Events,
FALSE,
INFINITE);
if (WaitResult == WAIT_OBJECT_0) {
//
// Defragger exit,
//
break;
} else if (WaitResult == WAIT_OBJECT_0 + 1) {
//
// Service exiting, cannot wait for the defragger anymore.
//
ErrorCode = ERROR_SHUTDOWN_IN_PROGRESS;
goto cleanup;
} else {
ErrorCode = GetLastError();
goto cleanup;
}
break;
default:
ErrorCode = GetLastError();
goto cleanup;
}
//
// If we came here, the defragger exit. Determine its exit code and
// propagate it. If the defragger exit because we told it to, this should
// be ENGERR_RETRY.
//
if (!GetExitCodeProcess(ProcessHandle, &ExitCode)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// If the defragger needs us to launch it again do so.
//
if (DefraggerExitOnItsOwn && (ExitCode == 9)) { // ENGERR_RETRY
//
// Reset the event that tells the defragger to stop.
//
ResetEvent(StopDefraggerEvent);
//
// Close to handle to the old defragger process.
//
CloseHandle(ProcessHandle);
ProcessHandle = NULL;
//
// Setup the error code to return. If we've already retried
// too many times, this is the error that we'll return when
// we end the retry loop.
//
ErrorCode = ERROR_REQUEST_ABORTED;
continue;
}
//
// If the defragger is crashing, note it so we don't attempt to run
// it again. When the defragger crashes, its exit code is an NT status
// code that will be error, e.g. 0xC0000005 for AV etc.
//
if (NT_ERROR(ExitCode)) {
PfSvcGlobals.DefraggerErrorCode = ExitCode;
}
//
// Translate the return value of the defragger to a Win32 error code.
// These codes are defined in base\fs\utils\dfrg\inc\dfrgcmn.h.
// I wish they were in a file I could include.
//
switch(ExitCode) {
case 0: ErrorCode = ERROR_SUCCESS; break; // ENG_NOERR
case 1: ErrorCode = ERROR_RETRY; break; // ENG_USER_CANCELLED
case 2: ErrorCode = ERROR_INVALID_PARAMETER; break; // ENGERR_BAD_PARAM
//
// If the defragger's children processes AV / die, it will return
// ENGERR_UNKNOWN == 3.
//
case 3:
ErrorCode = ERROR_INVALID_FUNCTION;
PfSvcGlobals.DefraggerErrorCode = STATUS_UNSUCCESSFUL;
break;
case 4: ErrorCode = ERROR_NOT_ENOUGH_MEMORY; break; // ENGERR_NOMEM
case 7: ErrorCode = ERROR_DISK_FULL; break; // ENGERR_LOW_FREESPACE
//
// There is no good translation for the other exit codes or we just
// don't understand them.
//
default: ErrorCode = ERROR_INVALID_FUNCTION;
}
//
// The defragger returned success or an error other than retry.
//
break;
}
//
// Fall through with error code.
//
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: LaunchDefragger(%p)=%x\n",Task,ErrorCode));
if (CommandLine) {
PFSVC_FREE(CommandLine);
}
if (StopDefraggerEvent) {
CloseHandle(StopDefraggerEvent);
}
if (ProcessHandle) {
CloseHandle(ProcessHandle);
}
return ErrorCode;
}
DWORD
PfSvGetBuildDefragStatusValueName (
OSVERSIONINFOEXW *OsVersion,
PWCHAR *ValueName
)
/*++
Routine Description:
This routine translates OsVersion to a string allocated with
PFSVC_ALLOC. The returned string should be freed by caller.
Arguments:
OsVersion - Version info to translate to string.
ValueName - Pointer to output string is returned here.
Return Value:
Win32 error code.
--*/
{
PWCHAR BuildName;
PWCHAR BuildNameFormat;
ULONG BuildNameMaxLength;
DWORD ErrorCode;
//
// Initialize locals.
//
BuildName = NULL;
BuildNameFormat = L"%x.%x.%x.%hx.%hx.%hx.%hx_DefragStatus";
BuildNameMaxLength = 80;
//
// Allocate the string.
//
BuildName = PFSVC_ALLOC((BuildNameMaxLength + 1) * sizeof(WCHAR));
if (!BuildName) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
_snwprintf(BuildName,
BuildNameMaxLength,
BuildNameFormat,
OsVersion->dwBuildNumber,
OsVersion->dwMajorVersion,
OsVersion->dwMinorVersion,
(WORD) OsVersion->wSuiteMask,
(WORD) OsVersion->wProductType,
(WORD) OsVersion->wServicePackMajor,
(WORD) OsVersion->wServicePackMinor);
//
// Make sure the string is terminated.
//
BuildName[BuildNameMaxLength] = 0;
*ValueName = BuildName;
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ErrorCode != ERROR_SUCCESS) {
if (BuildName) {
PFSVC_FREE(BuildName);
}
}
DBGPR((PFID,PFTRC,"PFSVC: GetBuildName(%.80ws)=%x\n",BuildName,ErrorCode));
return ErrorCode;
}
DWORD
PfSvSetBuildDefragStatus(
OSVERSIONINFOEXW *OsVersion,
PWCHAR BuildDefragStatus,
ULONG Size
)
/*++
Routine Description:
This routine will set the information on which drives have been
defragmented and such for the specified build (OsVersion).
Defrag status is in REG_MULTI_SZ format. Each element is a drive
path that has been defragged for this build. If all drives were
defragged, than the first element is PFSVC_DEFRAG_DRIVES_DONE.
Arguments:
OsVersion - Build & SP we are setting defrag status for.
BuildDefragStatus - A string that describes the status in
REG_MULTI_SZ format.
Size - Size in bytes of the data that has to be saved to the registry.
Return Value:
Win32 error code.
--*/
{
PWCHAR ValueName;
DWORD ErrorCode;
//
// Initialize locals.
//
ValueName = NULL;
//
// Build the value name from OS version info.
//
ErrorCode = PfSvGetBuildDefragStatusValueName(OsVersion, &ValueName);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
ErrorCode = RegSetValueEx(PfSvcGlobals.ServiceDataKey,
ValueName,
0,
REG_MULTI_SZ,
(PVOID) BuildDefragStatus,
Size);
//
// Fall through with error code.
//
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: SetBuildDefragStatus(%ws)=%x\n",BuildDefragStatus,ErrorCode));
if (ValueName) {
PFSVC_FREE(ValueName);
}
return ErrorCode;
}
DWORD
PfSvGetBuildDefragStatus(
OSVERSIONINFOEXW *OsVersion,
PWCHAR *BuildDefragStatus,
PULONG ReturnSize
)
/*++
Routine Description:
This routine will get the information on which drives have been
defragmented and such for the specified build (OsVersion).
Defrag status is in REG_MULTI_SZ format. Each element is a drive
path that has been defragged for this build. If all drives were
defragged, than the first element is PFSVC_DEFRAG_DRIVES_DONE.
Arguments:
OsVersion - Build & SP we are querying defrag status for.
BuildDefragStatus - Output for defrag status. If the function returns
success this should be freed with a call to PFSVC_FREE().
ReturnSize - Size of the returned value in bytes.
Return Value:
Win32 error code.
--*/
{
PWCHAR ValueBuffer;
PWCHAR ValueName;
DWORD ErrorCode;
DWORD RegValueType;
ULONG ValueBufferSize;
ULONG Size;
ULONG NumTries;
ULONG DefaultValueSize;
BOOLEAN InvalidValue;
//
// Initialize locals.
//
ValueName = NULL;
ValueBuffer = NULL;
ValueBufferSize = 0;
InvalidValue = FALSE;
//
// Build the value name from OS version info.
//
ErrorCode = PfSvGetBuildDefragStatusValueName(OsVersion, &ValueName);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Try to allocate a right size buffer to read this value into.
//
NumTries = 0;
do {
Size = ValueBufferSize;
ErrorCode = RegQueryValueEx(PfSvcGlobals.ServiceDataKey,
ValueName,
NULL,
&RegValueType,
(PVOID) ValueBuffer,
&Size);
//
// API returns SUCCESS with required size in Size if ValueBuffer
// is NULL. We have to special case that out.
//
if (ValueBuffer && ErrorCode == ERROR_SUCCESS) {
//
// We got it. Check the type.
//
if (RegValueType != REG_MULTI_SZ) {
//
// Return default value.
//
InvalidValue = TRUE;
} else {
InvalidValue = FALSE;
*ReturnSize = Size;
}
break;
}
if (ErrorCode == ERROR_FILE_NOT_FOUND) {
//
// The value does not exist. Return default value.
//
InvalidValue = TRUE;
break;
}
if (ErrorCode != ERROR_MORE_DATA &&
!(ErrorCode == ERROR_SUCCESS && !ValueBuffer)) {
//
// This is a real error.
//
goto cleanup;
}
//
// Allocate a bigger buffer and try again.
//
PFSVC_ASSERT(ValueBufferSize < Size);
if (ValueBuffer) {
PFSVC_ASSERT(ValueBufferSize);
PFSVC_FREE(ValueBuffer);
ValueBufferSize = 0;
}
ValueBuffer = PFSVC_ALLOC(Size);
if (!ValueBuffer) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
ValueBufferSize = Size;
NumTries++;
} while (NumTries < 10);
//
// If we did not get a valid value from the registry, make up a default
// one: empty string.
//
if (InvalidValue) {
DefaultValueSize = sizeof(WCHAR);
if (ValueBufferSize < DefaultValueSize) {
if (ValueBuffer) {
PFSVC_ASSERT(ValueBufferSize);
PFSVC_FREE(ValueBuffer);
ValueBufferSize = 0;
}
ValueBuffer = PFSVC_ALLOC(DefaultValueSize);
if (!ValueBuffer) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
ValueBufferSize = DefaultValueSize;
}
ValueBuffer[0] = 0;
*ReturnSize = DefaultValueSize;
}
//
// We should get here only if we got a value in value buffer.
//
PFSVC_ASSERT(ValueBuffer && ValueBufferSize);
*BuildDefragStatus = ValueBuffer;
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: GetBuildDefragStatus(%.80ws)=%x\n",*BuildDefragStatus,ErrorCode));
if (ErrorCode != ERROR_SUCCESS) {
if (ValueBuffer) {
PFSVC_ASSERT(ValueBufferSize);
PFSVC_FREE(ValueBuffer);
}
}
if (ValueName) {
PFSVC_FREE(ValueName);
}
return ErrorCode;
}
DWORD
PfSvDefragDisks(
PPFSVC_IDLE_TASK Task
)
/*++
Routine Description:
If we have not defragged all disks after a setup/upgrade, do so.
Arguments:
Task - If specified the function will check Task every once in a
while to see if it should exit with ERROR_RETRY.
Return Value:
Win32 error code.
--*/
{
PNTPATH_TRANSLATION_LIST VolumeList;
PNTPATH_TRANSLATION_ENTRY VolumeEntry;
PWCHAR DefraggedVolumeName;
PWCHAR BuildDefragStatus;
PWCHAR NewBuildDefragStatus;
PLIST_ENTRY NextEntry;
ULONG NewBuildDefragStatusLength;
ULONG BuildDefragStatusSize;
ULONG NewBuildDefragStatusSize;
DWORD ErrorCode;
BOOLEAN AlreadyDefragged;
//
// Initialize locals.
//
NewBuildDefragStatus = NULL;
NewBuildDefragStatusSize = 0;
BuildDefragStatus = NULL;
BuildDefragStatusSize = 0;
VolumeList = NULL;
DBGPR((PFID,PFTRC,"PFSVC: DefragDisks(%p)\n",Task));
//
// Determine defrag status for the current build.
//
ErrorCode = PfSvGetBuildDefragStatus(&PfSvcGlobals.OsVersion,
&BuildDefragStatus,
&BuildDefragStatusSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Check if we are already done for this build.
//
if (!_wcsicmp(BuildDefragStatus, PFSVC_DEFRAG_DRIVES_DONE)) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// Build a list of volumes that are mounted.
//
ErrorCode = PfSvBuildNtPathTranslationList(&VolumeList);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Walk through the volumes defragging the ones we have not yet
// defragged after setup.
//
for (NextEntry = VolumeList->Flink;
NextEntry != VolumeList;
NextEntry = NextEntry->Flink) {
VolumeEntry = CONTAINING_RECORD(NextEntry,
NTPATH_TRANSLATION_ENTRY,
Link);
//
// Skip volumes that are not fixed disks.
//
if (DRIVE_FIXED != GetDriveType(VolumeEntry->VolumeName)) {
continue;
}
//
// Have we already defragged this volume?
//
AlreadyDefragged = FALSE;
for (DefraggedVolumeName = BuildDefragStatus;
DefraggedVolumeName[0] != 0;
DefraggedVolumeName += wcslen(DefraggedVolumeName) + 1) {
PFSVC_ASSERT((PCHAR) DefraggedVolumeName < (PCHAR) BuildDefragStatus + BuildDefragStatusSize);
if (!_wcsicmp(DefraggedVolumeName, VolumeEntry->DosPrefix)) {
AlreadyDefragged = TRUE;
break;
}
}
if (AlreadyDefragged) {
continue;
}
//
// Launch the defragger to defrag this volume.
//
ErrorCode = PfSvLaunchDefragger(Task, FALSE, VolumeEntry->DosPrefix);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Note that we have defragged this volume.
//
NewBuildDefragStatusSize = BuildDefragStatusSize;
NewBuildDefragStatusSize += (VolumeEntry->DosPrefixLength + 1) * sizeof(WCHAR);
NewBuildDefragStatus = PFSVC_ALLOC(NewBuildDefragStatusSize);
if (!NewBuildDefragStatus) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Start with new defragged drive path.
//
wcscpy(NewBuildDefragStatus, VolumeEntry->DosPrefix);
//
// Append original status.
//
RtlCopyMemory(NewBuildDefragStatus + VolumeEntry->DosPrefixLength + 1,
BuildDefragStatus,
BuildDefragStatusSize);
//
// The last character and the one before that should be NUL.
//
PFSVC_ASSERT(NewBuildDefragStatus[NewBuildDefragStatusSize/sizeof(WCHAR)-1] == 0);
PFSVC_ASSERT(NewBuildDefragStatus[NewBuildDefragStatusSize/sizeof(WCHAR)-2] == 0);
//
// Save the new status.
//
ErrorCode = PfSvSetBuildDefragStatus(&PfSvcGlobals.OsVersion,
NewBuildDefragStatus,
NewBuildDefragStatusSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Update the old variable.
//
PFSVC_ASSERT(BuildDefragStatus && BuildDefragStatusSize);
PFSVC_FREE(BuildDefragStatus);
BuildDefragStatus = NewBuildDefragStatus;
NewBuildDefragStatus = NULL;
BuildDefragStatusSize = NewBuildDefragStatusSize;
NewBuildDefragStatusSize = 0;
//
// Continue to check & defrag other volumes.
//
}
//
// If we came here, then we have successfully defragged all the drives
// we had to. Set the status in the registry. Note that defrag status
// value has to end with an additional NUL because it is REG_MULTI_SZ.
//
NewBuildDefragStatusSize = (wcslen(PFSVC_DEFRAG_DRIVES_DONE) + 1) * sizeof(WCHAR);
NewBuildDefragStatusSize += sizeof(WCHAR);
NewBuildDefragStatus = PFSVC_ALLOC(NewBuildDefragStatusSize);
if (!NewBuildDefragStatus) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
wcscpy(NewBuildDefragStatus, PFSVC_DEFRAG_DRIVES_DONE);
NewBuildDefragStatus[(NewBuildDefragStatusSize / sizeof(WCHAR)) - 1] = 0;
ErrorCode = PfSvSetBuildDefragStatus(&PfSvcGlobals.OsVersion,
NewBuildDefragStatus,
NewBuildDefragStatusSize);
//
// Fall through with error code.
//
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: DefragDisks(%p)=%x\n",Task,ErrorCode));
if (BuildDefragStatus) {
//
// We should have NULL'ed NewBuildDefragStatus, otherwise we will
// try to free the same memory twice.
//
PFSVC_ASSERT(BuildDefragStatus != NewBuildDefragStatus);
PFSVC_ASSERT(BuildDefragStatusSize);
PFSVC_FREE(BuildDefragStatus);
}
if (NewBuildDefragStatus) {
PFSVC_ASSERT(NewBuildDefragStatusSize);
PFSVC_FREE(NewBuildDefragStatus);
}
if (VolumeList) {
PfSvFreeNtPathTranslationList(VolumeList);
}
return ErrorCode;
}
//
// Routines to cleanup old scenario files in the prefetch directory.
//
DWORD
PfSvCleanupPrefetchDirectory(
PPFSVC_IDLE_TASK Task
)
/*++
Routine Description:
If we have too many scenario files in the prefetch directory, discard the
ones that are not as useful to make room for new files.
Arguments:
Task - If specified the function will check Task every once in a
while to see if it should exit with ERROR_RETRY.
Return Value:
Win32 error code.
--*/
{
PPFSVC_SCENARIO_AGE_INFO Scenarios;
PFSVC_SCENARIO_FILE_CURSOR FileCursor;
PPF_SCENARIO_HEADER Scenario;
ULONG NumPrefetchFiles;
ULONG AllocationSize;
ULONG NumScenarios;
ULONG ScenarioIdx;
ULONG PrefetchFileIdx;
ULONG FileSize;
ULONG FailedCheck;
ULONG MaxRemainingScenarioFiles;
ULONG NumLaunches;
ULONG HoursSinceLastLaunch;
FILETIME CurrentTime;
ULARGE_INTEGER CurrentTimeLI;
ULARGE_INTEGER LastLaunchTimeLI;
DWORD ErrorCode;
BOOLEAN AcquiredLock;
//
// Initialize locals.
//
AcquiredLock = FALSE;
NumScenarios = 0;
Scenarios = NULL;
Scenario = NULL;
GetSystemTimeAsFileTime(&CurrentTime);
PfSvInitializeScenarioFileCursor(&FileCursor);
CurrentTimeLI.LowPart = CurrentTime.dwLowDateTime;
CurrentTimeLI.HighPart = CurrentTime.dwHighDateTime;
DBGPR((PFID,PFTRC,"PFSVC: CleanupPrefetchDirectory(%p)\n",Task));
//
// Once we are done cleaning up, we should not have more than this many
// prefetch files remaining.
//
MaxRemainingScenarioFiles = PFSVC_MAX_PREFETCH_FILES / 4;
PFSVC_ACQUIRE_LOCK(PfSvcGlobals.PrefetchRootLock);
AcquiredLock = TRUE;
//
// Start the file cursor.
//
ErrorCode = PfSvStartScenarioFileCursor(&FileCursor, PfSvcGlobals.PrefetchRoot);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Count the number of files in the directory.
//
ErrorCode = PfSvCountFilesInDirectory(PfSvcGlobals.PrefetchRoot,
L"*." PF_PREFETCH_FILE_EXTENSION,
&NumPrefetchFiles);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
AcquiredLock = FALSE;
//
// Allocate an array that we will fill in with information from
// scenario files to determine which ones need to be discarded.
//
AllocationSize = NumPrefetchFiles * sizeof(PFSVC_SCENARIO_AGE_INFO);
Scenarios = PFSVC_ALLOC(AllocationSize);
if (!Scenarios) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Initialize the scenarios array so we know what to clean up.
//
RtlZeroMemory(Scenarios, AllocationSize);
NumScenarios = 0;
//
// Enumerate the scenario files:
//
ScenarioIdx = 0;
PrefetchFileIdx = 0;
do {
//
// Should we continue to run?
//
ErrorCode = PfSvContinueRunningTask(Task);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Get the file info for the next scenario file.
//
ErrorCode = PfSvGetNextScenarioFileInfo(&FileCursor);
if (ErrorCode == ERROR_NO_MORE_FILES) {
break;
}
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Is the file name longer than what a valid prefetch file can be max?
//
if (FileCursor.FileNameLength > PF_MAX_SCENARIO_FILE_NAME) {
//
// Bogus file. Remove it.
//
DeleteFile(FileCursor.FilePath);
goto NextPrefetchFile;
}
//
// Map the file.
//
ErrorCode = PfSvGetViewOfFile(FileCursor.FilePath,
&Scenario,
&FileSize);
if (ErrorCode != ERROR_SUCCESS) {
goto NextPrefetchFile;
}
//
// Verify the scenario file.
//
if (!PfSvVerifyScenarioBuffer(Scenario, FileSize, &FailedCheck) ||
Scenario->ServiceVersion != PFSVC_SERVICE_VERSION) {
DeleteFile(FileCursor.FilePath);
goto NextPrefetchFile;
}
//
// Skip boot scenario, we won't discard it.
//
if (Scenario->ScenarioType == PfSystemBootScenarioType) {
goto NextPrefetchFile;
}
//
// Determine the last time scenario was updated. I assume this
// corresponds to the last time scenario was launched...
//
LastLaunchTimeLI.LowPart = FileCursor.FileData.ftLastWriteTime.dwLowDateTime;
LastLaunchTimeLI.HighPart = FileCursor.FileData.ftLastWriteTime.dwHighDateTime;
if (CurrentTimeLI.QuadPart > LastLaunchTimeLI.QuadPart) {
HoursSinceLastLaunch = (ULONG) ((CurrentTimeLI.QuadPart - LastLaunchTimeLI.QuadPart) /
PFSVC_NUM_100NS_IN_AN_HOUR);
} else {
HoursSinceLastLaunch = 0;
}
//
// Calculate weight: bigger weight means scenario file won't get
// discarded. We calculate the weight by dividing the total number
// times a scenario was launched by how long has it been since the last
// launch of the scenario.
//
NumLaunches = Scenario->NumLaunches;
//
// For the calculations below limit how large NumLaunches can be, so
// values does not overflow.
//
if (NumLaunches > 1 * 1024 * 1024) {
NumLaunches = 1 * 1024 * 1024;
}
//
// Since we are going divide by number of hours (e.g. 7*24 for a program
// launched a week ago) multiplying the number of launches with a number
// allows us to give a weight other than 0 to scenarios launched long ago.
//
Scenarios[ScenarioIdx].Weight = NumLaunches * 256;
if (HoursSinceLastLaunch) {
Scenarios[ScenarioIdx].Weight /= HoursSinceLastLaunch;
}
//
// Copy over the file path.
//
AllocationSize = (FileCursor.FilePathLength + 1) * sizeof(WCHAR);
Scenarios[ScenarioIdx].FilePath = PFSVC_ALLOC(AllocationSize);
if (!Scenarios[ScenarioIdx].FilePath) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
wcscpy(Scenarios[ScenarioIdx].FilePath, FileCursor.FilePath);
ScenarioIdx++;
NextPrefetchFile:
PrefetchFileIdx++;
if (Scenario) {
UnmapViewOfFile(Scenario);
Scenario = NULL;
}
} while (PrefetchFileIdx < NumPrefetchFiles);
//
// If we do not have too many scenario files, we don't have to do anything.
//
NumScenarios = ScenarioIdx;
if (NumScenarios <= MaxRemainingScenarioFiles) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// Sort the age information.
//
qsort(Scenarios,
NumScenarios,
sizeof(PFSVC_SCENARIO_AGE_INFO),
PfSvCompareScenarioAgeInfo);
//
// Delete the files with the smallest weight until we reach our goal.
//
for (ScenarioIdx = 0;
(ScenarioIdx < NumScenarios) &&
((NumScenarios - ScenarioIdx) > MaxRemainingScenarioFiles);
ScenarioIdx++) {
//
// Should we continue to run?
//
ErrorCode = PfSvContinueRunningTask(Task);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
DeleteFile(Scenarios[ScenarioIdx].FilePath);
}
//
// Count the files in the directory now and update the global.
//
ErrorCode = PfSvCountFilesInDirectory(PfSvcGlobals.PrefetchRoot,
L"*." PF_PREFETCH_FILE_EXTENSION,
&NumPrefetchFiles);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Note that global NumPrefetchFiles is not protected, so the new value
// we are setting it to may be overwritten with an older value. It should
// not be a big problem though, maybe resulting in this task being requeued.
//
PfSvcGlobals.NumPrefetchFiles = NumPrefetchFiles;
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: CleanupPrefetchDirectory(%p)=%x\n",Task,ErrorCode));
if (AcquiredLock) {
PFSVC_RELEASE_LOCK(PfSvcGlobals.PrefetchRootLock);
}
PfSvCleanupScenarioFileCursor(&FileCursor);
if (Scenarios) {
for (ScenarioIdx = 0; ScenarioIdx < NumScenarios; ScenarioIdx++) {
if (Scenarios[ScenarioIdx].FilePath) {
PFSVC_FREE(Scenarios[ScenarioIdx].FilePath);
}
}
PFSVC_FREE(Scenarios);
}
if (Scenario) {
UnmapViewOfFile(Scenario);
}
return ErrorCode;
}
int
__cdecl
PfSvCompareScenarioAgeInfo(
const void *Param1,
const void *Param2
)
/*++
Routine Description:
Qsort comparison function for PFSVC_SCENARIO_AGE_INFO structure.
Arguments:
Param1, Param2 - pointer to PFSVC_SCENARIO_AGE_INFO structures
Return Value:
Qsort comparison function return value.
--*/
{
PFSVC_SCENARIO_AGE_INFO *Elem1;
PFSVC_SCENARIO_AGE_INFO *Elem2;
Elem1 = (PVOID) Param1;
Elem2 = (PVOID) Param2;
//
// Compare precalculated weights.
//
if (Elem1->Weight > Elem2->Weight) {
return 1;
} else if (Elem1->Weight < Elem2->Weight) {
return -1;
} else {
return 0;
}
}
//
// Routines to enumerate scenario files.
//
VOID
PfSvInitializeScenarioFileCursor (
PPFSVC_SCENARIO_FILE_CURSOR FileCursor
)
/*++
Routine Description:
Initializes the cursor structure so it can be safely cleaned up.
Arguments:
FileCursor - Pointer to structure.
Return Value:
None.
--*/
{
FileCursor->FilePath = NULL;
FileCursor->FileNameLength = 0;
FileCursor->FilePathLength = 0;
FileCursor->CurrentFileIdx = 0;
FileCursor->PrefetchRoot = NULL;
FileCursor->FindFileHandle = INVALID_HANDLE_VALUE;
return;
}
VOID
PfSvCleanupScenarioFileCursor(
PPFSVC_SCENARIO_FILE_CURSOR FileCursor
)
/*++
Routine Description:
Cleans up an initialized and possibly started cursor structure.
Arguments:
FileCursor - Pointer to structure.
Return Value:
None.
--*/
{
if (FileCursor->FilePath) {
PFSVC_FREE(FileCursor->FilePath);
}
if (FileCursor->PrefetchRoot) {
PFSVC_FREE(FileCursor->PrefetchRoot);
}
if (FileCursor->FindFileHandle != INVALID_HANDLE_VALUE) {
FindClose(FileCursor->FindFileHandle);
}
return;
}
DWORD
PfSvStartScenarioFileCursor(
PPFSVC_SCENARIO_FILE_CURSOR FileCursor,
WCHAR *PrefetchRoot
)
/*++
Routine Description:
After making this call on an initialized FileCursor, you can start
enumerating the scenario files in that directory by calling the get
next file function.
You have to call the get next file function after starting the cursor
to get the information on the first file.
If this function fails, you should call cleanup on the FileCursor
structure and reinitialize it before trying to start the cursor again.
Arguments:
FileCursor - Pointer to initialized cursor structure.
PrefetchRoot - Directory path in which we'll look for prefetch
files.
Return Value:
Win32 error code.
--*/
{
WCHAR *PrefetchFileSearchPattern;
WCHAR *PrefetchFileSearchPath;
ULONG PrefetchRootLength;
ULONG PrefetchFileSearchPathLength;
ULONG FileNameMaxLength;
ULONG FilePathMaxLength;
DWORD ErrorCode;
//
// Initialize locals.
//
PrefetchRootLength = wcslen(PrefetchRoot);
PrefetchFileSearchPattern = L"\\*." PF_PREFETCH_FILE_EXTENSION;
PrefetchFileSearchPath = NULL;
//
// The file cursor should have been initialized.
//
PFSVC_ASSERT(!FileCursor->CurrentFileIdx);
PFSVC_ASSERT(!FileCursor->PrefetchRoot);
PFSVC_ASSERT(FileCursor->FindFileHandle == INVALID_HANDLE_VALUE);
//
// Copy the prefetch root directory path.
//
FileCursor->PrefetchRoot = PFSVC_ALLOC((PrefetchRootLength + 1) * sizeof(WCHAR));
if (!FileCursor->PrefetchRoot) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
wcscpy(FileCursor->PrefetchRoot, PrefetchRoot);
FileCursor->PrefetchRootLength = PrefetchRootLength;
//
// Build the path we will pass in to enumerate the prefetch files.
//
PrefetchFileSearchPathLength = PrefetchRootLength;
PrefetchFileSearchPathLength += wcslen(PrefetchFileSearchPattern);
PrefetchFileSearchPath = PFSVC_ALLOC((PrefetchFileSearchPathLength + 1) * sizeof(WCHAR));
if (!PrefetchFileSearchPath) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
wcscpy(PrefetchFileSearchPath, PrefetchRoot);
wcscat(PrefetchFileSearchPath, PrefetchFileSearchPattern);
//
// Allocate the string we will use to build the full path of the
// prefetch files. We can use it for prefetch files with names of
// max MAX_PATH. This works because that is the max file name that
// can fit into WIN32_FIND_DATA structure.
//
FileNameMaxLength = MAX_PATH;
FilePathMaxLength = PrefetchRootLength;
FilePathMaxLength += wcslen(L"\\");
FilePathMaxLength += FileNameMaxLength;
FileCursor->FilePath = PFSVC_ALLOC((FilePathMaxLength + 1) * sizeof(WCHAR));
if (!FileCursor->FilePath) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Initialize the first part of the file path and note where we will
// start copying file names from.
//
wcscpy(FileCursor->FilePath, PrefetchRoot);
wcscat(FileCursor->FilePath, L"\\");
FileCursor->FileNameStart = PrefetchRootLength + 1;
//
// Start enumerating the files. Note that this puts the data for the
// first file into FileData member.
//
FileCursor->FindFileHandle = FindFirstFile(PrefetchFileSearchPath,
&FileCursor->FileData);
if (FileCursor->FindFileHandle == INVALID_HANDLE_VALUE) {
ErrorCode = GetLastError();
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: StartFileCursor(%p,%ws)=%x\n",FileCursor,PrefetchRoot,ErrorCode));
if (PrefetchFileSearchPath) {
PFSVC_FREE(PrefetchFileSearchPath);
}
return ErrorCode;
}
DWORD
PfSvGetNextScenarioFileInfo(
PPFSVC_SCENARIO_FILE_CURSOR FileCursor
)
/*++
Routine Description:
Fills in public fields of the FileCursor with information on the next
scenario file.
You have to call the get next file function after starting the cursor
to get the information on the first file.
Files with *names* longer than MAX_PATH will be skipped because it
is not feasible to handle these with Win32 API.
Arguments:
FileCursor - Pointer to started cursor structure.
Return Value:
ERROR_NO_MORE_FILES - No more files to enumerate.
Win32 error code.
--*/
{
DWORD ErrorCode;
//
// File cursor should have been started.
//
PFSVC_ASSERT(FileCursor->PrefetchRoot);
PFSVC_ASSERT(FileCursor->FindFileHandle != INVALID_HANDLE_VALUE);
//
// If this it the first file, the FileData for it was already set when
// we started the cursor. Otherwise call FindNextFile.
//
if (FileCursor->CurrentFileIdx != 0) {
if (!FindNextFile(FileCursor->FindFileHandle, &FileCursor->FileData)) {
ErrorCode = GetLastError();
goto cleanup;
}
}
FileCursor->FileNameLength = wcslen(FileCursor->FileData.cFileName);
//
// We allocated a file path to hold MAX_PATH file name in addition to the
// directory path. FileData.cFileName is MAX_PATH sized.
//
PFSVC_ASSERT(FileCursor->FileNameLength < MAX_PATH);
if (FileCursor->FileNameLength >= MAX_PATH) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Copy the file name.
//
wcscpy(FileCursor->FilePath + FileCursor->FileNameStart,
FileCursor->FileData.cFileName);
FileCursor->FilePathLength = FileCursor->FileNameStart + FileCursor->FileNameLength;
FileCursor->CurrentFileIdx++;
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTRC,"PFSVC: GetNextScenarioFile(%p)=%ws,%x\n",FileCursor,FileCursor->FileData.cFileName,ErrorCode));
return ErrorCode;
}
//
// File I/O utility routines.
//
DWORD
PfSvGetViewOfFile(
IN WCHAR *FilePath,
OUT PVOID *BasePointer,
OUT PULONG FileSize
)
/*++
Routine Description:
Map the all of the specified file to memory.
Arguments:
FilePath - NUL terminated path to file to map.
BasePointer - Start address of mapping will be returned here.
FileSize - Size of the mapping/file will be returned here.
Return Value:
Win32 error code.
--*/
{
HANDLE InputHandle;
HANDLE InputMappingHandle;
DWORD ErrorCode;
DWORD SizeL;
DWORD SizeH;
BOOLEAN OpenedFile;
BOOLEAN CreatedFileMapping;
//
// Initialize locals.
//
OpenedFile = FALSE;
CreatedFileMapping = FALSE;
DBGPR((PFID,PFTRC,"PFSVC: GetViewOfFile(%ws)\n", FilePath));
//
// Note that we are opening the file exclusively. This guarantees
// that for trace files as long as the kernel is not done writing
// it we can't open the file, which guarantees we won't have an
// incomplete file to worry about.
//
InputHandle = CreateFile(FilePath,
GENERIC_READ,
0,
NULL,
OPEN_EXISTING,
0,
NULL);
if (INVALID_HANDLE_VALUE == InputHandle)
{
ErrorCode = GetLastError();
goto cleanup;
}
OpenedFile = TRUE;
SizeL = GetFileSize(InputHandle, &SizeH);
if (SizeL == INVALID_FILE_SIZE && (GetLastError() != NO_ERROR )) {
ErrorCode = GetLastError();
goto cleanup;
}
if (SizeH) {
ErrorCode = ERROR_BAD_LENGTH;
goto cleanup;
}
if (FileSize) {
*FileSize = SizeL;
}
InputMappingHandle = CreateFileMapping(InputHandle,
0,
PAGE_READONLY,
0,
0,
NULL);
if (NULL == InputMappingHandle)
{
ErrorCode = GetLastError();
goto cleanup;
}
CreatedFileMapping = TRUE;
*BasePointer = MapViewOfFile(InputMappingHandle,
FILE_MAP_READ,
0,
0,
0);
if (NULL == *BasePointer) {
ErrorCode = GetLastError();
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
if (OpenedFile) {
CloseHandle(InputHandle);
}
if (CreatedFileMapping) {
CloseHandle(InputMappingHandle);
}
DBGPR((PFID,PFTRC,"PFSVC: GetViewOfFile(%ws)=%x\n", FilePath, ErrorCode));
return ErrorCode;
}
DWORD
PfSvWriteBuffer(
PWCHAR FilePath,
PVOID Buffer,
ULONG Length
)
/*++
Routine Description:
This routine creats/overwrites the file at the specified path and
writes the contents of the buffer to it.
Arguments:
FilePath - Full path to the file.
Buffer - Buffer to write out.
Length - Number of bytes to write out from the buffer.
Return Value:
Win32 error code.
--*/
{
DWORD BytesWritten;
HANDLE OutputHandle;
DWORD ErrorCode;
BOOL Result;
//
// Initialize locals.
//
OutputHandle = INVALID_HANDLE_VALUE;
DBGPR((PFID,PFSTRC,"PFSVC: WriteBuffer(%p,%ws)\n", Buffer, FilePath));
//
// Open file overwriting any existing one. Don't share it so
// nobody tries to read a half-written file.
//
OutputHandle = CreateFile(FilePath,
GENERIC_READ | GENERIC_WRITE,
0,
NULL,
CREATE_ALWAYS,
0,
NULL);
if (INVALID_HANDLE_VALUE == OutputHandle)
{
ErrorCode = GetLastError();
goto cleanup;
}
//
// Write out the scenario.
//
Result = WriteFile(OutputHandle,
Buffer,
Length,
&BytesWritten,
NULL);
if (!Result || (BytesWritten != Length)) {
ErrorCode = GetLastError();
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
if (OutputHandle != INVALID_HANDLE_VALUE) {
CloseHandle(OutputHandle);
}
DBGPR((PFID,PFSTRC,"PFSVC: WriteBuffer(%p,%ws)=%x\n", Buffer, FilePath, ErrorCode));
return ErrorCode;
}
DWORD
PfSvGetLastWriteTime (
WCHAR *FilePath,
PFILETIME LastWriteTime
)
/*++
Routine Description:
This function attempts to get the last write time for the
specified file.
Arguments:
FilePath - Pointer to NUL terminated path.
LastWriteTime - Pointer to return buffer.
Return Value:
Win32 error code.
--*/
{
HANDLE FileHandle;
DWORD ErrorCode;
//
// Initialize locals.
//
FileHandle = INVALID_HANDLE_VALUE;
//
// Open the file.
//
FileHandle = CreateFile(FilePath,
GENERIC_READ,
FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
NULL,
OPEN_EXISTING,
FILE_FLAG_BACKUP_SEMANTICS,
NULL);
if (FileHandle == INVALID_HANDLE_VALUE) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Query last write time.
//
if (!GetFileTime(FileHandle, NULL, NULL, LastWriteTime)) {
ErrorCode = GetLastError();
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
if (FileHandle != INVALID_HANDLE_VALUE) {
CloseHandle(FileHandle);
}
return ErrorCode;
}
DWORD
PfSvReadLine (
FILE *File,
WCHAR **LineBuffer,
ULONG *LineBufferMaxChars,
ULONG *LineLength
)
/*++
Routine Description:
This function reads a line from the specified file into
LineBuffer. If LineBuffer is NULL or not big enough, it is
allocated or reallocated using PFSVC_ALLOC/FREE macros. It is the
caller's reponsibility to free the returned buffer.
Carriage return/Line feed characters are included in the returned
LineBuffer & LineLength. Thus, a LineLength of 0 means end of file
is hit. Returned LineBuffer is NUL terminated.
Arguments:
File - File to read from.
LineBuffer - Pointer to a buffer to read the line into.
LineBufferMaxChars - Pointer to size of LineBuffer in characters,
including room for NUL etc.
LineLength - Pointer to length of the read line in characters
including the carriage return/linefeed, excluding NUL.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
WCHAR *NewBuffer;
ULONG NewBufferMaxChars;
WCHAR *CurrentReadPosition;
ULONG MaxCharsToRead;
//
// Verify parameters.
//
PFSVC_ASSERT(LineBuffer && LineBufferMaxChars && LineLength);
if (*LineBufferMaxChars) {
PFSVC_ASSERT(*LineBuffer);
}
//
// If a zero length but non NULL buffer was passed in, free it so
// we can allocate a larger initial one.
//
if (((*LineBufferMaxChars) == 0) && (*LineBuffer)) {
PFSVC_FREE(*LineBuffer);
(*LineBuffer) = NULL;
}
//
// If no buffer was passed in, allocate one. We do not want to
// enter the read line loop with a zero length or NULL buffer.
//
if (!(*LineBuffer)) {
PFSVC_ASSERT((*LineBufferMaxChars) == 0);
(*LineBuffer) = PFSVC_ALLOC(MAX_PATH * sizeof(WCHAR));
if (!(*LineBuffer)) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
(*LineBufferMaxChars) = MAX_PATH;
}
//
// Initialize output length and NUL terminate the output line.
//
(*LineLength) = 0;
(*(*LineBuffer)) = 0;
do {
//
// Try to read a line from the file.
//
CurrentReadPosition = (*LineBuffer) + (*LineLength);
MaxCharsToRead = (*LineBufferMaxChars) - (*LineLength);
if (!fgetws(CurrentReadPosition,
MaxCharsToRead,
File)) {
//
// If we have not hit an EOF, we have hit an error.
//
if (!feof(File)) {
ErrorCode = ERROR_READ_FAULT;
goto cleanup;
} else {
//
// We hit end of file. Return what we have.
//
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
}
//
// Update line length.
//
(*LineLength) += wcslen(CurrentReadPosition);
//
// If we have read a carriage return, we are done. Check to
// see if we had room to read anything first!
//
if ((*LineLength) && (*LineBuffer)[(*LineLength) - 1] == L'\n') {
break;
}
//
// If we read up to the end of the buffer, resize it.
//
if ((*LineLength) == (*LineBufferMaxChars) - 1) {
//
// We should not enter this loop with a zero lengthed or NULL
// line buffer.
//
PFSVC_ASSERT((*LineBufferMaxChars) && (*LineBuffer));
NewBufferMaxChars = (*LineBufferMaxChars) * 2;
NewBuffer = PFSVC_ALLOC(NewBufferMaxChars * sizeof(WCHAR));
if (!NewBuffer) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Copy contents of the original buffer and free it.
//
RtlCopyMemory(NewBuffer,
(*LineBuffer),
((*LineLength) + 1) * sizeof(WCHAR));
PFSVC_FREE(*LineBuffer);
//
// Update line buffer.
//
(*LineBuffer) = NewBuffer;
(*LineBufferMaxChars) = NewBufferMaxChars;
}
//
// Continue reading this line and appending it to output
// buffer.
//
} while (TRUE);
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ErrorCode == ERROR_SUCCESS && (*LineBufferMaxChars)) {
//
// Returned length must fit into buffer.
//
PFSVC_ASSERT((*LineLength) < (*LineBufferMaxChars));
//
// Returned buffer should be NUL terminated.
//
PFSVC_ASSERT((*LineBuffer)[(*LineLength)] == 0);
}
return ErrorCode;
}
DWORD
PfSvGetFileBasicInformation (
WCHAR *FilePath,
PFILE_BASIC_INFORMATION FileInformation
)
/*++
Routine Description:
This routine queries the basic attributes for the specified file.
Arguments:
FilePath - Pointer to full NT file path, e.g.
\Device\HarddiskVolume1\boot.ini, NOT Win32 path, e.g. c:\boot.ini
FileInformation - If successful the basic file info is returned here.
Return Value:
Win32 error code.
--*/
{
OBJECT_ATTRIBUTES ObjectAttributes;
UNICODE_STRING FilePathU;
NTSTATUS Status;
DWORD ErrorCode;
//
// Query the file information.
//
RtlInitUnicodeString(&FilePathU, FilePath);
InitializeObjectAttributes(&ObjectAttributes,
&FilePathU,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
Status = NtQueryAttributesFile(&ObjectAttributes,
FileInformation);
if (NT_SUCCESS(Status)) {
//
// In the typical success case, don't call possibly an expensive
// routine to convert the error code.
//
ErrorCode = ERROR_SUCCESS;
} else {
ErrorCode = RtlNtStatusToDosError(Status);
}
return ErrorCode;
}
DWORD
PfSvGetFileIndexNumber(
WCHAR *FilePath,
PLARGE_INTEGER FileIndexNumber
)
/*++
Routine Description:
This routine queries the file system's IndexNumber for the specified
file.
Arguments:
FilePath - Pointer to full NT file path, e.g.
\Device\HarddiskVolume1\boot.ini, NOT Win32 path, e.g. c:\boot.ini
FileIndexNumber - If successful the index number is returned here.
Return Value:
Win32 error code.
--*/
{
HANDLE FileHandle;
BOOLEAN OpenedFile;
NTSTATUS Status;
OBJECT_ATTRIBUTES ObjectAttributes;
UNICODE_STRING FilePathU;
IO_STATUS_BLOCK IoStatusBlock;
FILE_INTERNAL_INFORMATION InternalInformation;
//
// Initialize locals.
//
OpenedFile = FALSE;
//
// Open the file.
//
RtlInitUnicodeString(&FilePathU, FilePath);
InitializeObjectAttributes(&ObjectAttributes,
&FilePathU,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
Status = NtCreateFile(&FileHandle,
STANDARD_RIGHTS_READ |
FILE_READ_ATTRIBUTES |
FILE_READ_EA,
&ObjectAttributes,
&IoStatusBlock,
0,
0,
FILE_SHARE_READ |
FILE_SHARE_WRITE |
FILE_SHARE_DELETE,
FILE_OPEN,
0,
NULL,
0);
if (!NT_SUCCESS(Status)) {
goto cleanup;
}
OpenedFile = TRUE;
//
// Query internal information.
//
Status = NtQueryInformationFile(FileHandle,
&IoStatusBlock,
&InternalInformation,
sizeof(InternalInformation),
FileInternalInformation);
if (!NT_SUCCESS(Status)) {
goto cleanup;
}
*FileIndexNumber = InternalInformation.IndexNumber;
Status = STATUS_SUCCESS;
cleanup:
if (OpenedFile) {
NtClose(FileHandle);
}
return RtlNtStatusToDosError(Status);
}
//
// String utility routines.
//
PFSV_SUFFIX_COMPARISON_RESULT
PfSvCompareSuffix(
WCHAR *String,
ULONG StringLength,
WCHAR *Suffix,
ULONG SuffixLength,
BOOLEAN CaseSensitive
)
/*++
Routine Description:
This compares the last characters of String to Suffix. The strings
don't have to be NUL terminated.
NOTE: The lexical ordering is done starting from the LAST
characters.
Arguments:
String - String to check suffix of.
StringLength - Number of characters in String.
Suffix - What the suffix of String should match.
SuffixLength - Number of characters in Suffix.
CaseSensitive - Whether the comparison should be case sensitive.
Return Value:
PFSV_SUFFIX_COMPARISON_RESULT
--*/
{
LONG StringCharIdx;
WCHAR StringChar;
LONG SuffixCharIdx;
WCHAR SuffixChar;
//
// If suffix is longer than the string itself, it cannot match.
//
if (SuffixLength > StringLength) {
return PfSvSuffixLongerThan;
}
//
// If the suffix is 0 length it matches anything.
//
if (SuffixLength == 0) {
return PfSvSuffixIdentical;
}
//
// If the suffix is not 0 length and it is greater than
// StringLength, StringLength cannot be 0.
//
PFSVC_ASSERT(StringLength);
//
// Start from the last character of the string and try to match
// the suffix.
//
StringCharIdx = StringLength - 1;
SuffixCharIdx = SuffixLength - 1;
while (SuffixCharIdx >= 0) {
SuffixChar = Suffix[SuffixCharIdx];
StringChar = String[StringCharIdx];
if (!CaseSensitive) {
SuffixChar = towupper(SuffixChar);
StringChar = towupper(StringChar);
}
//
// Is comparing the values of chars same comparing them
// lexically?
//
if (StringChar < SuffixChar) {
return PfSvSuffixGreaterThan;
} else if (StringChar > SuffixChar) {
return PfSvSuffixLessThan;
}
//
// Otherwise this character matches. Compare next one.
//
StringCharIdx--;
SuffixCharIdx--;
}
//
// All suffix characters matched.
//
return PfSvSuffixIdentical;
}
PFSV_PREFIX_COMPARISON_RESULT
PfSvComparePrefix(
WCHAR *String,
ULONG StringLength,
WCHAR *Prefix,
ULONG PrefixLength,
BOOLEAN CaseSensitive
)
/*++
Routine Description:
This compares the first characters of String to Prefix. The
strings don't have to be NUL terminated.
Arguments:
String - String to check prefix of.
StringLength - Number of characters in String.
Suffix - What the prefix of String should match.
SuffixLength - Number of characters in Prefix.
CaseSensitive - Whether the comparison should be case sensitive.
Return Value:
PFSV_PREFIX_COMPARISON_RESULT
--*/
{
LONG StrCmpResult;
//
// If prefix is longer than the string itself, it cannot match.
//
if (PrefixLength > StringLength) {
return PfSvPrefixLongerThan;
}
//
// If the prefix is 0 length it matches anything.
//
if (PrefixLength == 0) {
return PfSvPrefixIdentical;
}
//
// If the prefix is not 0 length and it is greater than
// StringLength, StringLength cannot be 0.
//
ASSERT(StringLength);
//
// Compare the prefix to the beginning of the string.
//
if (CaseSensitive) {
StrCmpResult = wcsncmp(Prefix, String, PrefixLength);
} else {
StrCmpResult = _wcsnicmp(Prefix, String, PrefixLength);
}
if (StrCmpResult == 0) {
return PfSvPrefixIdentical;
} else if (StrCmpResult > 0) {
return PfSvPrefixGreaterThan;
} else {
return PfSvPrefixLessThan;
}
}
VOID
FASTCALL
PfSvRemoveEndOfLineChars (
WCHAR *Line,
ULONG *LineLength
)
/*++
Routine Description:
If the Line ends with \n/\r\n, these characters are removed and
LineLength is adjusted accordingly.
Arguments:
Line - Pointer to line string.
LineLength - Pointer to length of line string in characters
excluding any terminating NULs. This is updated if carriage
return/linefeed characters are removed.
Return Value:
None.
--*/
{
if ((*LineLength) && (Line[(*LineLength) - 1] == L'\n')) {
Line[(*LineLength) - 1] = 0;
(*LineLength)--;
if ((*LineLength) && (Line[(*LineLength) - 1] == L'\r')) {
Line[(*LineLength) - 1] = 0;
(*LineLength)--;
}
}
}
PWCHAR
PfSvcAnsiToUnicode(
PCHAR str
)
/*++
Routine Description:
This routine converts an ANSI string into an allocated wide
character string. The returned string should be freed by
PfSvcFreeString.
Arguments:
str - Pointer to string to convert.
Return Value:
Allocated wide character string or NULL if there is a failure.
--*/
{
ULONG len;
wchar_t *retstr = NULL;
len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
retstr = (wchar_t *)PFSVC_ALLOC(len * sizeof(wchar_t));
if (!retstr)
{
return NULL;
}
MultiByteToWideChar(CP_ACP, 0, str, -1, retstr, len);
return retstr;
}
PCHAR
PfSvcUnicodeToAnsi(
PWCHAR wstr
)
/*++
Routine Description:
This routine converts a unicode string into an allocated ansi
string. The returned string should be freed by PfSvcFreeString.
Arguments:
wstr - Pointer to string to convert.
Return Value:
Allocated ANSI string or NULL if there is a failure.
--*/
{
ULONG len;
char *retstr = NULL;
len = WideCharToMultiByte(CP_ACP, 0, wstr, -1, NULL, 0, 0, 0);
retstr = (char *) PFSVC_ALLOC(len * sizeof(char));
if (!retstr)
{
return NULL;
}
WideCharToMultiByte(CP_ACP, 0, wstr, -1, retstr, len, 0, 0);
return retstr;
}
VOID
PfSvcFreeString(
PVOID String
)
/*++
Routine Description:
This routine frees a string allocated and returned by
PfSvcUnicodeToAnsi or PfSvcAnsiToUnicode.
Arguments:
String - Pointer to string to free.
Return Value:
None.
--*/
{
PFSVC_FREE(String);
}
//
// Routines that deal with information in the registry.
//
DWORD
PfSvSaveStartInfo (
HKEY ServiceDataKey
)
/*++
Routine Description:
This routine saves start time, prefetcher version etc. into the
registry.
Arguments:
ServiceDataKey - Key under which the values will be set.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
DWORD PrefetchVersion;
SYSTEMTIME LocalTime;
WCHAR CurrentTime[50];
ULONG CurrentTimeMaxChars;
ULONG CurrentTimeSize;
//
// Initialize locals.
//
PrefetchVersion = PF_CURRENT_VERSION;
CurrentTimeMaxChars = sizeof(CurrentTime) / sizeof(WCHAR);
//
// Save version.
//
ErrorCode = RegSetValueEx(ServiceDataKey,
PFSVC_VERSION_VALUE_NAME,
0,
REG_DWORD,
(PVOID) &PrefetchVersion,
sizeof(PrefetchVersion));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Get system time and convert it to a string.
//
GetLocalTime(&LocalTime);
_snwprintf(CurrentTime, CurrentTimeMaxChars,
L"%04d/%02d/%02d-%02d:%02d:%02d",
(ULONG)LocalTime.wYear,
(ULONG)LocalTime.wMonth,
(ULONG)LocalTime.wDay,
(ULONG)LocalTime.wHour,
(ULONG)LocalTime.wMinute,
(ULONG)LocalTime.wSecond);
//
// Make sure it is terminated.
//
CurrentTime[CurrentTimeMaxChars - 1] = 0;
//
// Save it to the registry.
//
CurrentTimeSize = (wcslen(CurrentTime) + 1) * sizeof(WCHAR);
ErrorCode = RegSetValueEx(ServiceDataKey,
PFSVC_START_TIME_VALUE_NAME,
0,
REG_SZ,
(PVOID) CurrentTime,
CurrentTimeSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Save the initial statistics (which should be mostly zeros).
//
ErrorCode = PfSvSaveTraceProcessingStatistics(ServiceDataKey);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvSaveExitInfo (
HKEY ServiceDataKey,
DWORD ExitCode
)
/*++
Routine Description:
This routine saves the prefetcher service exit information to the
registry.
Arguments:
ServiceDataKey - Key under which the values will be set.
ExitCode - Win32 error code the service is exiting with.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
SYSTEMTIME LocalTime;
WCHAR CurrentTime[50];
ULONG CurrentTimeMaxChars;
ULONG CurrentTimeSize;
//
// Initialize locals.
//
CurrentTimeMaxChars = sizeof(CurrentTime) / sizeof(WCHAR);
//
// Save exit code.
//
ErrorCode = RegSetValueEx(ServiceDataKey,
PFSVC_EXIT_CODE_VALUE_NAME,
0,
REG_DWORD,
(PVOID) &ExitCode,
sizeof(ExitCode));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Get system time and convert it to a string.
//
GetLocalTime(&LocalTime);
_snwprintf(CurrentTime, CurrentTimeMaxChars,
L"%04d/%02d/%02d-%02d:%02d:%02d",
(ULONG)LocalTime.wYear,
(ULONG)LocalTime.wMonth,
(ULONG)LocalTime.wDay,
(ULONG)LocalTime.wHour,
(ULONG)LocalTime.wMinute,
(ULONG)LocalTime.wSecond);
//
// Make sure it is terminated.
//
CurrentTime[CurrentTimeMaxChars - 1] = 0;
//
// Save it to the registry.
//
CurrentTimeSize = (wcslen(CurrentTime) + 1) * sizeof(WCHAR);
ErrorCode = RegSetValueEx(ServiceDataKey,
PFSVC_EXIT_TIME_VALUE_NAME,
0,
REG_SZ,
(PVOID) CurrentTime,
CurrentTimeSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Save the final statistics.
//
ErrorCode = PfSvSaveTraceProcessingStatistics(ServiceDataKey);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvSaveTraceProcessingStatistics (
HKEY ServiceDataKey
)
/*++
Routine Description:
This routine saves global trace processing statistics to the
registry.
Arguments:
ServiceDataKey - Key under which the values will be set.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
//
// Save the various global statistics.
//
ErrorCode = RegSetValueEx(ServiceDataKey,
PFSVC_TRACES_PROCESSED_VALUE_NAME,
0,
REG_DWORD,
(PVOID) &PfSvcGlobals.NumTracesProcessed,
sizeof(DWORD));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
ErrorCode = RegSetValueEx(ServiceDataKey,
PFSVC_TRACES_SUCCESSFUL_VALUE_NAME,
0,
REG_DWORD,
(PVOID) &PfSvcGlobals.NumTracesSuccessful,
sizeof(DWORD));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
ErrorCode = RegSetValueEx(ServiceDataKey,
PFSVC_LAST_TRACE_FAILURE_VALUE_NAME,
0,
REG_DWORD,
(PVOID) &PfSvcGlobals.LastTraceFailure,
sizeof(DWORD));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvGetLastDiskLayoutTime(
FILETIME *LastDiskLayoutTime
)
/*++
Routine Description:
This routine queries the last time disk layout was updated from
the registry under the service data key.
Arguments:
LastDiskLayoutTime - Pointer to output data.
Return Value:
Win32 error code.
--*/
{
ULONG Size;
DWORD ErrorCode;
DWORD RegValueType;
FILETIME CurrentFileTime;
SYSTEMTIME SystemTime;
//
// Query last disk layout time from the registry and adjust it if
// necessary.
//
Size = sizeof(FILETIME);
ErrorCode = RegQueryValueEx(PfSvcGlobals.ServiceDataKey,
PFSVC_LAST_DISK_LAYOUT_TIME_VALUE_NAME,
NULL,
&RegValueType,
(PVOID) LastDiskLayoutTime,
&Size);
if (ErrorCode != ERROR_SUCCESS) {
if (ErrorCode == ERROR_FILE_NOT_FOUND) {
//
// No successful runs of the defragger to update layout has
// been recorded in the registry.
//
RtlZeroMemory(LastDiskLayoutTime, sizeof(FILETIME));
} else {
//
// This is a real error.
//
goto cleanup;
}
} else {
//
// The query was successful, but if the value type is not
// REG_BINARY, we most likely read in trash.
//
if (RegValueType != REG_BINARY || (Size != sizeof(FILETIME))) {
RtlZeroMemory(LastDiskLayoutTime, sizeof(FILETIME));
} else {
//
// If the time we recorded looks greater than the current
// time (e.g. because the user played with the system time
// and such), adjust it.
//
GetSystemTime(&SystemTime);
if (!SystemTimeToFileTime(&SystemTime, &CurrentFileTime)) {
ErrorCode = GetLastError();
goto cleanup;
}
if (CompareFileTime(LastDiskLayoutTime, &CurrentFileTime) > 0) {
//
// The time in the registry looks bogus. We'll set it
// to 0, to drive our caller to run the defragger to
// update the layout again.
//
RtlZeroMemory(LastDiskLayoutTime, sizeof(FILETIME));
}
}
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvSetLastDiskLayoutTime(
FILETIME *LastDiskLayoutTime
)
/*++
Routine Description:
This routine saves the last time the disk layout was updated to
the registry under the service data key.
Arguments:
LastDiskLayoutTime - Pointer to new disk layout time.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
WCHAR CurrentTime[50];
ULONG CurrentTimeMaxChars;
ULONG CurrentTimeSize;
FILETIME LocalFileTime;
SYSTEMTIME LocalSystemTime;
//
// Initialize locals.
//
CurrentTimeMaxChars = sizeof(CurrentTime) / sizeof(WCHAR);
//
// Save the specified time.
//
ErrorCode = RegSetValueEx(PfSvcGlobals.ServiceDataKey,
PFSVC_LAST_DISK_LAYOUT_TIME_VALUE_NAME,
0,
REG_BINARY,
(PVOID) LastDiskLayoutTime,
sizeof(FILETIME));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Also save it in human readable format.
//
if (!FileTimeToLocalFileTime(LastDiskLayoutTime, &LocalFileTime)) {
ErrorCode = GetLastError();
goto cleanup;
}
if (!FileTimeToSystemTime(&LocalFileTime, &LocalSystemTime)) {
ErrorCode = GetLastError();
goto cleanup;
}
_snwprintf(CurrentTime, CurrentTimeMaxChars,
L"%04d/%02d/%02d-%02d:%02d:%02d",
(ULONG)LocalSystemTime.wYear,
(ULONG)LocalSystemTime.wMonth,
(ULONG)LocalSystemTime.wDay,
(ULONG)LocalSystemTime.wHour,
(ULONG)LocalSystemTime.wMinute,
(ULONG)LocalSystemTime.wSecond);
//
// Make sure it is terminated.
//
CurrentTime[CurrentTimeMaxChars - 1] = 0;
//
// Save it to the registry.
//
CurrentTimeSize = (wcslen(CurrentTime) + 1) * sizeof(WCHAR);
ErrorCode = RegSetValueEx(PfSvcGlobals.ServiceDataKey,
PFSVC_LAST_DISK_LAYOUT_TIME_STRING_VALUE_NAME,
0,
REG_SZ,
(PVOID) CurrentTime,
CurrentTimeSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
DWORD
PfSvGetDontRunDefragger(
DWORD *DontRunDefragger
)
/*++
Routine Description:
This routine queries the registry setting that disables launching
the defragger when the system is idle.
Arguments:
DontRunDefragger - Pointer to output data.
Return Value:
Win32 error code.
--*/
{
HKEY ParametersKey;
ULONG Size;
DWORD Value;
DWORD ErrorCode;
DWORD RegValueType;
BOOLEAN OpenedParametersKey;
//
// Initialize locals.
//
OpenedParametersKey = FALSE;
//
// Open the parameters key, creating it if necessary.
//
ErrorCode = RegCreateKey(HKEY_LOCAL_MACHINE,
PFSVC_OPTIMAL_LAYOUT_REG_KEY_PATH,
&ParametersKey);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
OpenedParametersKey = TRUE;
//
// Query whether auto layout is enabled.
//
Size = sizeof(Value);
ErrorCode = RegQueryValueEx(ParametersKey,
PFSVC_OPTIMAL_LAYOUT_ENABLE_VALUE_NAME,
NULL,
&RegValueType,
(PVOID) &Value,
&Size);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// The query was successful. Make sure value is a DWORD.
//
if (RegValueType != REG_DWORD) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// Set the value.
//
*DontRunDefragger = !(Value);
ErrorCode = ERROR_SUCCESS;
cleanup:
if (OpenedParametersKey) {
CloseHandle(ParametersKey);
}
return ErrorCode;
}
BOOLEAN
PfSvAllowedToRunDefragger(
BOOLEAN CheckRegistry
)
/*++
Routine Description:
This routine checks the global state/parameters to see if we
are allowed to try to run the defragger.
Arguments:
CheckRegistry - Whether to ignore auto-layout enable key in the registry.
Return Value:
TRUE - Go ahead and run the defragger.
FALSE - Don't run the defragger.
--*/
{
PF_SCENARIO_TYPE ScenarioType;
BOOLEAN AllowedToRunDefragger;
BOOLEAN PrefetchingEnabled;
//
// Initialize locals.
//
AllowedToRunDefragger = FALSE;
//
// On checked builds allow auto-layout on other sku's too.
//
#ifndef PFSVC_DBG
//
// Is this a server machine?
//
if (PfSvcGlobals.OsVersion.wProductType != VER_NT_WORKSTATION) {
goto cleanup;
}
#endif
//
// Is prefetching enabled for any scenario type?
//
PrefetchingEnabled = FALSE;
for(ScenarioType = 0; ScenarioType < PfMaxScenarioType; ScenarioType++) {
if (PfSvcGlobals.Parameters.EnableStatus[ScenarioType] == PfSvEnabled) {
PrefetchingEnabled = TRUE;
break;
}
}
if (!PrefetchingEnabled) {
goto cleanup;
}
//
// Did we try to run the defragger and it crashed before?
//
if (PfSvcGlobals.DefraggerErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// If in the registry we were not allowed to run the defragger, don't
// do so.
//
if (CheckRegistry) {
if (PfSvcGlobals.DontRunDefragger) {
goto cleanup;
}
}
//
// If we passed all checks, we are allowed to run the defragger.
//
AllowedToRunDefragger = TRUE;
cleanup:
return AllowedToRunDefragger;
}
//
// Routines that deal with security.
//
BOOL
PfSvSetPrivilege(
HANDLE hToken,
LPCTSTR lpszPrivilege,
ULONG ulPrivilege,
BOOL bEnablePrivilege
)
/*++
Routine Description:
Enables or disables a privilege in an access token.
Arguments:
hToken - Access token handle.
lpszPrivilege - Name of privilege to enable/disable.
ulPrivilege - If a name is not specified, then a ULONG privilege
should be specified.
bEnablePrivilege - Whether to enable or disable privilege
Return Value:
TRUE - Success.
FALSE - Failure.
--*/
{
TOKEN_PRIVILEGES tp;
LUID luid;
if (lpszPrivilege) {
if ( !LookupPrivilegeValue(NULL,
lpszPrivilege,
&luid)) {
return FALSE;
}
} else {
luid = RtlConvertUlongToLuid(ulPrivilege);
}
tp.PrivilegeCount = 1;
tp.Privileges[0].Luid = luid;
if (bEnablePrivilege)
tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
else
tp.Privileges[0].Attributes = 0;
//
// Enable the privilege or disable all privileges.
//
AdjustTokenPrivileges(
hToken,
FALSE,
&tp,
0,
(PTOKEN_PRIVILEGES) NULL,
(PDWORD) NULL);
//
// Call GetLastError to determine whether the function succeeded.
//
if (GetLastError() != ERROR_SUCCESS) {
return FALSE;
}
return TRUE;
}
DWORD
PfSvSetAdminOnlyPermissions(
WCHAR *ObjectPath,
HANDLE ObjectHandle,
SE_OBJECT_TYPE ObjectType
)
/*++
Routine Description:
This routine makes the built-in administrators group the owner and
only allowed in the DACL of the specified directory or event
object.
The calling thread must have the SE_TAKE_OWNERSHIP_NAME privilege.
Arguments:
ObjectPath - File/directory path or event name.
ObjectHandle - If this is a SE_KERNEL_OBJECT, handle to it,
otherwise NULL.
ObjectType - Security object type. Only SE_KERNEL_OBJECT and
SE_FILE_OBJECT are supported.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
SID_IDENTIFIER_AUTHORITY SIDAuthority = SECURITY_NT_AUTHORITY;
PSID AdministratorsSID;
PSECURITY_DESCRIPTOR SecurityDescriptor;
PACL DiscretionaryACL;
DWORD ACLRevision;
ULONG ACESize;
ULONG ACLSize;
PACCESS_ALLOWED_ACE AccessAllowedAce;
BOOL Result;
//
// Initialize locals.
//
AdministratorsSID = NULL;
SecurityDescriptor = NULL;
DiscretionaryACL = NULL;
ACLRevision = ACL_REVISION;
//
// Check parameters.
//
if (ObjectType == SE_KERNEL_OBJECT) {
if (ObjectHandle == NULL) {
ErrorCode = ERROR_INVALID_PARAMETER;
goto cleanup;
}
} else if (ObjectType == SE_FILE_OBJECT) {
if (ObjectHandle != NULL) {
ErrorCode = ERROR_INVALID_PARAMETER;
goto cleanup;
}
} else {
ErrorCode = ERROR_INVALID_PARAMETER;
goto cleanup;
}
//
// Create a SID for the BUILTIN\Administrators group.
//
if(!AllocateAndInitializeSid(&SIDAuthority,
2,
SECURITY_BUILTIN_DOMAIN_RID,
DOMAIN_ALIAS_RID_ADMINS,
0, 0, 0, 0, 0, 0,
&AdministratorsSID)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Make Administrators the owner.
//
ErrorCode = SetNamedSecurityInfo (ObjectPath,
ObjectType,
OWNER_SECURITY_INFORMATION,
AdministratorsSID,
NULL,
NULL,
NULL);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Setup a discretionary access control list:
//
//
// Determine size of an ACCESS_ALLOWED access control entry for
// the administrators group. (Subtract size of SidStart which is
// both part of ACE and SID.
//
ACESize = sizeof(ACCESS_ALLOWED_ACE);
ACESize -= sizeof (AccessAllowedAce->SidStart);
ACESize += GetLengthSid(AdministratorsSID);
//
// Determine size of the access control list.
//
ACLSize = sizeof(ACL);
ACLSize += ACESize;
//
// Allocate and initialize the access control list.
//
DiscretionaryACL = PFSVC_ALLOC(ACLSize);
if (!DiscretionaryACL) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
if (!InitializeAcl(DiscretionaryACL, ACLSize, ACLRevision)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Add an ACE to allow access for the Administrators group.
//
if (!AddAccessAllowedAce(DiscretionaryACL,
ACLRevision,
GENERIC_ALL,
AdministratorsSID)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Initialize a security descriptor.
//
SecurityDescriptor = PFSVC_ALLOC(SECURITY_DESCRIPTOR_MIN_LENGTH);
if (!SecurityDescriptor) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
if (!InitializeSecurityDescriptor(SecurityDescriptor,
SECURITY_DESCRIPTOR_REVISION)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Set the discretionary access control list on the security descriptor.
//
if (!SetSecurityDescriptorDacl(SecurityDescriptor,
TRUE,
DiscretionaryACL,
FALSE)) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Set the built security descriptor on the prefetch directory.
//
if (ObjectType == SE_FILE_OBJECT) {
Result = SetFileSecurity(ObjectPath,
DACL_SECURITY_INFORMATION,
SecurityDescriptor);
} else {
PFSVC_ASSERT(ObjectType == SE_KERNEL_OBJECT);
PFSVC_ASSERT(ObjectHandle);
Result = SetKernelObjectSecurity(ObjectHandle,
DACL_SECURITY_INFORMATION,
SecurityDescriptor);
}
if (!Result) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
if (AdministratorsSID) {
FreeSid(AdministratorsSID);
}
if (SecurityDescriptor) {
PFSVC_FREE(SecurityDescriptor);
}
if (DiscretionaryACL) {
PFSVC_FREE(DiscretionaryACL);
}
return ErrorCode;
}
DWORD
PfSvGetPrefetchServiceThreadPrivileges (
VOID
)
/*++
Routine Description:
This routine ensures there is a security token for the current
thread and sets the right privileges on it so the thread can
communicate with the kernel mode prefetcher. It should be called
right after a thread is created.
Arguments:
None.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
BOOLEAN ImpersonatedSelf;
BOOLEAN OpenedThreadToken;
HANDLE ThreadToken;
//
// Initialize locals.
//
ImpersonatedSelf = FALSE;
OpenedThreadToken = FALSE;
DBGPR((PFID,PFTRC,"PFSVC: GetThreadPriviliges()\n"));
//
// Obtain a security context for this thread so we can set
// privileges etc. without effecting the whole process.
//
if (!ImpersonateSelf(SecurityImpersonation)) {
DBGPR((PFID,PFERR,"PFSVC: GetThreadPriviliges()-FailedImpersonateSelf\n"));
ErrorCode = GetLastError();
goto cleanup;
}
ImpersonatedSelf = TRUE;
//
// Set the privileges we will need talk to the kernel mode
// prefetcher:
//
//
// Open thread's access token.
//
if (!OpenThreadToken(GetCurrentThread(),
TOKEN_ADJUST_PRIVILEGES,
FALSE,
&ThreadToken)) {
DBGPR((PFID,PFERR,"PFSVC: GetThreadPriviliges()-FailedOpenToken\n"));
ErrorCode = GetLastError();
goto cleanup;
}
OpenedThreadToken = TRUE;
//
// Enable the SE_PROF_SINGLE_PROCESS_PRIVILEGE privilege so the
// kernel mode prefetcher accepts our queries & set requests.
//
//
if (!PfSvSetPrivilege(ThreadToken, 0, SE_PROF_SINGLE_PROCESS_PRIVILEGE, TRUE)) {
DBGPR((PFID,PFERR,"PFSVC: GetThreadPriviliges()-FailedEnableProf\n"));
ErrorCode = GetLastError();
goto cleanup;
}
//
// Enable the SE_TAKE_OWNERSHIP_NAME privilege so we can get
// ownership of the prefetch directory.
//
if (!PfSvSetPrivilege(ThreadToken, SE_TAKE_OWNERSHIP_NAME, 0, TRUE)) {
DBGPR((PFID,PFERR,"PFSVC: GetThreadPriviliges()-FailedEnableOwn\n"));
ErrorCode = GetLastError();
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
if (OpenedThreadToken) {
CloseHandle(ThreadToken);
}
if (ErrorCode != ERROR_SUCCESS) {
if (ImpersonatedSelf) {
RevertToSelf();
}
}
DBGPR((PFID,PFTRC,"PFSVC: GetThreadPriviliges()=%x\n",ErrorCode));
return ErrorCode;
}
//
// Routines that deal with volume node structures.
//
DWORD
PfSvCreateVolumeNode (
PPFSVC_SCENARIO_INFO ScenarioInfo,
WCHAR *VolumePath,
ULONG VolumePathLength,
PLARGE_INTEGER CreationTime,
ULONG SerialNumber
)
/*++
Routine Description:
This routine creates a volume node with the specifed info if it
does not already exist. If a node already exists, it verifies
CreationTime and SerialNumber.
Arguments:
ScenarioInfo - Pointer to new scenario information.
VolumePath - UPCASE NT full path of the volume, NUL terminated.
VolumePathLength - Number of characters in VolumePath excluding NUL.
CreationTime & SerialNumber - For the volume.
Return Value:
ERROR_REVISION_MISMATCH - There already exists a volume node with
that path but with a different signature/creation time.
Win32 error code.
--*/
{
PLIST_ENTRY HeadEntry;
PLIST_ENTRY NextEntry;
PLIST_ENTRY FoundPosition;
PPFSVC_VOLUME_NODE VolumeNode;
DWORD ErrorCode;
LONG ComparisonResult;
PPFSVC_VOLUME_NODE NewVolumeNode;
PWCHAR NewVolumePath;
//
// Initialize locals.
//
NewVolumeNode = NULL;
NewVolumePath = NULL;
DBGPR((PFID,PFSTRC,"PFSVC: CreateVolumeNode(%ws)\n", VolumePath));
//
// Walk through the existing volume nodes list and try to find
// matching one.
//
HeadEntry = &ScenarioInfo->VolumeList;
NextEntry = HeadEntry->Flink;
FoundPosition = NULL;
while (NextEntry != HeadEntry) {
VolumeNode = CONTAINING_RECORD(NextEntry,
PFSVC_VOLUME_NODE,
VolumeLink);
NextEntry = NextEntry->Flink;
ComparisonResult = wcsncmp(VolumePath,
VolumeNode->VolumePath,
VolumePathLength);
if (ComparisonResult == 0) {
//
// Make sure VolumePathLengths are equal.
//
if (VolumeNode->VolumePathLength != VolumePathLength) {
//
// Continue searching.
//
continue;
}
//
// We found our volume. Verify magics.
//
if (VolumeNode->SerialNumber != SerialNumber ||
VolumeNode->CreationTime.QuadPart != CreationTime->QuadPart) {
ErrorCode = ERROR_REVISION_MISMATCH;
goto cleanup;
} else {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
} else if (ComparisonResult < 0) {
//
// The volume paths are sorted lexically. The file path
// would be less than other volumes too. The new node
// would go right before this node.
//
FoundPosition = &VolumeNode->VolumeLink;
break;
}
//
// Continue looking...
//
}
//
// If we could not find an entry to put the new entry before, it
// goes before the list head.
//
if (!FoundPosition) {
FoundPosition = HeadEntry;
}
//
// If we break out of the while loop, we could not find a
// volume. We should create a new node.
//
NewVolumeNode = PfSvChunkAllocatorAllocate(&ScenarioInfo->VolumeNodeAllocator);
if (!NewVolumeNode) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
NewVolumePath = PfSvStringAllocatorAllocate(&ScenarioInfo->PathAllocator,
(VolumePathLength + 1) * sizeof(WCHAR));
if (!NewVolumePath) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Copy file's volume path.
//
wcsncpy(NewVolumePath, VolumePath, VolumePathLength);
NewVolumePath[VolumePathLength] = 0;
//
// Initialize volume node.
//
NewVolumeNode->VolumePath = NewVolumePath;
NewVolumeNode->VolumePathLength = VolumePathLength;
NewVolumeNode->SerialNumber = SerialNumber;
NewVolumeNode->CreationTime = (*CreationTime);
InitializeListHead(&NewVolumeNode->SectionList);
PfSvInitializePathList(&NewVolumeNode->DirectoryList, &ScenarioInfo->PathAllocator, TRUE);
NewVolumeNode->NumSections = 0;
NewVolumeNode->NumAllSections = 0;
NewVolumeNode->MFTSectionNode = NULL;
//
// Add it to the scenario's volume list before the found position.
//
InsertTailList(FoundPosition, &NewVolumeNode->VolumeLink);
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ErrorCode != ERROR_SUCCESS) {
if (NewVolumePath) {
PfSvStringAllocatorFree(&ScenarioInfo->PathAllocator, NewVolumePath);
}
if (NewVolumeNode) {
PfSvChunkAllocatorFree(&ScenarioInfo->VolumeNodeAllocator, NewVolumeNode);
}
}
DBGPR((PFID,PFSTRC,"PFSVC: CreateVolumeNode()=%x\n", ErrorCode));
return ErrorCode;
}
PPFSVC_VOLUME_NODE
PfSvGetVolumeNode (
PPFSVC_SCENARIO_INFO ScenarioInfo,
WCHAR *FilePath,
ULONG FilePathLength
)
/*++
Routine Description:
This routine looks for a volume node for the specified file path.
Arguments:
ScenarioInfo - Pointer to new scenario information.
FilePath - NT full path of the file, NUL terminated.
FilePathLength - Number of characters in FilePath excluding NUL.
Return Value:
Pointer to found VolumeNode, or NULL.
--*/
{
PLIST_ENTRY HeadEntry;
PLIST_ENTRY NextEntry;
PPFSVC_VOLUME_NODE VolumeNode;
DWORD ErrorCode;
PFSV_PREFIX_COMPARISON_RESULT ComparisonResult;
DBGPR((PFID,PFSTRC,"PFSVC: GetVolumeNode(%ws)\n", FilePath));
//
// Walk through the existing volume nodes list and try to find
// matching one.
//
HeadEntry = &ScenarioInfo->VolumeList;
NextEntry = HeadEntry->Flink;
while (NextEntry != HeadEntry) {
VolumeNode = CONTAINING_RECORD(NextEntry,
PFSVC_VOLUME_NODE,
VolumeLink);
NextEntry = NextEntry->Flink;
ComparisonResult = PfSvComparePrefix(FilePath,
FilePathLength,
VolumeNode->VolumePath,
VolumeNode->VolumePathLength,
TRUE);
if (ComparisonResult == PfSvPrefixIdentical) {
//
// Make sure that there is a slash in the file
// path after the volume path.
//
if (FilePath[VolumeNode->VolumePathLength] != L'\\') {
//
// Continue searching.
//
continue;
}
//
// We found our volume.
//
ErrorCode = ERROR_SUCCESS;
goto cleanup;
} else if (ComparisonResult == PfSvPrefixGreaterThan) {
//
// The volume paths are sorted lexically. The volume path
// would be less than other volumes too.
//
break;
}
//
// Continue looking...
//
}
//
// If we break out of the while loop, we could not find a
// volume.
//
VolumeNode = NULL;
ErrorCode = ERROR_NOT_FOUND;
cleanup:
if (ErrorCode != ERROR_SUCCESS) {
VolumeNode = NULL;
}
DBGPR((PFID,PFSTRC,"PFSVC: GetVolumeNode()=%p\n", VolumeNode));
return VolumeNode;
}
VOID
PfSvCleanupVolumeNode(
PPFSVC_SCENARIO_INFO ScenarioInfo,
PPFSVC_VOLUME_NODE VolumeNode
)
/*++
Routine Description:
This function cleans up a volume node structure. It does not free
the structure itself.
Arguments:
ScenarioInfo - Pointer to scenario info context this volume node
belongs to.
VolumeNode - Pointer to structure.
Return Value:
None.
--*/
{
PLIST_ENTRY SectListEntry;
PPFSVC_SECTION_NODE SectionNode;
//
// Cleanup directory list.
//
PfSvCleanupPathList(&VolumeNode->DirectoryList);
//
// If there is a volume path, free it.
//
if (VolumeNode->VolumePath) {
PfSvStringAllocatorFree(&ScenarioInfo->PathAllocator, VolumeNode->VolumePath);
VolumeNode->VolumePath = NULL;
}
//
// Remove the section nodes from our list and re-initialize their
// links so they know they have been removed.
//
while (!IsListEmpty(&VolumeNode->SectionList)) {
SectListEntry = RemoveHeadList(&VolumeNode->SectionList);
SectionNode = CONTAINING_RECORD(SectListEntry,
PFSVC_SECTION_NODE,
SectionVolumeLink);
InitializeListHead(&SectionNode->SectionVolumeLink);
}
return;
}
DWORD
PfSvAddParentDirectoriesToList(
PPFSVC_PATH_LIST DirectoryList,
ULONG VolumePathLength,
WCHAR *FilePath,
ULONG FilePathLength
)
/*++
Routine Description:
This function will parse a fully qualified NT file path and add
all parent directories to the specified directory list. The part
of the path that is the volume path is skipped.
Arguments:
DirectoryList - Pointer to list to update.
VolumePathLength - Position in the file path at which the volume
path ends and the root directory starts.
FilePath - Pointer to NT file path, NUL terminated.
FullPathLength - Length of FilePath in characters excluding NUL.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
WCHAR DirectoryPath[MAX_PATH];
ULONG DirectoryPathLength;
WCHAR *CurrentChar;
WCHAR *FilePathEnd;
//
// Initialize locals.
//
FilePathEnd = FilePath + FilePathLength;
PFSVC_ASSERT(*FilePathEnd == 0);
//
// Skip the volume path and start from the root directory.
//
CurrentChar = FilePath + VolumePathLength;
while (CurrentChar < FilePathEnd) {
if (*CurrentChar == L'\\') {
//
// We got a directory.
//
DirectoryPathLength = (ULONG) (CurrentChar - FilePath + 1);
if (DirectoryPathLength >= MAX_PATH) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Copy directory path to buffer and NUL terminate it.
//
wcsncpy(DirectoryPath, FilePath, DirectoryPathLength);
DirectoryPath[DirectoryPathLength] = 0;
PFSVC_ASSERT(DirectoryPath[DirectoryPathLength - 1] == L'\\');
//
// Add it to the list.
//
ErrorCode = PfSvAddToPathList(DirectoryList,
DirectoryPath,
DirectoryPathLength);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Continue looking for more directories in the path.
//
}
CurrentChar++;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
//
// Routines used to allocate / free section & page nodes etc. efficiently.
//
// FUTURE-2002/03/29-ScottMa -- The "optional" Buffer parameter is always
// supplied in both the chunk and string allocators. You could remove the
// code that handles a NULL Buffer throughout these functions.
VOID
PfSvChunkAllocatorInitialize (
PPFSVC_CHUNK_ALLOCATOR Allocator
)
/*++
Routine Description:
Initializes the allocator structure. Must be called before other allocator
routines.
Arguments:
Allocator - Pointer to structure.
Return Value:
None.
--*/
{
//
// Zero the structure. This effectively initializes the following fields:
// Buffer
// BufferEnd
// FreePointer
// ChunkSize
// MaxHeapAllocs
// NumHeapAllocs
// UserSpecifiedBuffer
//
RtlZeroMemory(Allocator, sizeof(PFSVC_CHUNK_ALLOCATOR));
return;
}
DWORD
PfSvChunkAllocatorStart (
IN PPFSVC_CHUNK_ALLOCATOR Allocator,
OPTIONAL IN PVOID Buffer,
IN ULONG ChunkSize,
IN ULONG MaxChunks
)
/*++
Routine Description:
Must be called before calling alloc/free on an allocator that has
been initialized.
Arguments:
Allocator - Pointer to initialized structure.
Buffer - If specified, it is the buffer that will be divided up into
MaxChunks of ChunkSize and given away. Otherwise a buffer will be
allocated. If specified, the user has to free the buffer after the
chunk allocator has been cleaned up. It should be aligned right.
ChunkSize - In bytes how big each allocated chunk will be.
e.g. sizeof(PFSVC_PAGE_NODE) It should be greater than
sizeof(DWORD).
MaxChunks - Max number of allocs that will be made from the allocator.
Return Value:
Win32 error code.
--*/
{
ULONG AllocationSize;
DWORD ErrorCode;
//
// Initialize locals.
//
AllocationSize = ChunkSize * MaxChunks;
//
// We should be initialized and we should not get started twice.
//
PFSVC_ASSERT(Allocator->Buffer == NULL);
//
// Chunk size should not be too small.
//
if (ChunkSize < sizeof(DWORD) || !MaxChunks) {
ErrorCode = ERROR_INVALID_PARAMETER;
goto cleanup;
}
//
// Did the user specify the buffer to use?
//
if (Buffer) {
Allocator->Buffer = Buffer;
Allocator->UserSpecifiedBuffer = TRUE;
} else {
Allocator->Buffer = PFSVC_ALLOC(AllocationSize);
if (!Allocator->Buffer) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
Allocator->UserSpecifiedBuffer = FALSE;
}
Allocator->BufferEnd = (PCHAR) Buffer + (ULONG_PTR) AllocationSize;
Allocator->FreePointer = Buffer;
Allocator->ChunkSize = ChunkSize;
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
PVOID
PfSvChunkAllocatorAllocate (
PPFSVC_CHUNK_ALLOCATOR Allocator
)
/*++
Routine Description:
Returns a ChunkSize chunk allocated from Allocator. ChunkSize was specified
when Allocator was started. If a chunk is return the caller should free it
to this Allocator before uninitializing the Allocator.
Arguments:
Allocator - Pointer to started allocator.
Return Value:
NULL or chunk allocated from Allocator.
--*/
{
PVOID ReturnChunk;
//
// We should not be trying to make allocations before we start the
// allocator.
//
PFSVC_ASSERT(Allocator->Buffer && Allocator->ChunkSize);
//
// If we can allocate from our preallocated buffer do so. Otherwise we
// have to hit the heap.
//
if (Allocator->FreePointer >= Allocator->BufferEnd) {
Allocator->MaxHeapAllocs++;
ReturnChunk = PFSVC_ALLOC(Allocator->ChunkSize);
if (ReturnChunk) {
Allocator->NumHeapAllocs++;
}
} else {
ReturnChunk = Allocator->FreePointer;
Allocator->FreePointer += (ULONG_PTR) Allocator->ChunkSize;
}
return ReturnChunk;
}
VOID
PfSvChunkAllocatorFree (
PPFSVC_CHUNK_ALLOCATOR Allocator,
PVOID Allocation
)
/*++
Routine Description:
Frees a chunk allocated from the allocator. This may not make it available
for use by further allocations from the allocator.
Arguments:
Allocator - Pointer to started allocator.
Allocation - Allocation to free.
Return Value:
None.
--*/
{
//
// Is this within the preallocated block?
//
if ((PUCHAR) Allocation >= Allocator->Buffer &&
(PUCHAR) Allocation < Allocator->BufferEnd) {
//
// Mark this chunk freed.
//
*(PULONG)Allocation = PFSVC_CHUNK_ALLOCATOR_FREED_MAGIC;
} else {
//
// This chunk was allocated from heap.
//
PFSVC_ASSERT(Allocator->NumHeapAllocs && Allocator->MaxHeapAllocs);
Allocator->NumHeapAllocs--;
PFSVC_FREE(Allocation);
}
return;
}
VOID
PfSvChunkAllocatorCleanup (
PPFSVC_CHUNK_ALLOCATOR Allocator
)
/*++
Routine Description:
Cleans up resources associated with the allocator. There should not be
any outstanding allocations from the allocator when this function is
called.
Arguments:
Allocator - Pointer to initialized allocator.
Return Value:
None.
--*/
{
PCHAR CurrentChunk;
ULONG Magic;
if (Allocator->Buffer) {
#ifdef PFSVC_DBG
//
// Make sure all real heap allocations have been freed.
//
PFSVC_ASSERT(Allocator->NumHeapAllocs == 0);
//
// Make sure all allocated chunks have been freed. Check
// ChunkSize first, if it's corrupted we'd loop forever.
//
PFSVC_ASSERT(Allocator->ChunkSize);
for (CurrentChunk = Allocator->Buffer;
CurrentChunk < Allocator->FreePointer;
CurrentChunk += (ULONG_PTR) Allocator->ChunkSize) {
Magic = *(PULONG)CurrentChunk;
PFSVC_ASSERT(Magic == PFSVC_CHUNK_ALLOCATOR_FREED_MAGIC);
}
#endif // PFSVC_DBG
//
// If the buffer was allocated by us (and not specified by
// the user), free it.
//
if (!Allocator->UserSpecifiedBuffer) {
PFSVC_FREE(Allocator->Buffer);
}
#ifdef PFSVC_DBG
//
// Setup the fields so if we try to make allocations after cleaning up
// an allocator we'll hit an assert.
//
Allocator->FreePointer = Allocator->Buffer;
Allocator->Buffer = NULL;
#endif // PFSVC_DBG
}
return;
}
//
// Routines used to allocate / free path strings efficiently.
//
VOID
PfSvStringAllocatorInitialize (
PPFSVC_STRING_ALLOCATOR Allocator
)
/*++
Routine Description:
Initializes the allocator structure. Must be called before other allocator
routines.
Arguments:
Allocator - Pointer to structure.
Return Value:
None.
--*/
{
//
// Zero the structure. This effectively initializes the following fields:
// Buffer
// BufferEnd
// FreePointer
// MaxHeapAllocs
// NumHeapAllocs
// LastAllocationSize
// UserSpecifiedBuffer
//
RtlZeroMemory(Allocator, sizeof(PFSVC_STRING_ALLOCATOR));
return;
}
DWORD
PfSvStringAllocatorStart (
IN PPFSVC_STRING_ALLOCATOR Allocator,
OPTIONAL IN PVOID Buffer,
IN ULONG MaxSize
)
/*++
Routine Description:
Must be called before calling alloc/free on an allocator that has
been initialized.
Arguments:
Allocator - Pointer to initialized structure.
Buffer - If specified, it is the buffer that we will allocate strings
from. Otherwise a buffer will be allocated. If specified, the user
has to free the buffer after the string allocator has been cleaned up.
It should be aligned right.
MaxSize - Max valid size of buffer in bytes.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
//
// We should be initialized and we should not get started twice.
//
PFSVC_ASSERT(Allocator->Buffer == NULL);
//
// Did the user specify the buffer to use?
//
if (Buffer) {
Allocator->Buffer = Buffer;
Allocator->UserSpecifiedBuffer = TRUE;
} else {
Allocator->Buffer = PFSVC_ALLOC(MaxSize);
if (!Allocator->Buffer) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
Allocator->UserSpecifiedBuffer = FALSE;
}
Allocator->BufferEnd = (PCHAR) Buffer + (ULONG_PTR) MaxSize;
Allocator->FreePointer = Buffer;
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
PVOID
PfSvStringAllocatorAllocate (
PPFSVC_STRING_ALLOCATOR Allocator,
ULONG NumBytes
)
/*++
Routine Description:
Returns a NumBytes size buffer allocated from Allocator.
This buffer should be freed back to the Allocator before the it is
uninitialized.
Arguments:
Allocator - Pointer to started allocator.
NumBytes - Number of bytes to allocate.
Return Value:
NULL or buffer allocated from Allocator.
--*/
{
PVOID ReturnChunk;
PPFSVC_STRING_ALLOCATION_HEADER AllocationHeader;
ULONG_PTR RealAllocationSize;
//
// We should not be trying to make allocations before we start the
// allocator.
//
PFSVC_ASSERT(Allocator->Buffer);
//
// Calculate how much we have to reserve from the buffer to make this
// allocation.
//
RealAllocationSize = 0;
RealAllocationSize += sizeof(PFSVC_STRING_ALLOCATION_HEADER);
RealAllocationSize += NumBytes;
RealAllocationSize = (ULONG_PTR) PF_ALIGN_UP(RealAllocationSize, _alignof(PFSVC_STRING_ALLOCATION_HEADER));
//
// We can't allocate from our buffer and have to go to the heap if
// - We've run out of space.
// - Allocation size is too big to fit in a USHORT.
// - It is a 0 sized allocation.
//
if (Allocator->FreePointer + RealAllocationSize > Allocator->BufferEnd ||
NumBytes > PFSVC_STRING_ALLOCATOR_MAX_BUFFER_ALLOCATION_SIZE ||
NumBytes == 0) {
//
// Hit the heap.
//
Allocator->MaxHeapAllocs++;
ReturnChunk = PFSVC_ALLOC(NumBytes);
if (ReturnChunk) {
Allocator->NumHeapAllocs++;
}
} else {
AllocationHeader = (PVOID) Allocator->FreePointer;
AllocationHeader->PrecedingAllocationSize = Allocator->LastAllocationSize;
PFSVC_ASSERT(RealAllocationSize < USHRT_MAX);
AllocationHeader->AllocationSize = (USHORT) RealAllocationSize;
Allocator->FreePointer += RealAllocationSize;
Allocator->LastAllocationSize = (USHORT) RealAllocationSize;
//
// The user's allocation comes right after the allocation header.
// (Using pointer arithmetic...)
//
ReturnChunk = AllocationHeader + 1;
}
return ReturnChunk;
}
VOID
PfSvStringAllocatorFree (
PPFSVC_STRING_ALLOCATOR Allocator,
PVOID Allocation
)
/*++
Routine Description:
Frees a string allocated from the allocator. This may not make it available
for use by further allocations from the allocator.
Arguments:
Allocator - Pointer to started allocator.
Allocation - Allocation to free.
Return Value:
None.
--*/
{
//
// Is this within the preallocated block?
//
if ((PUCHAR) Allocation >= Allocator->Buffer &&
(PUCHAR) Allocation < Allocator->BufferEnd) {
//
// Mark this chunk freed.
//
*((PWCHAR)Allocation) = PFSVC_STRING_ALLOCATOR_FREED_MAGIC;
} else {
//
// This chunk was allocated from heap.
//
PFSVC_ASSERT(Allocator->NumHeapAllocs && Allocator->MaxHeapAllocs);
Allocator->NumHeapAllocs--;
PFSVC_FREE(Allocation);
}
return;
}
VOID
PfSvStringAllocatorCleanup (
PPFSVC_STRING_ALLOCATOR Allocator
)
/*++
Routine Description:
Cleans up resources associated with the allocator. There should not be
any outstanding allocations from the allocator when this function is
called.
Arguments:
Allocator - Pointer to initialized allocator.
Return Value:
None.
--*/
{
PPFSVC_STRING_ALLOCATION_HEADER AllocationHeader;
PCHAR NextAllocationHeader;
WCHAR Magic;
if (Allocator->Buffer) {
#ifdef PFSVC_DBG
//
// Make sure all real heap allocations have been freed.
//
PFSVC_ASSERT(Allocator->NumHeapAllocs == 0);
//
// Make sure all allocated strings have been freed.
//
for (AllocationHeader = (PVOID) Allocator->Buffer;
(PCHAR) AllocationHeader < (PCHAR) Allocator->FreePointer;
AllocationHeader = (PVOID) NextAllocationHeader) {
Magic = *((PWCHAR)(AllocationHeader + 1));
PFSVC_ASSERT(Magic == PFSVC_STRING_ALLOCATOR_FREED_MAGIC);
//
// Calculate where the NextAllocationHeader will be.
//
NextAllocationHeader = (PCHAR) AllocationHeader +
(ULONG_PTR) AllocationHeader->AllocationSize;
}
#endif // PFSVC_DBG
//
// If the buffer was allocated by us (and not specified by
// the user), free it.
//
if (!Allocator->UserSpecifiedBuffer) {
PFSVC_FREE(Allocator->Buffer);
}
#ifdef PFSVC_DBG
//
// Setup the fields so if we try to make allocations after cleaning up
// an allocator we'll hit an assert.
//
Allocator->FreePointer = Allocator->Buffer;
Allocator->Buffer = NULL;
#endif // PFSVC_DBG
}
return;
}
//
// Routines that deal with section node structures.
//
VOID
PfSvCleanupSectionNode(
PPFSVC_SCENARIO_INFO ScenarioInfo,
PPFSVC_SECTION_NODE SectionNode
)
/*++
Routine Description:
This function cleans up a section node structure. It does not free
the structure itself.
Arguments:
ScenarioInfo - Pointer to scenario info.
SectionNode - Pointer to structure.
Return Value:
None.
--*/
{
PPFSVC_PAGE_NODE PageNode;
PLIST_ENTRY ListHead;
//
// If there is an allocated file name, free it.
//
if (SectionNode->FilePath) {
PfSvStringAllocatorFree(&ScenarioInfo->PathAllocator, SectionNode->FilePath);
SectionNode->FilePath = NULL;
}
//
// Free all the page nodes for this section.
//
while (!IsListEmpty(&SectionNode->PageList)) {
ListHead = RemoveHeadList(&SectionNode->PageList);
PageNode = CONTAINING_RECORD(ListHead, PFSVC_PAGE_NODE, PageLink);
PfSvChunkAllocatorFree(&ScenarioInfo->PageNodeAllocator, PageNode);
}
//
// We should not be on a volume node's list if we are being
// cleaned up.
//
PFSVC_ASSERT(IsListEmpty(&SectionNode->SectionVolumeLink));
}
//
// Routines used to sort scenario's section nodes.
//
DWORD
PfSvSortSectionNodesByFirstAccess(
PLIST_ENTRY SectionNodeList
)
/*++
Routine Description:
This routine will sort the specified section node list by first
access using NewSectionIndex and OrgSectionIndex of the section
nodes.
Arguments:
SectionNodeList - Pointer to list of section nodes to be sorted.
Return Value:
Win32 error code.
--*/
{
PFSV_SECTNODE_PRIORITY_QUEUE SortQueue;
PLIST_ENTRY SectHead;
PPFSVC_SECTION_NODE SectionNode;
DWORD ErrorCode;
//
// Initialize locals.
//
PfSvInitializeSectNodePriorityQueue(&SortQueue);
DBGPR((PFID,PFSTRC,"PFSVC: SortByFirstAccess(%p)\n", SectionNodeList));
//
// We have to sort the section nodes by first access. Remove
// section nodes from the scenario list and put them on a priority
// queue. [Bummer, it may have been a little faster if we had
// built a binary tree and traversed that in the rest of the code]
//
while (!IsListEmpty(SectionNodeList)) {
//
// The section list is sorted by name. It is more likely that
// we also accessed files by name. So to make the priority
// queue act better in such cases, remove from the tail of the
// list to insert into the priority queue.
//
SectHead = RemoveTailList(SectionNodeList);
SectionNode = CONTAINING_RECORD(SectHead,
PFSVC_SECTION_NODE,
SectionLink);
PfSvInsertSectNodePriorityQueue(&SortQueue, SectionNode);
}
//
// Remove the section nodes from the priority queue sorted by
// first access and put them to the tail of the section node list.
//
while (SectionNode = PfSvRemoveMinSectNodePriorityQueue(&SortQueue)) {
InsertTailList(SectionNodeList, &SectionNode->SectionLink);
}
ErrorCode = ERROR_SUCCESS;
DBGPR((PFID,PFSTRC,"PFSVC: SortByFirstAccess(%p)=%x\n", SectionNodeList, ErrorCode));
return ErrorCode;
}
PFSV_SECTION_NODE_COMPARISON_RESULT
FASTCALL
PfSvSectionNodeComparisonRoutine(
PPFSVC_SECTION_NODE Element1,
PPFSVC_SECTION_NODE Element2
)
/*++
Routine Description:
This routine is called to compare to elements when sorting the
section nodes array by first access.
Arguments:
Element1, Element2 - The two elements to compare.
Return Value:
PFSVC_SECTION_NODE_COMPARISON_RESULT
--*/
{
//
// First compare first-access index in the new trace.
//
if (Element1->NewSectionIndex < Element2->NewSectionIndex) {
return PfSvSectNode1LessThanSectNode2;
} else if (Element1->NewSectionIndex > Element2->NewSectionIndex) {
return PfSvSectNode1GreaterThanSectNode2;
} else {
//
// Next compare first-access index in the current scenario
// file.
//
if (Element1->OrgSectionIndex < Element2->OrgSectionIndex) {
return PfSvSectNode1LessThanSectNode2;
} else if (Element1->OrgSectionIndex > Element2->OrgSectionIndex) {
return PfSvSectNode1GreaterThanSectNode2;
} else {
return PfSvSectNode1EqualToSectNode2;
}
}
}
//
// Routines that implement a priority queue used to sort section nodes
// for a scenario.
//
VOID
PfSvInitializeSectNodePriorityQueue(
PPFSV_SECTNODE_PRIORITY_QUEUE PriorityQueue
)
/*++
Routine Description:
Initialize a section node priority queue.
Arguments:
PriorityQueue - Pointer to the queue.
Return Value:
None.
--*/
{
PriorityQueue->Head = NULL;
}
VOID
PfSvInsertSectNodePriorityQueue(
PPFSV_SECTNODE_PRIORITY_QUEUE PriorityQueue,
PPFSVC_SECTION_NODE NewElement
)
/*++
Routine Description:
Insert a section node in the a section node priority queue.
Arguments:
PriorityQueue - Pointer to the queue.
NewElement - Pointer to new element.
Return Value:
None.
--*/
{
PPFSVC_SECTION_NODE *CurrentPosition;
//
// Initialize the link fields of NewElement.
//
NewElement->LeftChild = NULL;
NewElement->RightChild = NULL;
//
// If the queue is empty, insert this at the head.
//
if (PriorityQueue->Head == NULL) {
PriorityQueue->Head = NewElement;
return;
}
//
// If we are less than the current min element, put us at the
// head.
//
if (PfSvSectionNodeComparisonRoutine(NewElement, PriorityQueue->Head) <= 0) {
NewElement->RightChild = PriorityQueue->Head;
PriorityQueue->Head = NewElement;
return;
}
//
// Insert this node into the tree rooted at the right child of the
// head node.
//
CurrentPosition = &PriorityQueue->Head->RightChild;
while (*CurrentPosition) {
if (PfSvSectionNodeComparisonRoutine(NewElement, *CurrentPosition) <= 0) {
CurrentPosition = &(*CurrentPosition)->LeftChild;
} else {
CurrentPosition = &(*CurrentPosition)->RightChild;
}
}
//
// We found the place.
//
*CurrentPosition = NewElement;
}
PPFSVC_SECTION_NODE
PfSvRemoveMinSectNodePriorityQueue(
PPFSV_SECTNODE_PRIORITY_QUEUE PriorityQueue
)
/*++
Routine Description:
Remove the head element of the queue.
Arguments:
PriorityQueue - Pointer to the queue.
Return Value:
Pointer to head element of the queue.
NULL if queue is empty.
--*/
{
PPFSVC_SECTION_NODE *CurrentPosition;
PPFSVC_SECTION_NODE OrgHeadNode;
PPFSVC_SECTION_NODE NewHeadNode;
PPFSVC_SECTION_NODE TreeRoot;
//
// If the queue is empty return NULL.
//
if (PriorityQueue->Head == NULL) {
return NULL;
}
//
// Save pointer to original head node.
//
OrgHeadNode = PriorityQueue->Head;
//
// Find the minimum element of the tree rooted at the right child
// of the head node. CurrentPosition points to the link of the
// parent to the smaller child.
//
TreeRoot = OrgHeadNode->RightChild;
CurrentPosition = &TreeRoot;
while (*CurrentPosition && (*CurrentPosition)->LeftChild) {
CurrentPosition = &(*CurrentPosition)->LeftChild;
}
NewHeadNode = *CurrentPosition;
//
// Check if there is really a new head node that we have to remove
// from its current position.
//
if (NewHeadNode) {
//
// We are removing this node to put it at the head. In its
// place, we'll put its right child. Since we know that this
// node does not have a left child, that's all we have to do.
//
*CurrentPosition = NewHeadNode->RightChild;
//
// Set the tree rooted at the head's right child.
//
NewHeadNode->RightChild = TreeRoot;
}
//
// Set the new head.
//
PriorityQueue->Head = NewHeadNode;
//
// Return the original head node.
//
return OrgHeadNode;
}
//
// Implementation of the Nt path to Dos path translation API.
//
DWORD
PfSvBuildNtPathTranslationList(
PNTPATH_TRANSLATION_LIST *NtPathTranslationList
)
/*++
Routine Description:
This routine is called to build a list that can be used to
translate Nt paths to Dos paths. If successful, the returned list
should be freed by calling PfSvFreeNtPathTranslationList.
Arguments:
TranslationList - Pointer to where a pointer to the built
translation list is going to be put.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
ULONG VolumeNameLength;
ULONG VolumeNameMaxLength;
PWCHAR VolumeName;
ULONG NTDevicePathMaxLength;
ULONG NTDevicePathLength;
PWCHAR NTDevicePath;
HANDLE FindVolumeHandle;
ULONG RequiredLength;
ULONG VolumePathNamesLength;
WCHAR *VolumePathNames;
ULONG MountPathNameLength;
WCHAR *MountPathName;
ULONG ShortestMountPathLength;
WCHAR *ShortestMountPathName;
ULONG NumMountPoints;
ULONG NumResizes;
BOOL Result;
ULONG NumChars;
ULONG Length;
PNTPATH_TRANSLATION_LIST TranslationList;
PNTPATH_TRANSLATION_ENTRY TranslationEntry;
PNTPATH_TRANSLATION_ENTRY NextTranslationEntry;
ULONG AllocationSize;
PUCHAR DestinationPointer;
ULONG CopySize;
PLIST_ENTRY HeadEntry;
PLIST_ENTRY NextEntry;
PLIST_ENTRY InsertPosition;
BOOLEAN TrimmedTerminatingSlash;
//
// Initialize locals.
//
FindVolumeHandle = INVALID_HANDLE_VALUE;
VolumePathNames = NULL;
VolumePathNamesLength = 0;
VolumeName = NULL;
VolumeNameMaxLength = 0;
NTDevicePath = NULL;
NTDevicePathMaxLength = 0;
TranslationList = NULL;
DBGPR((PFID,PFTRC,"PFSVC: BuildTransList()\n"));
//
// Allocate intermediate buffers.
//
Length = MAX_PATH + 1;
VolumeName = PFSVC_ALLOC(Length * sizeof(WCHAR));
NTDevicePath = PFSVC_ALLOC(Length * sizeof(WCHAR));
if (!VolumeName || !NTDevicePath) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
VolumeNameMaxLength = Length;
NTDevicePathMaxLength = Length;
//
// Allocate and initialize a translation list.
//
TranslationList = PFSVC_ALLOC(sizeof(NTPATH_TRANSLATION_LIST));
if (!TranslationList) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
InitializeListHead(TranslationList);
//
// Start enumerating the volumes.
//
FindVolumeHandle = FindFirstVolume(VolumeName, VolumeNameMaxLength);
if (FindVolumeHandle == INVALID_HANDLE_VALUE) {
ErrorCode = GetLastError();
goto cleanup;
}
do {
//
// Update volume length after the FindFirst/NextVolume call.
//
VolumeNameLength = wcslen(VolumeName);
//
// Get list of where this volume is mounted.
//
NumResizes = 0;
do {
Result = GetVolumePathNamesForVolumeName(VolumeName,
VolumePathNames,
VolumePathNamesLength,
&RequiredLength);
if (Result) {
//
// We got the mount points.
//
break;
}
//
// Check why we failed.
//
ErrorCode = GetLastError();
if (ErrorCode != ERROR_MORE_DATA) {
//
// A real error...
//
goto cleanup;
}
//
// We need to increase the size of our buffer. If there is
// an existing buffer, first free it.
//
if (VolumePathNames) {
PFSVC_FREE(VolumePathNames);
VolumePathNames = NULL;
VolumePathNamesLength = 0;
}
//
// Try to allocate a new buffer.
//
VolumePathNames = PFSVC_ALLOC(RequiredLength * sizeof(WCHAR));
if (!VolumePathNames) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
VolumePathNamesLength = RequiredLength;
//
// Retry with the resized buffer but make sure we don't
// loop forever!
//
NumResizes++;
if (NumResizes > 1000) {
ErrorCode = ERROR_INVALID_FUNCTION;
goto cleanup;
}
} while (TRUE);
//
// Loop through the mount points to find the shortest one. It
// is possible that the depth of it is more.
//
MountPathName = VolumePathNames;
NumMountPoints = 0;
ShortestMountPathName = NULL;
ShortestMountPathLength = ULONG_MAX;
while (*MountPathName) {
MountPathNameLength = wcslen(MountPathName);
if (MountPathNameLength < ShortestMountPathLength) {
ShortestMountPathName = MountPathName;
ShortestMountPathLength = MountPathNameLength;
}
NumMountPoints++;
//
// Update the pointer to next mount point path.
//
MountPathName += MountPathNameLength + 1;
}
//
// Check if we got a mount point path.
//
if (ShortestMountPathName == NULL) {
//
// Skip this volume.
//
continue;
}
//
// Remove the terminating slash if there is one.
//
if (ShortestMountPathName[ShortestMountPathLength - 1] == L'\\') {
ShortestMountPathName[ShortestMountPathLength - 1] = 0;
ShortestMountPathLength--;
}
//
// Get NT device that is the target of the volume link in
// Win32 object namespace. We get the dos device name by
// trimming the first 4 characters [i.e. \\?\] of the
// VolumeName. Also trim the \ at the very end of the volume
// name.
//
if (VolumeNameLength <= 4) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
if (VolumeName[VolumeNameLength - 1] == L'\\') {
VolumeName[VolumeNameLength - 1] = 0;
TrimmedTerminatingSlash = TRUE;
} else {
TrimmedTerminatingSlash = FALSE;
}
NumChars = QueryDosDevice(&VolumeName[4],
NTDevicePath,
NTDevicePathMaxLength);
if (TrimmedTerminatingSlash) {
VolumeName[VolumeNameLength - 1] = L'\\';
}
if (NumChars == 0) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// We are interested only in the current mapping.
//
NTDevicePath[NTDevicePathMaxLength - 1] = 0;
NTDevicePathLength = wcslen(NTDevicePath);
if (NTDevicePathLength == 0) {
//
// Skip this volume.
//
continue;
}
//
// Remove terminating slash if there is one.
//
if (NTDevicePath[NTDevicePathLength - 1] == L'\\') {
NTDevicePath[NTDevicePathLength - 1] = 0;
NTDevicePathLength--;
}
//
// Allocate a translation entry big enough to contain both
// path names and the volume string.
//
AllocationSize = sizeof(NTPATH_TRANSLATION_ENTRY);
AllocationSize += (ShortestMountPathLength + 1) * sizeof(WCHAR);
AllocationSize += (NTDevicePathLength + 1) * sizeof(WCHAR);
AllocationSize += (VolumeNameLength + 1) * sizeof(WCHAR);
TranslationEntry = PFSVC_ALLOC(AllocationSize);
if (!TranslationEntry) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
DestinationPointer = (PUCHAR) TranslationEntry;
DestinationPointer += sizeof(NTPATH_TRANSLATION_ENTRY);
//
// Copy the NT path name and the terminating NUL.
//
TranslationEntry->NtPrefix = (PVOID) DestinationPointer;
TranslationEntry->NtPrefixLength = NTDevicePathLength;
CopySize = (NTDevicePathLength + 1) * sizeof(WCHAR);
RtlCopyMemory(DestinationPointer, NTDevicePath, CopySize);
DestinationPointer += CopySize;
//
// Copy the DOS mount point name and the terminating NUL.
//
TranslationEntry->DosPrefix = (PVOID) DestinationPointer;
TranslationEntry->DosPrefixLength = ShortestMountPathLength;
CopySize = (ShortestMountPathLength + 1) * sizeof(WCHAR);
RtlCopyMemory(DestinationPointer, ShortestMountPathName, CopySize);
DestinationPointer += CopySize;
//
// Copy the volume name and the terminating NUL.
//
TranslationEntry->VolumeName = (PVOID) DestinationPointer;
TranslationEntry->VolumeNameLength = VolumeNameLength;
CopySize = (VolumeNameLength + 1) * sizeof(WCHAR);
RtlCopyMemory(DestinationPointer, VolumeName, CopySize);
DestinationPointer += CopySize;
//
// Find the position for this entry in the sorted translation
// list.
//
HeadEntry = TranslationList;
NextEntry = HeadEntry->Flink;
InsertPosition = HeadEntry;
while (NextEntry != HeadEntry) {
NextTranslationEntry = CONTAINING_RECORD(NextEntry,
NTPATH_TRANSLATION_ENTRY,
Link);
if (_wcsicmp(TranslationEntry->NtPrefix,
NextTranslationEntry->NtPrefix) <= 0) {
break;
}
InsertPosition = NextEntry;
NextEntry = NextEntry->Flink;
}
//
// Insert it after the found position.
//
InsertHeadList(InsertPosition, &TranslationEntry->Link);
} while (FindNextVolume(FindVolumeHandle, VolumeName, VolumeNameMaxLength));
//
// We will break out of the loop when FindNextVolume does not
// return success. Check if it failed for a reason other than that
// we have enumerated all volumes.
//
ErrorCode = GetLastError();
if (ErrorCode != ERROR_NO_MORE_FILES) {
goto cleanup;
}
//
// Set return value.
//
*NtPathTranslationList = TranslationList;
ErrorCode = ERROR_SUCCESS;
cleanup:
if (FindVolumeHandle != INVALID_HANDLE_VALUE) {
FindVolumeClose(FindVolumeHandle);
}
if (VolumePathNames) {
PFSVC_FREE(VolumePathNames);
}
if (ErrorCode != ERROR_SUCCESS) {
if (TranslationList) {
PfSvFreeNtPathTranslationList(TranslationList);
}
}
if (VolumeName) {
PFSVC_FREE(VolumeName);
}
if (NTDevicePath) {
PFSVC_FREE(NTDevicePath);
}
DBGPR((PFID,PFTRC,"PFSVC: BuildTransList()=%x,%p\n", ErrorCode, TranslationList));
return ErrorCode;
}
VOID
PfSvFreeNtPathTranslationList(
PNTPATH_TRANSLATION_LIST TranslationList
)
/*++
Routine Description:
This routine is called to free a translation list returned by
PfSvBuildNtPathTranslationList.
Arguments:
TranslationList - Pointer to list to free.
Return Value:
None.
--*/
{
PLIST_ENTRY HeadEntry;
PNTPATH_TRANSLATION_ENTRY TranslationEntry;
DBGPR((PFID,PFTRC,"PFSVC: FreeTransList(%p)\n", TranslationList));
//
// Free all entries in the list.
//
while (!IsListEmpty(TranslationList)) {
HeadEntry = RemoveHeadList(TranslationList);
TranslationEntry = CONTAINING_RECORD(HeadEntry,
NTPATH_TRANSLATION_ENTRY,
Link);
PFSVC_FREE(TranslationEntry);
}
//
// Free the list itself.
//
PFSVC_FREE(TranslationList);
}
DWORD
PfSvTranslateNtPath(
PNTPATH_TRANSLATION_LIST TranslationList,
WCHAR *NtPath,
ULONG NtPathLength,
PWCHAR *DosPathBuffer,
PULONG DosPathBufferSize
)
/*++
Routine Description:
This routine is called to free a translation list returned by
PfSvBuildNtPathTranslationList. Note that it may not be possible to
translate all Nt path's to a Dos path.
Arguments:
TranslationList - Pointer to list built by PfSvBuildNtPathTranslationList.
NtPath - Path to translate.
NtPathLength - Length of NtPath in characters excluding terminating NUL.
DosPathBuffer - Buffer to put the translation into. If it is NULL
or not big enough it will get reallocated. If a buffer is passed
in, it should be allocated by PFSVC_ALLOC. It is the callers
responsibility to free the buffer with PFSVC_FREE when done.
DosPathBufferSize - Size of DosPathBuffer in bytes. Updated if the
buffer is reallocated.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
PLIST_ENTRY HeadEntry;
PLIST_ENTRY NextEntry;
PNTPATH_TRANSLATION_ENTRY CurrentTranslationEntry;
PNTPATH_TRANSLATION_ENTRY FoundTranslationEntry;
PFSV_PREFIX_COMPARISON_RESULT ComparisonResult;
ULONG RequiredSize;
//
// Initialize locals.
//
FoundTranslationEntry = NULL;
DBGPR((PFID,PFPATH,"PFSVC: TranslateNtPath(%ws)\n", NtPath));
//
// Walk through the sorted translation list to find an entry that
// applies.
//
HeadEntry = TranslationList;
NextEntry = HeadEntry->Flink;
while (NextEntry != HeadEntry) {
CurrentTranslationEntry = CONTAINING_RECORD(NextEntry,
NTPATH_TRANSLATION_ENTRY,
Link);
//
// Do a case insensitive comparison.
//
ComparisonResult = PfSvComparePrefix(NtPath,
NtPathLength,
CurrentTranslationEntry->NtPrefix,
CurrentTranslationEntry->NtPrefixLength,
FALSE);
if (ComparisonResult == PfSvPrefixIdentical) {
//
// Check to see if the character in NtPath after the
// prefix is a path seperator [i.e. '\']. Otherwise we may
// match \Device\CdRom10\DirName\FileName to \Device\Cdrom1.
//
if (NtPathLength == CurrentTranslationEntry->NtPrefixLength ||
NtPath[CurrentTranslationEntry->NtPrefixLength] == L'\\') {
//
// We found a translation entry that applies to us.
//
FoundTranslationEntry = CurrentTranslationEntry;
break;
}
} else if (ComparisonResult == PfSvPrefixGreaterThan) {
//
// Since the translation list is sorted in increasing
// order, following entries will also be greater than
// NtPath.
//
FoundTranslationEntry = NULL;
break;
}
//
// Continue looking for a matching prefix.
//
NextEntry = NextEntry->Flink;
}
//
// If we could not find an entry that applies we cannot translate
// the path.
//
if (FoundTranslationEntry == NULL) {
ErrorCode = ERROR_PATH_NOT_FOUND;
goto cleanup;
}
//
// Calculate required size: We will replace the NtPrefix with
// DosPrefix. Don't forget the terminating NUL character.
//
RequiredSize = (NtPathLength + 1) * sizeof(WCHAR);
RequiredSize += (FoundTranslationEntry->DosPrefixLength * sizeof(WCHAR));
RequiredSize -= (FoundTranslationEntry->NtPrefixLength * sizeof(WCHAR));
if (RequiredSize > (*DosPathBufferSize)) {
//
// Reallocate the buffer. First free it if there is one.
//
if (*DosPathBufferSize) {
PFSVC_ASSERT(*DosPathBuffer);
PFSVC_FREE(*DosPathBuffer);
(*DosPathBuffer) = NULL;
(*DosPathBufferSize) = 0;
}
PFSVC_ASSERT((*DosPathBuffer) == NULL);
(*DosPathBuffer) = PFSVC_ALLOC(RequiredSize);
if (!(*DosPathBuffer)) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
(*DosPathBufferSize) = RequiredSize;
}
//
// We should have enough room now.
//
PFSVC_ASSERT(RequiredSize <= (*DosPathBufferSize));
//
// Copy the DosPrefix.
//
wcscpy((*DosPathBuffer), FoundTranslationEntry->DosPrefix);
//
// Concatenate the remaining path.
//
wcscat((*DosPathBuffer), NtPath + CurrentTranslationEntry->NtPrefixLength);
//
// We are done.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFPATH,"PFSVC: TranslateNtPath(%ws)=%x,%ws\n",
NtPath,ErrorCode,(ErrorCode==ERROR_SUCCESS)?(*DosPathBuffer):L"Failed"));
return ErrorCode;
}
//
// Implementation of the path list API.
//
VOID
PfSvInitializePathList(
PPFSVC_PATH_LIST PathList,
OPTIONAL IN PPFSVC_STRING_ALLOCATOR PathAllocator,
IN BOOLEAN CaseSensitive
)
/*++
Routine Description:
This function initializes a path list structure.
Arguments:
PathList - Pointer to structure.
PathAllocator - If specified path allocations will be made from it.
CaseSenstive - Whether list will be case senstive.
Return Value:
None.
--*/
{
InitializeListHead(&PathList->InOrderList);
InitializeListHead(&PathList->SortedList);
PathList->NumPaths = 0;
PathList->TotalLength = 0;
PathList->Allocator = PathAllocator;
PathList->CaseSensitive = CaseSensitive;
}
VOID
PfSvCleanupPathList(
PPFSVC_PATH_LIST PathList
)
/*++
Routine Description:
This function cleans up a path list structure. It does not free
the structure itself. The structure should have been initialized
by PfSvInitializePathList.
Arguments:
PathList - Pointer to structure.
Return Value:
None.
--*/
{
PLIST_ENTRY ListEntry;
PPFSVC_PATH Path;
while (!IsListEmpty(&PathList->InOrderList)) {
PFSVC_ASSERT(PathList->NumPaths);
PathList->NumPaths--;
ListEntry = RemoveHeadList(&PathList->InOrderList);
Path = CONTAINING_RECORD(ListEntry,
PFSVC_PATH,
InOrderLink);
if (PathList->Allocator) {
PfSvStringAllocatorFree(PathList->Allocator, Path);
} else {
PFSVC_FREE(Path);
}
}
}
BOOLEAN
PfSvIsInPathList(
PPFSVC_PATH_LIST PathList,
WCHAR *Path,
ULONG PathLength
)
/*++
Routine Description:
This function checks if the specified path is already in the path
list.
Arguments:
PathList - Pointer to list.
Path - Path to look for. Does not have to be NUL terminated.
PathLength - Length of Path in characters excluding NUL if there
is one.
Return Value:
Win32 error code.
--*/
{
PLIST_ENTRY HeadEntry;
PLIST_ENTRY NextEntry;
PPFSVC_PATH PathEntry;
INT ComparisonResult;
BOOLEAN PathIsInPathList;
//
// Walk through the list.
//
HeadEntry = &PathList->SortedList;
NextEntry = HeadEntry->Flink;
while (NextEntry != HeadEntry) {
PathEntry = CONTAINING_RECORD(NextEntry,
PFSVC_PATH,
SortedLink);
if (PathList->CaseSensitive) {
ComparisonResult = wcsncmp(Path,
PathEntry->Path,
PathLength);
} else {
ComparisonResult = _wcsnicmp(Path,
PathEntry->Path,
PathLength);
}
//
// Adjust comparison result so we don't match "abcde" to
// "abcdefg". If string comparison says the first PathLength
// characters match, check to see if PathEntry's length is
// longer, which would make it "greater" than the new path.
//
if (ComparisonResult == 0 && PathEntry->Length != PathLength) {
//
// The string comparison would not say The path entry's
// path is equal to path if its length was smaller.
//
PFSVC_ASSERT(PathEntry->Length > PathLength);
//
// Path is actually less than this path entry.
//
ComparisonResult = -1;
}
//
// Based on comparison result determine what to do:
//
if (ComparisonResult == 0) {
//
// We found it.
//
PathIsInPathList = TRUE;
goto cleanup;
} else if (ComparisonResult < 0) {
//
// We will be less than the rest of the strings in the
// list after this too.
//
PathIsInPathList = FALSE;
goto cleanup;
}
//
// Continue looking for the path or an available position.
//
NextEntry = NextEntry->Flink;
}
//
// If we came here, we could not find the path in the list.
//
PathIsInPathList = FALSE;
cleanup:
return PathIsInPathList;
}
DWORD
PfSvAddToPathList(
PPFSVC_PATH_LIST PathList,
WCHAR *Path,
ULONG PathLength
)
/*++
Routine Description:
This function adds a path to a path list. If the path already
exists in the list, it is not added again.
Arguments:
PathList - Pointer to list.
Path - Path to add. Does not need to be NUL terminated.
PathLength - Length of Path in characters excluding NUL if there
is one.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
PLIST_ENTRY HeadEntry;
PLIST_ENTRY NextEntry;
PPFSVC_PATH PathEntry;
PPFSVC_PATH NewPathEntry;
INT ComparisonResult;
ULONG AllocationSize;
//
// Initialize locals.
//
NewPathEntry = NULL;
//
// Walk through the list to check if path is already in the list,
// or to find where it should be so we can insert it there.
//
HeadEntry = &PathList->SortedList;
NextEntry = HeadEntry->Flink;
while (NextEntry != HeadEntry) {
PathEntry = CONTAINING_RECORD(NextEntry,
PFSVC_PATH,
SortedLink);
if (PathList->CaseSensitive) {
ComparisonResult = wcsncmp(Path,
PathEntry->Path,
PathLength);
} else {
ComparisonResult = _wcsnicmp(Path,
PathEntry->Path,
PathLength);
}
//
// Adjust comparison result so we don't match "abcde" to
// "abcdefg". If string comparison says the first PathLength
// characters match, check to see if PathEntry's length is
// longer, which would make it "greater" than the new path.
//
if (ComparisonResult == 0 && PathEntry->Length != PathLength) {
//
// The string comparison would not say The path entry's
// path is equal to path if its length was smaller.
//
PFSVC_ASSERT(PathEntry->Length > PathLength);
//
// Path is actually less than this path entry.
//
ComparisonResult = -1;
}
//
// Based on comparison result determine what to do:
//
if (ComparisonResult == 0) {
//
// The path already exists in the list.
//
ErrorCode = ERROR_SUCCESS;
goto cleanup;
} else if (ComparisonResult < 0) {
//
// We will be less than the rest of the strings in the
// list after this too. We should be inserted before this
// one.
//
break;
}
//
// Continue looking for the path or an available position.
//
NextEntry = NextEntry->Flink;
}
//
// We will insert the path before NextEntry. First create an entry
// we can insert.
//
AllocationSize = sizeof(PFSVC_PATH);
AllocationSize += PathLength * sizeof(WCHAR);
//
// Note that PFSVC_PATH already contains space for the terminating
// NUL character.
//
if (PathList->Allocator) {
NewPathEntry = PfSvStringAllocatorAllocate(PathList->Allocator, AllocationSize);
} else {
NewPathEntry = PFSVC_ALLOC(AllocationSize);
}
if (!NewPathEntry) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
//
// Copy path and terminate it.
//
NewPathEntry->Length = PathLength;
RtlCopyMemory(NewPathEntry->Path,
Path,
PathLength * sizeof(WCHAR));
NewPathEntry->Path[PathLength] = 0;
//
// Insert it into the sorted list before the current entry.
//
InsertTailList(NextEntry, &NewPathEntry->SortedLink);
//
// Insert it at the end of in-order list.
//
InsertTailList(&PathList->InOrderList, &NewPathEntry->InOrderLink);
PathList->NumPaths++;
PathList->TotalLength += NewPathEntry->Length;
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ErrorCode != ERROR_SUCCESS) {
if (NewPathEntry) {
if (PathList->Allocator) {
PfSvStringAllocatorFree(PathList->Allocator, NewPathEntry);
} else {
PFSVC_FREE(NewPathEntry);
}
}
}
return ErrorCode;
}
PPFSVC_PATH
PfSvGetNextPathSorted (
PPFSVC_PATH_LIST PathList,
PPFSVC_PATH CurrentPath
)
/*++
Routine Description:
This function is used to walk through paths in a path list in
lexically sorted order.
Arguments:
PathList - Pointer to list.
CurrentPath - The current path entry. The function will return the
next entry in the list. If this is NULL, the first entry in the
list is returned.
Return Value:
NULL - There are no more entries in the list.
or Pointer to next path in the list.
--*/
{
PLIST_ENTRY EndOfList;
PLIST_ENTRY NextEntry;
PPFSVC_PATH NextPath;
//
// Initialize locals.
//
EndOfList = &PathList->SortedList;
//
// Determine NextEntry based on whether CurrentPath is specified.
//
if (CurrentPath) {
NextEntry = CurrentPath->SortedLink.Flink;
} else {
NextEntry = PathList->SortedList.Flink;
}
//
// Check if the NextEntry points to the end of list.
//
if (NextEntry == EndOfList) {
NextPath = NULL;
} else {
NextPath = CONTAINING_RECORD(NextEntry,
PFSVC_PATH,
SortedLink);
}
return NextPath;
}
PPFSVC_PATH
PfSvGetNextPathInOrder (
PPFSVC_PATH_LIST PathList,
PPFSVC_PATH CurrentPath
)
/*++
Routine Description:
This function is used to walk through paths in a path list in
the order they were inserted into the list.
Arguments:
PathList - Pointer to list.
CurrentPath - The current path entry. The function will return the
next entry in the list. If this is NULL, the first entry in the
list is returned.
Return Value:
NULL - There are no more entries in the list.
or Pointer to next path in the list.
--*/
{
PLIST_ENTRY EndOfList;
PLIST_ENTRY NextEntry;
PPFSVC_PATH NextPath;
//
// Initialize locals.
//
EndOfList = &PathList->InOrderList;
//
// Determine NextEntry based on whether CurrentPath is specified.
//
if (CurrentPath) {
NextEntry = CurrentPath->InOrderLink.Flink;
} else {
NextEntry = PathList->InOrderList.Flink;
}
//
// Check if the NextEntry points to the end of list.
//
if (NextEntry == EndOfList) {
NextPath = NULL;
} else {
NextPath = CONTAINING_RECORD(NextEntry,
PFSVC_PATH,
InOrderLink);
}
return NextPath;
}
//
// Routines to build the list of files accessed by the boot loader.
//
DWORD
PfSvBuildBootLoaderFilesList (
PPFSVC_PATH_LIST PathList
)
/*++
Routine Description:
This function attempts to add the list of files loaded in the boot
loader to the specified file list.
Arguments:
PathList - Pointer to initialized list.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
SC_HANDLE ScHandle;
SC_HANDLE ServiceHandle;
LPENUM_SERVICE_STATUS_PROCESS EnumBuffer;
LPENUM_SERVICE_STATUS_PROCESS ServiceInfo;
ULONG EnumBufferMaxSize;
ULONG NumResizes;
BOOL Result;
ULONG RequiredAdditionalSize;
ULONG RequiredSize;
ULONG NumServicesEnumerated;
ULONG ResumeHandle;
ULONG ServiceIdx;
LPQUERY_SERVICE_CONFIG ServiceConfigBuffer;
ULONG ServiceConfigBufferMaxSize;
WCHAR FilePath[MAX_PATH + 1];
ULONG FilePathLength;
ULONG SystemDirLength;
ULONG RequiredLength;
WCHAR *KernelName;
WCHAR *HalName;
WCHAR *SystemHive;
WCHAR *SoftwareHive;
//
// Initialize locals.
//
ScHandle = NULL;
ServiceHandle = NULL;
EnumBuffer = NULL;
EnumBufferMaxSize = 0;
NumServicesEnumerated = 0;
ServiceConfigBuffer = NULL;
ServiceConfigBufferMaxSize = 0;
KernelName = L"ntoskrnl.exe";
HalName = L"hal.dll";
SystemHive = L"config\\system";
SoftwareHive = L"config\\software";
//
// Add kernel & hal to known files list:
//
//
// Get path to system directory.
//
SystemDirLength = GetSystemDirectory(FilePath, MAX_PATH);
if (!SystemDirLength) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Append a trailing \.
//
if (SystemDirLength + 1 < MAX_PATH) {
FilePath[SystemDirLength] = '\\';
SystemDirLength++;
FilePath[SystemDirLength] = 0;
} else {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Append kernel name and add it to the list.
//
FilePathLength = SystemDirLength;
FilePathLength += wcslen(KernelName);
if (FilePathLength < MAX_PATH) {
wcscat(FilePath, KernelName);
ErrorCode = PfSvAddBootImageAndImportsToList(PathList,
FilePath,
FilePathLength);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
} else {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Roll FilePath back to system directory. Append hal name and add
// it to the list.
//
FilePathLength = SystemDirLength;
FilePathLength += wcslen(HalName);
if (FilePathLength < MAX_PATH) {
FilePath[SystemDirLength] = 0;
wcscat(FilePath, HalName);
ErrorCode = PfSvAddBootImageAndImportsToList(PathList,
FilePath,
FilePathLength);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
} else {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Roll FilePath back to system directory. Append system hive path
// and add it to the list.
//
FilePathLength = SystemDirLength;
FilePathLength += wcslen(SystemHive);
if (FilePathLength < MAX_PATH) {
FilePath[SystemDirLength] = 0;
wcscat(FilePath, SystemHive);
ErrorCode = PfSvAddToPathList(PathList,
FilePath,
FilePathLength);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
} else {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Roll FilePath back to system directory. Append software hive path
// and add it to the list.
//
FilePathLength = SystemDirLength;
FilePathLength += wcslen(SoftwareHive);
if (FilePathLength < MAX_PATH) {
FilePath[SystemDirLength] = 0;
wcscat(FilePath, SoftwareHive);
ErrorCode = PfSvAddToPathList(PathList,
FilePath,
FilePathLength);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
} else {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Note that we will use FilePath & FilePathLength to add the
// software hive after we add all the other boot loader files. The
// software hive is not accessed in the boot loader, but during
// boot. It is not put into the boot scenario file, however. We
// don't want to mix it in with the boot loader files, so we don't
// hurt the boot loader performance.
//
//
// Add file paths for NLS data & fonts loaded by the boot loader.
//
PfSvGetBootLoaderNlsFileNames(PathList);
//
// Open service controller.
//
// FUTURE-2002/03/29-ScottMa -- The dwDesiredAccess parameter may be able
// to be set to a lesser set of priveleges.
ScHandle = OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
if (ScHandle == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Get the list of boot services we are interested in.
//
NumResizes = 0;
do {
//
// We want to get all of services at one call.
//
ResumeHandle = 0;
Result = EnumServicesStatusEx (ScHandle,
SC_ENUM_PROCESS_INFO,
SERVICE_DRIVER,
SERVICE_ACTIVE,
(PVOID)EnumBuffer,
EnumBufferMaxSize,
&RequiredAdditionalSize,
&NumServicesEnumerated,
&ResumeHandle,
NULL);
if (Result) {
//
// We got it.
//
break;
}
//
// Check why our call failed.
//
ErrorCode = GetLastError();
//
// If we failed for some other reason than that our buffer was
// too small, we cannot go on.
//
if (ErrorCode != ERROR_MORE_DATA) {
goto cleanup;
}
//
// Free the old buffer if it exists, and allocate a bigger one.
//
RequiredSize = EnumBufferMaxSize + RequiredAdditionalSize;
if (EnumBuffer) {
PFSVC_FREE(EnumBuffer);
EnumBuffer = NULL;
EnumBufferMaxSize = 0;
}
EnumBuffer = PFSVC_ALLOC(RequiredSize);
if (EnumBuffer == NULL) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
EnumBufferMaxSize = RequiredSize;
//
// Make sure we don't loop for ever.
//
NumResizes++;
if (NumResizes > 100) {
ErrorCode = ERROR_INVALID_FUNCTION;
goto cleanup;
}
} while (TRUE);
//
// Identify the enumerated services that may be loaded by the boot
// loader.
//
for (ServiceIdx = 0; ServiceIdx < NumServicesEnumerated; ServiceIdx++) {
ServiceInfo = &EnumBuffer[ServiceIdx];
//
// Open the service to get its configuration info.
//
ServiceHandle = OpenService(ScHandle,
ServiceInfo->lpServiceName,
SERVICE_QUERY_CONFIG);
if (ServiceHandle == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
//
// Query service configuration.
//
NumResizes = 0;
do {
Result = QueryServiceConfig(ServiceHandle,
ServiceConfigBuffer,
ServiceConfigBufferMaxSize,
&RequiredSize);
if (Result) {
//
// We got it.
//
break;
}
ErrorCode = GetLastError();
if (ErrorCode != ERROR_INSUFFICIENT_BUFFER) {
//
// This is a real error.
//
goto cleanup;
}
//
// Resize the buffer and try again.
//
if (ServiceConfigBuffer) {
PFSVC_FREE(ServiceConfigBuffer);
ServiceConfigBuffer = NULL;
ServiceConfigBufferMaxSize = 0;
}
ServiceConfigBuffer = PFSVC_ALLOC(RequiredSize);
if (ServiceConfigBuffer == NULL) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
ServiceConfigBufferMaxSize = RequiredSize;
//
// Make sure we don't loop forever.
//
NumResizes++;
if (NumResizes > 100) {
ErrorCode = ERROR_INVALID_FUNCTION;
goto cleanup;
}
} while (TRUE);
//
// We are interested in this service only if it starts as a
// boot driver or if it is a file system.
//
if (ServiceConfigBuffer->dwStartType == SERVICE_BOOT_START ||
ServiceConfigBuffer->dwServiceType == SERVICE_FILE_SYSTEM_DRIVER) {
//
// Try to locate the real service binary path.
//
ErrorCode = PfSvGetBootServiceFullPath(ServiceInfo->lpServiceName,
ServiceConfigBuffer->lpBinaryPathName,
FilePath,
MAX_PATH,
&RequiredLength);
if (ErrorCode == ERROR_SUCCESS) {
PfSvAddBootImageAndImportsToList(PathList,
FilePath,
wcslen(FilePath));
}
}
//
// Close the handle and continue.
//
CloseServiceHandle(ServiceHandle);
ServiceHandle = NULL;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ServiceHandle) {
CloseServiceHandle(ServiceHandle);
}
if (ScHandle) {
CloseServiceHandle(ScHandle);
}
if (EnumBuffer) {
PFSVC_FREE(EnumBuffer);
}
if (ServiceConfigBuffer) {
PFSVC_FREE(ServiceConfigBuffer);
}
return ErrorCode;
}
DWORD
PfSvAddBootImageAndImportsToList(
PPFSVC_PATH_LIST PathList,
WCHAR *FilePath,
ULONG FilePathLength
)
/*++
Routine Description:
This function attempts to add the image file whose fully qualified
path is in FilePath as well as the modules it imports from to the
file list, if those modules can be located.
NOTE: Ntoskrnl.exe and Hal.dll are special cased out and not added
to the file list, since most drivers will import from them. They
can be added to the list seperately. Also note that the file list
is not checked for duplicates when adding new entries.
Arguments:
PathList - Pointer to list.
FilePath - Fully qualified path of an image file.
FilePathLength - Length of the file path in characters excluding NUL.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
ULONG MaxNumImports;
ULONG NumImports;
WCHAR **ImportNames;
ULONG ImportIdx;
ULONG BufferSize;
WCHAR *FileName;
WCHAR ParentDir[MAX_PATH + 1];
ULONG ParentDirLength;
WCHAR *ImportName;
ULONG ImportNameLength;
WCHAR ImportPath[MAX_PATH + 1];
PUCHAR ImportBase;
ULONG RequiredLength;
ULONG FileSize;
PIMAGE_IMPORT_DESCRIPTOR NewImportDescriptor;
CHAR *NewImportNameAnsi;
WCHAR *NewImportName;
ULONG NewImportNameRva;
BOOLEAN AddedToTable;
ULONG ImportTableSize;
PIMAGE_NT_HEADERS NtHeaders;
ULONG NextImport;
ULONG FailedCheck;
//
// Initialize locals.
//
MaxNumImports = 256;
ImportNames = NULL;
NumImports = 0;
NextImport = 0;
//
// Find the file name from the path.
//
if (FilePathLength == 0 || FilePath[FilePathLength - 1] == L'\\') {
ErrorCode = ERROR_BAD_LENGTH;
goto cleanup;
}
FileName = &FilePath[FilePathLength - 1];
while (FileName > FilePath) {
//
// It is OK to decrement FileName the first time in the loop because
// we checked for the terminating slash above.
//
FileName--;
if (*FileName == L'\\') {
FileName++;
break;
}
}
//
// Extract the parent directory.
//
ParentDirLength = (ULONG) (FileName - FilePath);
if (ParentDirLength >= MAX_PATH) {
ErrorCode = ERROR_BAD_LENGTH;
goto cleanup;
}
wcsncpy(ParentDir, FilePath, ParentDirLength);
ParentDir[ParentDirLength] = 0;
//
// Allocate a table for keeping track of imported modules.
//
BufferSize = MaxNumImports * sizeof(WCHAR *);
ImportNames = PFSVC_ALLOC(BufferSize);
if (ImportNames == NULL) {
ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
RtlZeroMemory(ImportNames, BufferSize);
//
// Insert the file into the table and kick off import enumeration
// on the table. Each enumerated import gets appended to the table
// if it is not already present. Enumeration continues until all
// appended entries are processed.
//
ImportNames[NumImports] = FileName;
NumImports++;
while (NextImport < NumImports) {
//
// Initialize loop locals.
//
ImportBase = NULL;
ImportName = ImportNames[NextImport];
ImportNameLength = wcslen(ImportName);
//
// Locate the file. First look in ParentDir.
//
if (ImportNameLength + ParentDirLength >= MAX_PATH) {
goto NextImport;
}
wcscpy(ImportPath, ParentDir);
wcscat(ImportPath, ImportName);
if (GetFileAttributes(ImportPath) == INVALID_FILE_ATTRIBUTES) {
//
// Look for this file in other known directories.
//
ErrorCode = PfSvLocateBootServiceFile(ImportName,
ImportNameLength,
ImportPath,
MAX_PATH,
&RequiredLength);
if (ErrorCode != ERROR_SUCCESS) {
goto NextImport;
}
}
//
// Add the file to the file list.
//
PfSvAddToPathList(PathList,
ImportPath,
wcslen(ImportPath));
//
// Map the file.
//
ErrorCode = PfSvGetViewOfFile(ImportPath, &ImportBase, &FileSize);
if (ErrorCode != ERROR_SUCCESS) {
goto NextImport;
}
//
// Make sure this is an image file.
//
__try {
//
// This is the first access to the mapped file. Under stress we might not be
// able to page this in and an exception might be raised. This protects us from
// the most common failure case.
//
//
// Verify the image and its import table. If the image is corrupt it might
// result in an AV when we try to walk its imports.
//
FailedCheck = PfVerifyImageImportTable(ImportBase, FileSize, FALSE);
if (FailedCheck) {
ErrorCode = ERROR_BAD_FORMAT;
goto NextImport;
}
NtHeaders = ImageNtHeader(ImportBase);
} __except (EXCEPTION_EXECUTE_HANDLER) {
NtHeaders = NULL;
}
if (NtHeaders == NULL) {
goto NextImport;
}
//
// Walk through the imports for this binary.
//
NewImportDescriptor = ImageDirectoryEntryToData(ImportBase,
FALSE,
IMAGE_DIRECTORY_ENTRY_IMPORT,
&ImportTableSize);
while (NewImportDescriptor &&
(NewImportDescriptor->Name != 0) &&
(NewImportDescriptor->FirstThunk != 0)) {
//
// Initialize loop locals.
//
AddedToTable = FALSE;
NewImportName = NULL;
//
// Get the name for this import.
//
NewImportNameRva = NewImportDescriptor->Name;
NewImportNameAnsi = ImageRvaToVa(NtHeaders,
ImportBase,
NewImportNameRva,
NULL);
ErrorCode = GetLastError();
if (NewImportNameAnsi) {
NewImportName = PfSvcAnsiToUnicode(NewImportNameAnsi);
}
if (NewImportName == NULL) {
goto NextImportDescriptor;
}
//
// Skip the kernel and hal imports. See comment in
// function description.
//
if (!_wcsicmp(NewImportName, L"ntoskrnl.exe") ||
!_wcsicmp(NewImportName, L"hal.dll")) {
goto NextImportDescriptor;
}
//
// Check to see if this import is already in our table.
//
for (ImportIdx = 0; ImportIdx < NumImports; ImportIdx++) {
if (!_wcsicmp(NewImportName, ImportNames[ImportIdx])) {
goto NextImportDescriptor;
}
}
//
// Append this import to the table.
//
if (NumImports < MaxNumImports) {
ImportNames[NumImports] = NewImportName;
NumImports++;
AddedToTable = TRUE;
}
NextImportDescriptor:
if (!AddedToTable && NewImportName) {
PFSVC_FREE(NewImportName);
}
if (NumImports >= MaxNumImports) {
break;
}
NewImportDescriptor++;
}
NextImport:
if (ImportBase) {
UnmapViewOfFile(ImportBase);
}
NextImport++;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
if (ImportNames) {
//
// The first entry in the table is the filename from FilePath,
// which is not allocated and which should not be freed.
//
for (ImportIdx = 1; ImportIdx < NumImports; ImportIdx++) {
PfSvcFreeString(ImportNames[ImportIdx]);
}
PFSVC_FREE(ImportNames);
}
return ErrorCode;
}
DWORD
PfSvLocateBootServiceFile(
IN WCHAR *FileName,
IN ULONG FileNameLength,
OUT WCHAR *FullPathBuffer,
IN ULONG FullPathBufferLength,
OUT PULONG RequiredLength
)
/*++
Routine Description:
This function looks at known directories in an *attempt* locate
the file whose name is specified. The logic may have to be
improved.
Arguments:
FileName - File name to look for.
FileNameLength - Length of file name in characters excluding NUL.
FullPathBuffer - The full path will be put here.
FullPathBufferLength - Length of the FullPathBuffer in characters.
RequiredLength - If FullPathBuffer is too small, this is how big it
should be in characters.
Return Value:
ERROR_INSUFFICIENT_BUFFER - The FullPathBuffer is not big enough.
Win32 error code.
--*/
{
DWORD ErrorCode;
WCHAR *DriversDirName;
ULONG SystemDirLength;
//
// Initialize locals.
//
DriversDirName = L"drivers\\";
//
// Copy system root path and a trailing \.
//
SystemDirLength = GetSystemDirectory(FullPathBuffer, FullPathBufferLength);
if (!SystemDirLength) {
ErrorCode = GetLastError();
goto cleanup;
}
SystemDirLength++;
//
// Calculate maximum size of required length.
//
(*RequiredLength) = SystemDirLength;
(*RequiredLength) += wcslen(DriversDirName);
(*RequiredLength) += FileNameLength;
(*RequiredLength) += 1; // terminating NUL.
if ((*RequiredLength) > FullPathBufferLength) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Append slash.
//
wcscat(FullPathBuffer, L"\\");
//
// Append drivers path.
//
wcscat(FullPathBuffer, DriversDirName);
//
// Append file name.
//
wcscat(FullPathBuffer, FileName);
if (GetFileAttributes(FullPathBuffer) != INVALID_FILE_ATTRIBUTES) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// Roll back and look for the file in the system
// directory. SystemDirLength includes the slash after system
// directory path.
//
FullPathBuffer[SystemDirLength] = 0;
wcscat(FullPathBuffer, FileName);
if (GetFileAttributes(FullPathBuffer) != INVALID_FILE_ATTRIBUTES) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
ErrorCode = ERROR_FILE_NOT_FOUND;
cleanup:
return ErrorCode;
}
DWORD
PfSvGetBootServiceFullPath(
IN WCHAR *ServiceName,
IN WCHAR *BinaryPathName,
OUT WCHAR *FullPathBuffer,
IN ULONG FullPathBufferLength,
OUT PULONG RequiredLength
)
/*++
Routine Description:
This function *attempts* to locate specified boot service. The
logic may have to be improved.
Arguments:
ServiceName - Name of the service.
BinaryPathName - From service configuration info. This is supposed
to be the full path, but it is not.
FullPathBuffer - The full path will be put here.
FullPathBufferLength - Length of the FullPathBuffer in characters.
RequiredLength - If FullPathBuffer is too small, this is how big it
should be in characters.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
WCHAR FileName[MAX_PATH];
ULONG BinaryPathLength;
BOOLEAN GotFileNameFromBinaryPath;
LONG CharIdx;
ULONG CopyLength;
WCHAR *SysExtension;
WCHAR *DllExtension;
WCHAR *FileNamePart;
//
// Initialize locals.
//
GotFileNameFromBinaryPath = FALSE;
SysExtension = L".sys";
DllExtension = L".dll";
//
// Check if a binary path was specified.
//
if (BinaryPathName && BinaryPathName[0]) {
//
// See if the file is really there.
//
if (GetFileAttributes(BinaryPathName) != INVALID_FILE_ATTRIBUTES) {
//
// BinaryPathName may not be a fully qualified path. Make
// sure it is.
//
(*RequiredLength) = GetFullPathName(BinaryPathName,
FullPathBufferLength,
FullPathBuffer,
&FileNamePart);
if ((*RequiredLength) == 0) {
ErrorCode = GetLastError();
goto cleanup;
}
if ((*RequiredLength) > FullPathBufferLength) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// Try to extract a file name from the binary path.
//
BinaryPathLength = wcslen(BinaryPathName);
for (CharIdx = BinaryPathLength - 1;
CharIdx >= 0;
CharIdx --) {
if (BinaryPathName[CharIdx] == L'\\') {
//
// Check length and copy it.
//
CopyLength = BinaryPathLength - CharIdx;
if (CopyLength < MAX_PATH &&
CopyLength > 1) {
//
// Copy name starting after the \ character.
//
wcscpy(FileName, &BinaryPathName[CharIdx + 1]);
GotFileNameFromBinaryPath = TRUE;
}
break;
}
}
//
// There was not a slash. Maybe the BinaryPathLength is just
// the file name.
//
if (GotFileNameFromBinaryPath == FALSE &&
BinaryPathLength &&
BinaryPathLength < MAX_PATH) {
wcscpy(FileName, BinaryPathName);
GotFileNameFromBinaryPath = TRUE;
}
}
//
// After this point we will base our search on file name hints.
//
//
// If we got a file name from the binary path try that first.
//
if (GotFileNameFromBinaryPath) {
ErrorCode = PfSvLocateBootServiceFile(FileName,
wcslen(FileName),
FullPathBuffer,
FullPathBufferLength,
RequiredLength);
if (ErrorCode != ERROR_FILE_NOT_FOUND) {
//
// If we found a path or if the buffer length was not
// enough we will bubble up that to our caller.
//
goto cleanup;
}
}
//
// Build a file name from service name by appending a .sys.
//
CopyLength = wcslen(ServiceName);
CopyLength += wcslen(SysExtension);
if (CopyLength >= MAX_PATH) {
//
// The service name is too long!
//
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
wcscpy(FileName, ServiceName);
wcscat(FileName, SysExtension);
ErrorCode = PfSvLocateBootServiceFile(FileName,
wcslen(FileName),
FullPathBuffer,
FullPathBufferLength,
RequiredLength);
if (ErrorCode != ERROR_FILE_NOT_FOUND) {
//
// If we found a path or if the buffer length was not
// enough we will bubble up that to our caller.
//
goto cleanup;
}
//
// Build a file name from service name by appending a .dll.
//
CopyLength = wcslen(ServiceName);
CopyLength += wcslen(DllExtension);
if (CopyLength >= MAX_PATH) {
//
// The service name is too long!
//
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
wcscpy(FileName, ServiceName);
wcscat(FileName, DllExtension);
ErrorCode = PfSvLocateBootServiceFile(FileName,
wcslen(FileName),
FullPathBuffer,
FullPathBufferLength,
RequiredLength);
if (ErrorCode != ERROR_FILE_NOT_FOUND) {
//
// If we found a path or if the buffer length was not
// enough we will bubble up that to our caller.
//
goto cleanup;
}
//
// We could not find the file...
//
ErrorCode = ERROR_FILE_NOT_FOUND;
cleanup:
return ErrorCode;
}
DWORD
PfSvGetBootLoaderNlsFileNames (
PPFSVC_PATH_LIST PathList
)
/*++
Routine Description:
This function attempts to add the list of NLS files loaded in the
boot loader to the specified file list.
Arguments:
PathList - Pointer to list.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
HKEY NlsKeyHandle;
WCHAR *CodePageKeyName;
HKEY CodePageKeyHandle;
WCHAR *LanguageKeyName;
HKEY LanguageKeyHandle;
ULONG BufferSize;
ULONG RequiredSize;
ULONG RequiredLength;
WCHAR FileName[MAX_PATH + 1];
WCHAR FilePath[MAX_PATH + 1];
WCHAR *AnsiCodePageName;
WCHAR *OemCodePageName;
WCHAR *OemHalName;
WCHAR *DefaultLangName;
ULONG RegValueType;
//
// Initialize locals.
//
NlsKeyHandle = NULL;
CodePageKeyHandle = NULL;
LanguageKeyHandle = NULL;
CodePageKeyName = L"CodePage";
LanguageKeyName = L"Language";
AnsiCodePageName = L"ACP";
OemCodePageName = L"OEMCP";
DefaultLangName = L"Default";
OemHalName = L"OEMHAL";
//
// Open NLS key.
//
ErrorCode = RegOpenKey(HKEY_LOCAL_MACHINE,
PFSVC_NLS_REG_KEY_PATH,
&NlsKeyHandle);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Open CodePage key.
//
ErrorCode = RegOpenKey(NlsKeyHandle,
CodePageKeyName,
&CodePageKeyHandle);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// Open Language key.
//
ErrorCode = RegOpenKey(NlsKeyHandle,
LanguageKeyName,
&LanguageKeyHandle);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
//
// AnsiCodePage:
//
ErrorCode = PfSvQueryNlsFileName(CodePageKeyHandle,
AnsiCodePageName,
FileName,
MAX_PATH * sizeof(WCHAR),
&RequiredSize);
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvLocateNlsFile(FileName,
FilePath,
MAX_PATH,
&RequiredLength);
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvAddToPathList(PathList, FilePath, wcslen(FilePath));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
}
//
// OemCodePage:
//
ErrorCode = PfSvQueryNlsFileName(CodePageKeyHandle,
OemCodePageName,
FileName,
MAX_PATH * sizeof(WCHAR),
&RequiredSize);
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvLocateNlsFile(FileName,
FilePath,
MAX_PATH,
&RequiredLength);
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvAddToPathList(PathList, FilePath, wcslen(FilePath));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
}
//
// Default language case conversion.
//
ErrorCode = PfSvQueryNlsFileName(LanguageKeyHandle,
DefaultLangName,
FileName,
MAX_PATH * sizeof(WCHAR),
&RequiredSize);
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvLocateNlsFile(FileName,
FilePath,
MAX_PATH,
&RequiredLength);
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvAddToPathList(PathList, FilePath, wcslen(FilePath));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
}
//
// OemHal:
//
BufferSize = MAX_PATH * sizeof(WCHAR);
ErrorCode = RegQueryValueEx(CodePageKeyHandle,
OemHalName,
NULL,
&RegValueType,
(PVOID) FileName,
&BufferSize);
FileName[MAX_PATH - 1] = 0;
if (ErrorCode == ERROR_SUCCESS && RegValueType == REG_SZ) {
ErrorCode = PfSvLocateNlsFile(FileName,
FilePath,
MAX_PATH,
&RequiredLength);
if (ErrorCode == ERROR_SUCCESS) {
ErrorCode = PfSvAddToPathList(PathList, FilePath, wcslen(FilePath));
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
}
}
ErrorCode = ERROR_SUCCESS;
cleanup:
if (NlsKeyHandle) {
RegCloseKey(NlsKeyHandle);
}
if (CodePageKeyHandle) {
RegCloseKey(CodePageKeyHandle);
}
if (LanguageKeyHandle) {
RegCloseKey(LanguageKeyHandle);
}
return ErrorCode;
}
DWORD
PfSvLocateNlsFile(
WCHAR *FileName,
WCHAR *FilePathBuffer,
ULONG FilePathBufferLength,
ULONG *RequiredLength
)
/*++
Routine Description:
This function attempts to locate a nls/font related file in known
directories.
Arguments:
FileName - File name to look for.
FullPathBuffer - The full path will be put here.
FullPathBufferLength - Length of the FullPathBuffer in characters.
RequiredLength - If FullPathBuffer is too small, this is how big it
should be in characters.
Return Value:
ERROR_INSUFFICIENT_BUFFER - The FullPathBuffer is not big enough.
Win32 error code.
--*/
{
DWORD ErrorCode;
ULONG SystemRootLength;
WCHAR *System32DirName;
WCHAR *FontsDirName;
WCHAR *SystemDirName;
WCHAR *LongestDirName;
//
// Initialize locals. NOTE: The length of the longest directory
// name to concatenate to SystemRoot is used in RequiredLength
// calculation.
//
System32DirName = L"System32\\";
SystemDirName = L"System\\";
FontsDirName = L"Fonts\\";
LongestDirName = System32DirName;
//
// Get system root path.
//
SystemRootLength = ExpandEnvironmentStrings(L"%SystemRoot%\\",
FilePathBuffer,
FilePathBufferLength);
if (SystemRootLength == 0) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
//
// SystemRootLength includes the terminating NUL. Adjust it.
//
SystemRootLength--;
//
// Calculate required length with space for terminating NUL.
//
(*RequiredLength) = SystemRootLength;
(*RequiredLength) += wcslen(LongestDirName);
(*RequiredLength) += wcslen(FileName);
(*RequiredLength) ++;
if ((*RequiredLength) > FilePathBufferLength) {
ErrorCode = ERROR_INSUFFICIENT_BUFFER;
goto cleanup;
}
//
// Look for it under system32 dir.
//
FilePathBuffer[SystemRootLength] = 0;
wcscat(FilePathBuffer, System32DirName);
wcscat(FilePathBuffer, FileName);
if (GetFileAttributes(FilePathBuffer) != INVALID_FILE_ATTRIBUTES) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// Look for it under fonts dir.
//
FilePathBuffer[SystemRootLength] = 0;
wcscat(FilePathBuffer, FontsDirName);
wcscat(FilePathBuffer, FileName);
if (GetFileAttributes(FilePathBuffer) != INVALID_FILE_ATTRIBUTES) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// Look for it under system dir.
//
FilePathBuffer[SystemRootLength] = 0;
wcscat(FilePathBuffer, SystemDirName);
wcscat(FilePathBuffer, FileName);
if (GetFileAttributes(FilePathBuffer) != INVALID_FILE_ATTRIBUTES) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// Look for it at SystemRoot.
//
FilePathBuffer[SystemRootLength] = 0;
wcscat(FilePathBuffer, FileName);
if (GetFileAttributes(FilePathBuffer) != INVALID_FILE_ATTRIBUTES) {
ErrorCode = ERROR_SUCCESS;
goto cleanup;
}
//
// Could not find the file.
//
ErrorCode = ERROR_FILE_NOT_FOUND;
cleanup:
return ErrorCode;
}
DWORD
PfSvQueryNlsFileName (
HKEY Key,
WCHAR *ValueName,
WCHAR *FileNameBuffer,
ULONG FileNameBufferSize,
ULONG *RequiredSize
)
/*++
Routine Description:
This function attempts to get a file name from an NLS
CodePage/Language registry key.
Arguments:
Key - CodePage or Language key handle.
ValueName - What we are trying to get the file name for.
FileNameBuffer - Where the file name will be put.
FileNameBufferSize - Size in bytes of the file name buffer.
RequiredSize - If FileNameBuffer is too small, this is what its
size should be.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
WCHAR FileValueName[MAX_PATH + 1];
ULONG BufferSize;
ULONG RegValueType;
//
// First we first get the valuename under which the file name is
// stored, then we get the file name:
//
BufferSize = MAX_PATH * sizeof(WCHAR);
ErrorCode = RegQueryValueEx(Key,
ValueName,
NULL,
&RegValueType,
(PVOID) FileValueName,
&BufferSize);
if (ErrorCode == ERROR_MORE_DATA) {
ErrorCode = ERROR_INVALID_FUNCTION;
goto cleanup;
}
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
if (RegValueType != REG_SZ) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
FileValueName[MAX_PATH - 1] = 0;
*RequiredSize = FileNameBufferSize;
ErrorCode = RegQueryValueEx(Key,
FileValueName,
NULL,
&RegValueType,
(PVOID) FileNameBuffer,
RequiredSize);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
if (RegValueType != REG_SZ) {
ErrorCode = ERROR_BAD_FORMAT;
goto cleanup;
}
if (FileNameBufferSize >= sizeof(WCHAR)) {
FileNameBuffer[(FileNameBufferSize/sizeof(WCHAR)) - 1] = 0;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
//
// Routines to manage / run idle tasks.
//
VOID
PfSvInitializeTask (
PPFSVC_IDLE_TASK Task
)
/*++
Routine Description:
Initialize the task structure. Should be called before any other
task functions are called. You should call the cleanup routine
on the initialized task.
Arguments:
Task - Pointer to structure.
Return Value:
None.
--*/
{
//
// Zero out the structure initializing the following to
// the right values:
//
// Registered
// WaitUnregisteredEvent
// CallbackStoppedEvent
// StartedUnregisteringEvent
// CompletedUnregisteringEvent
// Unregistering
// CallbackRunning
//
RtlZeroMemory(Task, sizeof(PFSVC_IDLE_TASK));
Task->Initialized = TRUE;
}
DWORD
PfSvRegisterTask (
PPFSVC_IDLE_TASK Task,
IT_IDLE_TASK_ID TaskId,
WAITORTIMERCALLBACK Callback,
PFSVC_IDLE_TASK_WORKER_FUNCTION DoWorkFunction
)
/*++
Routine Description:
Registers the Callback to be called when it is the turn of this
idle task to run. IFF this function returns success, you should
call unregister function before calling the cleanup function.
Arguments:
Task - Pointer to initialized task structure.
TaskId - Idle task ID to register.
Callback - We'll register a wait on the start event returned by
idle task registration with this callback. The callback should
call start/stop task callback functions appropriately.
DoWorkFunction - If the caller wants the common callback function
to be used, then this function will be called to do the actual
work in the common callback.
Return Value:
Win32 error code.
--*/
{
DWORD ErrorCode;
BOOL Success;
BOOLEAN CreatedWaitUnregisteredEvent;
BOOLEAN CreatedStartedUnregisteringEvent;
BOOLEAN CreatedCompletedUnregisteringEvent;
BOOLEAN CreatedCallbackStoppedEvent;
BOOLEAN RegisteredIdleTask;
//
// Initialize locals.
//
RegisteredIdleTask = FALSE;
CreatedWaitUnregisteredEvent = FALSE;
CreatedStartedUnregisteringEvent = FALSE;
CreatedCompletedUnregisteringEvent = FALSE;
CreatedCallbackStoppedEvent = FALSE;
DBGPR((PFID,PFTASK,"PFSVC: RegisterTask(%p,%d,%p,%p)\n",Task,TaskId,Callback,DoWorkFunction));
//
// The task should be initialized and not registered.
//
PFSVC_ASSERT(Task->Initialized);
PFSVC_ASSERT(!Task->Registered);
PFSVC_ASSERT(!Task->Unregistering);
PFSVC_ASSERT(!Task->CallbackRunning);
//
// Create the event that cleanup waits on to make sure
// the registered wait is fully unregistered.
//
Task->WaitUnregisteredEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
if (Task->WaitUnregisteredEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
CreatedWaitUnregisteredEvent = TRUE;
//
// Create the event that will get signaled when we start
// unregistering the task.
//
Task->StartedUnregisteringEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
if (Task->StartedUnregisteringEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
CreatedStartedUnregisteringEvent = TRUE;
//
// Create the event that will get signaled when we complete
// unregistering the task.
//
Task->CompletedUnregisteringEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
if (Task->CompletedUnregisteringEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
CreatedCompletedUnregisteringEvent = TRUE;
//
// Create the event we may wait on for the current running
// callback to go away.
//
Task->CallbackStoppedEvent = CreateEvent(NULL, TRUE, TRUE, NULL);
if (Task->CallbackStoppedEvent == NULL) {
ErrorCode = GetLastError();
goto cleanup;
}
CreatedCallbackStoppedEvent = TRUE;
//
// Register the idle task.
//
ErrorCode = RegisterIdleTask(TaskId,
&Task->ItHandle,
&Task->StartEvent,
&Task->StopEvent);
if (ErrorCode != ERROR_SUCCESS) {
goto cleanup;
}
RegisteredIdleTask = TRUE;
//
// Register the callback: Note that once this call succeeds the task has to
// be unregistered via PfSvUnregisterTask.
//
//
// The callback might fire right away so note that we registered it and set
// up its fields upfront.
//
// FUTURE-2002/03/29-ScottMa -- For consistency, the Task->Registered
// variable should be set to TRUE & FALSE instead of 0 & 1.
Task->Registered = 1;
Task->Callback = Callback;
Task->DoWorkFunction = DoWorkFunction;
//
// If the common task callback was specified, a worker function should also
// be specified.
//
if (Callback == PfSvCommonTaskCallback) {
PFSVC_ASSERT(DoWorkFunction);
}
Success = RegisterWaitForSingleObject(&Task->WaitHandle,
Task->StartEvent,
Task->Callback,
Task,
INFINITE,
WT_EXECUTEONLYONCE | WT_EXECUTELONGFUNCTION);
if (!Success) {
//
// We failed to really register the task.
//
Task->Registered = 0;
ErrorCode = GetLastError();
goto cleanup;
}
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTASK,"PFSVC: RegisterTask(%p)=%x\n",Task,ErrorCode));
if (ErrorCode != ERROR_SUCCESS) {
if (CreatedWaitUnregisteredEvent) {
CloseHandle(Task->WaitUnregisteredEvent);
Task->WaitUnregisteredEvent = NULL;
}
if (CreatedStartedUnregisteringEvent) {
CloseHandle(Task->StartedUnregisteringEvent);
Task->StartedUnregisteringEvent = NULL;
}
if (CreatedCompletedUnregisteringEvent) {
CloseHandle(Task->CompletedUnregisteringEvent);
Task->CompletedUnregisteringEvent = NULL;
}
if (CreatedCallbackStoppedEvent) {
CloseHandle(Task->CallbackStoppedEvent);
Task->CallbackStoppedEvent = NULL;
}
if (RegisteredIdleTask) {
UnregisterIdleTask(Task->ItHandle,
Task->StartEvent,
Task->StopEvent);
Task->StartEvent = NULL;
Task->StopEvent = NULL;
}
}
return ErrorCode;
}
DWORD
PfSvUnregisterTask (
PPFSVC_IDLE_TASK Task,
BOOLEAN CalledFromCallback
)
/*++
Routine Description:
Unregisters the idle task and the registered wait / callback. You should
call this function before calling the cleanup routine IFF the register
function returned success.
Arguments:
Task - Pointer to registered task.
CalledFromCallback - Whether this function is being called from inside
the queued callback of the task.
Return Value:
Win32 error code.
--*/
{
LONG OldValue;
LONG NewValue;
DWORD ErrorCode;
DBGPR((PFID,PFTASK,"PFSVC: UnregisterTask(%p,%d)\n",Task,(DWORD)CalledFromCallback));
//
// The task should be initialized. It may already be unregistered.
//
PFSVC_ASSERT(Task->Initialized);
if (Task->Registered == 0) {
ErrorCode = ERROR_SHUTDOWN_IN_PROGRESS;
goto cleanup;
}
//
// Distinguish whether we are unregistering the task from a callback.
//
if (CalledFromCallback) {
NewValue = PfSvcUnregisteringTaskFromCallback;
} else {
NewValue = PfSvcUnregisteringTaskFromMainThread;
}
//
// Is this task already being unregistered?
//
OldValue = InterlockedCompareExchange(&Task->Unregistering,
NewValue,
PfSvcNotUnregisteringTask);
if (OldValue != PfSvcNotUnregisteringTask) {
ErrorCode = ERROR_SHUTDOWN_IN_PROGRESS;
goto cleanup;
}
//
// *We* will be unregistering the task. There is no turning back.
//
SetEvent(Task->StartedUnregisteringEvent);
//
// If we are not inside a callback, wait for no callbacks to be running
// and cause new ones that start to bail out. We do this so we can safely
// unregister the wait.
//
if (!CalledFromCallback) {
do {
OldValue = InterlockedCompareExchange(&Task->CallbackRunning,
PfSvcTaskCallbackDisabled,
PfSvcTaskCallbackNotRunning);
if (OldValue == PfSvcTaskCallbackNotRunning) {
//
// We did it. No callbacks are running and new ones that try to
// start will bail out.
//
PFSVC_ASSERT(Task->CallbackRunning == PfSvcTaskCallbackDisabled);
break;
}
//
// A callback might be active right now. It will see that we are unregistering and
// go away. Sleep for a while and try again.
//
PFSVC_ASSERT(OldValue == PfSvcTaskCallbackRunning);
//
// We wait on this event with a timeout, because signaling of it is not
// 100% reliable because it is not under a lock etc.
//
WaitForSingleObject(Task->CallbackStoppedEvent, 1000);
} while (TRUE);
} else {
//
// We already have control of this variable as the running callback: just
// update it.
//
Task->CallbackRunning = PfSvcTaskCallbackDisabled;
}
//
// Unregister the wait. Note that in cleanup we have to wait to for
// WaitUnregisteredEvent to be signaled.
//
UnregisterWaitEx(Task->WaitHandle, Task->WaitUnregisteredEvent);
//
// Unregister the idle task.
//
UnregisterIdleTask(Task->ItHandle,
Task->StartEvent,
Task->StopEvent);
Task->StartEvent = NULL;
Task->StopEvent = NULL;
//
// Note that the task is no longer registered.
//
Task->Registered = FALSE;
SetEvent(Task->CompletedUnregisteringEvent);
ErrorCode = ERROR_SUCCESS;
cleanup:
DBGPR((PFID,PFTASK,"PFSVC: UnregisterTask(%p)=%x\n",Task,ErrorCode));
return ErrorCode;
}
VOID
PfSvCleanupTask (
PPFSVC_IDLE_TASK Task
)
/*++
Routine Description:
Cleans up all fields of an unregistered task or a task that was never
registered.
Arguments:
Task - Pointer to task.
Return Value:
None.
--*/
{
//
// The task should have been initialized.
//
PFSVC_ASSERT(Task->Initialized);
//
// If there is a WaitUnregisteredEvent, we have to wait on it
// to make sure the unregister operation is fully complete.
//
if (Task->WaitUnregisteredEvent) {
WaitForSingleObject(Task->WaitUnregisteredEvent, INFINITE);
CloseHandle(Task->WaitUnregisteredEvent);
Task->WaitUnregisteredEvent = NULL;
}
//
// If there is CompletedUnregisteringEvent, wait for it to
// be signalled to make sure both that the wait is unregistered,
// and the idle task is unregistered.
//
if (Task->CompletedUnregisteringEvent) {
WaitForSingleObject(Task->CompletedUnregisteringEvent, INFINITE);
}
//
// The task should be unregistered before it is cleaned up.
//
PFSVC_ASSERT(Task->Registered == FALSE);
//
// Cleanup task unregistering events.
//
if (Task->StartedUnregisteringEvent) {
CloseHandle(Task->StartedUnregisteringEvent);
Task->StartedUnregisteringEvent = NULL;
}
if (Task->CompletedUnregisteringEvent) {
CloseHandle(Task->CompletedUnregisteringEvent);
Task->CompletedUnregisteringEvent = NULL;
}
//
// Clean up callback stopped event.
//
if (Task->CallbackStoppedEvent) {
CloseHandle(Task->CallbackStoppedEvent);
Task->CallbackStoppedEvent = NULL;
}
Task->Initialized = FALSE;
return;
}
BOOL
PfSvStartTaskCallback(
PPFSVC_IDLE_TASK Task
)
/*++
Routine Description:
Callbacks registered via register task function should call this as the
first thing. If this function returns FALSE, the callback should go away
immediately without calling the stop-callback function.
Arguments:
Task - Pointer to task.
Return Value:
TRUE - Everything is cool.
FALSE - The task is being unregistered. Exit the callback asap.
--*/
{
BOOL ReturnValue;
LONG OldValue;
DBGPR((PFID,PFTASK,"PFSVC: StartTaskCallback(%p)\n",Task));
//
// We should not be called if the task is not initialized.
//
PFSVC_ASSERT(Task->Initialized);
do {
//
// First check if we are trying to unregister.
//
if (Task->Unregistering) {
ReturnValue = FALSE;
goto cleanup;
}
//
// Try to mark the callback running.
//
OldValue = InterlockedCompareExchange(&Task->CallbackRunning,
PfSvcTaskCallbackRunning,
PfSvcTaskCallbackNotRunning);
if (OldValue == PfSvcTaskCallbackNotRunning) {
//
// We are the running callback now. Reset the event that says
// the current callback stopped running.
//
ResetEvent(Task->CallbackStoppedEvent);
ReturnValue = TRUE;
goto cleanup;
}
//
// Either another callback is running or we are unregistering.
//
//
// Are we unregistering?
//
if (Task->Unregistering) {
ReturnValue = FALSE;
goto cleanup;
} else {
PFSVC_ASSERT(OldValue == PfSvcTaskCallbackRunning);
}
//
// Sleep for a while and try again. There should not be much conflict in this
// code, so we should hardly ever need to sleep.
//
Sleep(15);
} while (TRUE);
//
// We should not come here.
//
PFSVC_ASSERT(FALSE);
cleanup:
//
// If we are starting a callback, the task should not be in unregistered state.
//
PFSVC_ASSERT(!ReturnValue || Task->Registered);
DBGPR((PFID,PFTASK,"PFSVC: StartTaskCallback(%p)=%d\n",Task,ReturnValue));
return ReturnValue;
}
VOID
PfSvStopTaskCallback(
PPFSVC_IDLE_TASK Task
)
/*++
Routine Description:
Callbacks registered via register task function should call this as the
last thing, only if they successfully called the start callback function and
they did not unregister the task.
Arguments:
Task - Pointer to task.
Return Value:
None.
--*/
{
DBGPR((PFID,PFTASK,"PFSVC: StopTaskCallback(%p)\n",Task));
//
// The task should be registered.
//
PFSVC_ASSERT(Task->Registered);
//
// There should be a running callback.
//
PFSVC_ASSERT(Task->CallbackRunning == PfSvcTaskCallbackRunning);
Task->CallbackRunning = PfSvcTaskCallbackNotRunning;
//
// Signal the event the main thread may be waiting on to unregister
// this task.
//
SetEvent(Task->CallbackStoppedEvent);
return;
}
VOID
CALLBACK
PfSvCommonTaskCallback(
PVOID lpParameter,
BOOLEAN TimerOrWaitFired
)
/*++
Routine Description:
This is the callback for the idle tasks. It is called when the system is idle,
and it is this tasks turn to run.
Note that you cannot call PfSvCleanupTask from this thread, as it would cause
a deadlock when that function waits for registered wait callbacks to exit.
Arguments:
lpParameter - Pointer to task.
TimerOrWaitFired - Whether the callback was initiated by a timeout or the start
event getting signaled by the idle task service.
Return Value:
None.
--*/
{
HANDLE NewWaitHandle;
PPFSVC_IDLE_TASK Task;
BOOL StartedCallback;
BOOL Success;
DWORD ErrorCode;
//
// Initialize locals.
//
Task = lpParameter;
StartedCallback = FALSE;
DBGPR((PFID,PFTASK,"PFSVC: CommonTaskCallback(%p)\n",Task));
//
// Enter task callback.
//
StartedCallback = PfSvStartTaskCallback(Task);
if (!StartedCallback) {
goto cleanup;
}
//
// Do the task.
//
ErrorCode = Task->DoWorkFunction(Task);
if (ErrorCode == ERROR_RETRY) {
//
// The stop event was signaled. We will queue another callback.
//
Success = RegisterWaitForSingleObject(&NewWaitHandle,
Task->StartEvent,
Task->Callback,
Task,
INFINITE,
WT_EXECUTEONLYONCE | WT_EXECUTELONGFUNCTION);
if (Success) {
//
// Unregister the current wait handle and update it.
//
UnregisterWaitEx(Task->WaitHandle, NULL);
Task->WaitHandle = NewWaitHandle;
goto cleanup;
} else {
//
// We could not queue another callback. We will unregister,
// since we would not be able to respond to start signals
// from the idle task service. Unregister may fail only if
// the main thread is already trying to unregister.
//
ErrorCode = PfSvUnregisterTask(Task, TRUE);
if (ErrorCode == ERROR_SUCCESS) {
//
// Since *we* unregistered, we should not call stop callback.
//
StartedCallback = FALSE;
}
goto cleanup;
}
} else {
//
// The task completed. Let's unregister.
//
ErrorCode = PfSvUnregisterTask(Task, TRUE);
if (ErrorCode == ERROR_SUCCESS) {
//
// Since *we* unregistered, we should not call stop callback.
//
StartedCallback = FALSE;
}
goto cleanup;
}
//
// We should not come here.
//
PFSVC_ASSERT(FALSE);
cleanup:
DBGPR((PFID,PFTASK,"PFSVC: CommonTaskCallback(%p)=%x\n",Task,ErrorCode));
if (StartedCallback) {
PfSvStopTaskCallback(Task);
}
}
DWORD
PfSvContinueRunningTask(
PPFSVC_IDLE_TASK Task
)
/*++
Routine Description:
This is called from a running task to determine if we should continue
running this task. The task should continue running if ERROR_SUCCESS is
returned. ERROR_RETRY may be returned if the task is unregistering or
was asked to stop.
Arguments:
Task - Pointer to task. If NULL, this parameter is ignored.
Return Value:
Win32 error code.
--*/
{
DWORD WaitResult;
DWORD ErrorCode;
if (Task) {
//
// Is the task being unregistered?
//
if (Task->Unregistering) {
ErrorCode = ERROR_RETRY;
goto cleanup;
}
//
// Is the stop event signaled? We don't really wait here since
// the timeout is 0.
//
WaitResult = WaitForSingleObject(Task->StopEvent, 0);
if (WaitResult == WAIT_OBJECT_0) {
ErrorCode = ERROR_RETRY;
goto cleanup;
} else if (WaitResult != WAIT_TIMEOUT) {
//
// There was an error.
//
ErrorCode = GetLastError();
goto cleanup;
}
}
//
// Check if the service is exiting...
//
if (PfSvcGlobals.TerminateServiceEvent) {
WaitResult = WaitForSingleObject(PfSvcGlobals.TerminateServiceEvent, 0);
if (WaitResult == WAIT_OBJECT_0) {
ErrorCode = ERROR_RETRY;
goto cleanup;
} else if (WaitResult != WAIT_TIMEOUT) {
//
// There was an error.
//
ErrorCode = GetLastError();
goto cleanup;
}
}
//
// The task should continue to run.
//
ErrorCode = ERROR_SUCCESS;
cleanup:
return ErrorCode;
}
//
// ProcessIdleTasks notify routine and its dependencies.
//
VOID
PfSvProcessIdleTasksCallback(
VOID
)
/*++
Routine Description:
This is routine is registered with the idle task server as a notify
routine that is called when processing of all idle tasks is requested.
ProcessIdleTasks is usually called to prepare the system for a benchmark
run by performing the optimization tasks that would have been performed
when the system is idle.
Arguments:
None.
Return Value:
None.
--*/
{
HANDLE Events[2];
DWORD NumEvents;
DWORD WaitResult;
BOOLEAN ResetOverrideIdleEvent;
//
// First flush the idle tasks the prefetcher may have queued:
//
//
// Determine the current status of the override-idle event.
//
WaitResult = WaitForSingleObject(PfSvcGlobals.OverrideIdleProcessingEvent,
0);
if (WaitResult != WAIT_OBJECT_0) {
//
// Override idle event is not already set. Set it and note to reset
// it once tasks are completed.
//
SetEvent(PfSvcGlobals.OverrideIdleProcessingEvent);
ResetOverrideIdleEvent = TRUE;
} else {
ResetOverrideIdleEvent = FALSE;
}
//
// Wait for processing complete event to get signaled.
//
Events[0] = PfSvcGlobals.ProcessingCompleteEvent;
Events[1] = PfSvcGlobals.TerminateServiceEvent;
NumEvents = 2;
WaitForMultipleObjects(NumEvents, Events, FALSE, 30 * 60 * 1000);
//
// If we set the override idle event, reset it.
//
if (ResetOverrideIdleEvent) {
ResetEvent(PfSvcGlobals.OverrideIdleProcessingEvent);
}
//
// Force an update of the disk layout in case it did not happen.
// If we notice no changes we will not launch the defragger again.
//
PfSvUpdateOptimalLayout(NULL);
//
// Signal WMI to complete its idle tasks if it has pending tasks.
//
PfSvForceWMIProcessIdleTasks();
return;
}
DWORD
PfSvForceWMIProcessIdleTasks(
VOID
)
/*++
Routine Description:
This is routine is called to force WMI to process all of its idle tasks.
Arguments:
None.
Return Value:
Win32 error code.
--*/
{
HANDLE StartEvent;
HANDLE DoneEvent;
HANDLE Events[2];
DWORD NumEvents;
DWORD ErrorCode;
DWORD WaitResult;
BOOL Success;
//
// Initialize locals.
//
StartEvent = NULL;
DoneEvent = NULL;
//
// Wait until WMI service is started.
//
Success = PfSvWaitForServiceToStart(L"WINMGMT", 5 * 60 * 1000);
if (!Success) {
ErrorCode = ERROR_SERVICE_NEVER_STARTED;
goto cleanup;
}
//
// Open the start and done events.
//
StartEvent = OpenEvent(EVENT_ALL_ACCESS, FALSE, L"WMI_ProcessIdleTasksStart");
DoneEvent = OpenEvent(EVENT_ALL_ACCESS, FALSE, L"WMI_ProcessIdleTasksComplete");
if (!StartEvent || !DoneEvent) {
ErrorCode = ERROR_FILE_NOT_FOUND;
goto cleanup;
}
//
// Reset the done event.
//
ResetEvent(DoneEvent);
//
// Signal the start event.
//
SetEvent(StartEvent);
//
// Wait for the done event to be signaled.
//
Events[0] = DoneEvent;
Events[1] = PfSvcGlobals.TerminateServiceEvent;
NumEvents = 2;
WaitResult = WaitForMultipleObjects(NumEvents, Events, FALSE, 25 * 60 * 1000);
switch(WaitResult) {
case WAIT_OBJECT_0 : ErrorCode = ERROR_SUCCESS; break;
case WAIT_OBJECT_0 + 1 : ErrorCode = ERROR_SHUTDOWN_IN_PROGRESS; break;
case WAIT_FAILED : ErrorCode = GetLastError(); break;
case WAIT_TIMEOUT : ErrorCode = WAIT_TIMEOUT; break;
default : ErrorCode = ERROR_INVALID_FUNCTION;
}
//
// Fall through with error code.
//
cleanup:
if (StartEvent) {
CloseHandle(StartEvent);
}
if (DoneEvent) {
CloseHandle(DoneEvent);
}
return ErrorCode;
}
BOOL
PfSvWaitForServiceToStart (
WCHAR *lpServiceName,
DWORD dwMaxWait
)
/*++
Routine Description:
Waits for the service to start.
Arguments:
lpServiceName - Service to wait for.
dwMaxWait - Timeout in ms.
Return Value:
Whether the service was started.
--*/
{
BOOL bStarted = FALSE;
DWORD dwSize = 512;
DWORD StartTickCount;
SC_HANDLE hScManager = NULL;
SC_HANDLE hService = NULL;
SERVICE_STATUS ServiceStatus;
LPQUERY_SERVICE_CONFIG lpServiceConfig = NULL;
//
// OpenSCManager and the service.
//
hScManager = OpenSCManager(NULL, NULL, SC_MANAGER_CONNECT);
if (!hScManager) {
goto Exit;
}
hService = OpenService(hScManager, lpServiceName,
SERVICE_QUERY_CONFIG | SERVICE_QUERY_STATUS);
if (!hService) {
goto Exit;
}
//
// Query if the service is going to start
//
lpServiceConfig = LocalAlloc (LPTR, dwSize);
if (!lpServiceConfig) {
goto Exit;
}
if (!QueryServiceConfig (hService, lpServiceConfig, dwSize, &dwSize)) {
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER) {
goto Exit;
}
LocalFree (lpServiceConfig);
lpServiceConfig = LocalAlloc (LPTR, dwSize);
if (!lpServiceConfig) {
goto Exit;
}
if (!QueryServiceConfig (hService, lpServiceConfig, dwSize, &dwSize)) {
goto Exit;
}
}
if (lpServiceConfig->dwStartType != SERVICE_AUTO_START) {
goto Exit;
}
//
// Loop until the service starts or we think it never will start
// or we've exceeded our maximum time delay.
//
StartTickCount = GetTickCount();
while (!bStarted) {
if (WAIT_OBJECT_0 == WaitForSingleObject(PfSvcGlobals.TerminateServiceEvent, 0)) {
break;
}
if ((GetTickCount() - StartTickCount) > dwMaxWait) {
break;
}
if (!QueryServiceStatus(hService, &ServiceStatus )) {
break;
}
if (ServiceStatus.dwCurrentState == SERVICE_STOPPED) {
if (ServiceStatus.dwWin32ExitCode == ERROR_SERVICE_NEVER_STARTED) {
Sleep(500);
} else {
break;
}
} else if ( (ServiceStatus.dwCurrentState == SERVICE_RUNNING) ||
(ServiceStatus.dwCurrentState == SERVICE_CONTINUE_PENDING) ||
(ServiceStatus.dwCurrentState == SERVICE_PAUSE_PENDING) ||
(ServiceStatus.dwCurrentState == SERVICE_PAUSED) ) {
bStarted = TRUE;
} else if (ServiceStatus.dwCurrentState == SERVICE_START_PENDING) {
Sleep(500);
} else {
Sleep(500);
}
}
Exit:
if (lpServiceConfig) {
LocalFree (lpServiceConfig);
}
if (hService) {
CloseServiceHandle(hService);
}
if (hScManager) {
CloseServiceHandle(hScManager);
}
return bStarted;
}
//
// Wrappers around the verify routines.
//
BOOLEAN
PfSvVerifyScenarioBuffer(
PPF_SCENARIO_HEADER Scenario,
ULONG BufferSize,
PULONG FailedCheck
)
/*++
Routine Description:
This wrapper arounding PfVerifyScenarioBuffer traps exceptions such as in-page errors that
may happen when the system is under stress. Otherwise these non-fatal failures may take
down a service-host full of important system services.
Arguments:
See PfVerifyScenarioBuffer.
Return Value:
See PfVerifyScenarioBuffer.
--*/
{
BOOLEAN Success;
__try {
Success = PfVerifyScenarioBuffer(Scenario, BufferSize, FailedCheck);
} __except (EXCEPTION_EXECUTE_HANDLER) {
//
// We should not be masking other types of exceptions.
//
PFSVC_ASSERT(GetExceptionCode() == EXCEPTION_IN_PAGE_ERROR);
Success = FALSE;
*FailedCheck = (ULONG) GetExceptionCode();
}
return Success;
}
ULONG
PfVerifyImageImportTable (
IN PVOID BaseAddress,
IN ULONG MappingSize,
IN BOOLEAN MappedAsImage
)
/*++
Routine Description:
This function tries to verify that walking the imports table of
the mapped base portion of an image file won't result in accesses beyond
the mapped region.
This includes checks to make sure DOS & NT header magics match and
they are within bounds and that strings in all import table descriptors
are NUL terminated etc.
Arguments:
BaseAddress - Pointer to base of the mapping of the image file.
MappingSize - Size in bytes of the mapping starting at BaseAddress.
MappedAsImage - Whether the image is mapping as image or as file.
Return Value:
0 - It is safe to walk the imports of the image mapped as data.
Non 0 - It is not safe to walk the imports of the image mapped as data.
Returned value is the ID of the check failed.
--*/
{
PIMAGE_DOS_HEADER DosHeader;
PIMAGE_NT_HEADERS NtHeaders;
PIMAGE_OPTIONAL_HEADER OptionalHeader;
PIMAGE_SECTION_HEADER NtSection;
PIMAGE_IMPORT_DESCRIPTOR ImportDescriptor;
PIMAGE_IMPORT_DESCRIPTOR FirstImportDescriptor;
PCHAR ImportedModuleName;
ULONG ImportDescriptorRVA;
ULONG ImportedModuleNameLength;
ULONG FailedCheckId;
BOOLEAN Success;
//
// Initialize locals.
//
FailedCheckId = 1;
//
// Is the mapping big enough to contain the DOS header?
//
if (MappingSize <= sizeof(IMAGE_DOS_HEADER)) {
FailedCheckId = 10;
goto cleanup;
}
DosHeader = BaseAddress;
//
// Check DOS signature and sanity check where it says NT headers are.
// The checked size is the same constant used in RtlpImageNtHeader.
//
if (DosHeader->e_magic != IMAGE_DOS_SIGNATURE ||
DosHeader->e_lfanew >= 256 * 1024 * 1024) {
FailedCheckId = 20;
goto cleanup;
}
//
// Is the NtHeaders within the bounds of mapping?
//
NtHeaders = (PIMAGE_NT_HEADERS) ((PUCHAR)BaseAddress + DosHeader->e_lfanew);
if(!PfWithinBounds(NtHeaders, BaseAddress, MappingSize)) {
FailedCheckId = 30;
goto cleanup;
}
//
// Is the full NtHeaders within the bounds of mapping?
//
if(!PfWithinBounds(NtHeaders + 1, BaseAddress, MappingSize)) {
FailedCheckId = 40;
goto cleanup;
}
//
// Check NT signature.
//
if (NtHeaders->Signature != IMAGE_NT_SIGNATURE) {
FailedCheckId = 50;
goto cleanup;
}
//
// Check the optional header size in file header.
//
if (NtHeaders->FileHeader.SizeOfOptionalHeader != IMAGE_SIZEOF_NT_OPTIONAL_HEADER) {
FailedCheckId = 60;
goto cleanup;
}
OptionalHeader = &NtHeaders->OptionalHeader;
//
// Check the optional header magic.
//
if (OptionalHeader->Magic != IMAGE_NT_OPTIONAL_HDR_MAGIC) {
FailedCheckId = 70;
goto cleanup;
}
//
// Verify that the section headers are within the mapped region, since
// they will be used to translate RVAs to VAs.
//
NtSection = IMAGE_FIRST_SECTION(NtHeaders);
if(!PfWithinBounds(NtSection, BaseAddress, MappingSize)) {
FailedCheckId = 80;
goto cleanup;
}
if(!PfWithinBounds(NtSection + NtHeaders->FileHeader.NumberOfSections,
BaseAddress,
MappingSize)) {
FailedCheckId = 90;
goto cleanup;
}
//
// Verify that the import descriptors are within bounds.
//
ImportDescriptorRVA = OptionalHeader->DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress;
if (MappedAsImage) {
ImportDescriptor = (PVOID) ((PUCHAR) BaseAddress + ImportDescriptorRVA);
} else {
ImportDescriptor = ImageRvaToVa(NtHeaders, BaseAddress, ImportDescriptorRVA, NULL);
}
FirstImportDescriptor = ImportDescriptor;
//
// Make sure that the first import descriptor is fully within bounds.
//
if (!PfWithinBounds(ImportDescriptor, BaseAddress, MappingSize)) {
FailedCheckId = 100;
goto cleanup;
}
if (!PfWithinBounds(((PCHAR)(ImportDescriptor + 1)) - 1, BaseAddress, MappingSize)) {
FailedCheckId = 100;
goto cleanup;
}
while (ImportDescriptor->Name && ImportDescriptor->FirstThunk) {
//
// Is the first thunk within bounds?
//
if (!PfWithinBounds((PUCHAR)BaseAddress + ImportDescriptor->FirstThunk,
BaseAddress,
MappingSize)) {
FailedCheckId = 110;
goto cleanup;
}
//
// Is the Name within bounds?
//
if (MappedAsImage) {
ImportedModuleName = (PCHAR) BaseAddress + ImportDescriptor->Name;
} else {
ImportedModuleName = ImageRvaToVa(NtHeaders, BaseAddress, ImportDescriptor->Name, NULL);
}
//
// Is the first character within bounds?
//
if (!PfWithinBounds(ImportedModuleName, BaseAddress, MappingSize)) {
FailedCheckId = 120;
goto cleanup;
}
ImportedModuleNameLength = 0;
while (*ImportedModuleName) {
ImportedModuleName++;
ImportedModuleNameLength++;
//
// If the import name is too long bail out.
//
if (ImportedModuleNameLength >= MAX_PATH) {
FailedCheckId = 130;
goto cleanup;
}
//
// Check the next character before accessing it.
//
if (!PfWithinBounds(ImportedModuleName, BaseAddress, MappingSize)) {
FailedCheckId = 140;
goto cleanup;
}
}
ImportDescriptor++;
//
// Make sure that the next import descriptor is fully within bounds
// before we access its fields in the while loop condition check.
//
if (!PfWithinBounds(((PCHAR)(ImportDescriptor + 1)) - 1, BaseAddress, MappingSize)) {
FailedCheckId = 150;
goto cleanup;
}
}
//
// We passed all the checks.
//
FailedCheckId = 0;
cleanup:
return FailedCheckId;
}
//
// Try to keep the verification code below at the end of the file so it is
// easier to copy.
//
//
// Verification code shared between the kernel and user mode
// components. This code should be kept in sync with a simple copy &
// paste, so don't add any kernel/user specific code/macros. Note that
// the prefix on the function names are Pf, just like it is with
// shared structures / constants.
//
BOOLEAN
__forceinline
PfWithinBounds(
PVOID Pointer,
PVOID Base,
ULONG Length
)
/*++
Routine Description:
Check whether the pointer is within Length bytes from the base.
Arguments:
Pointer - Pointer to check.
Base - Pointer to base of mapping/array etc.
Length - Number of bytes that are valid starting from Base.
Return Value:
TRUE - Pointer is within bounds.
FALSE - Pointer is not within bounds.
--*/
{
if (((PCHAR)Pointer < (PCHAR)Base) ||
((PCHAR)Pointer >= ((PCHAR)Base + Length))) {
return FALSE;
} else {
return TRUE;
}
}
BOOLEAN
PfVerifyScenarioId (
PPF_SCENARIO_ID ScenarioId
)
/*++
Routine Description:
Verify that the scenario id is sensible.
Arguments:
ScenarioId - Scenario Id to verify.
Return Value:
TRUE - ScenarioId is fine.
FALSE - ScenarioId is corrupt.
--*/
{
LONG CurCharIdx;
//
// Make sure the scenario name is NUL terminated.
//
for (CurCharIdx = PF_SCEN_ID_MAX_CHARS; CurCharIdx >= 0; CurCharIdx--) {
if (ScenarioId->ScenName[CurCharIdx] == 0) {
break;
}
}
if (ScenarioId->ScenName[CurCharIdx] != 0) {
return FALSE;
}
//
// Make sure there is a scenario name.
//
if (CurCharIdx == 0) {
return FALSE;
}
//
// Checks passed.
//
return TRUE;
}
// ISSUE-2002/03/29-ScottMa -- FilePaths and VolumePaths must be upcased in
// the scenario file, or subsequent comparisons will fail. Verify the case
// of the embedded names in this function.
BOOLEAN
PfVerifyScenarioBuffer(
PPF_SCENARIO_HEADER Scenario,
ULONG BufferSize,
PULONG FailedCheck
)
/*++
Routine Description:
Verify offset and indices in a scenario file are not beyond
bounds. This code is shared between the user mode service and
kernel mode component. If you update this function, update it in
both.
Arguments:
Scenario - Base of mapped view of the whole file.
BufferSize - Size of the scenario buffer.
FailedCheck - If verify failed, Id for the check that was failed.
Return Value:
TRUE - Scenario is fine.
FALSE - Scenario is corrupt.
--*/
{
PPF_SECTION_RECORD Sections;
PPF_SECTION_RECORD pSection;
ULONG SectionIdx;
PPF_PAGE_RECORD Pages;
PPF_PAGE_RECORD pPage;
LONG PageIdx;
PCHAR FileNames;
PCHAR pFileNameStart;
PCHAR pFileNameEnd;
PWCHAR pwFileName;
LONG FailedCheckId;
ULONG NumRemainingPages;
ULONG NumPages;
LONG PreviousPageIdx;
ULONG FileNameSize;
BOOLEAN ScenarioVerified;
PCHAR MetadataInfoBase;
PPF_METADATA_RECORD MetadataRecordTable;
PPF_METADATA_RECORD MetadataRecord;
ULONG MetadataRecordIdx;
PWCHAR VolumePath;
PFILE_PREFETCH FilePrefetchInfo;
ULONG FilePrefetchInfoSize;
PPF_COUNTED_STRING DirectoryPath;
ULONG DirectoryIdx;
//
// Initialize locals.
//
FailedCheckId = 0;
//
// Initialize return value to FALSE. It will be set to TRUE only
// after all the checks pass.
//
ScenarioVerified = FALSE;
//
// The buffer should at least contain the scenario header.
//
if (BufferSize < sizeof(PF_SCENARIO_HEADER)) {
FailedCheckId = 10;
goto cleanup;
}
if ((ULONG_PTR)Scenario & (_alignof(PF_SCENARIO_HEADER) - 1)) {
FailedCheckId = 15;
goto cleanup;
}
//
// Check version and magic on the header.
//
if (Scenario->Version != PF_CURRENT_VERSION ||
Scenario->MagicNumber != PF_SCENARIO_MAGIC_NUMBER) {
FailedCheckId = 20;
goto cleanup;
}
//
// The buffer should not be greater than max allowed size.
//
if (BufferSize > PF_MAXIMUM_SCENARIO_SIZE) {
FailedCheckId = 25;
goto cleanup;
}
if (BufferSize != Scenario->Size) {
FailedCheckId = 26;
goto cleanup;
}
//
// Check for legal scenario type.
//
if (Scenario->ScenarioType < 0 || Scenario->ScenarioType >= PfMaxScenarioType) {
FailedCheckId = 27;
goto cleanup;
}
//
// Check limits on number of pages, sections etc.
//
if (Scenario->NumSections > PF_MAXIMUM_SECTIONS ||
Scenario->NumMetadataRecords > PF_MAXIMUM_SECTIONS ||
Scenario->NumPages > PF_MAXIMUM_PAGES ||
Scenario->FileNameInfoSize > PF_MAXIMUM_FILE_NAME_DATA_SIZE) {
FailedCheckId = 30;
goto cleanup;
}
if (Scenario->NumSections == 0 ||
Scenario->NumPages == 0 ||
Scenario->FileNameInfoSize == 0) {
FailedCheckId = 33;
goto cleanup;
}
//
// Check limit on sensitivity.
//
if (Scenario->Sensitivity < PF_MIN_SENSITIVITY ||
Scenario->Sensitivity > PF_MAX_SENSITIVITY) {
FailedCheckId = 35;
goto cleanup;
}
//
// Make sure the scenario id is valid.
//
if (!PfVerifyScenarioId(&Scenario->ScenarioId)) {
FailedCheckId = 37;
goto cleanup;
}
//
// Initialize pointers to tables.
//
Sections = (PPF_SECTION_RECORD) ((PCHAR)Scenario + Scenario->SectionInfoOffset);
if ((ULONG_PTR)Sections & (_alignof(PF_SECTION_RECORD) - 1)) {
FailedCheckId = 38;
goto cleanup;
}
if (!PfWithinBounds(Sections, Scenario, BufferSize)) {
FailedCheckId = 40;
goto cleanup;
}
if (!PfWithinBounds((PCHAR) &Sections[Scenario->NumSections] - 1,
Scenario,
BufferSize)) {
FailedCheckId = 45;
goto cleanup;
}
Pages = (PPF_PAGE_RECORD) ((PCHAR)Scenario + Scenario->PageInfoOffset);
if ((ULONG_PTR)Pages & (_alignof(PF_PAGE_RECORD) - 1)) {
FailedCheckId = 47;
goto cleanup;
}
if (!PfWithinBounds(Pages, Scenario, BufferSize)) {
FailedCheckId = 50;
goto cleanup;
}
if (!PfWithinBounds((PCHAR) &Pages[Scenario->NumPages] - 1,
Scenario,
BufferSize)) {
FailedCheckId = 55;
goto cleanup;
}
FileNames = (PCHAR)Scenario + Scenario->FileNameInfoOffset;
if ((ULONG_PTR)FileNames & (_alignof(WCHAR) - 1)) {
FailedCheckId = 57;
goto cleanup;
}
if (!PfWithinBounds(FileNames, Scenario, BufferSize)) {
FailedCheckId = 60;
goto cleanup;
}
if (!PfWithinBounds(FileNames + Scenario->FileNameInfoSize - 1,
Scenario,
BufferSize)) {
FailedCheckId = 70;
goto cleanup;
}
MetadataInfoBase = (PCHAR)Scenario + Scenario->MetadataInfoOffset;
MetadataRecordTable = (PPF_METADATA_RECORD) MetadataInfoBase;
if ((ULONG_PTR)MetadataRecordTable & (_alignof(PF_METADATA_RECORD) - 1)) {
FailedCheckId = 72;
goto cleanup;
}
if (!PfWithinBounds(MetadataInfoBase, Scenario, BufferSize)) {
FailedCheckId = 73;
goto cleanup;
}
if (!PfWithinBounds(MetadataInfoBase + Scenario->MetadataInfoSize - 1,
Scenario,
BufferSize)) {
FailedCheckId = 74;
goto cleanup;
}
if (!PfWithinBounds(((PCHAR) &MetadataRecordTable[Scenario->NumMetadataRecords]) - 1,
Scenario,
BufferSize)) {
FailedCheckId = 75;
goto cleanup;
}
//
// Verify that sections contain valid information.
//
NumRemainingPages = Scenario->NumPages;
for (SectionIdx = 0; SectionIdx < Scenario->NumSections; SectionIdx++) {
pSection = &Sections[SectionIdx];
//
// Check if file name is within bounds.
//
pFileNameStart = FileNames + pSection->FileNameOffset;
if ((ULONG_PTR)pFileNameStart & (_alignof(WCHAR) - 1)) {
FailedCheckId = 77;
goto cleanup;
}
if (!PfWithinBounds(pFileNameStart, Scenario, BufferSize)) {
FailedCheckId = 80;
goto cleanup;
}
//
// Make sure there is a valid sized file name.
//
if (pSection->FileNameLength == 0) {
FailedCheckId = 90;
goto cleanup;
}
//
// Check file name max length.
//
if (pSection->FileNameLength > PF_MAXIMUM_SECTION_FILE_NAME_LENGTH) {
FailedCheckId = 100;
goto cleanup;
}
//
// Note that pFileNameEnd gets a -1 so it is the address of
// the last byte.
//
FileNameSize = (pSection->FileNameLength + 1) * sizeof(WCHAR);
pFileNameEnd = pFileNameStart + FileNameSize - 1;
if (!PfWithinBounds(pFileNameEnd, Scenario, BufferSize)) {
FailedCheckId = 110;
goto cleanup;
}
//
// Check if the file name is NUL terminated.
//
pwFileName = (PWCHAR) pFileNameStart;
if (pwFileName[pSection->FileNameLength] != 0) {
FailedCheckId = 120;
goto cleanup;
}
//
// Check max number of pages in a section.
//
if (pSection->NumPages > PF_MAXIMUM_SECTION_PAGES) {
FailedCheckId = 140;
goto cleanup;
}
//
// Make sure NumPages for the section is at least less
// than the remaining pages in the scenario. Then update the
// remaining pages.
//
if (pSection->NumPages > NumRemainingPages) {
FailedCheckId = 150;
goto cleanup;
}
NumRemainingPages -= pSection->NumPages;
//
// Verify that there are NumPages pages in our page list and
// they are sorted by file offset.
//
PageIdx = pSection->FirstPageIdx;
NumPages = 0;
PreviousPageIdx = PF_INVALID_PAGE_IDX;
while (PageIdx != PF_INVALID_PAGE_IDX) {
//
// Check that page idx is within range.
//
if (PageIdx < 0 || (ULONG) PageIdx >= Scenario->NumPages) {
FailedCheckId = 160;
goto cleanup;
}
//
// If this is not the first page record, make sure it
// comes after the previous one. We also check for
// duplicate offset here.
//
if (PreviousPageIdx != PF_INVALID_PAGE_IDX) {
if (Pages[PageIdx].FileOffset <=
Pages[PreviousPageIdx].FileOffset) {
FailedCheckId = 165;
goto cleanup;
}
}
//
// Update the last page index.
//
PreviousPageIdx = PageIdx;
//
// Get the next page index.
//
pPage = &Pages[PageIdx];
PageIdx = pPage->NextPageIdx;
//
// Update the number of pages we've seen on the list so
// far. If it is greater than what there should be on the
// list we have a problem. We may have even hit a list.
//
NumPages++;
if (NumPages > pSection->NumPages) {
FailedCheckId = 170;
goto cleanup;
}
}
//
// Make sure the section has exactly the number of pages it
// says it does.
//
if (NumPages != pSection->NumPages) {
FailedCheckId = 180;
goto cleanup;
}
}
//
// We should have accounted for all pages in the scenario.
//
if (NumRemainingPages) {
FailedCheckId = 190;
goto cleanup;
}
//
// Make sure metadata prefetch records make sense.
//
for (MetadataRecordIdx = 0;
MetadataRecordIdx < Scenario->NumMetadataRecords;
MetadataRecordIdx++) {
MetadataRecord = &MetadataRecordTable[MetadataRecordIdx];
//
// Make sure that the volume path is within bounds and NUL
// terminated.
//
VolumePath = (PWCHAR)(MetadataInfoBase + MetadataRecord->VolumeNameOffset);
if ((ULONG_PTR)VolumePath & (_alignof(WCHAR) - 1)) {
FailedCheckId = 195;
goto cleanup;
}
if (!PfWithinBounds(VolumePath, Scenario, BufferSize)) {
FailedCheckId = 200;
goto cleanup;
}
if (!PfWithinBounds(((PCHAR)(VolumePath + MetadataRecord->VolumeNameLength + 1)) - 1,
Scenario,
BufferSize)) {
FailedCheckId = 210;
goto cleanup;
}
if (VolumePath[MetadataRecord->VolumeNameLength] != 0) {
FailedCheckId = 220;
goto cleanup;
}
//
// Make sure that FilePrefetchInformation is within bounds.
//
FilePrefetchInfo = (PFILE_PREFETCH)
(MetadataInfoBase + MetadataRecord->FilePrefetchInfoOffset);
if ((ULONG_PTR)FilePrefetchInfo & (_alignof(FILE_PREFETCH) - 1)) {
FailedCheckId = 225;
goto cleanup;
}
if (!PfWithinBounds(FilePrefetchInfo, Scenario, BufferSize)) {
FailedCheckId = 230;
goto cleanup;
}
//
// Its size should be greater than size of a FILE_PREFETCH
// structure (so we can safely access the fields).
//
if (MetadataRecord->FilePrefetchInfoSize < sizeof(FILE_PREFETCH)) {
FailedCheckId = 240;
goto cleanup;
}
if (!PfWithinBounds((PCHAR)FilePrefetchInfo + MetadataRecord->FilePrefetchInfoSize - 1,
Scenario,
BufferSize)) {
FailedCheckId = 245;
goto cleanup;
}
//
// It should be for prefetching file creates.
//
if (FilePrefetchInfo->Type != FILE_PREFETCH_TYPE_FOR_CREATE) {
FailedCheckId = 250;
goto cleanup;
}
//
// There should not be more entries then are files and
// directories. The number of inidividual directories may be
// more than what we allow for, but it would be highly rare to
// be suspicious and thus ignored.
//
if (FilePrefetchInfo->Count > PF_MAXIMUM_DIRECTORIES + PF_MAXIMUM_SECTIONS) {
FailedCheckId = 260;
goto cleanup;
}
//
// Its size should match the size calculated by number of file
// index numbers specified in the header.
//
FilePrefetchInfoSize = sizeof(FILE_PREFETCH);
if (FilePrefetchInfo->Count) {
FilePrefetchInfoSize += (FilePrefetchInfo->Count - 1) * sizeof(ULONGLONG);
}
if (FilePrefetchInfoSize != MetadataRecord->FilePrefetchInfoSize) {
FailedCheckId = 270;
goto cleanup;
}
//
// Make sure that the directory paths for this volume make
// sense.
//
if (MetadataRecord->NumDirectories > PF_MAXIMUM_DIRECTORIES) {
FailedCheckId = 280;
goto cleanup;
}
DirectoryPath = (PPF_COUNTED_STRING)
(MetadataInfoBase + MetadataRecord->DirectoryPathsOffset);
if ((ULONG_PTR)DirectoryPath & (_alignof(PF_COUNTED_STRING) - 1)) {
FailedCheckId = 283;
goto cleanup;
}
for (DirectoryIdx = 0;
DirectoryIdx < MetadataRecord->NumDirectories;
DirectoryIdx ++) {
//
// Make sure head of the structure is within bounds.
//
if (!PfWithinBounds(DirectoryPath, Scenario, BufferSize)) {
FailedCheckId = 285;
goto cleanup;
}
if (!PfWithinBounds((PCHAR)DirectoryPath + sizeof(PF_COUNTED_STRING) - 1,
Scenario,
BufferSize)) {
FailedCheckId = 290;
goto cleanup;
}
//
// Check the length of the string.
//
if (DirectoryPath->Length >= PF_MAXIMUM_SECTION_FILE_NAME_LENGTH) {
FailedCheckId = 300;
goto cleanup;
}
//
// Make sure end of the string is within bounds.
//
if (!PfWithinBounds((PCHAR)(&DirectoryPath->String[DirectoryPath->Length + 1]) - 1,
Scenario,
BufferSize)) {
FailedCheckId = 310;
goto cleanup;
}
//
// Make sure the string is NUL terminated.
//
if (DirectoryPath->String[DirectoryPath->Length] != 0) {
FailedCheckId = 320;
goto cleanup;
}
//
// Set pointer to next DirectoryPath.
//
DirectoryPath = (PPF_COUNTED_STRING)
(&DirectoryPath->String[DirectoryPath->Length + 1]);
}
}
//
// We've passed all the checks.
//
ScenarioVerified = TRUE;
cleanup:
*FailedCheck = FailedCheckId;
return ScenarioVerified;
}
BOOLEAN
PfVerifyTraceBuffer(
PPF_TRACE_HEADER Trace,
ULONG BufferSize,
PULONG FailedCheck
)
/*++
Routine Description:
Verify offset and indices in a trace buffer are not beyond
bounds. This code is shared between the user mode service and
kernel mode component. If you update this function, update it in
both.
Arguments:
Trace - Base of Trace buffer.
BufferSize - Size of the scenario file / mapping.
FailedCheck - If verify failed, Id for the check that was failed.
Return Value:
TRUE - Trace is fine.
FALSE - Trace is corrupt;
--*/
{
LONG FailedCheckId;
PPF_LOG_ENTRY LogEntries;
PPF_SECTION_INFO Section;
PPF_VOLUME_INFO VolumeInfo;
ULONG SectionLength;
ULONG EntryIdx;
ULONG SectionIdx;
ULONG TotalFaults;
ULONG PeriodIdx;
ULONG VolumeIdx;
BOOLEAN TraceVerified;
ULONG VolumeInfoSize;
//
// Initialize locals:
//
FailedCheckId = 0;
//
// Initialize return value to FALSE. It will be set to TRUE only
// after all the checks pass.
//
TraceVerified = FALSE;
//
// The buffer should at least contain the scenario header.
//
if (BufferSize < sizeof(PF_TRACE_HEADER)) {
FailedCheckId = 10;
goto cleanup;
}
//
// Check trace header alignment.
//
if ((ULONG_PTR)Trace & (_alignof(PF_TRACE_HEADER) - 1)) {
FailedCheckId = 15;
goto cleanup;
}
//
// Check version and magic on the header.
//
if (Trace->Version != PF_CURRENT_VERSION ||
Trace->MagicNumber != PF_TRACE_MAGIC_NUMBER) {
FailedCheckId = 20;
goto cleanup;
}
//
// The buffer should not be greater than max allowed size.
//
if (BufferSize > PF_MAXIMUM_TRACE_SIZE) {
FailedCheckId = 23;
goto cleanup;
}
//
// Check for legal scenario type.
//
if (Trace->ScenarioType < 0 || Trace->ScenarioType >= PfMaxScenarioType) {
FailedCheckId = 25;
goto cleanup;
}
//
// Check limits on number of pages, sections etc.
//
if (Trace->NumSections > PF_MAXIMUM_SECTIONS ||
Trace->NumEntries > PF_MAXIMUM_LOG_ENTRIES ||
Trace->NumVolumes > PF_MAXIMUM_SECTIONS) {
FailedCheckId = 30;
goto cleanup;
}
//
// Check buffer size and the size of the trace.
//
if (Trace->Size != BufferSize) {
FailedCheckId = 35;
goto cleanup;
}
//
// Make sure the scenario id is valid.
//
if (!PfVerifyScenarioId(&Trace->ScenarioId)) {
FailedCheckId = 37;
goto cleanup;
}
//
// Check Bounds of Trace Buffer
//
LogEntries = (PPF_LOG_ENTRY) ((PCHAR)Trace + Trace->TraceBufferOffset);
if ((ULONG_PTR)LogEntries & (_alignof(PF_LOG_ENTRY) - 1)) {
FailedCheckId = 38;
goto cleanup;
}
if (!PfWithinBounds(LogEntries, Trace, BufferSize)) {
FailedCheckId = 40;
goto cleanup;
}
if (!PfWithinBounds((PCHAR)&LogEntries[Trace->NumEntries] - 1,
Trace,
BufferSize)) {
FailedCheckId = 50;
goto cleanup;
}
//
// Verify pages contain valid information.
//
for (EntryIdx = 0; EntryIdx < Trace->NumEntries; EntryIdx++) {
//
// Make sure sequence number is within bounds.
//
if (LogEntries[EntryIdx].SectionId >= Trace->NumSections) {
FailedCheckId = 60;
goto cleanup;
}
}
//
// Verify section info entries are valid.
//
Section = (PPF_SECTION_INFO) ((PCHAR)Trace + Trace->SectionInfoOffset);
if ((ULONG_PTR)Section & (_alignof(PF_SECTION_INFO) - 1)) {
FailedCheckId = 65;
goto cleanup;
}
for (SectionIdx = 0; SectionIdx < Trace->NumSections; SectionIdx++) {
//
// Make sure the section is within bounds.
//
if (!PfWithinBounds(Section, Trace, BufferSize)) {
FailedCheckId = 70;
goto cleanup;
}
if (!PfWithinBounds((PCHAR)Section + sizeof(PF_SECTION_INFO) - 1,
Trace,
BufferSize)) {
FailedCheckId = 75;
goto cleanup;
}
//
// Make sure the file name is not too big.
//
if(Section->FileNameLength > PF_MAXIMUM_SECTION_FILE_NAME_LENGTH) {
FailedCheckId = 80;
goto cleanup;
}
//
// Calculate size of this section entry.
//
SectionLength = sizeof(PF_SECTION_INFO) +
(Section->FileNameLength) * sizeof(WCHAR);
//
// Make sure all of the data in the section info is within
// bounds.
//
if (!PfWithinBounds((PUCHAR)Section + SectionLength - 1,
Trace,
BufferSize)) {
FailedCheckId = 90;
goto cleanup;
}
//
// Make sure the file name is NUL terminated.
//
if (Section->FileName[Section->FileNameLength] != 0) {
FailedCheckId = 100;
goto cleanup;
}
//
// Set pointer to next section.
//
Section = (PPF_SECTION_INFO) ((PUCHAR) Section + SectionLength);
}
//
// Check FaultsPerPeriod information.
//
TotalFaults = 0;
for (PeriodIdx = 0; PeriodIdx < PF_MAX_NUM_TRACE_PERIODS; PeriodIdx++) {
TotalFaults += Trace->FaultsPerPeriod[PeriodIdx];
}
if (TotalFaults != Trace->NumEntries) {
FailedCheckId = 120;
goto cleanup;
}
//
// Verify the volume information block.
//
VolumeInfo = (PPF_VOLUME_INFO) ((PCHAR)Trace + Trace->VolumeInfoOffset);
if ((ULONG_PTR)VolumeInfo & (_alignof(PF_VOLUME_INFO) - 1)) {
FailedCheckId = 125;
goto cleanup;
}
if (!PfWithinBounds(VolumeInfo, Trace, BufferSize)) {
FailedCheckId = 130;
goto cleanup;
}
if (!PfWithinBounds((PCHAR)VolumeInfo + Trace->VolumeInfoSize - 1,
Trace,
BufferSize)) {
FailedCheckId = 140;
goto cleanup;
}
//
// If there are sections, we should have at least one volume.
//
if (Trace->NumSections && !Trace->NumVolumes) {
FailedCheckId = 150;
goto cleanup;
}
//
// Verify the volume info structures per volume.
//
for (VolumeIdx = 0; VolumeIdx < Trace->NumVolumes; VolumeIdx++) {
//
// Make sure the whole volume structure is within bounds. Note
// that VolumeInfo structure contains space for the
// terminating NUL.
//
if (!PfWithinBounds(VolumeInfo, Trace, BufferSize)) {
FailedCheckId = 155;
goto cleanup;
}
if (!PfWithinBounds((PCHAR) VolumeInfo + sizeof(PF_VOLUME_INFO) - 1,
Trace,
BufferSize)) {
FailedCheckId = 160;
goto cleanup;
}
VolumeInfoSize = sizeof(PF_VOLUME_INFO);
VolumeInfoSize += VolumeInfo->VolumePathLength * sizeof(WCHAR);
if (!PfWithinBounds((PCHAR) VolumeInfo + VolumeInfoSize - 1,
Trace,
BufferSize)) {
FailedCheckId = 165;
goto cleanup;
}
//
// Verify that the volume path string is terminated.
//
if (VolumeInfo->VolumePath[VolumeInfo->VolumePathLength] != 0) {
FailedCheckId = 170;
goto cleanup;
}
//
// Get the next volume.
//
VolumeInfo = (PPF_VOLUME_INFO) ((PCHAR) VolumeInfo + VolumeInfoSize);
//
// Make sure VolumeInfo is aligned.
//
VolumeInfo = PF_ALIGN_UP(VolumeInfo, _alignof(PF_VOLUME_INFO));
}
//
// We've passed all the checks.
//
TraceVerified = TRUE;
cleanup:
*FailedCheck = FailedCheckId;
return TraceVerified;
}