Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

3821 lines
99 KiB

/*++
Copyright (c) 1996 Microsoft Corporation
Module Name:
migmain.c
Abstract:
MigMain is called from w95upgnt.dll, which is called from SYSSETUP.DLL.
It is the main migration loop on the NT side of setup. MigMain loops
through all users on the Win95 side of configuration, migrates their
registry, creates their account, and fixes up their profile folders.
Then MigMain migrates all machine-specific settings such as link changes
and file deletion.
Author:
Jim Schmidt (jimschm) 12-Jul-1996
Revision History:
marcw 26-Mar-1999 More boot16 fixes -- hide msdos7 dir, fix to msdos.sys
marcw 18-Mar-1999 fixes for boot16 environment in localized case.
jimschm 23-Sep-1998 Changes for fileops & shell.c
calinn 23-Sep-1998 Changes for memdb fixup
jimschm 02-Jul-1998 Support for progress bar
jimschm 11-Jun-1998 Support for dynamic user profile paths in memdb
jimschm 05-May-1998 Migration of Default User if unattend option is enabled
jimschm 27-Apr-1998 Added icon preservation
jimschm 18-Mar-1998 Added pProcessAutoLogon
calinn 19-Nov-1997 Added pEnable16Boot, will create a 16 bit environment boot entry in boot.ini
jimschm 01-Oct-1997 Localized Everyone group
jimschm 13-Sep-1997 Reg Quota for beta 1 workaround
jimschm 21-Jul-1997 Use of fileops for ConvertWin9xPath (to be moved later)
jimschm 28-May-1997 Cleaned up
marcw 21-Mar-1997 added Pathmapping
jimschm 04-Feb-1997 Moved code into usermig.c and wkstamig.c
jimschm 15-Jan-1997 Plug-in spec modifications (now in migdlls.c)
jimschm 03-Jan-1997 Added g_UserName
jimschm 18-Dec-1996 Extracted code from miginf
mikeco O4-Dec-1996 Enumerate/modify PIF and LNK files
jimschm 23-Oct-1996 Joined ProcessUserInfs and ApplyChanges
jimschm 02-Oct-1996 Added default user migration
--*/
#include "pch.h"
#include "migmainp.h"
#include "fileops.h"
#include "quota.h"
#ifndef UNICODE
#error UNICODE required
#endif
#ifdef DEBUG
BOOL g_NoReloadsAllowed = FALSE;
#define DBG_VALIDNTFILES "NtFiles"
#endif
typedef BOOL (*PROFILE_PATH_PROVIDER)(OUT PTSTR AccountName, OUT PTSTR PathProfile);
//
// Globals for migmain.lib
//
HKEY g_hKeyRoot95, g_hKeyRootNT;
PCTSTR g_DomainUserName;
PCTSTR g_Win9xUserName;
PCTSTR g_FixedUserName;
PVOID g_HiveTable;
POOLHANDLE g_HivePool;
PVOID g_NulSessionTable;
PCTSTR g_EveryoneStr;
PCTSTR g_AdministratorsGroupStr;
PCTSTR g_PowerUsersGroupStr;
PCTSTR g_DomainUsersGroupStr;
PCTSTR g_NoneGroupStr;
TCHAR g_IconBin[MAX_TCHAR_PATH];
TCHAR g_DefaultUserName[MAX_USER_NAME];
TCHAR g_ComputerName[MAX_SERVER_NAME];
BOOL g_BlowAwayTempShellFolders = FALSE;
UINT g_Boot16 = BOOT16_AUTOMATIC;
//
// Buffer for GetString's messages
//
static TCHAR g_MsgBuf[2048];
//
// Flag identifying if the SKU is Personal
//
BOOL g_PersonalSKU = FALSE;
//
// Prototypes for migmain.c only
//
BOOL
pSetWin9xUpgValue (
VOID
);
VOID
pCountUsers (
OUT PDWORD TotalUsersPtr,
OUT PDWORD ActiveUsersPtr
);
BOOL
pMigrateUsers (
VOID
);
VOID
pRaiseRegistryQuota (
PCTSTR Win9xSystemDatSpec
);
VOID
pEnable16Boot (
VOID
);
VOID
pProcessAutoLogon (
BOOL Final
);
VOID
pFixUpMemDb (
VOID
);
BOOL
WINAPI
MigMain_Entry (
IN HINSTANCE hinstDLL,
IN DWORD dwReason,
IN PVOID lpv
)
/*++
Routine Description:
MigMain_Entry is called at DLL initialization time
Arguments:
hinstDLL - (OS-supplied) Instance handle for the DLL
dwReason - (OS-supplied) Type of initialization or termination
lpv - (OS-supplied) Unused
Return Value:
TRUE because LIB always initializes properly.
--*/
{
DWORD Size;
OSVERSIONINFOEX osviex;
switch (dwReason) {
case DLL_PROCESS_ATTACH:
if(!pSetupInitializeUtils()) {
return FALSE;
}
g_hKeyRoot95 = g_hKeyRootNT = NULL;
g_HivePool = PoolMemInitNamedPool ("Hive path pool");
if (!g_HivePool) {
return FALSE;
}
// Alloc string tables
g_HiveTable = pSetupStringTableInitializeEx (MAX_TCHAR_PATH,0);
if (!g_HiveTable) {
return FALSE;
}
g_NulSessionTable = pSetupStringTableInitializeEx (sizeof(PCWSTR), 0);
if (!g_NulSessionTable) {
return FALSE;
}
//
// Determine if upgrading to Personal SKU
//
osviex.dwOSVersionInfoSize = sizeof (OSVERSIONINFOEX);
if (!GetVersionEx ((LPOSVERSIONINFO)&osviex)) {
MYASSERT (FALSE);
}
if (osviex.wProductType == VER_NT_WORKSTATION &&
(osviex.wSuiteMask & VER_SUITE_PERSONAL)
) {
g_PersonalSKU = TRUE;
}
#if 0
if (g_PersonalSKU) {
g_EveryoneStr = GetStringResource (MSG_EVERYONE_GROUP);
g_AdministratorsGroupStr = GetStringResource (MSG_OWNERS_GROUP);
g_PowerUsersGroupStr = GetStringResource (MSG_POWER_USERS_GROUP);
g_DomainUsersGroupStr = GetStringResource (MSG_DOMAIN_USERS_GROUP);
g_NoneGroupStr = GetStringResource (MSG_NONE_GROUP);
} else {
#endif
g_EveryoneStr = GetStringResource (MSG_EVERYONE_GROUP);
g_AdministratorsGroupStr = GetStringResource (MSG_ADMINISTRATORS_GROUP);
g_PowerUsersGroupStr = GetStringResource (MSG_POWER_USERS_GROUP);
g_DomainUsersGroupStr = GetStringResource (MSG_DOMAIN_USERS_GROUP);
g_NoneGroupStr = GetStringResource (MSG_NONE_GROUP);
Size = ARRAYSIZE(g_ComputerName);
if (!GetComputerName (g_ComputerName, &Size)) {
g_ComputerName[0] = 0;
}
MYASSERT (
g_ComputerName[0] &&
g_EveryoneStr &&
g_AdministratorsGroupStr &&
g_PowerUsersGroupStr &&
g_DomainUsersGroupStr &&
g_NoneGroupStr
);
FindAccountInit();
break;
case DLL_PROCESS_DETACH:
if (g_HiveTable) {
pSetupStringTableDestroy (g_HiveTable);
}
if (g_NulSessionTable) {
pSetupStringTableDestroy (g_NulSessionTable);
}
if (g_HivePool) {
PoolMemDestroyPool (g_HivePool);
}
FreeStringResource (g_EveryoneStr);
FreeStringResource (g_AdministratorsGroupStr);
FreeStringResource (g_DomainUsersGroupStr);
FindAccountTerminate();
pSetupUninitializeUtils();
break;
}
return TRUE;
}
#ifdef DEBUG
BOOL
pValidateNtFiles (
VOID
)
/*++
Routine Description:
pValidateNtFiles validates the list of files that are supposed to be installed
by NT. We check for the flag set on Win95 side and for each entry we check to
see if the file is realy present (e.g. was installed by NT). If not then we delete
the entry.
Arguments:
none
Return Value:
Always returns TRUE
--*/
{
MEMDB_ENUMW enumFiles;
WCHAR key[MEMDB_MAX];
PWSTR fileName;
TREE_ENUMW enumTree;
DWORD value;
if (MemDbEnumFirstValue (
&enumFiles,
TEXT(MEMDB_CATEGORY_NT_FILESA)TEXT("\\*"),
MEMDB_ALL_SUBLEVELS,
MEMDB_ENDPOINTS_ONLY
)) {
do {
if (MemDbBuildKeyFromOffsetW (enumFiles.dwValue, key, 1, NULL)) {
fileName = JoinPaths (key, enumFiles.szName);
if (!DoesFileExistW (fileName)) {
MemDbSetValueEx (
MEMDB_CATEGORY_NT_FILES_BAD,
enumFiles.szName,
NULL,
NULL,
enumFiles.dwValue,
NULL
);
}
FreePathString (fileName);
}
ELSE_DEBUGMSG ((DBG_WHOOPS, "NT_FILES : cannot find installation directory."));
}
while (MemDbEnumNextValue (&enumFiles));
}
// now we have in MEMDB_CATEGORY_NT_FILES_BAD all files that should be installed
// by NT but are not. Now we are going to scan the file system and see if they are
// installed in a different place.
if (EnumFirstFileInTreeEx (&enumTree, g_WinDrive, TEXT("*.*"), FALSE, FALSE, FILE_ENUM_ALL_LEVELS)) {
do {
MemDbBuildKey (key, MEMDB_CATEGORY_NT_FILES_BAD, enumTree.Name, NULL, NULL);
if (MemDbGetValue (key, &value) && (value != 0)) {
MemDbSetValue (key, 0);
MemDbBuildKeyFromOffsetW (value, key, 1, NULL);
DEBUGMSG ((
DBG_VALIDNTFILES,
"%s listed to be installed in %s but installed in %s",
enumTree.Name,
key,
enumTree.FullPath));
}
} while (EnumNextFileInTree (&enumTree));
}
MemDbBuildKey (key, MEMDB_CATEGORY_NT_FILES_BAD, TEXT("*"), NULL, NULL);
if (MemDbEnumFirstValue (
&enumFiles,
key,
MEMDB_ALL_SUBLEVELS,
MEMDB_ENDPOINTS_ONLY
)) {
do {
if (enumFiles.dwValue) {
MemDbBuildKeyFromOffsetW (enumFiles.dwValue, key, 1, NULL);
DEBUGMSG ((
DBG_VALIDNTFILES,
"%s listed to be installed in %s but never installed",
enumFiles.szName,
key,
enumTree.FullPath));
}
}
while (MemDbEnumNextValue (&enumFiles));
}
return TRUE;
}
#endif
DWORD
pGetState (
IN PCTSTR Item
)
{
DWORD Value;
TCHAR Node[MEMDB_MAX];
MemDbBuildKey (Node, MEMDB_CATEGORY_STATE, Item, NULL, NULL);
if (MemDbGetValue (Node, &Value)) {
return Value;
}
return 0;
}
BOOL
MigMain_Init (
VOID
)
/*++
Routine Description:
MigMain_Init is called for initialization, and has a better opportunity
to fail than MigMain_Entry (which is called during DllMain).
Arguments:
none
Return Value:
TRUE if initialization succeeded, or FALSE if an error occurred.
Call GetLastError() for error code.
--*/
{
DWORD rc; // Temp: return code
TCHAR RelocWinDir[MAX_TCHAR_PATH];
TCHAR SrcResBin[MAX_TCHAR_PATH];
TCHAR IconFile[MAX_TCHAR_PATH];
ICON_EXTRACT_CONTEXT Context;
WORD CodePage;
LCID Locale;
TCHAR Node[MEMDB_MAX];
DWORD minorVersion;
#ifdef DEBUG
HANDLE hFile;
HKEY DebugKey = NULL;
CHAR Buf[32];
DWORD Value;
#endif
#ifdef PRERELEASE
//
// !!! This is for internal use only !!! It is used for auto stress.
//
if (g_ConfigOptions.AutoStress) {
SuppressAllLogPopups (TRUE);
}
#endif
//
// Dev: load c:\windbg.reg if it exists
//
#ifdef DEBUG
__try {
TCHAR WindbgRegPath[MAX_PATH] = TEXT("c:\\windbg.reg");
//
// Intentional hard-coded path!! This is for dev purposes only.
//
WindbgRegPath[0] = g_System32Dir[0];
if (!DoesFileExist (WindbgRegPath)) {
StringCopy (WindbgRegPath, TEXT("d:\\tools\\windbg.reg"));
}
hFile = CreateFile (
WindbgRegPath,
GENERIC_READ,
0,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL
);
if (hFile != INVALID_HANDLE_VALUE) {
CloseHandle (hFile);
rc = TrackedRegOpenKey (HKEY_CURRENT_USER, TEXT("Software\\Microsoft\\Windbg"), &DebugKey);
if (rc == ERROR_SUCCESS) {
DEBUGMSG ((DBG_VERBOSE, "Not restoring windbg.reg because it was already restored."));
__leave;
}
else {
rc = TrackedRegCreateKey (HKEY_CURRENT_USER, TEXT("Software\\Microsoft\\Windbg"), &DebugKey);
if (rc == ERROR_SUCCESS) {
if (!pSetupEnablePrivilege (SE_BACKUP_NAME, TRUE)) {
DEBUGMSG ((DBG_ERROR, "Windbg restore: pSetupEnablePrivilege SE_BACKUP_NAME failed"));
//__leave;
}
if (!pSetupEnablePrivilege (SE_RESTORE_NAME, TRUE)) {
DEBUGMSG ((DBG_ERROR, "Windbg restore: pSetupEnablePrivilege SE_RESTORE_NAME failed"));
//__leave;
}
rc = RegRestoreKey (DebugKey, WindbgRegPath, 0);
if (rc != ERROR_SUCCESS) {
DEBUGMSG ((DBG_WARNING, "Unable to restore windbg.reg, gle=%u", rc));
}
}
}
}
}
__finally {
if (DebugKey) {
CloseRegKey (DebugKey);
}
}
//
// If debug.inf has a line UserEnv=1, then add a registry key to debug userenv.dll
//
if (GetPrivateProfileStringA (
"Debug",
"UserEnv",
"0",
Buf,
sizeof (Buf) / sizeof (Buf[0]),
g_DebugInfPath
)
) {
if (atoi (Buf)) {
rc = TrackedRegCreateKey (
HKEY_LOCAL_MACHINE,
TEXT("Software\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon"),
&DebugKey
);
if (rc == ERROR_SUCCESS) {
Value = 0x00010002;
RegSetValueEx (
DebugKey,
TEXT("UserEnvDebugLevel"),
0,
REG_DWORD,
(PBYTE) &Value,
sizeof (DWORD)
);
CloseRegKey (DebugKey);
}
}
}
#endif
//
// Initialize the registry APIs
//
// We look in memdb for the location of .default
//
if (!MemDbLoad (GetMemDbDat())) {
LOG ((LOG_ERROR, "MigMain Init: MemDbLoad could not load %s", GetMemDbDat()));
return FALSE;
}
//
// Get platform name
//
if (!MemDbGetEndpointValueEx (
MEMDB_CATEGORY_STATE,
MEMDB_ITEM_PLATFORM_NAME,
NULL,
g_Win95Name
)) {
LOG ((LOG_ERROR, "Could not find product name for OS being upgraded."));
StringCopy (g_Win95Name, TEXT("Windows 95"));
}
// Try Paths\Windir first
if (!MemDbGetEndpointValueEx (
MEMDB_CATEGORY_PATHS,
MEMDB_ITEM_RELOC_WINDIR,
NULL,
RelocWinDir
)) {
LOG ((LOG_ERROR, "Could not find relocated windir!"));
return FALSE;
}
//
// if upgrading from Millennium, also map classes.dat
//
MemDbBuildKey (Node, MEMDB_CATEGORY_STATE, MEMDB_ITEM_MINOR_VERSION, NULL, NULL);
if (!MemDbGetValue (Node, &minorVersion)) {
LOG ((LOG_ERROR, "Could not get previous OS version information!"));
minorVersion = 0;
}
rc = Win95RegInit (RelocWinDir, minorVersion == 90);
if (rc != ERROR_SUCCESS) {
SetLastError (rc);
LOG ((LOG_ERROR, "Init Processor: Win95RegInit failed (path: %s)", RelocWinDir));
return FALSE;
}
//
// Update locale
//
CodePage = (WORD) pGetState (MEMDB_ITEM_CODE_PAGE);
Locale = (LCID) pGetState (MEMDB_ITEM_LOCALE);
SetGlobalCodePage (CodePage, Locale);
//
// Prepare path to system.dat, then raise registry quota if necessary
//
StringCopy (AppendWack (RelocWinDir), TEXT("system.dat"));
pRaiseRegistryQuota (RelocWinDir);
//
// Copy migisol.exe to migicons.exe
//
wsprintf (g_IconBin, TEXT("%s\\migicons.exe"), g_System32Dir);
wsprintf (SrcResBin, TEXT("%s\\migisol.exe"), g_TempDir);
if (!CopyFile (SrcResBin, g_IconBin, FALSE)) {
LOG ((LOG_ERROR, "Can't copy %s to %s", SrcResBin, g_IconBin));
}
else {
//
// Insert all icons from migicons.dat into g_IconBin
//
__try {
wsprintf (IconFile, TEXT("%s\\%s"), g_TempDir, S_MIGICONS_DAT);
if (!BeginIconExtraction (&Context, g_IconBin)) {
LOG ((LOG_ERROR, "Can't begin icon extraction"));
__leave;
}
if (!OpenIconImageFile (&Context, IconFile, FALSE)) {
LOG ((LOG_ERROR, "Can't open %s", IconFile));
__leave;
}
while (CopyIcon (&Context, NULL, NULL, 0)) {
// empty
}
}
__finally {
if (!EndIconExtraction (&Context)) {
DEBUGMSG ((DBG_WARNING, "EndIconExtraction failed"));
}
}
}
#ifdef DEBUG
// Validate MEMDB_CATEGORY_NT_FILES category. We need to find if the files
// that were supposed to be installed by NT are really there.
if (g_ConfigOptions.CheckNtFiles) {
pValidateNtFiles ();
}
#endif
return TRUE;
}
BOOL
MigMain_Migrate (
VOID
)
/*++
Routine Description:
MigMain_Migrate is the main migration function in NT GUI mode setup.
w95upgnt.dll calls this function, and it is here that users are migrated,
the local machine settings are migrated, and files are adjusted appropriately.
See the file progress.c for a list of functions that are called.
Arguments:
none
Return Value:
TRUE if migration succeeded, or FALSE if an error occurred. Call
GetLastError() for error code.
--*/
{
BOOL Result;
InitializeProgressBar (
g_ProgressBar,
NULL,
NULL,
NULL
);
PrepareMigrationProgressBar();
pProcessAutoLogon (FALSE);
g_BlowAwayTempShellFolders = TRUE;
Result = CallAllMigrationFunctions();
PushError();
if (Result) {
//
// Save logon prompt settings and set up auto-logon
//
pProcessAutoLogon (TRUE);
} else {
g_BlowAwayTempShellFolders = FALSE;
ClearAdminPassword();
}
//
// All done!
//
TerminateProgressBar();
PopError();
return Result;
}
DWORD
ResolveDomains (
DWORD Request
)
{
DWORD rc = ERROR_SUCCESS;
TCHAR unattendFile[MAX_TCHAR_PATH];
TCHAR buffer[32];
switch (Request) {
case REQUEST_QUERYTICKS:
if (!IsMemberOfDomain()) {
return 1;
}
return TICKS_DOMAIN_SEARCH;
case REQUEST_RUN:
//
// If autologon is enabled, then force classic mode
//
wsprintf (unattendFile, TEXT("%s\\system32\\$winnt$.inf"), g_WinDir);
if (GetPrivateProfileString (
TEXT("GuiUnattended"),
TEXT("AutoLogon"),
TEXT(""),
buffer,
ARRAYSIZE(buffer),
unattendFile
)) {
if (StringIMatch (buffer, TEXT("Yes"))) {
DEBUGMSG ((DBG_VERBOSE, "Found autologon; forcing classic logon type"));
SetClassicLogonType();
}
}
//
// Resolve the domains
//
if (!SearchDomainsForUserAccounts()) {
LOG ((LOG_ERROR, "An error occurred searching for user domains. The upgrade failed."));
rc = GetLastError();
} else {
//
// Fix up memdb for dynamic user profile paths
//
pFixUpMemDb();
}
if (IsMemberOfDomain()) {
TickProgressBarDelta (TICKS_DOMAIN_SEARCH);
} else {
TickProgressBar();
}
break;
}
return rc;
}
DWORD
PrepareEnvironment (
IN DWORD Request
)
{
DWORD rc = ERROR_SUCCESS;
switch (Request) {
case REQUEST_QUERYTICKS:
return TICKS_INIT;
case REQUEST_RUN:
//
// Disable Win 3.1 migration dialog
//
pSetWin9xUpgValue();
// Enable 16 bit environment boot
pEnable16Boot();
//
// Enable privileges (req'd for several things)
//
if (!pSetupEnablePrivilege (SE_BACKUP_NAME, TRUE)) {
LOG ((LOG_ERROR, "MigMain Migrate: pSetupEnablePrivilege SE_BACKUP_NAME failed"));
//rc = GetLastError();
//break;
}
if (!pSetupEnablePrivilege (SE_RESTORE_NAME, TRUE)) {
LOG ((LOG_ERROR, "MigMain Migrate: pSetupEnablePrivilege SE_RESTORE_NAME failed"));
//rc = GetLastError();
//break;
}
TickProgressBarDelta (TICKS_INIT);
break;
}
return rc;
}
BOOL
MigMain_Cleanup (
VOID
)
/*++
Routine Description:
MigMain_Cleanup is called to perform file removal. We delete everything
that is in the memdb category DelFile, and we also try to clean up after
MSN and other empty Win9x directories. Before exiting we delete our
temporary directory.
This function is called very last in Setup and is part of syssetup's
cleanup.
Arguments:
none
Return Value:
TRUE when all file deletes were successful, FALSE if an error occurred.
Call GetLastError for the reason of failure.
--*/
{
BOOL b = TRUE;
PCTSTR TempDir;
TCHAR normalPath[] = S_SHELL_TEMP_NORMAL_PATH;
TCHAR longPath[] = S_SHELL_TEMP_LONG_PATH;
DRIVELETTERS drives;
UINT u;
#ifdef DEBUG
INT n = 0;
#endif
// Remove everything in memdb's DelFile category
b = DoFileDel();
//
// Clean up any remaining directories that are now empty, including shell
// folder temp dirs
//
InitializeDriveLetterStructure (&drives);
if (!g_BlowAwayTempShellFolders) {
for (u = 0 ; u < NUMDRIVELETTERS ; u++) {
if (drives.ExistsOnSystem[u] && drives.Type[u] == DRIVE_FIXED) {
normalPath[0] = drives.Letter[u];
longPath[0] = drives.Letter[u];
MemDbSetValueEx (MEMDB_CATEGORY_CLEAN_UP_DIR, normalPath, NULL, NULL, 1, NULL);
MemDbSetValueEx (MEMDB_CATEGORY_CLEAN_UP_DIR, longPath, NULL, NULL, 1, NULL);
}
}
}
RemoveEmptyDirs();
if (!g_BlowAwayTempShellFolders) {
//
// Setup failed, clean up temp dir but leave it in place
//
for (u = 0 ; u < NUMDRIVELETTERS ; u++) {
if (drives.ExistsOnSystem[u] && drives.Type[u] == DRIVE_FIXED) {
normalPath[0] = drives.Letter[u];
longPath[0] = drives.Letter[u];
RemoveDirectory (normalPath);
if (DoesFileExist (normalPath)) {
LOG ((LOG_ERROR, (PCSTR) MSG_LEFT_TEMP_SHELL_FOLDERS, normalPath));
}
RemoveDirectory (longPath);
if (DoesFileExist (longPath)) {
LOG ((LOG_ERROR, (PCSTR) MSG_LEFT_TEMP_SHELL_FOLDERS, longPath));
}
}
}
} else {
//
// Setup was successful, blow away entire temp dir regardless of its content
//
for (u = 0 ; u < NUMDRIVELETTERS ; u++) {
if (drives.ExistsOnSystem[u] && drives.Type[u] == DRIVE_FIXED) {
normalPath[0] = drives.Letter[u];
longPath[0] = drives.Letter[u];
RemoveCompleteDirectory (normalPath);
DEBUGMSG_IF ((
DoesFileExist (normalPath),
DBG_ERROR,
"Temp dir cannot be removed: %s",
normalPath
));
RemoveCompleteDirectory (longPath);
DEBUGMSG_IF ((
DoesFileExist (longPath),
DBG_ERROR,
"Temp dir cannot be removed: %s",
longPath
));
}
}
}
#ifdef DEBUG
n = GetPrivateProfileIntA ("debug", "keeptempfiles", n, g_DebugInfPath);
if (n) {
return b;
}
#endif
if (g_ConfigOptions.KeepTempFiles) {
return b;
}
//
// Blow away temp dir
//
TempDir = JoinPaths (g_WinDir, S_SETUP);
b = DeleteDirectoryContents (TempDir);
if (b) {
b = RemoveDirectory (TempDir);
if (!b) {
LOG ((LOG_ERROR, "Could not delete the tree %s.", TempDir));
}
}
else {
LOG ((LOG_ERROR, "Could not delete the contents of %s.", TempDir));
}
FreePathString (TempDir);
return b;
}
PCTSTR
GetMemDbDat (
VOID
)
/*++
Routine Description:
Returns a pointer to the path of the DAT file holding the Win9x memdb tree.
Arguments:
none
Return Value:
Returns a pointer to the Win32 path of ntsetup.dat.
--*/
{
static TCHAR FileName[MAX_TCHAR_PATH];
MYASSERT (!g_NoReloadsAllowed);
StringCopy (FileName, g_TempDir);
StringCopy (AppendWack (FileName), S_NTSETUPDAT);
return FileName;
}
PCTSTR
GetUserDatLocation (
IN PCTSTR User,
OUT PBOOL CreateOnlyFlag OPTIONAL
)
/*++
Routine Description:
Looks in memdb to locate the user.dat file for the specified user. On
Win9x, migapp.lib writes a line to memdb giving the location of user.dat
for each user, and the default user. This function retrieves that
location to guarantee the same file is used on both NT and Win9x.
Arguments:
User - The fixed name of the user to process, or NULL for the default user.
CreateOnlyFlag - Receives the create-only flag specified on the Win9x side
of the upgrade. If this flag is TRUE, then the account
should not be migrated.
Return Value:
Returns a pointer to the Win32 path of user.dat for the given user.
If the entry does not exist, NULL will be returned, and the user
will not be processed.
--*/
{
MEMDB_ENUM e;
static TCHAR UserDatLocation[MAX_TCHAR_PATH];
if (!MemDbGetValueEx (&e, MEMDB_CATEGORY_USER_DAT_LOC, User, NULL)) {
if (!StringIMatch (User, g_AdministratorStr)) {
DEBUGMSG ((DBG_WARNING, "'UserDatLocation' for %s does not exist.", User?User:S_DOT_DEFAULT));
}
return NULL;
}
StringCopy (UserDatLocation, e.szName);
if (CreateOnlyFlag) {
*CreateOnlyFlag = (BOOL) e.dwValue;
}
return UserDatLocation;
}
VOID
pSaveVersionStr (
IN HKEY Key,
IN PCTSTR Name
)
{
TCHAR Data[MEMDB_MAX];
if (MemDbGetEndpointValueEx (MEMDB_CATEGORY_STATE, Name, NULL, Data)) {
RegSetValueEx (
Key,
Name,
0,
REG_SZ,
(PBYTE) Data,
SizeOfString (Data)
);
}
}
VOID
pSaveVersionDword (
IN HKEY Key,
IN PCTSTR Name
)
{
DWORD Data;
TCHAR Node[MEMDB_MAX];
MemDbBuildKey (Node, MEMDB_CATEGORY_STATE, Name, NULL, NULL);
if (MemDbGetValue (Node, &Data)) {
RegSetValueEx (
Key,
Name,
0,
REG_DWORD,
(PBYTE) &Data,
sizeof (Data)
);
}
}
BOOL
pSetWin9xUpgValue (
VOID
)
/*++
Routine Description:
Create the value entry Win9xUpg on
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\WinLogon
This routine should always be called when setup is installing an NT system
on top of Win9x, otherwise NT will think it has to migrate Win 3.1.
Arguments:
None.
Return Value:
Returns TRUE if the opearation succeeds.
--*/
{
ULONG Error;
HKEY Key;
DWORD Value;
HKEY VersionKey;
Key = OpenRegKeyStr (S_WINLOGON_REGKEY);
if (!Key) {
return FALSE;
}
Value = 1;
Error = RegSetValueEx (
Key,
S_WIN9XUPG_FLAG_VALNAME,
0,
REG_DWORD,
(PBYTE) &Value,
sizeof (DWORD)
);
//
// Save the version info
//
VersionKey = CreateRegKey (Key, TEXT("PrevOsVersion"));
if (VersionKey) {
pSaveVersionStr (VersionKey, MEMDB_ITEM_PLATFORM_NAME);
pSaveVersionStr (VersionKey, MEMDB_ITEM_VERSION_TEXT);
pSaveVersionDword (VersionKey, MEMDB_ITEM_MAJOR_VERSION);
pSaveVersionDword (VersionKey, MEMDB_ITEM_MINOR_VERSION);
pSaveVersionDword (VersionKey, MEMDB_ITEM_BUILD_NUMBER);
pSaveVersionDword (VersionKey, MEMDB_ITEM_PLATFORM_ID);
CloseRegKey (VersionKey);
}
CloseRegKey (Key);
if (Error != ERROR_SUCCESS) {
SetLastError (Error);
return FALSE;
}
return TRUE;
}
PCTSTR
GetString (
WORD wMsg
)
/*++
Routine Description:
Load the string resource given in wMsg and copy it to a global string
buffer. Return the a pointer to the buffer.
Arguments:
wMsg - The identifier of the message to load.
Return Value:
Returns a pointer to the loaded message, or NULL. The message must be
smaller than 2048 characters.
--*/
{
PCTSTR String;
String = GetStringResource (wMsg);
if (!String) {
return TEXT("Error: String resource could not be loaded");
}
_tcssafecpy (g_MsgBuf, String, ARRAYSIZE(g_MsgBuf));
FreeStringResource (String);
return g_MsgBuf;
}
VOID
pCountUsers (
OUT PDWORD TotalUsersPtr,
OUT PDWORD ActiveUsersPtr
)
/*++
Routine Description:
Counts all Win9x users, and determines how many of them are active
for migration. The count includes the Administrator account, the
logon prompt account, and optional default user account.
NOTE: Administrator may be counted twice in ActiveUsersPtr, once for
a real Win9x user named Administrator, and again for the
NT Administrator account that is always migrated. The caller
must handle this special case.
Arguments:
TotalUsersPtr - A DWORD that receives the total number of Win9x users,
including the NT-only users.
ActiveUsersPtr - A DWORD that receives the number of users that require
migration. Migration may or may not be enabled for any
user.
Return Value:
none
--*/
{
USERPOSITION up;
TCHAR User[MAX_TCHAR_PATH];
DWORD rc;
PCTSTR UserDatLocation;
*ActiveUsersPtr = 0;
*TotalUsersPtr = 3; // include logon, default and administrator in the total
rc = Win95RegGetFirstUser (&up, User);
if (rc != ERROR_SUCCESS) {
*TotalUsersPtr = 0;
return;
}
while (Win95RegHaveUser (&up)) {
GetFixedUserName (User);
// see if this user requires migration
UserDatLocation = GetUserDatLocation (User, NULL);
if (UserDatLocation) {
*ActiveUsersPtr += 1;
}
// count all users, migrated and non-migrated
*TotalUsersPtr += 1;
Win95RegGetNextUser (&up, User);
}
// test migration requirement of default user and adminsistrator
UserDatLocation = GetUserDatLocation (g_AdministratorStr, NULL);
if (UserDatLocation) {
*ActiveUsersPtr += 1;
}
UserDatLocation = GetUserDatLocation (S_DOT_DEFAULT, NULL);
if (UserDatLocation) {
*ActiveUsersPtr += 1;
}
if (g_ConfigOptions.MigrateDefaultUser) {
*ActiveUsersPtr += 1;
}
DEBUGMSG ((DBG_VERBOSE, "pCountUsers: %u users, %u require migration", *TotalUsersPtr, *ActiveUsersPtr));
}
CONVERTPATH_RC
ConvertWin9xPath (
PTSTR PathBuf
)
{
TCHAR Buffer[MEMDB_MAX];
DWORD status;
status = GetFileInfoOnNt (PathBuf, Buffer, MEMDB_MAX);
if (status & FILESTATUS_REPLACED) {
if (status & FILESTATUS_MOVED) {
_tcssafecpy (PathBuf, Buffer, MAX_TCHAR_PATH);
return CONVERTPATH_REMAPPED;
}
return CONVERTPATH_NOT_REMAPPED;
}
if (status & FILESTATUS_MOVED) {
_tcssafecpy (PathBuf, Buffer, MAX_TCHAR_PATH);
return CONVERTPATH_REMAPPED;
}
if (status & FILESTATUS_DELETED) {
return CONVERTPATH_DELETED;
}
return CONVERTPATH_NOT_REMAPPED;
}
VOID
pRaiseRegistryQuota (
PCTSTR Win9xSystemDatSpec
)
{
NTSTATUS Status;
SYSTEM_REGISTRY_QUOTA_INFORMATION RegQuotaInfo;
HANDLE FileHandle;
DWORD QuotaNeeded;
ULARGE_INTEGER FreeBytes, dc1, dc2;
LONGLONG FreeBytesNeeded;
HKEY SaveKey;
DWORD rc;
#ifdef PAGED_POOL_INCREASE
SYSTEM_POOL_INFORMATION PoolInfo;
//
// Obtain current system settings
//
Status = NtQuerySystemInformation (
SystemPagedPoolInformation,
(PVOID) &PoolInfo,
sizeof(PoolInfo),
NULL
);
if (Status != ERROR_SUCCESS) {
LOG ((LOG_ERROR, "Cannot obtain PoolInfo"));
return;
}
#endif
pSetupEnablePrivilege (SE_INCREASE_QUOTA_NAME, TRUE);
Status = NtQuerySystemInformation (
SystemRegistryQuotaInformation,
(PVOID) &RegQuotaInfo,
sizeof(RegQuotaInfo),
NULL
);
if (Status != ERROR_SUCCESS) {
LOG ((LOG_ERROR, "Cannot obtain RegQuotaInfo"));
return;
}
//
// Obtain Win9x registry system.dat size
//
FileHandle = CreateFile (
Win9xSystemDatSpec,
GENERIC_READ,
FILE_SHARE_READ,
NULL, // security attributes
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL // template file
);
if (FileHandle == INVALID_HANDLE_VALUE) {
LOG ((LOG_ERROR, "Cannot open %s; cannot raise registry quota", Win9xSystemDatSpec));
return;
}
QuotaNeeded = GetFileSize (FileHandle, NULL);
CloseHandle (FileHandle);
if (QuotaNeeded > 0x3fffffff) {
LOG ((LOG_ERROR, "Cannot obtain size for %s; cannot raise registry quota", Win9xSystemDatSpec));
return;
}
QuotaNeeded *= 6;
//
// Get free disk space on boot drive
//
if (!GetDiskFreeSpaceEx (
g_WinDir,
&FreeBytes,
&dc1,
&dc2
)) {
LOG ((LOG_ERROR, "Can't get free space on drive holding %s; cannot raise registry quota", g_WinDir));
return;
}
//
// Lots of disk space? Raise paged pool by 5 times the size of system.dat.
// Example: Win9x system.dat is 5M; must have 150M free to raise paged pool.
//
FreeBytesNeeded = (LONGLONG) QuotaNeeded * (LONGLONG) 6;
if (FreeBytes.QuadPart >= (DWORDLONG) FreeBytesNeeded) {
//
// Unimplemented: Raise the paged pool and return
//
DEBUGMSG ((DBG_WARNING, "RegQuota: Really should be raising paged pool -- this machine has %u bytes free", FreeBytes.LowPart));
}
//
// Last resort: raise the registry quota (if necessary)
//
if (RegQuotaInfo.RegistryQuotaAllowed < QuotaNeeded) {
DEBUGMSG ((DBG_VERBOSE, "Raising registry quota from %u to %u", RegQuotaInfo.RegistryQuotaAllowed, QuotaNeeded));
RegQuotaInfo.RegistryQuotaAllowed = QuotaNeeded;
Status = NtSetSystemInformation (
SystemRegistryQuotaInformation,
&RegQuotaInfo,
sizeof (RegQuotaInfo)
);
if (Status != ERROR_SUCCESS) {
LOG ((LOG_ERROR, "Can't set raised registry quota"));
}
//
// Set a permanent value in the registry
//
SaveKey = OpenRegKeyStr (TEXT("HKLM\\System\\CurrentControlSet\\Control"));
if (SaveKey) {
rc = RegSetValueEx (
SaveKey,
TEXT("RegistrySizeLimit"),
0,
REG_DWORD,
(PBYTE) &QuotaNeeded,
sizeof (DWORD)
);
CloseRegKey (SaveKey);
if (rc != ERROR_SUCCESS) {
LOG ((LOG_ERROR, "Could not set HKLM\\System\\CurrentControlSet\\Control [RegistrySizeLimit]"));
}
}
ELSE_DEBUGMSG ((DBG_ERROR, "Can't open HKLM\\System\\CurrentControlSet\\Control"));
}
}
BOOL
pCopyDosFile (
IN PCTSTR FileName,
IN BOOL InRootDir
)
/*++
Routine Description:
Copies a file from %windir%\setup\msdos7 into the designated DOS directory
Arguments:
FileName - file to copy (no path).
Return Value:
TRUE if succeeded, FALSE if not
--*/
{
PTSTR sourcePath;
PTSTR sourceFileName;
PTSTR destPath;
PTSTR destFileName;
BOOL result;
sourcePath = JoinPaths (g_TempDir, S_BOOT16_DOS_DIR);
sourceFileName = JoinPaths (sourcePath, FileName);
if (InRootDir) {
destPath = NULL;
destFileName = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
FileName);
}
else {
destPath = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_DOS_DIR);
destFileName = JoinPaths (destPath, FileName);
}
SetFileAttributes (destFileName, FILE_ATTRIBUTE_NORMAL);
result = CopyFile (sourceFileName, destFileName, FALSE);
FreePathString (sourcePath);
FreePathString (sourceFileName);
if (destPath != NULL) {
FreePathString (destPath);
}
FreePathString (destFileName);
return result;
}
VOID
pWriteBoot16ConfigLines (
IN HANDLE File,
IN PCTSTR BaseSection,
IN PCTSTR DosPath,
IN BOOL Localized
)
/*++
Routine Description:
pWriteBoot16ConfigLines reads configuration lines from wkstamig.inf and
writes them to the specified file handle. The caller can control wether the
lines should contain first boot only items or not and can control wether to
read in the base dos lines (same for all languages) or special lines used
for specific languages.
Arguments:
File - An opened handle with appropriate access to the file where
the data should be written.
BaseSection - Contains the Base Section name to read from the INF. This
section may be modified with a code page if Localized is TRUE.
DosPath - Contains the full path to the dos boot files (typically
c:\msdos7)
Localized - Controls wether data from the localized section is read. If
this parameter is TRUE, then the code page will be appended to
the BaseSection string for purposes of reading from wkstamig.inf.
Return Value:
none
++*/
{
INFSTRUCT is = INITINFSTRUCT_GROWBUFFER;
GROWLIST list = GROWLIST_INIT;
PTSTR line;
TCHAR codePageSection[MAX_TCHAR_PATH];
USHORT codePage;
PCTSTR infSection;
//
// Add boot16 line specific environment variables.
//
GrowListAppendString (&list, TEXT("BOOTDRIVE"));
GrowListAppendString (&list, g_BootDrive);
GrowListAppendString (&list, TEXT("BOOT16DIR"));
GrowListAppendString (&list, DosPath);
//
// Terminate the arg list with two NULLs
//
GrowListAppendEmptyItem (&list);
GrowListAppendEmptyItem (&list);
if (Localized) {
//
// Caller wants data from the localized section.
//
GetGlobalCodePage (&codePage, NULL);
wsprintf (codePageSection, TEXT("%s %u"), BaseSection, codePage);
infSection = codePageSection;
}
else {
infSection = BaseSection;
}
//
// Write lines from base section.
//
if (InfFindFirstLine (g_WkstaMigInf, infSection, NULL, &is)) {
do {
//
// Get the line from the section and expand any environment
// variables.
//
line = InfGetLineText (&is);
MYASSERT (line);
line = ExpandEnvironmentTextEx (line,GrowListGetStringPtrArray (&list));
MYASSERT (line);
//
// Write the line to the file.
//
WriteFileString (File, line);
WriteFileString (File, TEXT("\r\n"));
FreeText (line);
} while (InfFindNextLine (&is));
}
FreeGrowList (&list);
InfCleanUpInfStruct (&is);
}
BOOL
pCreateConfigFile(
IN PCTSTR DosPath
)
/*++
Routine Description:
Creates a CONFIG.SYS file containing default settings.
Arguments:
DosPath - Contains the path to the dos files. (e.g. c:\msdos7)
Return Value:
TRUE if file was created, FALSE if not
--*/
{
PTSTR configName = NULL;
HANDLE handle;
configName = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_CONFIG_FILE);
SetFileAttributes (configName, FILE_ATTRIBUTE_NORMAL);
handle = CreateFile (
configName,
GENERIC_READ | GENERIC_WRITE,
0,
NULL,
CREATE_ALWAYS,
FILE_ATTRIBUTE_ARCHIVE | FILE_ATTRIBUTE_NORMAL
,
NULL
);
if (handle == INVALID_HANDLE_VALUE) {
return FALSE;
}
//
// Read lines from wkstamig.inf into this file.
//
pWriteBoot16ConfigLines (handle, S_BOOT16_CONFIGSYS_SECTION, DosPath, FALSE);
pWriteBoot16ConfigLines (handle, S_BOOT16_CONFIGSYS_SECTION, DosPath, TRUE);
CloseHandle (handle);
FreePathString (configName);
return TRUE;
}
BOOL
pCreateStartupFile(
IN PCTSTR DosPath
)
/*++
Routine Description:
Creates an AUTOEXEC.BAT file containing default settings.
Arguments:
DosPath - Contains the path to the dos files. (e.g. c:\msdos7)
Return Value:
TRUE if file was created, FALSE if not
--*/
{
PTSTR startupName = NULL;
PCTSTR comment = NULL;
HANDLE handle;
PCTSTR args[2];
args[0] = DosPath;
args[1] = NULL;
startupName = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_STARTUP_FILE);
SetFileAttributes (startupName, FILE_ATTRIBUTE_NORMAL);
handle = CreateFile (
startupName,
GENERIC_READ | GENERIC_WRITE,
0,
NULL,
CREATE_ALWAYS,
FILE_ATTRIBUTE_ARCHIVE | FILE_ATTRIBUTE_HIDDEN,
NULL
);
if (handle == INVALID_HANDLE_VALUE) {
return FALSE;
}
comment = ParseMessageID (MSG_BOOT16_STARTUP_COMMENT, args);
//
// Read lines from wkstamig.inf into this file.
//
pWriteBoot16ConfigLines (handle, S_BOOT16_AUTOEXEC_SECTION, DosPath, FALSE);
pWriteBoot16ConfigLines (handle, S_BOOT16_AUTOEXEC_SECTION, DosPath, TRUE);
//
// Write localized comment.
//
WriteFileString (handle, comment);
WriteFileString (handle, TEXT("\r\n"));
FreeStringResource (comment);
CloseHandle (handle);
FreePathString (startupName);
return TRUE;
}
VOID
pEliminateCollision (
IN PCTSTR FileSpec
)
/*++
Routine Description:
pEliminateCollision checks to see if the specified file spec already
exists. If it does, the file is renamed with a numeric .nnn extension. If
the file can't be renamed, it is removed.
Arguments:
FileSpec - Specifies the file spec that is going to be used for a new file.
If this file already exists, it is renamed.
Return Value:
None.
--*/
{
PTSTR p;
PCTSTR NewFileSpec;
UINT u;
BOOL b;
if (DoesFileExist (FileSpec)) {
NewFileSpec = DuplicatePathString (FileSpec, 0);
p = _tcsrchr (NewFileSpec, TEXT('.'));
if (!p || _tcschr (p, TEXT('\\'))) {
p = GetEndOfString (NewFileSpec);
}
u = 0;
do {
wsprintf (p, TEXT(".%03u"), u);
u++;
} while (DoesFileExist (NewFileSpec));
b = OurMoveFile (FileSpec, NewFileSpec);
LOG_IF ((
!b,
LOG_ERROR,
"Could not rename %s to %s; source file might be lost",
FileSpec,
NewFileSpec
));
if (!b) {
SetFileAttributes (FileSpec, FILE_ATTRIBUTE_NORMAL);
b = DeleteFile (FileSpec);
LOG_IF ((
!b,
LOG_ERROR,
"Could not remove %s to make room for a new file. The new file is lost.",
FileSpec
));
}
FreePathString (NewFileSpec);
}
}
BOOL
pRenameCfgFiles (
IN PCTSTR DosDrive
)
/*++
Routine Description:
Renames old CONFIG.SYS and AUTOEXEC.BAT to make room for automatically generated ones.
Arguments:
DosDirectory - Contains the directory where the msdos files live (typeically c:\msdos7)
Return Value:
TRUE if rename succeeded, FALSE if not
--*/
{
PTSTR fileName1 = NULL;
PTSTR fileName2 = NULL;
fileName1 = JoinPaths (
ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_CONFIG_FILE
);
fileName2 = JoinPaths (
DosDrive,
S_BOOT16_CONFIGUPG_FILE
);
OurMoveFile (fileName1, fileName2);
SetFileAttributes (fileName2, FILE_ATTRIBUTE_NORMAL);
FreePathString (fileName1);
FreePathString (fileName2);
fileName1 = JoinPaths (
ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_STARTUP_FILE
);
fileName2 = JoinPaths (
DosDrive,
S_BOOT16_STARTUPUPG_FILE
);
OurMoveFile (fileName1, fileName2);
SetFileAttributes (fileName2, FILE_ATTRIBUTE_NORMAL);
FreePathString (fileName1);
FreePathString (fileName2);
return TRUE;
}
VOID
pCleanRootDir (
VOID
)
/*++
Routine Description:
Blows away dos files in root directory.
Arguments:
none
Return Value:
none
--*/
{
PTSTR fileName = NULL;
fileName = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_SYSMAIN_FILE);
MarkFileForDelete (fileName);
FreePathString (fileName);
fileName = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_DOSINI_FILE);
MarkFileForDelete (fileName);
FreePathString (fileName);
}
#define IoFile TEXT("IO.SYS")
VOID
pEnable16Boot (
VOID
)
/*++
Routine Description:
Creates a 16 bit environment boot option.
First we will check to see if everything is OK, we have all the files we need etc.
Then create DOS directory, rename old AUTOEXEC and CONFIG, create new ones and
add an entry in BOOT.INI
Arguments:
none
Return Value:
TRUE if file was created, FALSE if not
--*/
{
PTSTR fileName = NULL;
PTSTR dosPath = NULL;
INFCONTEXT infContext;
DWORD oldFileAttr;
BOOL result = TRUE;
HANDLE fileHandle = INVALID_HANDLE_VALUE;
if (g_Boot16 == BOOT16_NO) {
pCleanRootDir ();
return;
}
__try {
//
// first thing. Copy IO.SYS in root directory (BOOTSECT.DOS should be there)
//
pCopyDosFile (IoFile, TRUE);
fileName = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
IoFile);
SetFileAttributes (fileName, FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_READONLY | FILE_ATTRIBUTE_SYSTEM);
FreePathString (fileName);
//
// Create DOS7 directory and copy dos files there
//
dosPath = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_DOS_DIR);
if (!CreateDirectory (dosPath, NULL) && (GetLastError()!=ERROR_ALREADY_EXISTS)) {
LOG ((LOG_ERROR,"BOOT16 : Unable to create DOS directory %s",dosPath));
__leave;
}
//
// If we find autoexec.bat and config.sys rename them as *.upg
//
if (!pRenameCfgFiles (dosPath)) {
__leave;
}
if (g_WkstaMigInf == INVALID_HANDLE_VALUE) {
LOG ((LOG_ERROR,"BOOT16 : WKSTAMIG.INF is not opened"));
__leave;
}
//
// Read the section, for every file, we are trying to read it from our temp dir
// and copy it to the new DOS7 location
//
fileName = AllocPathString (MAX_TCHAR_PATH);
if (!SetupFindFirstLine (
g_WkstaMigInf,
S_BOOT16_SECTION,
NULL,
&infContext
)) {
LOG ((LOG_ERROR,"BOOT16 : Cannot read from %s section (WKSTAMIG.INF)",S_BOOT16_SECTION));
__leave;
}
do {
if (SetupGetStringField (
&infContext,
0,
fileName,
MAX_TCHAR_PATH/sizeof(fileName[0]),
NULL
)) {
pCopyDosFile (fileName, FALSE);
}
}
while (SetupFindNextLine (&infContext, &infContext));
//
// Hide the msdos7 directory (not our idea...)
//
SetFileAttributes (dosPath, FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM);
FreePathString (fileName);
fileName = NULL;
//
// Next step, build MSDOS.SYS file.
//
fileName = JoinPaths (ISPC98() ? g_Win9xBootDrivePath : g_BootDrivePath,
S_BOOT16_DOSINI_FILE);
if (SetFileAttributes (fileName, FILE_ATTRIBUTE_NORMAL)) {
if (!DeleteFile (fileName)) {
LOG ((LOG_ERROR, "BOOT16 : Unable to delete %s",fileName));
__leave;
}
}
result &= WritePrivateProfileString (TEXT("Paths"), TEXT("WinDir"), dosPath, fileName);
result &= WritePrivateProfileString (TEXT("Paths"), TEXT("WinBootDir"), dosPath, fileName);
result &= WritePrivateProfileString (TEXT("Options"), TEXT("LOGO"), TEXT("0"), fileName);
result &= WritePrivateProfileString (TEXT("Options"), TEXT("BootGUI"), TEXT("0"), fileName);
if (!result) {
LOG((LOG_ERROR,"Unable to write to %s",fileName));
__leave;
}
FreePathString (fileName);
fileName = NULL;
//
// Generate config.sys and autoexec.bat files.
//
if (!pCreateConfigFile (dosPath)) {
LOG ((LOG_ERROR, "BOOT16 : Unable to create %s",S_BOOT16_CONFIG_FILE));
__leave;
}
if (!pCreateStartupFile (dosPath)) {
LOG ((LOG_ERROR, "BOOT16 : Unable to create %s",S_BOOT16_STARTUP_FILE));
__leave;
}
if ((!ISPC98()) || (g_BootDrivePath[0] == g_Win9xBootDrivePath[0])) {
//
// If boot16 is set to BOOT16_AUTOMATIC, we create a boot.dos file,
// but don't actually modify boot.ini. If it is BOOT16_YES, then
// we modify boot.ini
//
// The result is that DOS will not show up as a boot option unless
// there was a specific reason it was turned on originally. However,
// there will be a way to enable it if needed.
//
if (g_Boot16 == BOOT16_AUTOMATIC) {
fileName = JoinPaths (g_BootDrivePath, S_BOOT16_BOOTDOS_FILE);
fileHandle = CreateFile (
fileName,
GENERIC_READ | GENERIC_WRITE,
0,
NULL,
CREATE_ALWAYS,
FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_ARCHIVE,
NULL
);
if (fileHandle != INVALID_HANDLE_VALUE) {
WriteFileString (fileHandle, ISPC98() ? L"C:\\" : g_BootDrivePath);
WriteFileString (fileHandle, TEXT("="));
WriteFileString (fileHandle, S_BOOT16_OS_ENTRY);
}
}
else {
fileName = JoinPaths (g_BootDrivePath, S_BOOT16_BOOTINI_FILE);
oldFileAttr = GetFileAttributes (fileName);
SetFileAttributes (fileName, FILE_ATTRIBUTE_NORMAL);
if (!WritePrivateProfileString (
S_BOOT16_OS_SECTION,
ISPC98() ? L"C:\\" : g_BootDrivePath,
S_BOOT16_OS_ENTRY,
fileName
)) {
LOG((LOG_ERROR,"Unable to write to %s",fileName));
SetFileAttributes (fileName, oldFileAttr);
__leave;
}
SetFileAttributes (fileName, oldFileAttr);
}
}
}
__finally {
if (fileName != NULL) {
FreePathString (fileName);
fileName = NULL;
}
if (dosPath != NULL) {
FreePathString (dosPath);
dosPath = NULL;
}
}
}
VOID
pCopyRegString (
IN HKEY DestKey,
IN HKEY SrcKey,
IN PCTSTR SrcValue
)
{
PCTSTR Data;
Data = GetRegValueString (SrcKey, SrcValue);
if (Data) {
RegSetValueEx (DestKey, SrcValue, 0, REG_SZ, (PBYTE) Data, SizeOfString (Data));
MemFree (g_hHeap, 0, Data);
}
}
#ifdef PRERELEASE
//
// !!! This is for internal use only !!! It is used for auto stress.
//
VOID
pTransferAutoStressVal (
IN HKEY StressKey,
IN PCTSTR ValueName
)
{
TCHAR Data[MEMDB_MAX];
LONG rc;
if (!MemDbGetEndpointValueEx (
MEMDB_CATEGORY_STATE,
ValueName,
NULL, // no field
Data
)) {
return;
}
rc = RegSetValueEx (
StressKey,
ValueName,
0,
REG_SZ,
(PBYTE) Data,
SizeOfString (Data)
);
DEBUGMSG_IF ((rc == ERROR_SUCCESS, DBG_VERBOSE, "Transferred autostress value %s", ValueName));
}
#endif
VOID
pProcessAutoLogon (
BOOL Final
)
/*++
Routine Description:
pProcessAutoLogon copies the logon defaults to a special key, so the
migpwd.exe tool can restore them if it runs. Then, the function calls
AutoStartProcessing to set up RunOnce and AutoAdminLogon.
This function is called early in migration to save the clean install
autologon, and then again at the end to prepare migpwd.exe.
Arguments:
Final - Specifies FALSE if this is the early call, TRUE if it is the
final call.
Return Value:
None.
--*/
{
HKEY SrcKey, DestKey;
PCTSTR Data;
BOOL copyNow = FALSE;
static BOOL alreadyCopied = FALSE;
//
// If autologon is enabled, preserve it in Win9xUpg key, so that
// migpwd.exe will restore it.
//
SrcKey = OpenRegKeyStr (S_WINLOGON_REGKEY);
if (SrcKey) {
if (!Final) {
//
// Early in migration, we get the clean install autologon values.
// If autologon is enabled, preserve the settings.
//
Data = GetRegValueString (SrcKey, S_AUTOADMIN_LOGON_VALUE);
if (Data) {
if (_ttoi (Data)) {
//
// on PER, don't want to preserve this value;
// instead, we need to preserve the name of
// the Win9x username as migrated via wkstamig.inf (see below)
//
copyNow = !g_PersonalSKU;
}
MemFree (g_hHeap, 0, Data);
}
} else if (!alreadyCopied) {
//
// Near the end of migration, we get the default logon prompt
// settings via wkstamig.inf migration. We want the attended case
// to work properly (preserving default user name & password).
//
// But if we've already preserved autologon, then we don't get
// here.
//
copyNow = TRUE;
}
if (copyNow) {
MYASSERT (!alreadyCopied);
alreadyCopied = TRUE;
DestKey = CreateRegKeyStr (S_WIN9XUPG_KEY);
if (DestKey) {
pCopyRegString (DestKey, SrcKey, S_AUTOADMIN_LOGON_VALUE);
pCopyRegString (DestKey, SrcKey, S_DEFAULT_PASSWORD_VALUE);
pCopyRegString (DestKey, SrcKey, S_DEFAULT_USER_NAME_VALUE);
pCopyRegString (DestKey, SrcKey, S_DEFAULT_DOMAIN_NAME_VALUE);
CloseRegKey (DestKey);
}
}
CloseRegKey (SrcKey);
}
if (!Final) {
return;
}
AutoStartProcessing();
#ifdef PRERELEASE
//
// !!! This is for internal use only !!! It is used for auto stress.
//
if (g_ConfigOptions.AutoStress) {
HKEY StressKey;
StressKey = CreateRegKeyStr (S_AUTOSTRESS_KEY);
MYASSERT (StressKey);
pTransferAutoStressVal (StressKey, S_AUTOSTRESS_USER);
pTransferAutoStressVal (StressKey, S_AUTOSTRESS_PASSWORD);
pTransferAutoStressVal (StressKey, S_AUTOSTRESS_OFFICE);
pTransferAutoStressVal (StressKey, S_AUTOSTRESS_DBG);
pTransferAutoStressVal (StressKey, S_AUTOSTRESS_FLAGS);
CloseRegKey (StressKey);
}
#endif
}
PCTSTR
GetProfilePathForAllUsers (
VOID
)
{
PTSTR result = NULL;
DWORD size = 0;
if (!GetAllUsersProfileDirectory (NULL, &size) &&
ERROR_INSUFFICIENT_BUFFER != GetLastError()) {
return NULL;
}
result = AllocPathString (size + 1);
if (!GetAllUsersProfileDirectory (result, &size)) {
FreePathString (result);
return NULL;
}
return result;
}
PCTSTR
pGetDefaultShellFolderLocationFromInf (
IN PCTSTR FolderName,
IN PCTSTR ProfilePath
)
{
INFSTRUCT is = INITINFSTRUCT_GROWBUFFER;
PCTSTR data;
PCTSTR result = NULL;
MYASSERT (g_WkstaMigInf && g_WkstaMigInf != INVALID_HANDLE_VALUE);
if (InfFindFirstLine (g_WkstaMigInf, TEXT("ShellFolders.DefaultNtLocation"), FolderName, &is)) {
data = InfGetStringField (&is, 1);
if (data) {
result = StringSearchAndReplace (data, S_USERPROFILE_ENV, ProfilePath);
if (!result) {
result = DuplicatePathString (data, 0);
}
}
}
InfCleanUpInfStruct (&is);
return result;
}
VOID
pFixUpDynamicPaths (
PCTSTR Category
)
{
MEMDB_ENUM e;
TCHAR Pattern[MEMDB_MAX];
PTSTR p;
GROWBUFFER Roots = GROWBUF_INIT;
MULTISZ_ENUM e2;
TCHAR NewRoot[MEMDB_MAX];
TCHAR AllProfilePath[MAX_TCHAR_PATH];
PCTSTR ProfilePath;
DWORD Size;
PTSTR UserName;
HKEY sfKey = NULL;
PCTSTR sfPath = NULL;
PTSTR NtLocation;
PCTSTR tempExpand;
BOOL regFolder;
BOOL mkDir;
//
// Collect all the roots that need to be renamed
//
StringCopy (Pattern, Category);
p = AppendWack (Pattern);
StringCopy (p, TEXT("*"));
if (MemDbEnumFirstValue (&e, Pattern, MEMDB_THIS_LEVEL_ONLY, MEMDB_ALL_BUT_PROXY)) {
do {
if ((_tcsnextc (e.szName) == TEXT('>')) ||
(_tcsnextc (e.szName) == TEXT('<'))
) {
StringCopy (p, e.szName);
MultiSzAppend (&Roots, Pattern);
}
} while (MemDbEnumNextValue (&e));
}
//
// Now change each root
//
if (EnumFirstMultiSz (&e2, (PCTSTR) Roots.Buf)) {
do {
//
// Compute NewRoot
//
StringCopy (NewRoot, e2.CurrentString);
p = _tcschr (NewRoot, TEXT('<'));
if (p) {
UserName = _tcschr (p, TEXT('>'));
MYASSERT (UserName);
StringCopyAB (Pattern, _tcsinc (p), UserName);
UserName = _tcsinc (UserName);
regFolder = TRUE;
if (StringIMatch (Pattern, TEXT("Profiles"))) {
regFolder = FALSE;
}
if (StringIMatch (Pattern, TEXT("Common Profiles"))) {
regFolder = FALSE;
}
//
// Get the profile root
//
if (StringIMatch (UserName, S_DOT_ALLUSERS)) {
Size = MAX_TCHAR_PATH;
if (regFolder) {
if (!GetAllUsersProfileDirectory (AllProfilePath, &Size)) {
DEBUGMSG ((DBG_WHOOPS, "Cannot get All Users profile path."));
continue;
}
sfKey = OpenRegKeyStr (S_USHELL_FOLDERS_KEY_SYSTEM);
} else {
if (!GetProfilesDirectory (AllProfilePath, &Size)) {
DEBUGMSG ((DBG_WHOOPS, "Cannot get All Users profile path."));
continue;
}
}
} else if (StringMatch (UserName, S_DEFAULT_USER)) {
Size = MAX_TCHAR_PATH;
if (regFolder) {
if (!GetDefaultUserProfileDirectory (AllProfilePath, &Size)) {
DEBUGMSG ((DBG_WHOOPS, "Cannot get Default User profile path."));
continue;
}
sfKey = OpenRegKey (HKEY_CURRENT_USER, S_USHELL_FOLDERS_KEY_USER);
} else {
if (!GetProfilesDirectory (AllProfilePath, &Size)) {
DEBUGMSG ((DBG_WHOOPS, "Cannot get All Users profile path."));
continue;
}
}
} else {
ProfilePath = GetProfilePathForUser (UserName);
if (!ProfilePath) {
DEBUGMSG ((DBG_WHOOPS, "Cannot get profile path for user:%s", UserName));
continue;
}
StringCopy (AllProfilePath, ProfilePath);
if (regFolder) {
sfKey = OpenRegKey (HKEY_CURRENT_USER, S_USHELL_FOLDERS_KEY_USER);
}
}
//
// If a specific reg folder is specified, get its path
//
mkDir = FALSE;
if (regFolder) {
if (!sfKey) {
DEBUGMSG ((DBG_ERROR, "Could not open Shell folders key."));
continue;
}
sfPath = GetRegValueString (sfKey, Pattern);
CloseRegKey (sfKey);
if (!sfPath || *sfPath == 0) {
DEBUGMSG ((DBG_WARNING, "Could not get Shell Folder path for: %s", Pattern));
tempExpand = pGetDefaultShellFolderLocationFromInf (Pattern, AllProfilePath);
if (!tempExpand) {
DEBUGMSG ((
DBG_WHOOPS,
"Shell folder %s is not in registry nor is it in [ShellFolders.DefaultNtLocation] of wkstamig.inf",
Pattern
));
continue;
}
//
// Special case: Shell wants read-only on this folder. Create it now.
//
mkDir = TRUE;
} else {
tempExpand = StringSearchAndReplace (
sfPath,
S_USERPROFILE_ENV,
AllProfilePath
);
if (!tempExpand) {
tempExpand = DuplicatePathString (sfPath, 0);
}
}
if (sfPath) {
MemFree (g_hHeap, 0, sfPath);
}
} else {
tempExpand = DuplicatePathString (AllProfilePath, 0);
}
//
// Move symbolic name to full path
//
NtLocation = ExpandEnvironmentText (tempExpand);
if (mkDir) {
MakeSurePathExists (NtLocation, TRUE);
SetFileAttributes (NtLocation, FILE_ATTRIBUTE_READONLY);
}
StringCopy (p, NtLocation);
MemDbMoveTree (e2.CurrentString, NewRoot);
FreeText (NtLocation);
FreePathString (tempExpand);
} else {
p = _tcschr (NewRoot, TEXT('>'));
MYASSERT (p);
if (StringIMatch (_tcsinc (p), S_DOT_ALLUSERS)) {
Size = MAX_TCHAR_PATH;
if (!GetAllUsersProfileDirectory (AllProfilePath, &Size)) {
DEBUGMSG ((DBG_WARNING, "Dynamic path for %s could not be resolved", e2.CurrentString));
}
else {
StringCopy (p, AllProfilePath);
MemDbMoveTree (e2.CurrentString, NewRoot);
}
} else if (StringMatch (_tcsinc (p), S_DEFAULT_USER)) {
Size = MAX_TCHAR_PATH;
if (!GetDefaultUserProfileDirectory (AllProfilePath, &Size)) {
DEBUGMSG ((DBG_WARNING, "Dynamic path for %s could not be resolved", e2.CurrentString));
}
else {
StringCopy (p, AllProfilePath);
MemDbMoveTree (e2.CurrentString, NewRoot);
}
} else {
ProfilePath = GetProfilePathForUser (_tcsinc (p));
if (ProfilePath) {
StringCopy (p, ProfilePath);
MemDbMoveTree (e2.CurrentString, NewRoot);
}
else {
DEBUGMSG ((DBG_WARNING, "Dynamic path for %s could not be resolved", e2.CurrentString));
}
}
}
} while (EnumNextMultiSz (&e2));
}
FreeGrowBuffer (&Roots);
}
VOID
pFixUpMemDb (
VOID
)
{
MEMDB_ENUM e;
TCHAR node[MEMDB_MAX];
pFixUpDynamicPaths (MEMDB_CATEGORY_PATHROOT);
//pFixUpDynamicPaths (MEMDB_CATEGORY_DATA); OPTIMIZATION -- Data overlaps PathRoot
pFixUpDynamicPaths (MEMDB_CATEGORY_USERFILEMOVE_DEST);
pFixUpDynamicPaths (MEMDB_CATEGORY_SHELLFOLDERS_DEST);
pFixUpDynamicPaths (MEMDB_CATEGORY_SHELLFOLDERS_SRC);
pFixUpDynamicPaths (MEMDB_CATEGORY_LINKEDIT_TARGET);
pFixUpDynamicPaths (MEMDB_CATEGORY_LINKEDIT_WORKDIR);
pFixUpDynamicPaths (MEMDB_CATEGORY_LINKEDIT_ICONPATH);
pFixUpDynamicPaths (MEMDB_CATEGORY_LINKSTUB_TARGET);
pFixUpDynamicPaths (MEMDB_CATEGORY_LINKSTUB_WORKDIR);
pFixUpDynamicPaths (MEMDB_CATEGORY_LINKSTUB_ICONPATH);
//
// Enumerate each user in MyDocsMoveWarning, then update dynamic paths
//
// MyDocsMoveWarning\<user>\<path>
MemDbBuildKey (
node,
MEMDB_CATEGORY_MYDOCS_WARNING,
TEXT("*"),
NULL,
NULL
);
if (MemDbEnumFirstValue (&e, node, MEMDB_THIS_LEVEL_ONLY, MEMDB_ALL_MATCHES)) {
do {
MemDbBuildKey (
node,
MEMDB_CATEGORY_MYDOCS_WARNING,
e.szName, // <user>
NULL,
NULL
);
pFixUpDynamicPaths (node);
} while (MemDbEnumNextValue (&e));
}
}
BOOL
EnumFirstUserToMigrate (
OUT PMIGRATE_USER_ENUM e,
IN DWORD Flags
)
{
ZeroMemory (e, sizeof (MIGRATE_USER_ENUM));
e->Flags = Flags;
pCountUsers (&e->TotalUsers, &e->ActiveUsers);
e->UserNumber = e->TotalUsers;
Win95RegGetFirstUser (&e->up, e->Win95RegName);
return EnumNextUserToMigrate (e);
}
BOOL
EnumNextUserToMigrate (
IN OUT PMIGRATE_USER_ENUM e
)
{
LONG rc;
PCTSTR Domain;
TCHAR Win9xAccount[MEMDB_MAX];
TCHAR EnumAccount[MAX_TCHAR_PATH];
USERPOSITION *AdminPosPtr;
USERPOSITION AdminPos;
BOOL Loop = TRUE;
PCTSTR UserDatLocation;
while (Loop) {
if (e->UserNumber == 0) {
return FALSE;
}
Loop = FALSE;
e->UserNumber--;
__try {
e->UserDoingTheUpgrade = FALSE;
if (e->UserNumber == INDEX_ADMINISTRATOR) {
_tcssafecpy (e->FixedUserName, g_AdministratorStr, MAX_USER_NAME);
StringCopy (e->Win9xUserName, e->FixedUserName);
e->AccountType = ADMINISTRATOR_ACCOUNT;
} else if (e->UserNumber == INDEX_LOGON_PROMPT) {
StringCopy (e->FixedUserName, S_DOT_DEFAULT);
StringCopy (e->Win9xUserName, e->FixedUserName);
e->AccountType = LOGON_USER_SETTINGS;
} else if (e->UserNumber == INDEX_DEFAULT_USER) {
//
// Do not process unless default user migration is enabled
//
if (!g_ConfigOptions.MigrateDefaultUser) {
Loop = (e->Flags & ENUM_ALL_USERS) == 0;
__leave;
}
StringCopy (e->FixedUserName, S_DEFAULT_USER);
StringCopy (e->Win9xUserName, e->FixedUserName);
e->AccountType = DEFAULT_USER_ACCOUNT;
} else {
_tcssafecpy (e->Win9xUserName, e->Win95RegName, MAX_USER_NAME);
StringCopy (e->FixedUserName, e->Win95RegName);
GetFixedUserName (e->FixedUserName);
e->AccountType = WIN9X_USER_ACCOUNT;
//
// Special case: Account named Administrator exists. In this
// case, we'd have two Administrator users unless
// one was skipped. So here is the test to skip
// if the user is named Administrator.
//
if (StringIMatch (e->Win9xUserName, g_AdministratorStr)) {
Loop = TRUE;
__leave;
}
}
StringCopy (e->FixedDomainName, e->FixedUserName);
//
// See if we are to migrate this user, and if so, perpare
// the Win95 registry and call ProcessUser.
//
UserDatLocation = GetUserDatLocation (e->FixedUserName, &e->CreateOnly);
if (UserDatLocation && DoesFileExist (UserDatLocation)) {
e->Valid = TRUE;
StringCopy (e->UserDatLocation, UserDatLocation);
} else {
e->Valid = FALSE;
e->UserDatLocation[0] = 0;
}
if (e->Flags & ENUM_SET_WIN9X_HKR) {
//
// Make HKCU equal to the enumerated user
//
g_hKeyRoot95 = HKEY_CURRENT_USER;
}
if (e->Valid) {
//
// Is this user the user doing migration?
//
if (MemDbGetEndpointValueEx (
MEMDB_CATEGORY_ADMINISTRATOR_INFO,
MEMDB_ITEM_AI_USER_DOING_MIG,
NULL, // no field
Win9xAccount
)) {
//
// Win9xAccount is unfixed name, convert to fixed name then
// compare with the current enumerated user.
//
GetFixedUserName (Win9xAccount);
DEBUGMSG ((DBG_NAUSEA, "Comparing %s to %s", e->FixedUserName, Win9xAccount));
if (StringIMatch (e->FixedUserName, Win9xAccount)) {
e->UserDoingTheUpgrade = TRUE;
}
}
//
// Perform special init depending on the user type
//
if (e->AccountType == WIN9X_USER_ACCOUNT) {
if (e->Flags & ENUM_SET_WIN9X_HKR) {
//
// Map HKCU on Win95 to current user
//
rc = Win95RegSetCurrentUserNt (&e->up, e->UserDatLocation);
if (rc != ERROR_SUCCESS) {
SetLastError (rc);
LOG ((
LOG_ERROR,
"Migrate Users: Win95RegSetCurrentUserNt could not set user "
"to %s (user path %s)",
e->FixedUserName,
e->UserDatLocation
));
LOG ((LOG_ERROR, "Could not load %s", e->UserDatLocation));
Loop = (e->Flags & ENUM_ALL_USERS) == 0;
__leave;
}
}
// Obtain the full user name
Domain = GetDomainForUser (e->FixedUserName);
if (Domain) {
StringCopy (e->FixedDomainName, Domain);
StringCopy (AppendWack (e->FixedDomainName), e->FixedUserName);
}
}
else if (e->AccountType == ADMINISTRATOR_ACCOUNT) {
//
// Map Win9x registry appropriate for the Administrator hive
//
if (e->Flags & ENUM_SET_WIN9X_HKR) {
AdminPosPtr = NULL;
// Obtain user account from memdb and find USERPOSITION for it
if (MemDbGetEndpointValueEx (
MEMDB_CATEGORY_ADMINISTRATOR_INFO,
MEMDB_ITEM_AI_ACCOUNT,
NULL, // no field
Win9xAccount
)) {
// Search Win9x user list for user
Win95RegGetFirstUser (&AdminPos, EnumAccount);
while (Win95RegHaveUser (&AdminPos)) {
GetFixedUserName (EnumAccount);
if (StringIMatch (Win9xAccount, EnumAccount)) {
AdminPosPtr = &AdminPos;
break;
}
Win95RegGetNextUser (&AdminPos, EnumAccount);
}
if (!AdminPosPtr) {
DEBUGMSG ((
DBG_WARNING,
"pMigrateUsers: Account %s not found",
Win9xAccount
));
}
}
//
// Map HKCU on Win95 to match, or default user if no match or
// no memdb entry
//
rc = Win95RegSetCurrentUserNt (AdminPosPtr, e->UserDatLocation);
if (rc != ERROR_SUCCESS) {
SetLastError (rc);
LOG ((LOG_ERROR, "Could not load %s for Administrator", e->UserDatLocation));
Loop = (e->Flags & ENUM_ALL_USERS) == 0;
__leave;
}
}
}
else if (e->AccountType == LOGON_USER_SETTINGS || e->AccountType == DEFAULT_USER_ACCOUNT) {
//
// Map HKCU on Win95 to default user
//
if (e->Flags & ENUM_SET_WIN9X_HKR) {
rc = Win95RegSetCurrentUserNt (NULL, e->UserDatLocation);
if (rc != ERROR_SUCCESS) {
SetLastError (rc);
LOG ((LOG_ERROR, "Could not load default user hive"));
Loop = (e->Flags & ENUM_ALL_USERS) == 0;
__leave;
}
}
}
} /* if (e->Valid) */
else {
Loop = (e->Flags & ENUM_ALL_USERS) == 0;
}
} /* try */
__finally {
//
// Get the next user for next time through loop, ignore errors
//
if (e->AccountType == WIN9X_USER_ACCOUNT) {
Win95RegGetNextUser (&e->up, e->Win95RegName);
}
}
} /* while (Loop) */
DEBUGMSG_IF ((
e->Flags & ENUM_SET_WIN9X_HKR,
DBG_VERBOSE,
"--- User Info ---\n"
" User Name: %s (%s)\n"
" Domain User Name: %s\n"
" Win95Reg Name: %s\n"
" User Hive: %s\n"
" Account Type: %s\n"
" Create Only: %s\n"
" Valid: %s\n"
" UserDoingTheUpgrade: %s\n",
e->Win9xUserName,
e->FixedUserName,
e->FixedDomainName,
e->Win95RegName,
e->UserDatLocation,
e->AccountType == WIN9X_USER_ACCOUNT ? TEXT("User") :
e->AccountType == ADMINISTRATOR_ACCOUNT ? TEXT("Administrator") :
e->AccountType == LOGON_USER_SETTINGS ? TEXT("Logon User") :
e->AccountType == DEFAULT_USER_ACCOUNT ? TEXT("Default User") : TEXT("Unknown"),
e->CreateOnly ? TEXT("Yes") : TEXT("No"),
e->Valid ? TEXT("Yes") : TEXT("No"),
e->UserDoingTheUpgrade ? TEXT("Yes") : TEXT("No")
));
return TRUE;
}
VOID
RunExternalProcesses (
IN HINF Inf,
IN PMIGRATE_USER_ENUM EnumPtr OPTIONAL
)
{
INFSTRUCT is = INITINFSTRUCT_GROWBUFFER;
GROWLIST List = GROWLIST_INIT;
PCTSTR RawCmdLine;
PCTSTR ExpandedCmdLine;
BOOL ProcessResult;
STARTUPINFO si;
PROCESS_INFORMATION pi;
DWORD rc;
GrowListAppendString (&List, TEXT("SYSTEMDIR"));
GrowListAppendString (&List, g_System32Dir);
if (EnumPtr) {
GrowListAppendString (&List, TEXT("USERNAME"));
GrowListAppendString (&List, EnumPtr->FixedUserName);
GrowListAppendString (&List, TEXT("USERNAMEWITHDOMAIN"));
GrowListAppendString (&List, EnumPtr->FixedDomainName);
GrowListAppendString (&List, TEXT("PREVOS_USERNAME"));
GrowListAppendString (&List, EnumPtr->Win9xUserName);
if (EnumPtr->AccountType != LOGON_USER_SETTINGS) {
GrowListAppendString (&List, TEXT("USERHIVEROOT"));
GrowListAppendString (&List, S_FULL_TEMP_USER_KEY);
} else {
GrowListAppendString (&List, TEXT("USERHIVEROOT"));
GrowListAppendString (&List, S_DEFAULT_USER_KEY);
}
if (EnumPtr->ExtraData) {
GrowListAppendString (&List, TEXT("USERPROFILE"));
GrowListAppendString (&List, EnumPtr->ExtraData->TempProfile);
}
}
//
// Terminate the arg list with two NULLs
//
GrowListAppendEmptyItem (&List);
GrowListAppendEmptyItem (&List);
if (InfFindFirstLine (Inf, S_EXTERNAL_PROCESSES, NULL, (&is))) {
do {
//
// Get the command line
//
RawCmdLine = InfGetLineText (&is);
//
// Expand environment variables
//
ExpandedCmdLine = ExpandEnvironmentTextEx (
RawCmdLine,
GrowListGetStringPtrArray (&List)
);
//
// Launch the process
//
ZeroMemory (&si, sizeof (si));
si.cb = sizeof (si);
si.dwFlags = STARTF_FORCEOFFFEEDBACK;
ProcessResult = CreateProcess (
NULL,
(PTSTR) ExpandedCmdLine,
NULL,
NULL,
FALSE,
CREATE_DEFAULT_ERROR_MODE,
NULL,
NULL,
&si,
&pi
);
if (ProcessResult) {
CloseHandle (pi.hThread);
//
// Wait 60 seconds for the process to complete
//
rc = WaitForSingleObject (pi.hProcess, 60000);
if (rc != WAIT_OBJECT_0) {
TerminateProcess (pi.hProcess, 0);
DEBUGMSG ((DBG_ERROR, "Process %s timed out and was aborted", ExpandedCmdLine));
}
ELSE_DEBUGMSG ((DBG_VERBOSE, "External process completed: %s", ExpandedCmdLine));
CloseHandle (pi.hProcess);
}
ELSE_DEBUGMSG ((DBG_ERROR, "Cannot launch %s", ExpandedCmdLine));
FreeText (ExpandedCmdLine);
} while (InfFindNextLine (&is));
}
FreeGrowList (&List);
InfCleanUpInfStruct (&is);
}
DWORD
MigrateGhostSystemFiles (
IN DWORD Request
)
{
/*
TREE_ENUM e;
PCTSTR systemName;
DWORD status;
*/
if (Request == REQUEST_QUERYTICKS) {
return TICKS_GHOST_SYSTEM_MIGRATION;
} else if (Request != REQUEST_RUN) {
return ERROR_SUCCESS;
}
/*
if (EnumFirstFileInTreeEx (&e, g_System32Dir, NULL, FALSE, FALSE, FILE_ENUM_THIS_LEVEL)) {
do {
systemName = JoinPaths (g_SystemDir, e.Name);
status = GetFileStatusOnNt (systemName);
if ((status & FILESTATUS_NTINSTALLED) &&
!(status & FILESTATUS_MOVED)
) {
if (!DoesFileExist (systemName)) {
MarkFileForMove (systemName, e.FullPath);
}
}
FreePathString (systemName);
} while (EnumNextFileInTree (&e));
}
*/
return ERROR_SUCCESS;
}
typedef struct _KNOWN_DIRS {
PCTSTR DirId;
PCTSTR Translation;
}
KNOWN_DIRS, *PKNOWN_DIRS;
KNOWN_DIRS g_KnownDirs [] = {
{TEXT("10"), g_WinDir},
{NULL, NULL}
};
typedef struct {
PCTSTR ShellFolderName;
PCTSTR DirId;
PCTSTR ShellFolderNameDefault;
BOOL bUsed;
} SHELL_TO_DIRS, *PSHELL_TO_DIRS;
SHELL_TO_DIRS g_ShellToDirs[] = {
{TEXT("Administrative Tools"), TEXT("7501"), TEXT("7517\\Administrative Tools")},
{TEXT("Common Administrative Tools"), TEXT("7501"), TEXT("7517\\Administrative Tools")},
{TEXT("AppData"), TEXT("7502"), TEXT("Application Data")},
{TEXT("Common AppData"), TEXT("7502"), TEXT("Application Data")},
{TEXT("Cache"), TEXT("7503"), NULL},
{TEXT("Cookies"), TEXT("7504"), NULL},
{TEXT("Desktop"), TEXT("7505"), NULL},
{TEXT("Common Desktop"), TEXT("7505"), TEXT("Desktop")},
{TEXT("Favorites"), TEXT("7506"), NULL},
{TEXT("Common Favorites"), TEXT("7506"), TEXT("Favorites")},
{TEXT("Local Settings"), TEXT("7510"), NULL},
{TEXT("History"), TEXT("7508"), TEXT("7510\\History")},
{TEXT("Local AppData"), TEXT("7509"), TEXT("7510\\Application Data")},
{TEXT("Personal"), TEXT("7515"), TEXT("My Documents")},
{TEXT("Common Documents"), TEXT("7515"), TEXT("My Documents")},
{TEXT("My Music"), TEXT("7511"), TEXT("7515\\My Music")},
{TEXT("CommonMusic"), TEXT("7511"), TEXT("7515\\My Music")},
{TEXT("My Pictures"), TEXT("7512"), TEXT("7515\\My Pictures")},
{TEXT("CommonPictures"), TEXT("7512"), TEXT("7515\\My Pictures")},
{TEXT("My Video"), TEXT("7513"), TEXT("7515\\My Video")},
{TEXT("CommonVideo"), TEXT("7513"), TEXT("7515\\My Video")},
{TEXT("NetHood"), TEXT("7514"), NULL},
{TEXT("PrintHood"), TEXT("7516"), NULL},
{TEXT("Start Menu"), TEXT("7520"), NULL},
{TEXT("Common Start Menu"), TEXT("7520"), TEXT("Start Menu")},
{TEXT("Programs"), TEXT("7517"), TEXT("7520\\Programs")},
{TEXT("Common Programs"), TEXT("7517"), TEXT("7520\\Programs")},
{TEXT("Recent"), TEXT("7518"), NULL},
{TEXT("SendTo"), TEXT("7519"), NULL},
{TEXT("Startup"), TEXT("7521"), TEXT("7517\\Startup")},
{TEXT("Common Startup"), TEXT("7521"), TEXT("7517\\Startup")},
{TEXT("Templates"), TEXT("7522"), NULL},
{TEXT("Common Templates"), TEXT("7522"), TEXT("Templates")},
{TEXT("Fonts"), TEXT("7507"), TEXT("10\\Fonts")},
{NULL, NULL, NULL, FALSE}
};
GROWLIST g_KnownDirIds = GROWLIST_INIT;
GROWLIST g_KnownDirPaths = GROWLIST_INIT;
VOID
pAddKnownShellFolder (
IN PCTSTR ShellFolderName,
IN PCTSTR SrcPath
)
{
PSHELL_TO_DIRS p;
for (p = g_ShellToDirs ; p->ShellFolderName ; p++) {
if (StringIMatch (ShellFolderName, p->ShellFolderName)) {
break;
}
}
if (!p->ShellFolderName) {
DEBUGMSG ((DBG_ERROR, "This system has an unsupported shell folder tag: %s", ShellFolderName));
return;
}
p->bUsed = TRUE;
GrowListAppendString (&g_KnownDirIds, p->DirId);
GrowListAppendString (&g_KnownDirPaths, SrcPath);
}
typedef struct {
PCTSTR sfName;
PCTSTR sfPath;
HKEY SfKey;
REGVALUE_ENUM SfKeyEnum;
BOOL UserSf;
} SF_ENUM, *PSF_ENUM;
BOOL
EnumFirstRegShellFolder (
IN OUT PSF_ENUM e,
IN BOOL UserSf
);
BOOL
EnumNextRegShellFolder (
IN OUT PSF_ENUM e
);
BOOL
pConvertDirName (
PCTSTR OldDirName,
PTSTR NewDirName,
PINT NameNumber
);
VOID
pInitKnownDirs (
IN BOOL bUser
)
{
SF_ENUM e;
PCTSTR profileForAllUsers;
PCTSTR profileForAllUsersVar = TEXT("%ALLUSERSPROFILE%");
PCTSTR sfPathPtr;
TCHAR shellPartialPath[MAX_PATH];
UINT charCount;
UINT charCountProfileVar;
UINT charCountProfile;
PSHELL_TO_DIRS p;
KNOWN_DIRS * pKnownDirs;
INT nameNumber;
for (p = g_ShellToDirs ; p->ShellFolderName; p++){
p->bUsed = FALSE;
}
if(bUser){
if (EnumFirstRegShellFolder(&e, TRUE)) {
do {
pAddKnownShellFolder(e.sfName, e.sfPath);
DEBUGMSG((DBG_VERBOSE, "USER: ShellFolderPath=%s\nCutedFolderPath=%s", e.sfPath, e.sfPath));
} while (EnumNextRegShellFolder(&e));
}
}
else{
profileForAllUsers = GetProfilePathForAllUsers();
MYASSERT(profileForAllUsers);
if(profileForAllUsers){
charCountProfile = TcharCount(profileForAllUsers);
}
charCountProfileVar = TcharCount(profileForAllUsersVar);
if (EnumFirstRegShellFolder(&e, FALSE)) {
do {
if(profileForAllUsers){
charCount = 0;
if(StringIMatchCharCount(e.sfPath, profileForAllUsers, charCountProfile)){
charCount = charCountProfile;
}
else{
if(StringIMatchCharCount(e.sfPath, profileForAllUsersVar, charCountProfileVar)){
charCount = charCountProfileVar;
}
}
StringCopy(shellPartialPath, TEXT("%USERPROFILE%"));
StringCat(shellPartialPath, &e.sfPath[charCount]);
sfPathPtr = shellPartialPath;
}
else{
sfPathPtr = e.sfPath;
}
DEBUGMSG((DBG_VERBOSE, "SYSTEM: ShellFolderPath=%s\r\nCutedFolderPath=%s", e.sfPath, shellPartialPath));
pAddKnownShellFolder(e.sfName, sfPathPtr);
} while (EnumNextRegShellFolder(&e));
}
FreePathString (profileForAllUsers);
}
for (pKnownDirs = g_KnownDirs ; pKnownDirs->DirId ; pKnownDirs++) {
GrowListAppendString (&g_KnownDirIds, pKnownDirs->DirId);
GrowListAppendString (&g_KnownDirPaths, pKnownDirs->Translation);
}
for (p = g_ShellToDirs ; p->ShellFolderName; p++){
if(p->bUsed){
continue;
}
shellPartialPath[0] = '\0';
nameNumber = 0;
pConvertDirName(p->DirId, shellPartialPath, &nameNumber);
if(!StringMatch (p->DirId, shellPartialPath)){
p->bUsed = TRUE;
continue;
}
if(p->ShellFolderNameDefault){
if(_istdigit(p->ShellFolderNameDefault[0])){
nameNumber = 0;
pConvertDirName(
p->ShellFolderNameDefault,
shellPartialPath,
&nameNumber);
}
else{
StringCopy(shellPartialPath, TEXT("%USERPROFILE%\\"));
StringCat(shellPartialPath, p->ShellFolderNameDefault);
}
}
else{
StringCopy(shellPartialPath, TEXT("%USERPROFILE%\\"));
StringCat(shellPartialPath, p->ShellFolderName);
}
pAddKnownShellFolder(p->ShellFolderName, shellPartialPath);
DEBUGMSG((DBG_VERBOSE, "REST: ShellFolderPath=%s\nCutedFolderPath=%s", p->ShellFolderName, shellPartialPath));
}
}
VOID
pCleanUpKnownDirs (
VOID
)
{
FreeGrowList (&g_KnownDirPaths);
FreeGrowList (&g_KnownDirIds);
}
BOOL
pConvertDirName (
PCTSTR OldDirName,
PTSTR NewDirName,
PINT NameNumber
)
{
PCTSTR OldDirCurr = OldDirName;
PCTSTR OldDirNext;
BOOL match = FALSE;
INT index;
PCTSTR listStr;
if (*NameNumber == -1) {
return FALSE;
}
//
// Extract the dir id, keeping a pointer to the subdir
//
NewDirName[0] = 0;
OldDirNext = _tcschr (OldDirCurr, '\\');
if (OldDirNext == NULL) {
OldDirNext = GetEndOfString (OldDirCurr);
}
StringCopyAB (NewDirName, OldDirCurr, OldDirNext);
//
// Find the next match in the known dir ID list
//
listStr = GrowListGetString (&g_KnownDirIds, *NameNumber);
while (listStr) {
*NameNumber += 1;
if (StringMatch (NewDirName, listStr)) {
listStr = GrowListGetString (&g_KnownDirPaths, (*NameNumber) - 1);
MYASSERT (listStr);
StringCopy (NewDirName, listStr);
break;
}
listStr = GrowListGetString (&g_KnownDirIds, *NameNumber);
}
//
// Cat the subpath to the output string and return
//
StringCat (NewDirName, OldDirNext);
if (!listStr) {
*NameNumber = -1;
return FALSE;
}
return TRUE;
}
VOID
pUninstallUserProfileCleanupPreparation (
IN HINF Inf,
IN PTSTR UserNamePtr,
IN PCTSTR PathProfileRootPtr,
IN PCTSTR DocsAndSettingsRoot,
IN GROWLIST * ListOfLogicalPathsPtr,
IN OUT GROWLIST * ListOfPaths
)
{
INFSTRUCT is = INITINFSTRUCT_GROWBUFFER;
GROWLIST List = GROWLIST_INIT;
PTSTR rawDir;
TCHAR rawPath[MAX_PATH];
PTSTR ExpandedPath;
PTSTR fileName;
TCHAR shellPath[MAX_PATH];
INT nameNumber;
INT i;
INT listSize;
PCTSTR pathLogicalPath;
GrowListAppendString (&List, TEXT("USERPROFILE"));
GrowListAppendString (&List, PathProfileRootPtr);
GrowListAppendString (&List, TEXT("PROFILES"));
GrowListAppendString (&List, DocsAndSettingsRoot);
GrowListAppendString (&List, TEXT("USERNAME"));
GrowListAppendString (&List, UserNamePtr);
GrowListAppendEmptyItem (&List);
GrowListAppendEmptyItem (&List);
DEBUGMSG ((DBG_VERBOSE, "USERPROFILE.pathProfileRoot=%s\n", PathProfileRootPtr));
if (InfFindFirstLine (Inf, S_UNINSTALL_PROFILE_CLEAN_OUT, NULL, (&is))) {
do{
rawDir = InfGetStringField (&is, 1);
if(!rawDir || *rawDir == 0){
DEBUGMSG ((DBG_VERBOSE, "rawDir == NULL"));
continue;
}
StringCopy (rawPath, rawDir);
fileName = InfGetStringField (&is, 2);
if (fileName && *fileName) {
StringCopy (AppendWack(rawPath), fileName);
}
nameNumber = 0;
pConvertDirName(rawPath, shellPath, &nameNumber);
ExpandedPath = ExpandEnvironmentTextEx (
shellPath,
GrowListGetStringPtrArray (&List)
);
DEBUGMSG ((DBG_VERBOSE, "rawPath=%s\nExpandedPath=%s\nShellPath=%s", rawPath, ExpandedPath, shellPath));
GrowListAppendString (ListOfPaths, ExpandedPath);
FreeText (ExpandedPath);
} while (InfFindNextLine (&is));
}
if(ListOfLogicalPathsPtr){
for(i = 0, listSize = GrowListGetSize (ListOfLogicalPathsPtr); i < listSize; i++) {
pathLogicalPath = GrowListGetString(ListOfLogicalPathsPtr, i);
if(!pathLogicalPath){
continue;
}
nameNumber = 0;
pConvertDirName(pathLogicalPath, shellPath, &nameNumber);
ExpandedPath = ExpandEnvironmentTextEx (
shellPath,
GrowListGetStringPtrArray (&List)
);
GrowListAppendString (ListOfPaths, ExpandedPath);
FreeText (ExpandedPath);
}
}
FreeGrowList (&List);
InfCleanUpInfStruct (&is);
DEBUGMSG ((DBG_VERBOSE, "UninstallUserProfileCleanupPreparation end"));
}
BOOL
pGetProfilePathForAllUsers(
OUT PTSTR AccountName,
OUT PTSTR PathProfile
)
{
PCTSTR pathProfileForAllUser;
MYASSERT(AccountName && PathProfile);
if(!AccountName || !PathProfile){
MYASSERT(FALSE);
return FALSE;
}
pathProfileForAllUser = GetProfilePathForAllUsers();
if(!pathProfileForAllUser) {
return FALSE;
}
StringCopy (AccountName, S_ALL_USERS);
StringCopy (PathProfile, pathProfileForAllUser);
return TRUE;
}
BOOL
pGetProfilePathForDefaultUser(
OUT PTSTR AccountName,
OUT PTSTR PathProfile
)
{
DWORD bufferSize;
MYASSERT(AccountName && PathProfile);
if(!AccountName || !PathProfile){
MYASSERT(FALSE);
return FALSE;
}
bufferSize = MAX_PATH;
if(!GetDefaultUserProfileDirectory(PathProfile, &bufferSize) ||
!PathProfile[0]) {
return FALSE;
}
StringCopy (AccountName, S_DEFAULT_USER);
return TRUE;
}
BOOL
pGetProfilePathForUser(
IN PCTSTR UserName,
OUT PTSTR AccountName,
OUT PTSTR PathProfile
)
{
DWORD bufferSize;
MYASSERT(UserName && UserName[0] && AccountName && PathProfile);
if(!UserName || !UserName[0] || !AccountName || !PathProfile){
MYASSERT(FALSE);
return FALSE;
}
bufferSize = MAX_PATH;
if(!GetProfilesDirectory(PathProfile, &bufferSize) ||
!PathProfile[0]) {
MYASSERT(FALSE);
return FALSE;
}
StringCat(AppendWack(PathProfile), UserName);
StringCopy (AccountName, UserName);
return TRUE;
}
BOOL
pGetProfilePathForLocalService(
OUT PTSTR AccountName,
OUT PTSTR PathProfile
)
{
return pGetProfilePathForUser(S_LOCALSERVICE_USER, AccountName, PathProfile);
}
BOOL
pGetProfilePathForNetworkService(
OUT PTSTR AccountName,
OUT PTSTR PathProfile
)
{
return pGetProfilePathForUser(S_NETWORKSERVICE_USER, AccountName, PathProfile);
}
BOOL
pGetProfilePathForMachineName(
OUT PTSTR AccountName,
OUT PTSTR PathProfile
)
{
TCHAR machineName[MAX_COMPUTERNAME_LENGTH + 2];
PTSTR machineNamePtr = ExpandEnvironmentTextEx (TEXT("%COMPUTERNAME%"), NULL);
BOOL bResult;
if(!machineNamePtr || machineNamePtr[0] == '%'){
MYASSERT(FALSE);
DEBUGMSG((DBG_VERBOSE, "ComputerName is NULL"));
return FALSE;
}
DEBUGMSG ((DBG_VERBOSE, "machineName=%s", machineNamePtr? machineNamePtr: TEXT("NULL")));
StringCopy(machineName, machineNamePtr);
StringCat(machineName, TEXT("$"));
return pGetProfilePathForUser(machineName, AccountName, PathProfile);
}
VOID
UninstallUserProfileCleanupPreparation (
IN HINF Inf,
IN PMIGRATE_USER_ENUM EnumPtr,
IN BOOL Playback
)
{
static GROWLIST listOfPaths = GROWLIST_INIT;
static PROFILE_PATH_PROVIDER profilePathProviders[] =
{
pGetProfilePathForAllUsers,
pGetProfilePathForDefaultUser,
pGetProfilePathForLocalService,
pGetProfilePathForNetworkService,
pGetProfilePathForMachineName
};
TCHAR accountName[MAX_PATH];
TCHAR pathProfile[MAX_PATH];
TCHAR docsAndSettingsRoot[MAX_PATH];
PCTSTR pathProfileRootPtr;
UINT i;
UINT listSize;
DWORD bufferSize;
INT stringLen;
INT cleanOutType;
TCHAR pathDir[MAX_PATH];
bufferSize = ARRAYSIZE (docsAndSettingsRoot);
if (!GetProfilesDirectory (docsAndSettingsRoot, &bufferSize)) {
DEBUGMSG ((DBG_ERROR, "Can't get Documents and Settings root"));
*docsAndSettingsRoot = 0;
}
if (EnumPtr) {
pathProfileRootPtr = GetProfilePathForUser(EnumPtr->FixedUserName);
if(pathProfileRootPtr) {
pInitKnownDirs(TRUE);
pUninstallUserProfileCleanupPreparation(
Inf,
EnumPtr->FixedUserName,
pathProfileRootPtr,
docsAndSettingsRoot,
&g_StartMenuItemsForCleanUpPrivate,
&listOfPaths
);
pCleanUpKnownDirs();
}
} else {
pInitKnownDirs(FALSE);
for(i = 0; i < ARRAYSIZE(profilePathProviders); i++){
if(profilePathProviders[i](accountName, pathProfile)){
pUninstallUserProfileCleanupPreparation(
Inf,
accountName,
pathProfile,
docsAndSettingsRoot,
&g_StartMenuItemsForCleanUpCommon,
&listOfPaths
);
}
}
pCleanUpKnownDirs();
}
if (Playback) {
for(i = 0, listSize = GrowListGetSize (&listOfPaths); i < listSize; i++) {
pathProfileRootPtr = GrowListGetString(&listOfPaths, i);
if (pathProfileRootPtr){
stringLen = TcharCount(pathProfileRootPtr);
if(stringLen > 2 && '*' == pathProfileRootPtr[stringLen - 1]){
MYASSERT('\\' == pathProfileRootPtr[stringLen - 2] || '/' == pathProfileRootPtr[stringLen - 2]);
StringCopyTcharCount(pathDir, pathProfileRootPtr, stringLen - 1);
pathProfileRootPtr = pathDir;
cleanOutType = BACKUP_AND_CLEAN_TREE;
}
else{
cleanOutType = BACKUP_FILE;
}
if (!MemDbSetValueEx (
MEMDB_CATEGORY_CLEAN_OUT,
pathProfileRootPtr,
NULL,
NULL,
cleanOutType,
NULL
)){
DEBUGMSG ((DBG_VERBOSE, "MemDbSetValueEx - failed"));
}
}
}
FreeGrowList (&listOfPaths);
FreeGrowList (&g_StartMenuItemsForCleanUpCommon);
FreeGrowList (&g_StartMenuItemsForCleanUpPrivate);
}
}
VOID
SetClassicLogonType (
VOID
)
{
static BOOL logonTypeChanged = FALSE;
DWORD d;
HKEY key;
LONG regResult;
if (!logonTypeChanged) {
key = OpenRegKeyStr (S_WINLOGON_REGKEY);
if (key) {
d = 0; // classic logon style
regResult = RegSetValueEx (
key,
TEXT("LogonType"),
0,
REG_DWORD,
(PCBYTE)(&d),
sizeof (d)
);
if (regResult == ERROR_SUCCESS) {
logonTypeChanged = TRUE;
LOG ((LOG_INFORMATION, "Logon type set to classic style because of MigrateUserAs answer file settings"));
}
CloseRegKey (key);
}
if (!logonTypeChanged) {
LOG ((LOG_ERROR, "Failed to set logon type to classic style; users will not appear in the logon menu"));
logonTypeChanged = TRUE;
}
}
}