windows-nt/Source/XPSP1/NT/base/pnp/setupapi/dll.c

2279 lines
54 KiB
C
Raw Normal View History

2020-09-26 03:20:57 -05:00
/*++
Copyright (c) Microsoft Corporation. All rights reserved.
Module Name:
dll.c
Abstract:
Initialization/de-initialization of setupapi.dll
Author:
Lonny McMichael (lonnym) 10-May-1995
Revision History:
--*/
#include "precomp.h"
#pragma hdrstop
#ifdef ANSI_SETUPAPI
#include <locale.h>
#endif
HANDLE MyDllModuleHandle;
OSVERSIONINFOEX OSVersionInfo;
//
// TLS Data
//
DWORD TlsIndex = (DWORD)(-1); // no data
PSETUP_TLS pLastTlsAlloc = NULL; // for cleanup
//
// Static strings we retreive once. Listed in order they are retrieved
//
PCTSTR OsLoaderRelativePath = NULL; // (can be NULL)
PCTSTR OsSystemPartitionRoot = NULL; // eg: \\?\GLOBALROOT\Device\HarddiskVolume1
PCTSTR ProcessFileName = NULL; // Filename of app calling setupapi
PCTSTR WindowsDirectory = NULL; // %windir%, GetSystemWindowsDirectory()
PCTSTR InfDirectory = NULL; // %windir%\INF
PCTSTR System16Directory = NULL; // %windir%\SYSTEM
PCTSTR LastGoodDirectory = NULL; // %windir%\LastGood
PCTSTR SystemDirectory = NULL; // <sys>, %windir%\SYSTEM or %windir%\System32
PCTSTR WindowsBackupDirectory = NULL; // <sys>\ReinstallBackups
PCTSTR ConfigDirectory = NULL; // <sys>\CONFIG
PCTSTR DriversDirectory = NULL; // <sys>\DRIVERS
PCTSTR SystemSourcePath = NULL; // location system installed from
PCTSTR ServicePackSourcePath = NULL; // location service pack installed from (can be NULL)
PCTSTR DriverCacheSourcePath = NULL; // location of driver cache (can be NULL)
BOOL GuiSetupInProgress = FALSE; // set if we determine we're in GUI setup
PCTSTR InfSearchPaths = NULL; // Multi-sz list of fully-qualified directories where INFs are to be searched for.
#ifndef _WIN64
BOOL IsWow64 = FALSE; // set if we're running under WOW64
#endif
CRITICAL_SECTION InitMutex; // for one-time initializations
CRITICAL_SECTION ImageHlpMutex; // for dealing with IMAGEHLP library
CRITICAL_SECTION DelayedComponentMutex; // for any delayed initialization of certain components
CRITICAL_SECTION PlatformPathOverrideCritSect;
CRITICAL_SECTION LogUseCountCs;
CRITICAL_SECTION MruCritSect;
CRITICAL_SECTION NetConnectionListCritSect;
BOOL InInitialization = FALSE;
DWORD DoneInitialization = 0; // bitmask of items we've initialized
DWORD DoneCleanup = 0; // bitmask of items we've cleaned up
DWORD DoneComponentInitialize = 0; // bitmask of components we've done delayed initialization
DWORD FailedComponentInitialize = 0; // bitmask of components we've failed initialization
HANDLE GlobalNoDriverPromptsEventFlag = NULL; // event that acts as a flag during setup
BOOL DoneCriticalSections = FALSE;
#ifdef UNICODE
DWORD Seed;
#endif
#define DONEINIT_TLS (0x0000001)
#define DONEINIT_UTILS (0x0000002)
#define DONEINIT_MEM (0x0000004)
#define DONEINIT_CTRL (0x0000008)
#define DONEINIT_FUSION (0x0000010)
#define DONEINIT_STUBS (0x0000020)
#define DONEINIT_COMMON (0x0000040)
#define DONEINIT_DIAMOND (0x0000080)
#define DONEINIT_LOGGING (0x0000100)
#define DONEINIT_CFGMGR32 (0x0000200)
#define DONEINIT_COMPLETE (0x8000000)
//
// various control flags
//
DWORD GlobalSetupFlags = 0;
DWORD GlobalSetupFlagsOverride = PSPGF_MINIMAL_EMBEDDED | PSPGF_NO_SCE_EMBEDDED; // flags that cannot be modified
//
// Declare a (non-CONST) array of strings that specifies what lines to look for
// in an INF's [ControlFlags] section when determining whether a particular device
// ID should be excluded. These lines are of the form "ExcludeFromSelect[.<suffix>]",
// where <suffix> is determined and filled in during process attach as an optimization.
//
// The max string length (including NULL) is 32, and there can be a maximum of 3
// such strings. E.g.: ExcludeFromSelect, ExcludeFromSelect.NT, ExcludeFromSelect.NTAlpha
//
// WARNING!! Be very careful when mucking with the order/number of these entries. Check
// the assumptions made in devdrv.c!pSetupShouldDevBeExcluded.
//
TCHAR pszExcludeFromSelectList[3][32] = { INFSTR_KEY_EXCLUDEFROMSELECT,
INFSTR_KEY_EXCLUDEFROMSELECT,
INFSTR_KEY_EXCLUDEFROMSELECT
};
DWORD ExcludeFromSelectListUb; // contains the number of strings in the above list (2 or 3).
#ifndef _WIN64
BOOL
GetIsWow64 (
VOID
);
#endif
BOOL
CommonProcessAttach(
IN BOOL Attach
);
PCTSTR
GetDriverCacheSourcePath(
VOID
);
PCTSTR
pSetupGetOsLoaderPath(
VOID
);
PCTSTR
pSetupGetSystemPartitionRoot(
VOID
);
PCTSTR
pSetupGetProcessPath(
VOID
);
BOOL
pGetGuiSetupInProgress(
VOID
);
BOOL
CfgmgrEntry(
PVOID hModule,
ULONG Reason,
PCONTEXT pContext
);
BOOL
ThreadTlsInitialize(
IN BOOL Init
);
VOID
ThreadTlsCleanup(
);
BOOL
IsNoDriverPrompts(
VOID
);
DWORD
GetEmbeddedFlags(
VOID
);
DWORD
GetSeed(
VOID
);
BOOL
ProcessAttach(
IN HANDLE DllHandle,
IN DWORD Reason,
IN LPVOID Reserved
)
/*++
Routine Description:
Handles DLL_PROCESS_ATTACH in a way that can be unwound
Arguments:
Reserved = 'Reserved' value passed into DllMain
Return Value:
TRUE if processed initialized
FALSE if partially/not initialized
--*/
{
BOOL b = FALSE;
if(DoneCleanup) {
//
// if we get here, this means someone lied to us
//
MYASSERT(!DoneCleanup);
DoneInitialization &= ~DoneCleanup;
DoneCleanup = 0;
}
//
// nothing should already be initialized
//
MYASSERT(!DoneInitialization);
try {
MyDllModuleHandle = DllHandle;
//
// initialize TLS before anything else - hence why we use LocalAlloc
//
TlsIndex = TlsAlloc();
if (TlsIndex!=(DWORD)(-1)) {
DoneInitialization |= DONEINIT_TLS;
} else {
leave;
}
#ifndef UNICODE
//
// Initialize the C runtime locale (ANSI version of setupapi.dll only)
//
setlocale(LC_ALL,"");
#endif
//
// always do pSetupInitializeUtils and MemoryInitializeEx first
// (pSetupInitializeUtils sets up memory functions)
//
if(pSetupInitializeUtils()) {
DoneInitialization |= DONEINIT_UTILS;
} else {
leave;
}
if(MemoryInitializeEx(TRUE)) {
DoneInitialization |= DONEINIT_MEM;
} else {
leave;
}
#ifdef FUSIONAWARE
if(spFusionInitialize()) {
DoneInitialization |= DONEINIT_FUSION;
}
#else
InitCommonControls();
DoneInitialization |= DONEINIT_CTRL;
#endif
//
// Dynamically load proc addresses of NT-specific APIs
// must be before CommonProcessAttach etc
// however memory must be initialized
//
InitializeStubFnPtrs();
DoneInitialization |= DONEINIT_STUBS;
//
// most of the remaining initialization
//
if(CommonProcessAttach(TRUE)) {
DoneInitialization |= DONEINIT_COMMON;
} else {
leave;
}
if(DiamondProcessAttach(TRUE)) {
DoneInitialization |= DONEINIT_DIAMOND;
} else {
leave;
}
if(InitializeContextLogging(TRUE)) {
DoneInitialization |= DONEINIT_LOGGING;
} else {
leave;
}
#ifdef UNICODE
//
// Since we've incorporated cfgmgr32 into setupapi, we need
// to make sure it gets initialized just like it did when it was
// its own DLL. - must do AFTER everything else
//
if(CfgmgrEntry(DllHandle, Reason, Reserved)) {
DoneInitialization |= DONEINIT_CFGMGR32;
}
#endif
DoneInitialization |= DONEINIT_COMPLETE;
b = TRUE;
} except(EXCEPTION_EXECUTE_HANDLER) {
}
return b;
}
void
DestroySetupTlsData(
)
/*++
Routine Description:
Destroy all TLS data from every thread
calling any cleanup routines as required
Arguments:
NONE
Return Value:
NONE
--*/
{
PSETUP_TLS pTLS;
if(pLastTlsAlloc) {
pLastTlsAlloc->Prev->Next = NULL;
while(pLastTlsAlloc) {
pTLS = pLastTlsAlloc;
pLastTlsAlloc = pTLS->Next;
TlsSetValue(TlsIndex,pTLS); // switch specific data into this thread
ThreadTlsCleanup();
LocalFree(pTLS);
}
}
TlsSetValue(TlsIndex,NULL); // don't leave invalid pointer hanging around
}
void
ProcessDetach(
IN HANDLE DllHandle,
IN DWORD Reason,
IN LPVOID Reserved
)
/*++
Routine Description:
Handles DLL_PROCESS_DETACH
Arguments:
Reserved = 'Reserved' value passed into DllMain
Which is actually TRUE for Process Exit, FALSE otherwise
Return Value:
None
--*/
{
DWORD ToCleanup = DoneInitialization & ~ DoneCleanup;
if(!ToCleanup) {
//
// nothing to cleanup
//
return;
}
try {
if (ToCleanup & DONEINIT_COMPLETE) {
DoneCleanup |= DONEINIT_COMPLETE;
}
if(DoneInitialization & DONEINIT_TLS) {
//
// cleanup all remaining Tls data
//
if(!Reserved) {
//
// only do this for FreeLibrary/ failed LoadLibrary
//
DestroySetupTlsData();
}
}
if(ToCleanup & DONEINIT_TLS) {
//
// destroy our allocated TLS index
// do this now so that we don't try allocating
// TLS storage during this cleanup
//
TlsFree(TlsIndex);
TlsIndex = (DWORD)(-1);
DoneCleanup |= DONEINIT_TLS;
}
//
// do things generally in reverse order of ProcessAttach
//
#ifdef UNICODE
// Since we've incorporated cfgmgr32 into setupapi, we need
// to make sure it gets uninitialized just like it did when it was
// its own DLL. - must do BEFORE anything else
//
if(ToCleanup & DONEINIT_CFGMGR32) {
CfgmgrEntry(DllHandle, Reason, Reserved);
DoneCleanup |= DONEINIT_CFGMGR32;
}
#endif
if(ToCleanup & DONEINIT_DIAMOND) {
DiamondProcessAttach(FALSE);
DoneCleanup |= DONEINIT_DIAMOND;
}
#if 0 // see ComponentCleanup at end of file
ComponentCleanup(DoneComponentInitialize);
#endif
#ifdef FUSIONAWARE
if(ToCleanup & DONEINIT_FUSION) {
//
// Fusion cleanup
// only do full if this is FreeLibrary (or failed attach)
//
spFusionUninitialize((Reserved == NULL) ? TRUE : FALSE);
DoneCleanup |= DONEINIT_FUSION;
}
#endif
if(ToCleanup & DONEINIT_COMMON) {
//
// Most of remaining cleanup
//
CommonProcessAttach(FALSE);
DoneCleanup |= DONEINIT_COMMON;
}
if(ToCleanup & DONEINIT_STUBS) {
//
// Clean up stub functions
//
CleanUpStubFns();
DoneCleanup |= DONEINIT_STUBS;
}
if(ToCleanup & DONEINIT_LOGGING) {
//
// Clean up context logging
//
InitializeContextLogging(FALSE);
DoneCleanup |= DONEINIT_LOGGING;
}
//
// *THESE MUST ALWAYS BE* very last things, in this order
//
if(ToCleanup & DONEINIT_MEM) {
//
// Clean up context logging
//
MemoryInitializeEx(FALSE);
DoneCleanup |= DONEINIT_MEM;
}
if(ToCleanup & DONEINIT_UTILS) {
//
// Clean up context logging
//
pSetupUninitializeUtils();
DoneCleanup |= DONEINIT_UTILS;
}
} except(EXCEPTION_EXECUTE_HANDLER) {
}
}
//
// Called by CRT when _DllMainCRTStartup is the DLL entry point
//
BOOL
WINAPI
DllMain(
IN HANDLE DllHandle,
IN DWORD Reason,
IN LPVOID Reserved
)
{
BOOL b;
InInitialization = TRUE;
b = TRUE;
switch(Reason) {
case DLL_PROCESS_ATTACH:
b = ProcessAttach(DllHandle,Reason,Reserved);
if(!b) {
ProcessDetach(DllHandle,DLL_THREAD_DETACH,NULL);
}
break;
case DLL_THREAD_ATTACH:
//
// don't do anything here
// any TLS must be done on demand
// via SetupGetTlsData
//
break;
case DLL_PROCESS_DETACH:
//
// any TLS cleanup must be done in ThreadTlsCleanup
//
ProcessDetach(DllHandle,Reason,Reserved);
break;
case DLL_THREAD_DETACH:
ThreadTlsInitialize(FALSE);
break;
}
InInitialization = FALSE;
return(b);
}
PSETUP_TLS
SetupGetTlsData(
)
/*++
Routine Description:
Called to obtain a pointer to TLS data
Arguments:
NONE
Return Value:
Pointer to TLS data, or NULL
--*/
{
PSETUP_TLS pTLS;
if (TlsIndex==(DWORD)(-1)) {
return NULL;
}
pTLS = (PSETUP_TLS)TlsGetValue(TlsIndex);
if (!pTLS) {
ThreadTlsInitialize(TRUE);
pTLS = (PSETUP_TLS)TlsGetValue(TlsIndex);
}
return pTLS;
}
VOID
ThreadTlsCleanup(
)
/*++
Routine Description:
Called to uninitialize some pTLS data
might be a different thread to initialize
but SetupAPI TLS data will have been switched in
Arguments:
pTLS - data to cleanup
Return Value:
NONE.
--*/
{
DiamondTlsInit(FALSE);
ContextLoggingTlsInit(FALSE);
}
BOOL
ThreadTlsUnlink(
IN PSETUP_TLS pTLS
)
{
BOOL b;
try {
EnterCriticalSection(&InitMutex);
if(pTLS->Next == pTLS->Prev) {
pLastTlsAlloc = NULL;
} else {
pTLS->Prev->Next = pTLS->Next;
pTLS->Next->Prev = pTLS->Prev;
pLastTlsAlloc = pTLS->Prev; // anything but pTLS
}
LeaveCriticalSection(&InitMutex);
b = TRUE;
} except(EXCEPTION_EXECUTE_HANDLER) {
b = FALSE;
}
return b;
}
PSETUP_TLS
ThreadTlsCreate(
)
/*++
Routine Description:
Called to create pTLS data for this thread
Arguments:
NONE
Return Value:
per-thread data, or NULL on failure
--*/
{
BOOL b;
PSETUP_TLS pTLS;
if (TlsIndex==(DWORD)(-1)) {
return NULL;
}
pTLS = (PSETUP_TLS)LocalAlloc(LMEM_ZEROINIT,sizeof(SETUP_TLS));
if(!pTLS) {
return NULL;
}
b = TlsSetValue(TlsIndex,pTLS);
if(!b) {
LocalFree(pTLS);
return NULL;
}
try {
EnterCriticalSection(&InitMutex);
if(pLastTlsAlloc) {
pTLS->Prev = pLastTlsAlloc;
pTLS->Next = pTLS->Prev->Next;
pTLS->Prev->Next = pTLS;
pTLS->Next->Prev = pTLS;
} else {
pTLS->Next = pTLS;
pTLS->Prev = pTLS;
}
pLastTlsAlloc = pTLS;
LeaveCriticalSection(&InitMutex);
b = TRUE;
} except(EXCEPTION_EXECUTE_HANDLER) {
b = FALSE;
}
if(!b) {
LocalFree(pTLS);
TlsSetValue(TlsIndex,NULL);
return NULL;
}
//
// TLS data initialized, now specific Init routines
//
b = DiamondTlsInit(TRUE);
if(b) {
b = ContextLoggingTlsInit(TRUE);
if(b) {
//
// all done ok
//
return pTLS;
}
//
// cleanup DiamondTlsInit
//
DiamondTlsInit(FALSE);
}
//
// cleanup memory
//
TlsSetValue(TlsIndex,NULL);
if(ThreadTlsUnlink(pTLS)) {
LocalFree(pTLS);
}
return NULL;
}
BOOL
ThreadTlsInitialize(
IN BOOL Init
)
/*++
Routine Description:
Called with TRUE to initialize TLS, if FALSE, to uninitialize
Arguments:
Init - indicates if we are to initialize vs uninitialize
Return Value:
NONE.
--*/
{
BOOL b = FALSE;
PSETUP_TLS pTLS = NULL;
if (TlsIndex!=(DWORD)(-1)) {
if (Init) {
pTLS = ThreadTlsCreate();
b = pTLS ? TRUE : FALSE;
} else {
pTLS = (PSETUP_TLS)TlsGetValue(TlsIndex);
if(pTLS) {
ThreadTlsCleanup();
TlsSetValue(TlsIndex,NULL);
if(ThreadTlsUnlink(pTLS)) {
LocalFree(pTLS);
}
}
b = TRUE;
}
}
return b;
}
BOOL
IsInteractiveWindowStation(
)
/*++
Routine Description:
Determine if we are running on an interactive station vs non-interactive station (i.e., service)
Arguments:
none
Return Value:
True if interactive
--*/
{
HWINSTA winsta;
USEROBJECTFLAGS flags;
BOOL interactive = TRUE; // true unless we determine otherwise
DWORD lenNeeded;
winsta = GetProcessWindowStation();
if(!winsta) {
return interactive;
}
if(GetUserObjectInformation(winsta,UOI_FLAGS,&flags,sizeof(flags),&lenNeeded)) {
interactive = (flags.dwFlags & WSF_VISIBLE) ? TRUE : FALSE;
}
//
// don't call CLoseWindowStation
//
return interactive;
}
BOOL
CommonProcessAttach(
IN BOOL Attach
)
{
BOOL b;
TCHAR Buffer[MAX_PATH+32];
PTCHAR p;
UINT u;
b = !Attach;
if(Attach) {
//
// remaining critical sections
//
try {
InitializeCriticalSection(&InitMutex);
InitializeCriticalSection(&ImageHlpMutex);
InitializeCriticalSection(&DelayedComponentMutex);
InitializeCriticalSection(&PlatformPathOverrideCritSect);
InitializeCriticalSection(&LogUseCountCs);
InitializeCriticalSection(&MruCritSect);
InitializeCriticalSection(&NetConnectionListCritSect);
DoneCriticalSections = TRUE;
} except (EXCEPTION_EXECUTE_HANDLER) {
//
// DoneCriticalSections remains FALSE
//
}
if(!DoneCriticalSections) {
return FALSE;
}
try {
#ifndef _WIN64
IsWow64 = GetIsWow64();
#endif
//
// flag indicating we're running in context of GUI setup
//
GuiSetupInProgress = pGetGuiSetupInProgress();
//
// determine if we're interactive or not
//
if(!IsInteractiveWindowStation()) {
GlobalSetupFlagsOverride |= PSPGF_NONINTERACTIVE; // don't allow this to be changed
GlobalSetupFlags |= PSPGF_NONINTERACTIVE; // actual value
}
if(IsNoDriverPrompts()) {
GlobalSetupFlagsOverride |= PSPGF_UNATTENDED_SETUP; // don't allow this to be changed
GlobalSetupFlags |= PSPGF_UNATTENDED_SETUP; // actual value
}
GlobalSetupFlags |= GetEmbeddedFlags();
#ifdef UNICODE
Seed = GetSeed();
#endif
pSetupInitNetConnectionList(TRUE);
pSetupInitPlatformPathOverrideSupport(TRUE);
OsLoaderRelativePath = pSetupGetOsLoaderPath(); // ok to fail
OsSystemPartitionRoot = pSetupGetSystemPartitionRoot(); // ok to fail
//
// Fill in system and windows directories.
//
if ((ProcessFileName = pSetupGetProcessPath()) == NULL) {
goto cleanAll;
}
//
// determine %windir%
//
if(((u = GetSystemWindowsDirectory(Buffer,MAX_PATH)) == 0) || u>MAX_PATH) {
goto cleanAll;
}
p = Buffer + u; // offset past directory to do all the sub-directories
//
// %windir% ==> WindowsDirectory
//
if((WindowsDirectory = DuplicateString(Buffer)) == NULL) {
goto cleanAll;
}
//
// %windir%\INF ==> InfDirectory
//
*p = 0;
if(!pSetupConcatenatePaths(Buffer,TEXT("INF"),MAX_PATH,NULL)
|| ((InfDirectory = DuplicateString(Buffer)) == NULL)) {
goto cleanAll;
}
//
// %windir%\SYSTEM ==> System16Directory
//
*p = 0;
if(!pSetupConcatenatePaths(Buffer,TEXT("SYSTEM"),MAX_PATH,NULL)
|| ((System16Directory = DuplicateString(Buffer))==NULL)) {
goto cleanAll;
}
//
// %windir%\LastGood ==> LastGoodDirectory
//
*p = 0;
if(!pSetupConcatenatePaths(Buffer,SP_LASTGOOD_NAME,MAX_PATH,NULL)
|| ((LastGoodDirectory = DuplicateString(Buffer))==NULL)) {
goto cleanAll;
}
//
// determine system directory
//
if(((u = GetSystemDirectory(Buffer,MAX_PATH)) == 0) || u>MAX_PATH) {
goto cleanAll;
}
p = Buffer + u; // offset past directory to do all the sub-directories
//
// <sys> ==> SystemDirectory (%windir%\System or %windir%\System32)
//
if((SystemDirectory = DuplicateString(Buffer)) == NULL) {
goto cleanAll;
}
//
// <sys>\ReinstallBackups ==> WindowsBackupDirectory
//
*p = 0;
if(!pSetupConcatenatePaths(Buffer,TEXT("ReinstallBackups"),MAX_PATH,NULL)
|| ((WindowsBackupDirectory = DuplicateString(Buffer))==NULL)) {
goto cleanAll;
}
//
// <sys>\CONFIG ==> ConfigDirectory
//
*p = 0;
if(!pSetupConcatenatePaths(Buffer,TEXT("CONFIG"),MAX_PATH,NULL)
|| ((ConfigDirectory = DuplicateString(Buffer))==NULL)) {
goto cleanAll;
}
//
// <sys>\DRIVERS ==> DriversDirectory
//
*p = 0;
if(!pSetupConcatenatePaths(Buffer,TEXT("DRIVERS"),MAX_PATH,NULL)
|| ((DriversDirectory = DuplicateString(Buffer))==NULL)) {
goto cleanAll;
}
//
// location system installed from
//
if((SystemSourcePath = GetSystemSourcePath())==NULL) {
goto cleanAll;
}
//
// location service pack installed from (may be NULL)
//
ServicePackSourcePath = GetServicePackSourcePath();
//
// location of driver cache (may be NULL)
//
DriverCacheSourcePath = GetDriverCacheSourcePath();
//
// determine driver search path
//
if((InfSearchPaths = AllocAndReturnDriverSearchList(INFINFO_INF_PATH_LIST_SEARCH))==NULL) {
goto cleanAll;
}
//
// note that InitMiniIconList, InitDrvSearchInProgressList, and
// InitDrvSignPolicyList need to be explicitly cleaned up on failure
//
//
// initialize mini icons
//
if(!InitMiniIconList()) {
goto cleanAll;
}
//
// allows aborting of search
//
if(!InitDrvSearchInProgressList()) {
DestroyMiniIconList();
goto cleanAll;
}
//
// global list of device setup classes subject to driver signing policy
//
if(!InitDrvSignPolicyList()) {
DestroyMiniIconList();
DestroyDrvSearchInProgressList();
goto cleanAll;
}
//
// common version initialization
//
ZeroMemory(&OSVersionInfo,sizeof(OSVersionInfo));
OSVersionInfo.dwOSVersionInfoSize = sizeof(OSVersionInfo);
if(!GetVersionEx((LPOSVERSIONINFO)&OSVersionInfo)) {
OSVersionInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
if(!GetVersionEx((LPOSVERSIONINFO)&OSVersionInfo)) {
//
// should never get here
//
MYASSERT(FALSE);
DestroyMiniIconList();
DestroyDrvSearchInProgressList();
DestroyDrvSignPolicyList();
goto cleanAll;
}
}
//
// Fill in our ExcludeFromSelect string list which
// we pre-compute as an optimization.
//
if(OSVersionInfo.dwPlatformId == VER_PLATFORM_WIN32_NT) {
lstrcat(pszExcludeFromSelectList[1],
pszNtSuffix
);
lstrcat(pszExcludeFromSelectList[2],
pszNtPlatformSuffix
);
ExcludeFromSelectListUb = 3;
} else {
lstrcat(pszExcludeFromSelectList[1],
pszWinSuffix
);
ExcludeFromSelectListUb = 2;
}
//
// Now lower-case all the strings in this list, so that it
// doesn't have to be done at each string table lookup.
//
for(u = 0; u < ExcludeFromSelectListUb; u++) {
CharLower(pszExcludeFromSelectList[u]);
}
b = TRUE;
cleanAll: ;
} except (EXCEPTION_EXECUTE_HANDLER) {
//
// Unexpected exception occurred, drop into cleanup
//
}
if(b) {
//
// succeeded
//
goto Done;
}
} else {
//
// Detach
//
DestroyMiniIconList();
DestroyDrvSearchInProgressList();
DestroyDrvSignPolicyList();
if(DoneCriticalSections) {
DeleteCriticalSection(&InitMutex);
DeleteCriticalSection(&ImageHlpMutex);
DeleteCriticalSection(&DelayedComponentMutex);
DeleteCriticalSection(&PlatformPathOverrideCritSect);
DeleteCriticalSection(&LogUseCountCs);
DeleteCriticalSection(&MruCritSect);
DeleteCriticalSection(&NetConnectionListCritSect);
}
if(GlobalNoDriverPromptsEventFlag) {
CloseHandle(GlobalNoDriverPromptsEventFlag);
}
}
if (InfSearchPaths) {
MyFree(InfSearchPaths);
InfSearchPaths = NULL;
}
if (DriverCacheSourcePath) {
MyFree(DriverCacheSourcePath);
DriverCacheSourcePath = NULL;
}
if (ServicePackSourcePath) {
MyFree(ServicePackSourcePath);
ServicePackSourcePath = NULL;
}
if (SystemSourcePath) {
MyFree(SystemSourcePath);
SystemSourcePath = NULL;
}
if (SystemDirectory) {
MyFree(SystemDirectory);
SystemDirectory = NULL;
if (WindowsBackupDirectory) {
MyFree(WindowsBackupDirectory);
WindowsBackupDirectory = NULL;
}
if (ConfigDirectory) {
MyFree(ConfigDirectory);
ConfigDirectory = NULL;
}
if (DriversDirectory) {
MyFree(DriversDirectory);
DriversDirectory = NULL;
}
}
if (WindowsDirectory) {
MyFree(WindowsDirectory);
WindowsDirectory = NULL;
if (InfDirectory) {
MyFree(InfDirectory);
InfDirectory = NULL;
}
if (System16Directory) {
MyFree(System16Directory);
System16Directory = NULL;
}
if (LastGoodDirectory) {
MyFree(LastGoodDirectory);
} LastGoodDirectory = NULL;
}
if (ProcessFileName) {
MyFree(ProcessFileName);
ProcessFileName = NULL;
}
if (OsLoaderRelativePath) {
MyFree(OsLoaderRelativePath);
OsLoaderRelativePath = NULL;
}
if (OsSystemPartitionRoot) {
MyFree(OsSystemPartitionRoot);
OsSystemPartitionRoot = NULL;
}
pSetupInitNetConnectionList(FALSE);
pSetupInitPlatformPathOverrideSupport(FALSE);
Done:
return(b);
}
#if MEM_DBG
#undef GetSystemSourcePath // defined again below
#endif
PCTSTR
GetSystemSourcePath(
TRACK_ARG_DECLARE
)
/*++
Routine Description:
This routine returns a newly-allocated buffer containing the source path from
which the system was installed, or "A:\" if that value cannot be determined.
This value is retrieved from the following registry location:
\HKLM\Software\Microsoft\Windows\CurrentVersion\Setup
SourcePath : REG_SZ : "\\ntalpha1\1300fre.wks" // for example.
Arguments:
None.
Return Value:
If the function succeeds, the return value is a pointer to the path string.
This memory must be freed via MyFree().
If the function fails due to out-of-memory, the return value is NULL.
--*/
{
HKEY hKey;
TCHAR CharBuffer[CSTRLEN(REGSTR_PATH_SETUP) + SIZECHARS(REGSTR_KEY_SETUP)];
DWORD Err, DataType, DataSize;
PTSTR Value;
PCTSTR ReturnVal;
TRACK_PUSH
CopyMemory(CharBuffer,
pszPathSetup,
sizeof(pszPathSetup) - sizeof(TCHAR)
);
CopyMemory((PBYTE)CharBuffer + (sizeof(pszPathSetup) - sizeof(TCHAR)),
pszKeySetup,
sizeof(pszKeySetup)
);
if((Err = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
CharBuffer,
0,
KEY_READ,
&hKey)) == ERROR_SUCCESS) {
//
// Attempt to read the the "SourcePath" value.
//
Err = QueryRegistryValue(hKey, pszSourcePath, &Value, &DataType, &DataSize);
RegCloseKey(hKey);
}
ReturnVal = NULL;
if(Err == NO_ERROR) {
ReturnVal = Value;
}
if(!ReturnVal && Err != ERROR_NOT_ENOUGH_MEMORY) {
//
// We failed to retrieve the SourcePath value, and it wasn't due to an out-of-memory
// condition. Fall back to our default of "A:\".
//
ReturnVal = DuplicateString(pszOemInfDefaultPath);
}
TRACK_POP
return ReturnVal;
}
#if MEM_DBG
#define GetSystemSourcePath() GetSystemSourcePath(TRACK_ARG_CALL)
#endif
#if MEM_DBG
#undef GetServicePackSourcePath // defined again below
#endif
PCTSTR
GetServicePackSourcePath(
TRACK_ARG_DECLARE
)
/*++
Routine Description:
This routine returns a newly-allocated buffer containing the service pack source path
where we should look for service pack source files, or "CDM" if that value cannot be determined.
This value is retrieved from the following registry location:
\HKLM\Software\Microsoft\Windows\CurrentVersion\Setup
ServicePackSourcePath : REG_SZ : "\\ntalpha1\1300fre.wks" // for example.
Arguments:
None.
Return Value:
If the function succeeds, the return value is a pointer to the path string.
This memory must be freed via MyFree().
If the function fails due to out-of-memory, the return value is NULL.
--*/
{
HKEY hKey;
TCHAR CharBuffer[CSTRLEN(REGSTR_PATH_SETUP) + SIZECHARS(REGSTR_KEY_SETUP)];
DWORD Err, DataType, DataSize;
PTSTR Value;
PCTSTR ReturnStr = NULL;
TRACK_PUSH
CopyMemory(CharBuffer,
pszPathSetup,
sizeof(pszPathSetup) - sizeof(TCHAR)
);
CopyMemory((PBYTE)CharBuffer + (sizeof(pszPathSetup) - sizeof(TCHAR)),
pszKeySetup,
sizeof(pszKeySetup)
);
if((Err = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
CharBuffer,
0,
KEY_READ,
&hKey)) == ERROR_SUCCESS) {
//
// Attempt to read the the "ServicePackSourcePath" value.
//
Err = QueryRegistryValue(hKey, pszSvcPackPath, &Value, &DataType, &DataSize);
RegCloseKey(hKey);
}
if(Err == NO_ERROR) {
ReturnStr = Value;
}
if(!ReturnStr && Err != ERROR_NOT_ENOUGH_MEMORY) {
//
// We failed to retrieve the ServicePackSourcePath value, and it wasn't due to an out-of-memory
// condition. Fall back to the SourcePath value in the registry
//
ReturnStr = GetSystemSourcePath();
}
TRACK_POP
return ReturnStr;
}
#if MEM_DBG
#define GetServicePackSourcePath() GetServicePackSourcePath(TRACK_ARG_CALL)
#endif
PCTSTR
GetDriverCacheSourcePath(
VOID
)
/*++
Routine Description:
This routine returns a newly-allocated buffer containing the source path to the local driver cache
cab-file.
This value is retrieved from the following registry location:
\HKLM\Software\Microsoft\Windows\CurrentVersion\Setup
DriverCachePath : REG_SZ : "\\ntalpha1\1300fre.wks" // for example.
Arguments:
None.
Return Value:
If the function succeeds, the return value is a pointer to the path string.
This memory must be freed via MyFree().
If the function fails due to out-of-memory, the return value is NULL.
--*/
{
HKEY hKey;
TCHAR CharBuffer[CSTRLEN(REGSTR_PATH_SETUP) + SIZECHARS(REGSTR_KEY_SETUP)];
DWORD Err, DataType, DataSize;
PTSTR Value;
TCHAR Path[MAX_PATH];
CopyMemory(CharBuffer,
pszPathSetup,
sizeof(pszPathSetup) - sizeof(TCHAR)
);
CopyMemory((PBYTE)CharBuffer + (sizeof(pszPathSetup) - sizeof(TCHAR)),
pszKeySetup,
sizeof(pszKeySetup)
);
if((Err = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
CharBuffer,
0,
KEY_READ,
&hKey)) == ERROR_SUCCESS) {
//
// Attempt to read the the "DriverCachePath" value.
//
Err = QueryRegistryValue(hKey, pszDriverCachePath, &Value, &DataType, &DataSize);
RegCloseKey(hKey);
}
if(Err == NO_ERROR) {
if(Value) {
ExpandEnvironmentStrings(Value,Path,MAX_PATH);
MyFree(Value);
Value = NULL;
if (*Path) {
Value = DuplicateString( Path );
}
return (PCTSTR)Value;
}
} else if(Err == ERROR_NOT_ENOUGH_MEMORY) {
return NULL;
}
return NULL;
}
BOOL
pSetupSetSystemSourcePath(
IN PCTSTR NewSourcePath,
IN PCTSTR NewSvcPackSourcePath
)
/*++
Routine Description:
This routine is used to override the system source path used by setupapi (as
retrieved by GetSystemSourcePath during DLL initialization). This is used by
syssetup.dll to set the system source path appropriately during GUI-mode setup,
so that the device installer APIs will copy files from the correct source location.
We do the same thing for the service pack source path
NOTE: This routine IS NOT thread safe!
Arguments:
NewSourcePath - supplies the new source path to be used.
NewSvcPackSourcePath - supplies the new svcpack source path to be used.
Return Value:
If the function succeeds, the return value is TRUE.
If the function fails (due to out-of-memory), the return value is FALSE.
--*/
{
PCTSTR p,q;
p = (PCTSTR)DuplicateString(NewSourcePath);
q = (PCTSTR)DuplicateString(NewSvcPackSourcePath);
if(p) {
MyFree(SystemSourcePath);
SystemSourcePath = p;
}
if (q) {
MyFree(ServicePackSourcePath);
ServicePackSourcePath = q;
}
if (!p || !q) {
//
// failed due to out of memory!
//
return(FALSE);
}
return TRUE;
}
PCTSTR
pSetupGetOsLoaderPath(
VOID
)
/*++
Routine Description:
This routine returns a newly-allocated buffer containing the path to the OsLoader
(relative to the system partition drive). This value is retrieved from the
following registry location:
HKLM\System\Setup
OsLoaderPath : REG_SZ : <path> // e.g., "\os\winnt40"
Arguments:
None.
Return Value:
If the registry entry is found, the return value is a pointer to the string containing
the path. The caller must free this buffer via MyFree().
If the registry entry is not found, or memory cannot be allocated for the buffer, the
return value is NULL.
--*/
{
HKEY hKey;
PTSTR Value;
DWORD Err, DataType, DataSize;
if(RegOpenKeyEx(HKEY_LOCAL_MACHINE,
TEXT("SYSTEM\\Setup"),
0,
KEY_READ,
&hKey) == ERROR_SUCCESS) {
Err = QueryRegistryValue(hKey, TEXT("OsLoaderPath"), &Value, &DataType, &DataSize);
RegCloseKey(hKey);
return (Err == NO_ERROR) ? (PCTSTR)Value : NULL;
}
return NULL;
}
PCTSTR
pSetupGetSystemPartitionRoot(
VOID
)
/*++
Routine Description:
This routine returns a newly-allocated buffer containing the path to the OsLoader
(relative to the system partition drive). This value is retrieved from the
following registry location:
HKLM\System\Setup
SystemPartition : REG_SZ : <path> // e.g., "\Device\HarddiskVolume1"
Arguments:
None.
Return Value:
If the registry entry is found, the return value is a pointer to the string containing
the path. The caller must free this buffer via MyFree().
If the registry entry is not found, or memory cannot be allocated for the buffer, the
return value is NULL.
--*/
{
#ifdef UNICODE
HKEY hKey;
PTSTR Value;
DWORD Err, DataType, DataSize;
TCHAR Path[MAX_PATH];
if(RegOpenKeyEx(HKEY_LOCAL_MACHINE,
TEXT("SYSTEM\\Setup"),
0,
KEY_READ,
&hKey) == ERROR_SUCCESS) {
Err = QueryRegistryValue(hKey, TEXT("SystemPartition"), &Value, &DataType, &DataSize);
RegCloseKey(hKey);
if(Err == NO_ERROR) {
//
// prepend \\?\GLOBALROOT\
//
lstrcpy(Path,TEXT("\\\\?\\GLOBALROOT\\"));
if(pSetupConcatenatePaths(Path,Value,MAX_PATH,NULL)) {
MyFree(Value);
Value = DuplicateString(Path);
if(!Value) {
Err = ERROR_NOT_ENOUGH_MEMORY;
}
} else {
Err = GetLastError();
MyFree(Value);
}
}
return (Err == NO_ERROR) ? (PCTSTR)Value : NULL;
}
#endif
return NULL;
}
PCTSTR
pSetupGetProcessPath(
VOID
)
/*++
Routine Description:
Get the name of the EXE that we're running in.
Arguments:
NONE.
Return Value:
Pointer to a dynamically allocated string containing the name.
--*/
{
LPTSTR modname;
modname = MyMalloc(MAX_PATH * sizeof(TCHAR));
if(modname != NULL) {
if(GetModuleFileName(NULL, modname, MAX_PATH) > 0) {
LPTSTR modname2;
modname2 = MyRealloc(modname, (lstrlen(modname)+1)*sizeof(TCHAR));
if(modname2) {
modname = modname2;
}
return modname;
} else {
#ifdef PRERELEASE
OutputDebugStringA("GetModuleFileName returned 0\r\n");
DebugBreak();
#endif
MyFree(modname);
}
}
return NULL;
}
#ifdef UNICODE
BOOL
pGetGuiSetupInProgress(
VOID
)
/*++
Routine Description:
This routine determines if we're doing a gui-mode setup.
This value is retrieved from the following registry location:
\HKLM\System\Setup\
SystemSetupInProgress : REG_DWORD : 0x00 (where nonzero means we're doing a gui-setup)
Arguments:
None.
Return Value:
If the function succeeds, the return value is a pointer to the path string.
This memory must be freed via MyFree().
If the function fails due to out-of-memory, the return value is NULL.
--*/
{
HKEY hKey;
TCHAR CharBuffer[CSTRLEN(REGSTR_PATH_SETUP) + SIZECHARS(REGSTR_KEY_SETUP)];
DWORD Err, DataType, DataSize = sizeof(DWORD);
DWORD Value;
if((Err = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
TEXT("System\\Setup"),
0,
KEY_READ,
&hKey)) == ERROR_SUCCESS) {
//
// Attempt to read the the "DriverCachePath" value.
//
Err = RegQueryValueEx(
hKey,
TEXT("SystemSetupInProgress"),
NULL,
&DataType,
(LPBYTE)&Value,
&DataSize);
RegCloseKey(hKey);
}
if(Err == NO_ERROR) {
if(Value) {
return(TRUE);
}
}
return(FALSE);
}
#else
BOOL
pGetGuiSetupInProgress(
VOID
)
{
return FALSE;
}
#endif
VOID pSetupSetGlobalFlags(
IN DWORD Value
)
/*++
exported as a private function
Routine Description:
Sets global flags to change certain setupapi features,
such as "should we call runonce after installing every device" (set if we will manually call it)
or "should we backup every file"
Used to initialize value
Arguments:
Value: combination of:
PSPGF_NO_RUNONCE - set to inhibit runonce calls (e.g., during GUI-
mode setup)
PSPGF_NO_BACKUP - set to inhibit automatic backup (e.g., during
GUI-mode setup)
PSPGF_NONINTERACTIVE - set to inhibit _all_ UI (e.g., for server-side
device installation)
PSPGF_SERVER_SIDE_RUNONCE - batch RunOnce entries for server-side
processing (for use only by umpnpmgr)
PSPGF_NO_VERIFY_INF - set to inhibit verification (digital signature) of
INF files until after the cyrpto DLLs have been registered.
PSPGF_UNATTENDED_SETUP - similar to NONINTERACTIVE - but specific to unattended setup
PSPGF_AUTOFAIL_VERIFIES - automatically fail all calls to crypto
Return Value:
none
--*/
{
pSetupModifyGlobalFlags((DWORD)(-1),Value);
}
VOID
pSetupModifyGlobalFlags(
IN DWORD Flags,
IN DWORD Value
)
/*++
exported as a private function
Routine Description:
Modifies global setup flags
such as "should we call runonce after installing every device" (set if we will manually call it)
or "should we backup every file"
only modifies specified flags to given value
Arguments:
Flags: what actual flags to modify, combination of:
PSPGF_NO_RUNONCE - set to inhibit runonce calls (e.g., during GUI-
mode setup)
PSPGF_NO_BACKUP - set to inhibit automatic backup (e.g., during
GUI-mode setup)
PSPGF_NONINTERACTIVE - set to inhibit _all_ UI (e.g., for server-side
device installation)
PSPGF_SERVER_SIDE_RUNONCE - batch RunOnce entries for server-side
processing (for use only by umpnpmgr)
PSPGF_NO_VERIFY_INF - set to inhibit verification (digital signature) of
INF files until after the cyrpto DLLs have been registered.
PSPGF_UNATTENDED_SETUP - similar to PSPGF_NONINTERACTIVE, but specific to setup
PSPGF_AUTOFAIL_VERIFIES - automatically fail all calls to crypto
Value: new value of bits specified in Flags
Return Value:
none
--*/
{
Flags &= ~GlobalSetupFlagsOverride; // exclusion
#ifdef UNICODE
if((Flags & PSPGF_NO_VERIFY_INF) && !(Value & PSPGF_NO_VERIFY_INF) && (GlobalSetupFlags & PSPGF_NO_VERIFY_INF)) {
Seed = GetSeed();
}
#endif
GlobalSetupFlags = (Value & Flags) | (GlobalSetupFlags & ~Flags);
}
DWORD pSetupGetGlobalFlags(
VOID
)
/*++
exported as a private function, also called internally
Routine Description:
Return flags previously set
Arguments:
none
Return value:
Flags (combination of values described above for pSetupSetGlobalFlags)
--*/
{
return GlobalSetupFlags;
}
BOOL
WINAPI
SetupSetNonInteractiveMode(
IN BOOL NonInteractiveFlag
)
/*++
Global access to the flag PSPGF_NONINTERACTIVE
Routine Description:
Set/Reset the NonInteractiveMode flag, and return previous flag value
(can't clear override)
Arguments:
New flag value
Return value:
Old flag value
--*/
{
BOOL f = (GlobalSetupFlags & PSPGF_NONINTERACTIVE) ? TRUE : FALSE;
if (NonInteractiveFlag) {
pSetupModifyGlobalFlags(PSPGF_NONINTERACTIVE,PSPGF_NONINTERACTIVE);
} else {
pSetupModifyGlobalFlags(PSPGF_NONINTERACTIVE,0);
}
return f;
}
BOOL
WINAPI
SetupGetNonInteractiveMode(
VOID
)
/*++
Global access to the flag PSPGF_NONINTERACTIVE
Routine Description:
Get current flag value
Arguments:
none
Return value:
Current flag value
--*/
{
return (GlobalSetupFlags & PSPGF_NONINTERACTIVE) ? TRUE : FALSE;
}
#ifndef _WIN64
BOOL
GetIsWow64 (
VOID
)
/*++
Routine Description:
Determine if we're running on WOW64 or not (not supported on ANSI builds)
Arguments:
none
Return value:
TRUE if running under WOW64 (and special Wow64 features available)
--*/
{
#ifdef UNICODE
ULONG_PTR ul = 0;
NTSTATUS st;
//
// If this call succeeds and sets ul non-zero
// it's a 32-bit process running on Win64
//
st = NtQueryInformationProcess(NtCurrentProcess(),
ProcessWow64Information,
&ul,
sizeof(ul),
NULL);
if (NT_SUCCESS(st) && (0 != ul)) {
// 32-bit code running on Win64
return TRUE;
}
#endif
return FALSE;
}
#endif // _WIN64
#if 0 // deleted code
//
// this will be useful at somepoint, but not used right now
//
BOOL
InitComponents(
DWORD Components
)
/*++
Routine Description:
Called at a point we want certain components initialized
Arguments:
bitmask of components to initialize
Return value:
TRUE if all initialized ok
--*/
{
BOOL success = FALSE;
PSETUP_TLS pPerThread = SetupGetTlsData();
BOOL locked = FALSE;
if(!pPerThread) {
MYASSERT(pPerThread);
return FALSE;
}
try {
EnterCriticalSection(&DelayedComponentMutex);
locked = TRUE;
Components &= ~ DoneComponentInitialize;
Components &= ~ pPerThread->PerThreadDoneComponent;
if (!Components) {
//
// already done
//
success = TRUE;
leave;
}
if (Components & FailedComponentInitialize) {
//
// previously failed
//
leave;
}
if (Components & pPerThread->PerThreadFailedComponent) {
//
// previously failed
//
leave;
}
MYASSERT(((DoneComponentInitialize | pPerThread->PerThreadDoneComponent) & Components) == Components);
MYASSERT(((FailedComponentInitialize | pPerThread->PerThreadFailedComponent) & Components) == 0);
success = TRUE;
} except (EXCEPTION_EXECUTE_HANDLER) {
MYASSERT(FALSE);
}
if(locked) {
LeaveCriticalSection(&DelayedComponentMutex);
}
return success;
}
VOID
ComponentCleanup(
DWORD Components
)
/*++
Routine Description:
Called to cleanup components
Arguments:
bitmask of components to uninitialize
Return value:
none
--*/
{
PSETUP_TLS pPerThread = SetupGetTlsData();
BOOL locked = FALSE;
try {
EnterCriticalSection(&DelayedComponentMutex);
locked = TRUE;
Components &= (DoneComponentInitialize | (pPerThread? pPerThread->PerThreadDoneComponent : 0));
if (!Components) {
//
// already done
//
leave;
}
} except (EXCEPTION_EXECUTE_HANDLER) {
MYASSERT(FALSE);
}
MYASSERT(((DoneComponentInitialize | (pPerThread ? pPerThread->PerThreadDoneComponent : 0)) & Components) == 0);
if(locked) {
LeaveCriticalSection(&DelayedComponentMutex);
}
}
#endif // deleted code
#ifdef UNICODE
BOOL
pSetupSetNoDriverPrompts(
BOOL Flag
)
/*++
exported as a private function
Routine Description:
Sets system into headless/non-headless mode
Arguments:
Flag - indicating mode
--*/
{
//
//
if(!GuiSetupInProgress) {
return FALSE;
}
if(GlobalNoDriverPromptsEventFlag == NULL) {
GlobalNoDriverPromptsEventFlag = CreateEvent(NULL,TRUE,Flag,SETUP_NODRIVERPROMPTS_MODE);
if(GlobalNoDriverPromptsEventFlag == NULL) {
return FALSE;
}
}
if(Flag) {
//
// force this process's setupapi to be non-interative, and any future setupapi's
//
GlobalSetupFlagsOverride |= PSPGF_UNATTENDED_SETUP; // don't allow this to be changed
GlobalSetupFlags |= PSPGF_UNATTENDED_SETUP; // actual value
SetEvent(GlobalNoDriverPromptsEventFlag);
} else {
//
// can't reset flag for this/existing processes, but can reset it for all future processes
//
ResetEvent(GlobalNoDriverPromptsEventFlag);
}
return TRUE;
}
#endif
BOOL
IsNoDriverPrompts(
VOID
)
/*++
internal
Routine Description:
Obtains headless state
Arguments:
Flag - indicating mode
--*/
{
#ifdef UNICODE
if(!GuiSetupInProgress) {
return FALSE;
}
if(GlobalNoDriverPromptsEventFlag == NULL) {
GlobalNoDriverPromptsEventFlag = OpenEvent(SYNCHRONIZE,FALSE,SETUP_NODRIVERPROMPTS_MODE);
if(GlobalNoDriverPromptsEventFlag == NULL) {
return FALSE;
}
}
//
// poll event, returning TRUE if it's signalled
//
return WaitForSingleObject(GlobalNoDriverPromptsEventFlag,0) == WAIT_OBJECT_0;
#else
return FALSE;
#endif
}
DWORD
GetEmbeddedFlags(
VOID
)
/*++
Routine Description:
This routine determines whether or not we are running on an embedded
product, and if so, whether:
* The "minimize setupapi footprint" option is enabled. This causes us to
modify our default behaviors as follows:
1. Never call any crypto APIs, and just assume everything is signed
2. Never generate PNFs.
* The "disable SCE" option is enabled. This causes us to avoid all use of
the Security Configuration Editor (SCE) routines (as the corresponding
DLLs won't be available on that embedded configuration).
Arguments:
none
Return value:
Combination of the following flags:
PSPGF_MINIMAL_EMBEDDED if we're running in "minimize footprint" mode
PSPGF_NO_SCE_EMBEDDED if we're running in "disable SCE" mode
--*/
{
OSVERSIONINFOEX osvix;
DWORDLONG dwlConditionMask = 0;
HKEY hKey;
TCHAR CharBuffer[CSTRLEN(REGSTR_PATH_SETUP) + SIZECHARS(REGSTR_KEY_SETUP)];
DWORD RegDataType, Data, DataSize;
DWORD Flags;
//
// Are we on the embedded product suite?
//
ZeroMemory(&osvix, sizeof(OSVERSIONINFOEX));
osvix.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
osvix.wSuiteMask = VER_SUITE_EMBEDDEDNT;
VER_SET_CONDITION(dwlConditionMask, VER_SUITENAME, VER_OR);
if(!VerifyVersionInfo(&osvix,
VER_SUITENAME,
dwlConditionMask)) {
return 0;
}
Flags = 0;
//
// OK, we running on embedded. Now we need to find out whether or not we
// should run in "minimal footprint" mode. This is stored in a REG_DWORD
// value entry called "MinimizeFootprint" under
// HKLM\Software\Microsoft\Windows\CurrentVersion\Setup.
//
CopyMemory(CharBuffer,
pszPathSetup,
sizeof(pszPathSetup) - sizeof(TCHAR)
);
CopyMemory((PBYTE)CharBuffer + (sizeof(pszPathSetup) - sizeof(TCHAR)),
pszKeySetup,
sizeof(pszKeySetup)
);
if(ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE,
CharBuffer,
0,
KEY_READ,
&hKey)) {
DataSize = sizeof(Data);
if((ERROR_SUCCESS == RegQueryValueEx(hKey,
pszMinimizeFootprint,
NULL,
&RegDataType,
(LPBYTE)&Data,
&DataSize))
&& (RegDataType == REG_DWORD) && (DataSize == sizeof(Data))) {
Flags |= PSPGF_MINIMAL_EMBEDDED;
}
RegCloseKey(hKey);
}
//
// Now look under HKLM\Software\Microsoft\EmbeddedNT\Security for a
// DisableSCE REG_DWORD value entry, indicating we shouldn't call SCE.
//
if(ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE,
pszEmbeddedNTSecurity,
0,
KEY_READ,
&hKey)) {
DataSize = sizeof(Data);
if((ERROR_SUCCESS == RegQueryValueEx(hKey,
pszDisableSCE,
NULL,
&RegDataType,
(LPBYTE)&Data,
&DataSize))
&& (RegDataType == REG_DWORD) && (DataSize == sizeof(Data))) {
Flags |= PSPGF_NO_SCE_EMBEDDED;
}
RegCloseKey(hKey);
}
return Flags;
}
#ifdef UNICODE
DWORD
GetSeed(
VOID
)
{
HKEY hKey;
DWORD val = 0;
DWORD valsize, valdatatype;
if(GlobalSetupFlags & PSPGF_MINIMAL_EMBEDDED) {
return val;
}
if(ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE,
L"System\\WPA\\PnP",
0,
KEY_READ,
&hKey)) {
valsize = sizeof(val);
if((ERROR_SUCCESS != RegQueryValueEx(hKey,
L"seed",
NULL,
&valdatatype,
(PBYTE)&val,
&valsize))
|| (valdatatype != REG_DWORD) || (valsize != sizeof(val))) {
val = 0;
}
RegCloseKey(hKey);
}
return val;
}
#endif