Leaked source code of windows server 2003
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.
 
 
 
 
 
 

1104 lines
34 KiB

/*++
Copyright (c) 1996 Microsoft Corporation
Module Name:
filescan.c
Abstract:
This source file deals with file scanning phase.
Author:
Calin Negreanu (calinn) 09-Feb-1998
Revision History:
ovidiut 22-May-1999 Made AddMigrationPath globally accessible through w95upg.h
jimschm 23-Sep-1998 mapif.h removal,
calinn 31-Mar-1998 Dir recursion limited to ENUM_MAX_LEVELS (1024)
--*/
#include "pch.h"
#include "migappp.h"
#include "migdbp.h"
#define PROGBAR_DIR_LEVEL 2
#define MAX_BACKUPDIRS_IN_REPORT 5
GROWBUFFER g_OtherOsPaths = GROWBUF_INIT;
DWORD g_ProgressBarTicks;
BOOL g_OtherOsExists = FALSE;
BOOL g_IsFusionDir = FALSE;
HASHTABLE g_UseNtFileHashTable;
#define WINNT_FLAT TEXT("WINNT.EXE")
#define WIN9X_FLAT TEXT("WINSETUP.BIN")
#define WIN98_FLAT TEXT("W98SETUP.BIN")
#define WIN3X_FLAT_1 TEXT("KRNL386.EX_")
#define WIN3X_FLAT_2 TEXT("KERNEL.EXE")
#define WINNT_INSTALLATION TEXT("SYSTEM32\\NTOSKRNL.EXE")
#define WIN9X_INSTALLATION TEXT("SYSTEM\\KERNEL32.DLL")
#define WIN3X_INSTALLATION_1 TEXT("SYSTEM\\KRNL386.EXE")
#define WIN3X_INSTALLATION_2 TEXT("SYSTEM\\KRNL386.EXE")
#define WIN3X_INSTALLATION_3 TEXT("SYSTEM\\KERNEL.EXE")
#define WINNT_SETUP_1 TEXT(":\\$WIN_NT$.~BT")
#define WINNT_SETUP_2 TEXT(":\\$WIN_NT$.~LS")
#define RECYCLE_1 TEXT(":\\RECYCLED")
#define RECYCLE_2 TEXT(":\\RECYCLER")
BOOL
pSpecialExcludedDir (
IN PCTSTR FullFileSpec
)
{
PCTSTR testPath = NULL;
BOOL Result = TRUE;
__try {
if (StringIMatch (g_WinDir, FullFileSpec)) {
Result = FALSE;
__leave;
}
//
// let's see if it's one of our dirs.
//
if (((*SOURCEDIRECTORY(0))&&(StringIMatch (FullFileSpec, SOURCEDIRECTORY(0)) )) ||
((*g_TempDirWack )&&(StringIMatchTcharCount (FullFileSpec, g_TempDirWack, g_TempDirWackChars-1) )) ||
((*g_PlugInDirWack )&&(StringIMatchTcharCount (FullFileSpec, g_PlugInDirWack, g_PlugInDirWackChars-1) )) ||
((*g_RecycledDirWack )&&(StringIMatchTcharCount (FullFileSpec, g_RecycledDirWack, g_RecycledDirWackChars-1) ))) {
__leave;
}
//
//we are trying to see if we are entering a winnt, win95 or win3.x
//flat directory
//
testPath = JoinPaths (FullFileSpec, WINNT_FLAT);
if (DoesFileExist (testPath)) {
__leave;
}
FreePathString (testPath);
testPath = JoinPaths (FullFileSpec, WIN9X_FLAT);
if (DoesFileExist (testPath)) {
__leave;
}
FreePathString (testPath);
testPath = JoinPaths (FullFileSpec, WIN98_FLAT);
if (DoesFileExist (testPath)) {
__leave;
}
FreePathString (testPath);
testPath = JoinPaths (FullFileSpec, WIN3X_FLAT_1);
if (DoesFileExist (testPath)) {
__leave;
}
FreePathString (testPath);
testPath = JoinPaths (FullFileSpec, WIN3X_FLAT_2);
if (DoesFileExist (testPath)) {
__leave;
}
FreePathString (testPath);
//
//we are trying to see if we are entering a winnt installation,
//win95 installation or win3.x installation
//
testPath = JoinPaths (FullFileSpec, WINNT_INSTALLATION);
if (DoesFileExist (testPath)) {
MultiSzAppend (&g_OtherOsPaths, FullFileSpec);
__leave;
}
FreePathString (testPath);
if (FullFileSpec [0]) {
testPath = _tcsinc (FullFileSpec);
if ((StringIMatch (testPath, WINNT_SETUP_1)) ||
(StringIMatch (testPath, WINNT_SETUP_2)) ||
(StringIMatch (testPath, RECYCLE_1 )) ||
(StringIMatch (testPath, RECYCLE_2 ))) {
testPath = NULL;
__leave;
}
}
testPath = NULL;
Result = FALSE;
}
__finally {
if (testPath) {
FreePathString (testPath);
testPath = NULL;
}
}
return Result;
}
INT
pCountDirectories (
IN PCTSTR FullPath,
IN PCTSTR DontCare,
IN WIN32_FIND_DATA *FindData,
IN DWORD EnumHandle,
IN PVOID Params,
PDWORD CurrentDirData
)
{
if (pSpecialExcludedDir (FullPath)) {
ExcludePath (g_ExclusionValue, FullPath);
return CALLBACK_DO_NOT_RECURSE_THIS_DIRECTORY;
}
if (!(FindData->dwFileAttributes & FILE_ATTRIBUTE_HIDDEN)) {
g_ProgressBarTicks += TICKS_FILESCAN_DIR_INCREMENT;
}
return CALLBACK_CONTINUE;
}
DWORD
FileScan_GetProgressMax (
VOID
)
{
ACCESSIBLE_DRIVE_ENUM e;
DWORD Count = 0;
g_ProgressBarTicks = 0;
//
// Enumerate of the accessible drives. The callback routine will keep track of all
// of the directories on that drive to a depth of two.
//
if (GetFirstAccessibleDriveEx (&e, TRUE)) {
do {
//
// restrict file system analyze to the windows drive
//
if (!EnumerateTree (
e->Drive,
pCountDirectories,
NULL, // failure-logging callback
g_ExclusionValue ,
NULL, // unused
PROGBAR_DIR_LEVEL,
NULL, // Unused exclusion structure.
FILTER_DIRECTORIES
)) {
LOG((LOG_ERROR,"Error counting directories on drive %s.",e->Drive));
}
} while (GetNextAccessibleDrive(&e));
}
DEBUGLOGTIME (("FileScan_GetProgressMax estimation: %lu", g_ProgressBarTicks));
return g_ProgressBarTicks;
}
INT
pGetDirLevel (
IN PCTSTR DirName
)
{
INT result = 0;
PCTSTR dirPtr = DirName;
do {
dirPtr = _tcschr (dirPtr, TEXT('\\'));
if (dirPtr != NULL) {
result++;
dirPtr = _tcsinc (dirPtr);
if (dirPtr[0] == 0) {
result--;
dirPtr = NULL;
}
}
}
while (dirPtr);
return result;
}
INT
pProcessFileOrDir (
IN PCTSTR FullFileSpec,
IN PCTSTR DontCare,
IN WIN32_FIND_DATA *FindData,
IN DWORD EnumHandle,
IN LPVOID Params,
IN OUT PDWORD CurrentDirData
)
{
FILE_HELPER_PARAMS HelperParams;
INT result = CALLBACK_CONTINUE;
if (CANCELLED()) {
SetLastError (ERROR_CANCELLED);
return CALLBACK_FAILED;
}
#ifdef DEBUG
{
TCHAR DbgBuf[256];
if (GetPrivateProfileString ("FileScan", FullFileSpec, "", DbgBuf, 256, g_DebugInfPath)) {
DEBUGMSG ((DBG_NAUSEA, "%s found", FullFileSpec));
}
if (GetPrivateProfileString ("FileScan", "All", "", DbgBuf, 256, g_DebugInfPath)) {
DEBUGMSG ((DBG_NAUSEA, "%s", FullFileSpec));
}
}
#endif
if (!SafeModeActionCrashed (SAFEMODEID_FILES, FullFileSpec)) {
SafeModeRegisterAction(SAFEMODEID_FILES, FullFileSpec);
__try {
//
//prepare structure for calling helper functions
//
HelperParams.FullFileSpec = FullFileSpec;
HelperParams.Handled = 0;
HelperParams.FindData = FindData;
HelperParams.Extension = GetDotExtensionFromPath (HelperParams.FullFileSpec);
HelperParams.VirtualFile = (FindData == NULL);
HelperParams.CurrentDirData = CurrentDirData;
if (FindData) {
if ((FindData->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) != 0) {
if (pGetDirLevel (FullFileSpec) <= PROGBAR_DIR_LEVEL) {
// This is an increment point
if (!TickProgressBarDelta (TICKS_FILESCAN_DIR_INCREMENT)) {
SetLastError (ERROR_CANCELLED);
result = CALLBACK_FAILED;
__leave;
}
ProgressBar_SetSubComponent (FullFileSpec);
}
//
// If we know that this dir has something special and we must exclude it
// if you need to add code to exclude some dir plug it in pSpecialExcludedDir
//
if (pSpecialExcludedDir (FullFileSpec)) {
result = CALLBACK_DO_NOT_RECURSE_THIS_DIRECTORY;
__leave;
}
_tcssafecpy (HelperParams.DirSpec, FullFileSpec, MAX_TCHAR_PATH);
HelperParams.IsDirectory = TRUE;
}
else {
_tcssafecpyab (HelperParams.DirSpec, FullFileSpec, GetFileNameFromPath (FullFileSpec), MAX_TCHAR_PATH);
HelperParams.IsDirectory = FALSE;
}
} else {
HelperParams.IsDirectory = FALSE;
}
// calling the process helper functions
if (!ProcessFileHelpers (&HelperParams)) {
result = CALLBACK_FAILED;
__leave;
}
}
__finally {
}
SafeModeUnregisterAction();
}
return result;
}
static
BOOL
pExamineAccessibleDrive (
IN ACCESSIBLE_DRIVE_ENUM Enum
)
{
BOOL fRet = TRUE;
//
// Enumerate volume. FALSE retval ends enumeration; callback sets last error.
//
SetLastError (ERROR_SUCCESS);
if (!(fRet = EnumerateTree (
Enum -> Drive,
pProcessFileOrDir,
NULL, // No failure callback
g_ExclusionValue,
NULL, // Params - Unused.
ENUM_MAX_LEVELS,
NULL,
FILTER_ALL
))
) {
DEBUGMSG_IF ((
GetLastError() != ERROR_SUCCESS &&
GetLastError() != ERROR_PATH_NOT_FOUND &&
GetLastError() != ERROR_CANCELLED,
DBG_ERROR,
"pAccessibleDrivesEnum_Callback: EnumerateTree failed."
));
if (GetLastError() != ERROR_SUCCESS &&
GetLastError() != ERROR_PATH_NOT_FOUND &&
GetLastError() != ERROR_CANCELLED) {
LOG ((
LOG_ERROR,
"Failure while enumerating tree. rc: %u",
GetLastError()
));
}
}
return fRet;
}
VOID
pReportOtherOs (
VOID
)
/*
This function will report if other OS was found in the system. It will also report (with a strong message) if PATH variable points to
some directories that belong to some other OSes.
*/
{
MULTISZ_ENUM enumOsPaths;
PCTSTR Group;
PCTSTR Message;
if (g_ConfigOptions.IgnoreOtherOS) {
return;
}
if (g_OtherOsExists) {
//
// Already done.
//
return;
}
if (EnumFirstMultiSz (&enumOsPaths, g_OtherOsPaths.Buf)) {
g_OtherOsExists = TRUE;
Group = BuildMessageGroup (MSG_BLOCKING_ITEMS_ROOT, MSG_OTHER_OS_WARNING_SUBGROUP, NULL);
Message = GetStringResource (MSG_OTHER_OS_WARNING);
if (Message && Group) {
MsgMgr_ObjectMsg_Add (TEXT("*OtherOsFound"), Group, Message);
}
FreeText (Group);
FreeStringResource (Message);
}
}
VOID
AddMigrationPathEx (
IN PCTSTR PathLong,
IN DWORD Levels,
IN BOOL Win9xOsPath
)
/*++
Routine Description:
AddMigrationPath adds the specified path to MEMDB_CATEGORY_MIGRATION_PATHS,
ensuring that PathLong is not a drive root; in this case Levels is set to 0
Arguments:
PathLong - Specifies the path to be added (long file name format)
Levels - Specifies how many levels (subdirs) are valid
Win9xOsPath - Specifies TRUE if the path supplied should be treated as an OS path
Return Value:
none
--*/
{
TCHAR key[MEMDB_MAX];
//
// a drive spec is supposed to be one driveletter followed by a column (eg. l:)
//
if (_istalpha (PathLong[0]) && PathLong[1] == TEXT(':') &&
(PathLong[2] == 0 || PathLong[2] == TEXT('\\') && PathLong[3] == 0)) {
//
// this is the root of a drive, so override Levels
//
Levels = 0;
}
if (StringIMatch (PathLong, g_ProgramFilesDir)) {
Levels = 0;
}
MemDbBuildKey (key, MEMDB_CATEGORY_MIGRATION_PATHS, PathLong, NULL, NULL);
MemDbSetValueAndFlags (key, Levels, Win9xOsPath ? TRUE : FALSE, 0);
}
VOID
pAddValueEnumDirsAsMigDirs (
IN PCTSTR Key,
IN DWORD Levels
)
{
HKEY key;
REGVALUE_ENUM e;
TCHAR pathLong[MAX_TCHAR_PATH];
PTSTR Data;
PCTSTR pathExp;
PTSTR filePtr;
key = OpenRegKeyStr (Key);
if (key != NULL) {
if (EnumFirstRegValue (&e, key)) {
do {
Data = GetRegValueString (e.KeyHandle, e.ValueName);
if (Data) {
ExtractArgZeroEx (Data, pathLong, NULL, FALSE);
if (*pathLong) {
pathExp = ExpandEnvironmentTextA(pathLong);
if (pathExp) {
// eliminate the file name
filePtr = (PTSTR)GetFileNameFromPath (pathExp);
if (filePtr) {
filePtr = _tcsdec (pathExp, filePtr);
if (filePtr) {
*filePtr = 0;
}
}
if (filePtr && OurGetLongPathName (pathExp, pathLong, MAX_TCHAR_PATH)) {
AddMigrationPath (pathLong, Levels);
}
FreeText (pathExp);
}
}
MemFree (g_hHeap, 0, Data);
}
}
while (EnumNextRegValue (&e));
}
CloseRegKey (key);
}
}
VOID
pBuildMigrationPaths (
VOID
)
/*
This function will create a list with all the paths that are considered "ours". Any other path that has OS files in it is considered
a backup path. If an OS file is found in a "backup" path a warning is presented to the user.
*/
{
PCTSTR pathExp = NULL;
TCHAR pathLong [MAX_TCHAR_PATH];
CHAR pathExpAnsi [MAX_TCHAR_PATH];
PATH_ENUM pathEnum;
TCHAR dirName [MAX_TCHAR_PATH];
INFCONTEXT context;
PCTSTR argList[]={"ProgramFiles", g_ProgramFilesDir, "SystemDrive", g_WinDrive, NULL};
HKEY appPathsKey, currentAppKey;
REGKEY_ENUM appPathsEnum;
PCTSTR appPaths = NULL;
MEMDB_ENUM eFolder;
TREE_ENUM eFile;
PCTSTR extPtr;
TCHAR shortcutTarget [MEMDB_MAX];
TCHAR shortcutArgs [MEMDB_MAX];
TCHAR shortcutWorkDir [MEMDB_MAX];
TCHAR shortcutIconPath [MEMDB_MAX];
INT shortcutIcon;
WORD shortcutHotKey;
BOOL msDosMode;
IShellLink *shellLink;
IPersistFile *persistFile;
PTSTR filePtr;
HKEY sharedDllsKey;
REGVALUE_ENUM sharedDllsEnum;
DWORD Levels;
//
// First thing. Include PATH variable and root of the boot drive in our migration paths.
//
AddMigrationPath (g_BootDrive, 0);
if (EnumFirstPath (&pathEnum, NULL, g_WinDir, g_SystemDir)) {
do {
pathExp = ExpandEnvironmentTextA(pathEnum.PtrCurrPath);
filePtr = (PTSTR)GetFileNameFromPath (pathExp);
if (*filePtr == 0) {
filePtr = _tcsdec (pathExp, filePtr);
if (filePtr) {
*filePtr = 0;
}
}
if (OurGetLongPathName (pathExp, pathLong, MAX_TCHAR_PATH)) {
AddMigrationPathEx (pathLong, 2, TRUE);
}
FreeText (pathExp);
}
while (EnumNextPath (&pathEnum));
EnumPathAbort (&pathEnum);
}
//
// Then include temporary directory as a tree
//
if (GetTempPath (MAX_TCHAR_PATH, pathLong)) {
// eliminate \ from the end of a path
filePtr = GetEndOfString (pathLong);
filePtr = _tcsdec (pathLong, filePtr);
if ((filePtr) &&
(*filePtr == TEXT('\\'))
) {
*filePtr = 0;
}
AddMigrationPath (pathLong, MAX_DEEP_LEVELS);
}
//
// Then include known directories from win95upg.inf section "MigrationDirs".
//
MYASSERT (g_Win95UpgInf != INVALID_HANDLE_VALUE);
if (SetupFindFirstLine (g_Win95UpgInf, S_MIGRATION_DIRS, NULL, &context)) {
do {
if (SetupGetStringField (&context, 1, dirName, MAX_TCHAR_PATH, NULL)) {
pathExp = ExpandEnvironmentTextExA(dirName, argList);
if (pathExp) {
filePtr = (PTSTR)GetFileNameFromPath (pathExp);
if (*filePtr == 0) {
filePtr = _tcsdec (pathExp, filePtr);
if (filePtr) {
*filePtr = 0;
}
}
if (OurGetLongPathName (pathExp, pathLong, MAX_TCHAR_PATH)) {
AddMigrationPathEx (pathLong, MAX_DEEP_LEVELS, TRUE);
}
//
// also add the path translated to ANSI (if different)
//
if (TcharCountA (pathExp) < ARRAYSIZE(pathExpAnsi)) {
#pragma prefast(suppress:56, "OemToCharBuffA unsupported on Win9x")
OemToCharA (pathExp, pathExpAnsi);
if (OurGetLongPathName (pathExpAnsi, pathLong, MAX_TCHAR_PATH)) {
AddMigrationPathEx (pathLong, MAX_DEEP_LEVELS, TRUE);
}
}
FreeText (pathExp);
}
}
}
while (SetupFindNextLine (&context, &context));
}
//
// Then include known OEM directories from win95upg.inf section "OemMigrationDirs".
//
if (SetupFindFirstLine (g_Win95UpgInf, S_OEM_MIGRATION_DIRS, NULL, &context)) {
do {
if (SetupGetStringField (&context, 1, dirName, MAX_TCHAR_PATH, NULL)) {
pathExp = ExpandEnvironmentTextExA(dirName, argList);
if (pathExp) {
filePtr = (PTSTR)GetFileNameFromPath (pathExp);
if (*filePtr == 0) {
filePtr = _tcsdec (pathExp, filePtr);
if (filePtr) {
*filePtr = 0;
}
}
if (OurGetLongPathName (pathExp, pathLong, MAX_TCHAR_PATH)) {
Levels = 0;
if (SetupGetStringField (&context, 2, dirName, MAX_TCHAR_PATH, NULL)) {
if (_ttoi (dirName) != 0) {
Levels = MAX_DEEP_LEVELS;
}
}
AddMigrationPathEx (pathLong, Levels, TRUE);
}
FreeText (pathExp);
}
}
}
while (SetupFindNextLine (&context, &context));
}
//
// Then include paths listed in SharedDlls key.
//
sharedDllsKey = OpenRegKeyStr (S_REG_SHARED_DLLS);
if (sharedDllsKey != NULL) {
if (EnumFirstRegValue (&sharedDllsEnum, sharedDllsKey)) {
do {
pathExp = ExpandEnvironmentTextA(sharedDllsEnum.ValueName);
if (pathExp) {
// eliminate the file name
filePtr = (PTSTR)GetFileNameFromPath (pathExp);
if (filePtr) {
filePtr = _tcsdec (pathExp, filePtr);
if (filePtr) {
*filePtr = 0;
}
}
if (OurGetLongPathName (pathExp, pathLong, MAX_TCHAR_PATH)) {
AddMigrationPath (pathLong, 0);
}
FreeText (pathExp);
}
}
while (EnumNextRegValue (&sharedDllsEnum));
}
CloseRegKey (sharedDllsKey);
}
//
// Then include paths listed in AppPaths key.
//
appPathsKey = OpenRegKeyStr (S_SKEY_APP_PATHS);
if (appPathsKey != NULL) {
if (EnumFirstRegKey (&appPathsEnum, appPathsKey)) {
do {
currentAppKey = OpenRegKey (appPathsKey, appPathsEnum.SubKeyName);
if (currentAppKey != NULL) {
appPaths = GetRegValueString (currentAppKey, TEXT("Path"));
if (appPaths != NULL) {
if (EnumFirstPath (&pathEnum, appPaths, NULL, NULL)) {
do {
pathExp = ExpandEnvironmentTextA(pathEnum.PtrCurrPath);
if (pathExp) {
// eliminate \ from the end of a path
filePtr = GetEndOfString (pathExp);
filePtr = _tcsdec (pathExp, filePtr);
if ((filePtr) &&
(*filePtr == TEXT('\\'))
) {
*filePtr = 0;
}
if (OurGetLongPathName (pathExp, pathLong, MAX_TCHAR_PATH)) {
AddMigrationPath (pathLong, 2);
}
FreeText (pathExp);
}
}
while (EnumNextPath (&pathEnum));
EnumPathAbort (&pathEnum);
}
MemFree (g_hHeap, 0, appPaths);
}
CloseRegKey (currentAppKey);
}
}
while (EnumNextRegKey (&appPathsEnum));
}
CloseRegKey (appPathsKey);
}
//
// Then include paths listed in Run* keys.
//
pAddValueEnumDirsAsMigDirs (S_RUNKEY, 2);
pAddValueEnumDirsAsMigDirs (S_RUNONCEKEY, 2);
pAddValueEnumDirsAsMigDirs (S_RUNONCEEXKEY, 2);
pAddValueEnumDirsAsMigDirs (S_RUNSERVICESKEY, 2);
pAddValueEnumDirsAsMigDirs (S_RUNSERVICESONCEKEY, 2);
pAddValueEnumDirsAsMigDirs (S_RUNKEY_USER, 2);
pAddValueEnumDirsAsMigDirs (S_RUNONCEKEY_USER, 2);
pAddValueEnumDirsAsMigDirs (S_RUNONCEEXKEY_USER, 2);
pAddValueEnumDirsAsMigDirs (S_RUNONCEKEY_DEFAULTUSER, 2);
pAddValueEnumDirsAsMigDirs (S_RUNONCEEXKEY_DEFAULTUSER, 2);
pAddValueEnumDirsAsMigDirs (S_RUNKEY_DEFAULTUSER, 2);
//
// Finally include paths listed in all links from all user profiles.
//
if (InitCOMLink (&shellLink, &persistFile)) {
if (MemDbEnumFirstValue (&eFolder, MEMDB_CATEGORY_NICE_PATHS"\\*", MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY)) {
do {
if (*eFolder.szName == 0) {
continue;
}
// first: this directory is a migration directory
AddMigrationPath (eFolder.szName, MAX_DEEP_LEVELS);
//
// For each shell folder we enumerate all links trying to get the path where the links are running
//
if (EnumFirstFileInTreeEx (&eFile, eFolder.szName, TEXT("*.*"), FALSE, FALSE, 0)) {
do {
if (!eFile.Directory) {
extPtr = GetFileExtensionFromPath (eFile.Name);
if (extPtr) {
if (StringIMatch (extPtr, TEXT("LNK"))) {
if (ExtractShellLinkInfo (
shortcutTarget,
shortcutArgs,
shortcutWorkDir,
shortcutIconPath,
&shortcutIcon,
&shortcutHotKey,
NULL,
eFile.FullPath,
shellLink,
persistFile
)) {
if (shortcutWorkDir [0] != 0) {
AddMigrationPath (shortcutWorkDir, 2);
}
if (shortcutTarget [0] != 0) {
filePtr = (PTSTR)GetFileNameFromPath (shortcutTarget);
if (filePtr) {
filePtr = _tcsdec (shortcutTarget, filePtr);
if (filePtr) {
*filePtr = 0;
AddMigrationPath (shortcutTarget, 2);
}
}
}
}
}
if (StringIMatch (extPtr, TEXT("PIF"))) {
if (ExtractPifInfo (
shortcutTarget,
shortcutArgs,
shortcutWorkDir,
shortcutIconPath,
&shortcutIcon,
&msDosMode,
NULL,
eFile.FullPath
) == ERROR_SUCCESS) {
if (shortcutWorkDir [0] != 0) {
AddMigrationPath (shortcutWorkDir, 2);
}
if (shortcutTarget [0] != 0) {
filePtr = (PTSTR)GetFileNameFromPath (shortcutTarget);
if (filePtr) {
filePtr = _tcsdec (shortcutTarget, filePtr);
if (filePtr) {
*filePtr = 0;
AddMigrationPath (shortcutTarget, 2);
}
}
}
}
}
}
}
}
while (EnumNextFileInTree (&eFile));
}
}
while (MemDbEnumNextValue (&eFolder));
}
FreeCOMLink (&shellLink, &persistFile);
}
#ifdef DEBUG
if (GetPrivateProfileIntA ("Debug", "MigPaths", 0, g_DebugInfPathBufA) == 1) {
if (MemDbGetValueEx (&eFolder, MEMDB_CATEGORY_MIGRATION_PATHS, NULL, NULL)) {
DEBUGMSG(("Migration Paths:",""));
do {
DEBUGMSG(("","%s - %ld", eFolder.szName, eFolder.dwValue));
}
while (MemDbEnumNextValue (&eFolder));
}
}
#endif
}
VOID
pReportBackupDirs (
)
{
MEMDB_ENUM e;
PCTSTR BackupDirsGroup;
PCTSTR Message;
PCTSTR ArgArray[2];
TCHAR Buffer[12];
if (g_BackupDirCount <= MAX_BACKUPDIRS_IN_REPORT) {
if (MemDbGetValueEx (&e, MEMDB_CATEGORY_BACKUPDIRS, NULL, NULL)) {
do {
BackupDirsGroup = BuildMessageGroup (
MSG_INSTALL_NOTES_ROOT,
MSG_BACKUP_DETECTED_LIST_SUBGROUP,
e.szName
);
if (BackupDirsGroup) {
MsgMgr_ObjectMsg_Add(
e.szName,
BackupDirsGroup,
S_EMPTY
);
FreeText (BackupDirsGroup);
}
} while (MemDbEnumNextValue (&e));
} else {
MYASSERT (!g_BackupDirCount);
}
} else {
//
// just put a generic message
//
BackupDirsGroup = BuildMessageGroup (
MSG_INSTALL_NOTES_ROOT,
MSG_BACKUP_DETECTED_SUBGROUP,
NULL
);
if (BackupDirsGroup) {
ArgArray[0] = g_Win95Name;
wsprintf (Buffer, TEXT("%lu"), g_BackupDirCount);
ArgArray[1] = Buffer;
Message = ParseMessageID (MSG_BACKUP_DETECTED, ArgArray);
if (Message) {
MsgMgr_ObjectMsg_Add (
TEXT("*BackupDetected"),
BackupDirsGroup,
Message
);
FreeStringResource (Message);
//
// write all backup dirs to the log file
//
if (MemDbGetValueEx (&e, MEMDB_CATEGORY_BACKUPDIRS, NULL, NULL)) {
do {
//
// write it in the log
//
LOG ((LOG_WARNING, (PCSTR)MSG_BACKUP_DETECTED_LOG, e.szName, g_Win95Name));
} while (MemDbEnumNextValue (&e));
} else {
MYASSERT (FALSE);
}
}
FreeText (BackupDirsGroup);
} else {
MYASSERT (FALSE);
}
}
}
BOOL
pScanFileSystem (
VOID
)
{
BOOL fStatus = TRUE;
BOOL fRet = TRUE;
ACCESSIBLE_DRIVE_ENUM e;
LONG ReturnCode = ERROR_SUCCESS;
INFSTRUCT context = INITINFSTRUCT_GROWBUFFER;
PCTSTR virtualFile;
PCTSTR pathExp;
PCTSTR argList[3]={"ProgramFiles", g_ProgramFilesDir, NULL};
pReportOtherOs ();
pBuildMigrationPaths ();
if (!InitLinkAnnounce ()) {
return FALSE;
}
TickProgressBar ();
if (GetFirstAccessibleDriveEx (&e, TRUE)) {
do {
fStatus = pExamineAccessibleDrive(e);
} while (fStatus && GetNextAccessibleDrive(&e));
}
else {
fRet = FALSE;
}
//
// Act on status
//
if (!fRet || !fStatus) {
ReturnCode = GetLastError ();
if (ReturnCode != ERROR_CANCELLED && !CANCELLED()) {
ERROR_CRITICAL
LOG ((LOG_ERROR, (PCSTR)MSG_ENUMDRIVES_FAILED_LOG));
DEBUGMSG((DBG_ERROR,"FileScan: Error enumerating drives"));
}
return FALSE;
}
//
// OK, we are now at the end of filescan. Let's enumerate virtual files
//
MYASSERT (g_Win95UpgInf != INVALID_HANDLE_VALUE);
if (InfFindFirstLine (g_Win95UpgInf, S_VIRTUAL_FILES, NULL, &context)) {
do {
virtualFile = InfGetStringField (&context, 1);
if (virtualFile) {
pathExp = ExpandEnvironmentTextExA(virtualFile, argList);
if (pathExp) {
if (!DoesFileExist (pathExp)) {
pProcessFileOrDir (pathExp, NULL, NULL, 0, NULL, NULL);
}
FreeText (pathExp);
}
}
}
while (InfFindNextLine (&context));
InfCleanUpInfStruct (&context);
}
CleanupUseNtFilesMap ();
return TRUE;
}
DWORD
ScanFileSystem (
IN DWORD Request
)
{
DWORD Ticks;
switch (Request) {
case REQUEST_QUERYTICKS:
return FileScan_GetProgressMax ();
case REQUEST_RUN:
Ticks = GetTickCount();
if (!pScanFileSystem ()) {
return GetLastError ();
}
pReportBackupDirs ();
Ticks = GetTickCount() - Ticks;
g_ProgressBarTime += Ticks;
return ERROR_SUCCESS;
default:
DEBUGMSG ((DBG_ERROR, "Bad parameter in ScanFileSystem"));
}
return 0;
}
VOID
InitUseNtFilesMap (
VOID
)
{
INFSTRUCT context = INITINFSTRUCT_POOLHANDLE;
PTSTR name;
DWORD zero = 0;
g_UseNtFileHashTable = HtAllocWithData (sizeof (DWORD));
if (!g_UseNtFileHashTable) {
return;
}
if (InfFindFirstLine (g_MigDbInf, S_USENTFILES, NULL, &context)) {
do {
name = InfGetStringField (&context, 4);
if (name) {
HtAddStringAndData (g_UseNtFileHashTable, name, &zero);
}
} while (InfFindNextLine (&context));
InfCleanUpInfStruct (&context);
}
}
VOID
CleanupUseNtFilesMap (
VOID
)
{
if (g_UseNtFileHashTable) {
HtFree (g_UseNtFileHashTable);
g_UseNtFileHashTable = NULL;
}
}