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.
2245 lines
61 KiB
2245 lines
61 KiB
/*++
|
|
|
|
Copyright (c) 1997 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
stf.c
|
|
|
|
Abstract:
|
|
|
|
Applications that use the ACME Setup toolkit leave .STF files with their
|
|
installation, so ACME can reinstall or uninstall the application. During
|
|
the upgrade, we move paths around, and we confuse ACME Setup to the point
|
|
where it won't run.
|
|
|
|
The routines in this file update all STF files found on the system. Each
|
|
STF has an associated INF file, and the code here parses the STF and INF
|
|
files into memory structures, and then enumerates the structures in
|
|
various ways, updating all the paths.
|
|
|
|
Entry points:
|
|
|
|
ProcessStfFiles - Enumerates all STF files and processes those that have
|
|
not already been handled in another way. The older
|
|
STF files are overwritten.
|
|
|
|
See the ACME Setup specification for more details on the format of STF
|
|
files and their associated INFs.
|
|
|
|
Author:
|
|
|
|
Jim Schmidt (jimschm) 12-Sep-1997
|
|
|
|
Revision History:
|
|
|
|
jimschm 28-Sep-1998 Updated to change all altered dirs
|
|
|
|
--*/
|
|
|
|
#include "pch.h"
|
|
#include "migmainp.h"
|
|
|
|
#include "stftable.h"
|
|
#include "fileops.h"
|
|
#include "stfp.h"
|
|
|
|
|
|
#define DBG_STF "STF"
|
|
|
|
#define S_SECTIONNAME_SPRINTF TEXT("Win9xUpg_%u")
|
|
|
|
#define COLUMN_OBJECT_ID 0
|
|
#define COLUMN_COMMAND 4
|
|
#define COLUMN_OBJECT_DATA 5
|
|
#define COLUMN_DEST_DIR 10
|
|
#define COLUMN_INSTALL_DESTDIR 14
|
|
|
|
|
|
PVOID
|
|
pBuildObjectIdTable (
|
|
IN PSETUPTABLE TablePtr,
|
|
OUT PUINT FirstLinePtr, OPTIONAL
|
|
OUT PUINT LastLinePtr OPTIONAL
|
|
);
|
|
|
|
|
|
BOOL
|
|
pIsObjIdValid (
|
|
IN PVOID ObjIdTable,
|
|
IN UINT ObjId,
|
|
OUT PUINT Line OPTIONAL
|
|
);
|
|
|
|
|
|
|
|
BOOL
|
|
ProcessStfFiles (
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
ProcessStfFiles enumerates the memdb category Stf and converts
|
|
the paths in the STF to the new locations.
|
|
|
|
Arguments:
|
|
|
|
none
|
|
|
|
Return Value:
|
|
|
|
TRUE if success, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
MEMDB_ENUM e;
|
|
DWORD ops;
|
|
|
|
if (MemDbGetValueEx (&e, MEMDB_CATEGORY_STF, NULL, NULL)) {
|
|
do {
|
|
//
|
|
// Is file handled? If so, skip it.
|
|
//
|
|
|
|
ops = GetOperationsOnPath (e.szName);
|
|
if (ops & (OPERATION_MIGDLL_HANDLED|ALL_DELETE_OPERATIONS|OPERATION_FILE_DISABLED)) {
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// Process the file
|
|
//
|
|
|
|
DEBUGMSG ((DBG_STF, "Processing %s", e.szName));
|
|
|
|
if (!pProcessSetupTableFile (e.szName)) {
|
|
//
|
|
// Log the failure
|
|
//
|
|
|
|
LOG ((LOG_INFORMATION, (PCSTR)MSG_COULD_NOT_PROCESS_STF_LOG, e.szName));
|
|
} else {
|
|
TickProgressBar ();
|
|
}
|
|
|
|
} while (MemDbEnumNextValue (&e));
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
BOOL
|
|
pProcessSetupTable (
|
|
IN OUT PSETUPTABLE TablePtr
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pProcessSetupTable scans the entire setup table file specified, looking
|
|
for CopyFile, CopySection, RemoveFile or RemoveSection lines. If any are
|
|
found, any paths that point to moved or deleted files are adjusted, and
|
|
any STF group references are updated.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
Return Value:
|
|
|
|
TRUE if success, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT MaxObj;
|
|
UINT Line;
|
|
UINT Obj;
|
|
PCTSTR EntryStr;
|
|
PCTSTR DataStr;
|
|
PCTSTR InstallDestDir;
|
|
PCTSTR *ArgArray;
|
|
PCTSTR p;
|
|
UINT ArgCount;
|
|
PTABLEENTRY Entry;
|
|
TCHAR SystemDir[MAX_TCHAR_PATH];
|
|
UINT SystemDirLen;
|
|
PCTSTR UpdatedDir;
|
|
PVOID ObjTable;
|
|
TCHAR MovedPath[MAX_TCHAR_PATH];
|
|
DWORD MovedPathLen;
|
|
PCTSTR NewPath;
|
|
PTSTR q;
|
|
DWORD FileStatus;
|
|
PTSTR *ListOfWacks;
|
|
PTSTR *WackPos;
|
|
BOOL Result = TRUE;
|
|
|
|
MaxObj = TablePtr->MaxObj;
|
|
|
|
if (TablePtr->SourceInfFile != INVALID_HANDLE_VALUE) {
|
|
//
|
|
// If an INF is specified, scan the STF/INF pair for references to moved files.
|
|
// If found, correct the files.
|
|
//
|
|
|
|
for (Line = 0 ; Line < TablePtr->LineCount ; Line++) {
|
|
|
|
if (!pGetNonEmptyTableEntry (TablePtr, Line, COLUMN_OBJECT_ID, NULL, &EntryStr)) {
|
|
continue;
|
|
}
|
|
|
|
Obj = _ttoi (EntryStr);
|
|
if (Obj < 1 || Obj > MaxObj) {
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// CopySection or RemoveSection: Data column has INF section name
|
|
//
|
|
|
|
if (!pGetNonEmptyTableEntry (TablePtr, Line, COLUMN_COMMAND, NULL, &EntryStr)) {
|
|
continue;
|
|
}
|
|
|
|
if (!pGetNonEmptyTableEntry (TablePtr, Line, COLUMN_OBJECT_DATA, NULL, &DataStr)) {
|
|
continue;
|
|
}
|
|
|
|
InstallDestDir = GetDestDir (TablePtr, Line);
|
|
if (!InstallDestDir) {
|
|
continue;
|
|
}
|
|
|
|
ArgArray = ParseCommaList (TablePtr, DataStr);
|
|
if (!ArgArray) {
|
|
continue;
|
|
}
|
|
|
|
for (ArgCount = 0 ; ArgArray[ArgCount] ; ArgCount++) {
|
|
// empty
|
|
}
|
|
|
|
__try {
|
|
|
|
if (StringIMatch (EntryStr, TEXT("CopySection")) ||
|
|
StringIMatch (EntryStr, TEXT("RemoveSection"))
|
|
) {
|
|
if (ArgCount != 1) {
|
|
__leave;
|
|
}
|
|
|
|
if (!pProcessSectionCommand (TablePtr, Line, ArgArray[0], InstallDestDir)) {
|
|
DEBUGMSG ((DBG_STF, "%s [%s] could not be processed", EntryStr, ArgArray[0]));
|
|
Result = FALSE;
|
|
__leave;
|
|
}
|
|
}
|
|
|
|
else if (StringIMatch (EntryStr, TEXT("CopyFile")) ||
|
|
StringIMatch (EntryStr, TEXT("RemoveFile")) ||
|
|
StringIMatch (EntryStr, TEXT("InstallSysFile"))
|
|
) {
|
|
|
|
if (ArgCount != 2) {
|
|
__leave;
|
|
}
|
|
|
|
if (!pProcessLineCommand (TablePtr, Line, ArgArray[0], ArgArray[1], InstallDestDir)) {
|
|
DEBUGMSG ((DBG_STF, "%s [%s] %s could not be processed", EntryStr, ArgArray[0], ArgArray[1]));
|
|
Result = FALSE;
|
|
__leave;
|
|
}
|
|
}
|
|
|
|
else if (StringIMatch (EntryStr, TEXT("CompanionFile"))) {
|
|
|
|
if (ArgCount != 2) {
|
|
__leave;
|
|
}
|
|
|
|
// First arg has a colon -- skip past it
|
|
p = _tcschr (ArgArray[0], TEXT(':'));
|
|
if (!p) {
|
|
__leave;
|
|
}
|
|
p = SkipSpace (_tcsinc (p));
|
|
|
|
if (!pProcessLineCommand (TablePtr, Line, p, ArgArray[1], InstallDestDir)) {
|
|
DEBUGMSG ((DBG_STF, "%s [%s] %s could not be processed", EntryStr, ArgArray[0], ArgArray[1]));
|
|
Result = FALSE;
|
|
__leave;
|
|
}
|
|
}
|
|
|
|
else if (StringIMatch (EntryStr, TEXT("InstallShared"))) {
|
|
|
|
if (ArgCount != 5) {
|
|
__leave;
|
|
}
|
|
|
|
if (!pProcessLineCommand (TablePtr, Line, ArgArray[0], ArgArray[1], InstallDestDir)) {
|
|
DEBUGMSG ((DBG_STF, "%s [%s] %s could not be processed", EntryStr, ArgArray[0], ArgArray[1]));
|
|
Result = FALSE;
|
|
__leave;
|
|
}
|
|
}
|
|
}
|
|
__finally {
|
|
FreeDestDir (TablePtr, InstallDestDir);
|
|
FreeCommaList (TablePtr, ArgArray);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Perform STF-only processing
|
|
//
|
|
|
|
SystemDirLen = wsprintf (SystemDir, TEXT("%s\\system\\"), g_WinDir);
|
|
ObjTable = pBuildObjectIdTable (TablePtr, NULL, NULL);
|
|
|
|
__try {
|
|
for (Line = 0 ; Line < TablePtr->LineCount ; Line++) {
|
|
|
|
//
|
|
// Get InstallDestDir and Entry from the line that needs to be modified.
|
|
//
|
|
|
|
if (!pGetNonEmptyTableEntry (
|
|
TablePtr,
|
|
Line,
|
|
COLUMN_INSTALL_DESTDIR,
|
|
&Entry,
|
|
&InstallDestDir
|
|
)) {
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// If InstallDestDir has a %windir%\system in it, we must adjust the path
|
|
// to point to system32.
|
|
//
|
|
|
|
if (StringIMatchTcharCount (InstallDestDir, SystemDir, SystemDirLen)) {
|
|
UpdatedDir = JoinPaths (
|
|
g_System32Dir,
|
|
InstallDestDir + SystemDirLen - 1
|
|
);
|
|
|
|
if (!ReplaceTableEntryStr (TablePtr, Entry, UpdatedDir)) {
|
|
LOG ((LOG_ERROR, "Could not replace a %%M path"));
|
|
Result = FALSE;
|
|
}
|
|
|
|
FreePathString (UpdatedDir);
|
|
|
|
if (!Result) {
|
|
__leave;
|
|
}
|
|
}
|
|
|
|
//
|
|
// If InstallDestDir points to a moved dir, we must fix it
|
|
//
|
|
|
|
else if (*InstallDestDir && _tcsnextc (_tcsinc (InstallDestDir)) == TEXT(':')) {
|
|
//
|
|
// Build list of wacks in the path
|
|
//
|
|
|
|
ListOfWacks = (PTSTR *) MemAlloc (g_hHeap, 0, sizeof (PTSTR) * MAX_TCHAR_PATH);
|
|
MYASSERT (ListOfWacks);
|
|
|
|
StringCopy (MovedPath, InstallDestDir);
|
|
q = _tcschr (MovedPath, TEXT('\\'));
|
|
WackPos = ListOfWacks;
|
|
|
|
if (q) {
|
|
|
|
while (*q) {
|
|
if (_tcsnextc (q) == TEXT('\\')) {
|
|
*WackPos = q;
|
|
WackPos++;
|
|
}
|
|
|
|
q = _tcsinc (q);
|
|
}
|
|
|
|
//
|
|
// We assume the STF always has an extra wack at the end
|
|
// of the path.
|
|
//
|
|
|
|
//
|
|
// Test each path from longest to shortest, skipping the root
|
|
//
|
|
|
|
FileStatus = FILESTATUS_UNCHANGED;
|
|
|
|
while (WackPos > ListOfWacks) {
|
|
|
|
WackPos--;
|
|
q = *WackPos;
|
|
*q = 0;
|
|
|
|
FileStatus = GetFileStatusOnNt (MovedPath);
|
|
if (FileStatus == FILESTATUS_MOVED) {
|
|
break;
|
|
}
|
|
|
|
DEBUGMSG_IF ((
|
|
FileStatus != FILESTATUS_UNCHANGED,
|
|
DBG_WARNING,
|
|
"STF may point to changed dir: %s",
|
|
MovedPath
|
|
));
|
|
}
|
|
|
|
if (FileStatus == FILESTATUS_MOVED) {
|
|
//
|
|
// Adjust the STF path
|
|
//
|
|
|
|
NewPath = GetPathStringOnNt (MovedPath);
|
|
|
|
if (NewPath) {
|
|
MovedPathLen = (PBYTE) q - (PBYTE) MovedPath;
|
|
|
|
UpdatedDir = JoinPaths (
|
|
NewPath,
|
|
ByteCountToPointer (InstallDestDir, MovedPathLen)
|
|
);
|
|
|
|
DEBUGMSG ((
|
|
DBG_STF,
|
|
"Line %u has a new install destination: %s",
|
|
Line,
|
|
UpdatedDir
|
|
));
|
|
|
|
if (!ReplaceTableEntryStr (TablePtr, Entry, UpdatedDir)) {
|
|
LOG ((LOG_ERROR, "Could not replace a moved path"));
|
|
Result = FALSE;
|
|
}
|
|
|
|
FreePathString (UpdatedDir);
|
|
FreePathString (NewPath);
|
|
|
|
if (!Result) {
|
|
__leave;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
MemFree (g_hHeap, 0, ListOfWacks);
|
|
}
|
|
}
|
|
}
|
|
__finally {
|
|
if (ObjTable) {
|
|
pSetupStringTableDestroy (ObjTable);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Update all the lines that reference other OBJs. This includes GROUP,
|
|
// DEPEND and COMPANIONFILE lines.
|
|
//
|
|
|
|
return Result && pUpdateObjReferences (TablePtr);
|
|
}
|
|
|
|
|
|
PVOID
|
|
pBuildObjectIdTable (
|
|
IN PSETUPTABLE TablePtr,
|
|
OUT PUINT FirstLinePtr, OPTIONAL
|
|
OUT PUINT LastLinePtr OPTIONAL
|
|
)
|
|
{
|
|
PVOID ObjIds;
|
|
UINT FirstLine, LastLine;
|
|
UINT Line;
|
|
UINT Obj;
|
|
UINT MaxObj;
|
|
PCTSTR EntryStr;
|
|
TCHAR NumBuf[32];
|
|
|
|
MaxObj = TablePtr->MaxObj;
|
|
|
|
//
|
|
// Alloc string table
|
|
//
|
|
|
|
ObjIds = pSetupStringTableInitializeEx (sizeof (DWORD), 0);
|
|
if (!ObjIds) {
|
|
LOG ((LOG_ERROR, "STF: Can't init string table"));
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// Fill string table with list of ObjIDs
|
|
//
|
|
|
|
FirstLine = 0;
|
|
LastLine = TablePtr->LineCount;
|
|
|
|
for (Line = 0 ; Line < LastLine ; Line++) {
|
|
if (!pGetNonEmptyTableEntry (TablePtr, Line, COLUMN_OBJECT_ID, NULL, &EntryStr)) {
|
|
continue;
|
|
}
|
|
|
|
Obj = _ttoi (EntryStr);
|
|
if (Obj < 1 || Obj > MaxObj) {
|
|
continue;
|
|
}
|
|
|
|
if (!FirstLine) {
|
|
FirstLine = Line;
|
|
}
|
|
|
|
wsprintf (NumBuf, TEXT("%u"), Obj);
|
|
if (-1 == pSetupStringTableAddStringEx (
|
|
ObjIds,
|
|
NumBuf,
|
|
STRTAB_CASE_SENSITIVE,
|
|
(PBYTE) &Line,
|
|
sizeof (DWORD)
|
|
)) {
|
|
LOG ((LOG_ERROR, "STF: Can't add to string table"));
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (FirstLinePtr) {
|
|
*FirstLinePtr = FirstLine;
|
|
}
|
|
|
|
if (LastLinePtr) {
|
|
*LastLinePtr = LastLine;
|
|
}
|
|
|
|
return ObjIds;
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
pIsObjIdValid (
|
|
IN PVOID ObjIdTable,
|
|
IN UINT ObjId,
|
|
OUT PUINT Line OPTIONAL
|
|
)
|
|
{
|
|
TCHAR NumBuf[32];
|
|
LONG rc;
|
|
DWORD LineData;
|
|
|
|
wsprintf (NumBuf, TEXT("%u"), ObjId);
|
|
|
|
rc = pSetupStringTableLookUpStringEx (
|
|
ObjIdTable,
|
|
NumBuf,
|
|
STRTAB_CASE_SENSITIVE|STRTAB_BUFFER_WRITEABLE,
|
|
(PBYTE) &LineData,
|
|
sizeof (DWORD)
|
|
);
|
|
|
|
if (Line && rc != -1) {
|
|
*Line = LineData;
|
|
}
|
|
|
|
return rc != -1;
|
|
}
|
|
|
|
|
|
PCTSTR
|
|
pValidateGroup (
|
|
IN PSETUPTABLE TablePtr,
|
|
IN PCTSTR EntryStr,
|
|
IN PVOID ObjIds
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pValidateGroup parses all object IDs in the EntryStr, compares them against
|
|
the string table ObjIds, and adds only those IDs that are in both EntryStr
|
|
and ObjIds. The caller receives a text pool string, which may be
|
|
empty.
|
|
|
|
The buffer must be freed by the caller with FreeText.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table being processed
|
|
EntryStr - Specifies the entry string that contains zero or more numeric
|
|
object ID references, sparated by spaces.
|
|
ObjIds - Specifies the string table of valid object IDs.
|
|
|
|
Return Value:
|
|
|
|
A pointer to the validated object ID string, or NULL if memory allocation
|
|
failed.
|
|
|
|
--*/
|
|
|
|
{
|
|
PTSTR Buffer;
|
|
PTSTR p;
|
|
PCTSTR q;
|
|
CHARTYPE ch;
|
|
UINT Obj;
|
|
TCHAR NumBuf[32];
|
|
|
|
//
|
|
// Validate EntryStr
|
|
//
|
|
|
|
Buffer = AllocText (LcharCount (EntryStr) + 1);
|
|
if (!Buffer) {
|
|
return NULL;
|
|
}
|
|
|
|
p = Buffer;
|
|
*p = 0;
|
|
|
|
q = EntryStr;
|
|
while (*q) {
|
|
ch = (CHARTYPE)_tcsnextc (q);
|
|
|
|
if (ch >= TEXT('0') && ch <= TEXT('9')) {
|
|
//
|
|
// Extract object ID reference
|
|
//
|
|
|
|
Obj = 0;
|
|
|
|
for (;;) {
|
|
ch = (CHARTYPE)_tcsnextc (q);
|
|
|
|
if (ch >= TEXT('0') && ch <= TEXT('9')) {
|
|
Obj = Obj * 10 + (ch - TEXT('0'));
|
|
} else {
|
|
break;
|
|
}
|
|
|
|
q = _tcsinc (q);
|
|
}
|
|
|
|
//
|
|
// If match found, add obj ID to data
|
|
//
|
|
|
|
if (pIsObjIdValid (ObjIds, Obj, NULL)) {
|
|
wsprintf (NumBuf, TEXT("%u"), Obj);
|
|
p = _tcsappend (p, NumBuf);
|
|
}
|
|
} else {
|
|
_copytchar (p, q);
|
|
p = _tcsinc (p);
|
|
*p = 0;
|
|
q = _tcsinc (q);
|
|
}
|
|
}
|
|
|
|
return Buffer;
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
pUpdateObjReferences (
|
|
IN PSETUPTABLE TablePtr
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pUpdateObjReferences scans the specified table for GROUP, DEPEND and COMPANIONFILE
|
|
lines, and for each line found, the line is updated.
|
|
|
|
In the case of a GROUP line, the data argument is updated if it points to one or
|
|
more invalid object IDs. If the cleanup operation causes a group to have zero
|
|
items, the group line itself is deleted, and the update is restarted.
|
|
|
|
In the case of a DEPEND line, if the first object ID no longer exists, then
|
|
the line is deleted, and the update is restarted. If an obj in the group
|
|
following the ? no longer exists, then the obj reference is deleted. If the
|
|
delete causes no objects to be listed, then the line is deleted.
|
|
|
|
In the case of a COMPANIONFILE line, the object ID is extracted from the data
|
|
argument, and the line is deleted if its original line is gone.
|
|
|
|
NOTE: This routine has a lot of exit conditions that cause leaks, but all of them
|
|
can only be hit by memory allocation failures
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
Return Value:
|
|
|
|
TRUE if processing was successful, or FALSE if an error occured. FALSE will cause
|
|
STF processing to fail for the current STF, and will likely generate an error log
|
|
entry.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT Line;
|
|
PVOID ObjIds;
|
|
PCTSTR EntryStr;
|
|
UINT Obj;
|
|
BOOL b = FALSE;
|
|
UINT FirstLine, LastLine;
|
|
PTSTR Buffer;
|
|
PTSTR DependBuf;
|
|
PTSTR p;
|
|
BOOL StartOver;
|
|
PTABLEENTRY Entry;
|
|
BOOL GroupMode;
|
|
BOOL CompanionFileMode;
|
|
BOOL DependMode;
|
|
|
|
do {
|
|
|
|
StartOver = FALSE;
|
|
|
|
ObjIds = pBuildObjectIdTable (TablePtr, &FirstLine, &LastLine);
|
|
if (!ObjIds) {
|
|
return FALSE;
|
|
}
|
|
|
|
Line = TablePtr->LineCount;
|
|
if (!FirstLine) {
|
|
//
|
|
// Small table -- no object IDs at all! Return TRUE to caller.
|
|
//
|
|
|
|
b = TRUE;
|
|
Line = 0;
|
|
}
|
|
|
|
//
|
|
// Look for lines that have object ID references
|
|
//
|
|
|
|
if (Line == TablePtr->LineCount) {
|
|
for (Line = FirstLine ; !StartOver && Line < LastLine ; Line++) {
|
|
if (!pGetNonEmptyTableEntry (TablePtr, Line, COLUMN_COMMAND, NULL, &EntryStr)) {
|
|
continue;
|
|
}
|
|
|
|
GroupMode = StringIMatch (EntryStr, TEXT("Group"));
|
|
CompanionFileMode = StringIMatch (EntryStr, TEXT("CompanionFile"));
|
|
DependMode = StringIMatch (EntryStr, TEXT("Depend"));
|
|
|
|
if (!GroupMode && !CompanionFileMode && !DependMode) {
|
|
continue;
|
|
}
|
|
|
|
if (!pGetNonEmptyTableEntry (TablePtr, Line, COLUMN_OBJECT_DATA, NULL, &EntryStr)) {
|
|
continue;
|
|
}
|
|
|
|
if (GroupMode) {
|
|
|
|
Buffer = (PTSTR) pValidateGroup (TablePtr, EntryStr, ObjIds);
|
|
if (!Buffer) {
|
|
break;
|
|
}
|
|
|
|
//
|
|
// If Buffer is empty, delete the group line, then start over
|
|
//
|
|
|
|
if (*Buffer == 0) {
|
|
pDeleteStfLine (TablePtr, Line);
|
|
StartOver = TRUE;
|
|
|
|
DEBUGMSG ((
|
|
DBG_STF,
|
|
"Group line %u references only deleted lines, so it was deleted as well.",
|
|
Line
|
|
));
|
|
}
|
|
|
|
//
|
|
// If Buffer is not empty, replace the data on the current line
|
|
//
|
|
|
|
else if (!StringMatch (EntryStr, Buffer)) {
|
|
DEBUGMSG ((
|
|
DBG_STF,
|
|
"Group has reference to one or more deleted objects. Original: %s New: %s",
|
|
EntryStr,
|
|
Buffer
|
|
));
|
|
|
|
Entry = GetTableEntry (TablePtr, Line, COLUMN_OBJECT_DATA, NULL);
|
|
MYASSERT (Entry);
|
|
|
|
if (Entry) {
|
|
if (!ReplaceTableEntryStr (TablePtr, Entry, Buffer)) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
FreeText (Buffer);
|
|
}
|
|
|
|
if (!StartOver && (DependMode || CompanionFileMode)) {
|
|
//
|
|
// Extract the obj ID from the data arg
|
|
//
|
|
|
|
Obj = _ttoi (EntryStr);
|
|
if (Obj || EntryStr[0] == TEXT('0')) {
|
|
|
|
if (!pIsObjIdValid (ObjIds, Obj, NULL)) {
|
|
//
|
|
// CompanionFile/Depend is for a line that was deleted. Delete
|
|
// the line and start over.
|
|
//
|
|
|
|
pDeleteStfLine (TablePtr, Line);
|
|
StartOver = TRUE;
|
|
|
|
DEBUGMSG_IF ((
|
|
CompanionFileMode,
|
|
DBG_STF,
|
|
"CompanionFile line %u references a deleted line %u, so it was deleted as well.",
|
|
Line,
|
|
Obj
|
|
));
|
|
|
|
DEBUGMSG_IF ((
|
|
DependMode,
|
|
DBG_STF,
|
|
"Depend line %u references a deleted line %u, so it was deleted as well.",
|
|
Line,
|
|
Obj
|
|
));
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!StartOver && DependMode) {
|
|
//
|
|
// Go beyond question mark, then validate group
|
|
//
|
|
|
|
p = _tcschr (EntryStr, TEXT('?'));
|
|
if (p) {
|
|
p = _tcsinc (p);
|
|
while (*p == TEXT(' ')) {
|
|
p++;
|
|
}
|
|
|
|
Buffer = (PTSTR) pValidateGroup (TablePtr, p, ObjIds);
|
|
if (!Buffer) {
|
|
break;
|
|
}
|
|
|
|
if (*Buffer == 0) {
|
|
pDeleteStfLine (TablePtr, Line);
|
|
StartOver = TRUE;
|
|
|
|
DEBUGMSG ((
|
|
DBG_STF,
|
|
"Depend line %u references only deleted lines, so it was deleted as well.",
|
|
Line
|
|
));
|
|
}
|
|
|
|
//
|
|
// If Buffer is not empty, replace the data on the current line
|
|
//
|
|
|
|
else if (!StringMatch (p, Buffer)) {
|
|
DependBuf = AllocText (ByteCount (Buffer) + 32);
|
|
if (!DependBuf) {
|
|
break;
|
|
}
|
|
|
|
StringCopyAB (DependBuf, EntryStr, p);
|
|
StringCat (DependBuf, Buffer);
|
|
|
|
DEBUGMSG ((
|
|
DBG_STF,
|
|
"Depend line has reference to one or more deleted objects. Original: %s New: %s",
|
|
EntryStr,
|
|
DependBuf
|
|
));
|
|
|
|
Entry = GetTableEntry (TablePtr, Line, COLUMN_OBJECT_DATA, NULL);
|
|
MYASSERT (Entry);
|
|
|
|
if (Entry) {
|
|
if (!ReplaceTableEntryStr (TablePtr, Entry, DependBuf)) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
FreeText (DependBuf);
|
|
}
|
|
|
|
FreeText (Buffer);
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
//
|
|
// If we managed to get through the loop, we are done! Return TRUE to caller.
|
|
//
|
|
|
|
if (Line == LastLine) {
|
|
b = TRUE;
|
|
}
|
|
}
|
|
|
|
pSetupStringTableDestroy (ObjIds);
|
|
|
|
} while (StartOver);
|
|
|
|
return b;
|
|
}
|
|
|
|
|
|
BOOL
|
|
pGetNonEmptyTableEntry (
|
|
IN PSETUPTABLE TablePtr,
|
|
IN UINT Line,
|
|
IN UINT Col,
|
|
OUT PTABLEENTRY *EntryPtr, OPTIONAL
|
|
OUT PCTSTR *EntryStr
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pGetNonEmptyTableEntry is a wrapper routine that gets an entry in the
|
|
STF table and returns TRUE only if the string actually exists and is
|
|
not empty. If a non-empty string is found, the pointer is returned to
|
|
the caller.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
Line - Specifies the line to get the entry for
|
|
|
|
Col - Specifies the column on the line to get the entry for
|
|
|
|
EntryPtr - Receives a pointer to the entry struct
|
|
|
|
EntryStr - Receives a pointer to the entry string
|
|
|
|
Return Value:
|
|
|
|
TRUE if entry exists and is not empty, FALSE if the entry does not exist
|
|
or is empty.
|
|
|
|
--*/
|
|
|
|
{
|
|
PCTSTR String;
|
|
PTABLEENTRY Entry;
|
|
|
|
Entry = GetTableEntry (TablePtr, Line, Col, &String);
|
|
if (!Entry) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (!String || !String[0]) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (EntryPtr) {
|
|
*EntryPtr = Entry;
|
|
}
|
|
|
|
if (EntryStr) {
|
|
*EntryStr = String;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
PCTSTR
|
|
pQuoteThis (
|
|
IN PCTSTR String
|
|
)
|
|
{
|
|
static TCHAR Buffer[128];
|
|
|
|
MYASSERT (ByteCount (String) < (sizeof (Buffer) - 2));
|
|
Buffer[0] = TEXT('\"');
|
|
StringCopy (&Buffer[1], String);
|
|
StringCat (Buffer, TEXT("\""));
|
|
|
|
return Buffer;
|
|
}
|
|
|
|
|
|
BOOL
|
|
pProcessSectionCommand (
|
|
IN OUT PSETUPTABLE TablePtr,
|
|
IN UINT StfLine,
|
|
IN PCTSTR InfSection,
|
|
IN PCTSTR InstallDestDir
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pProcessSectionCommand scans an INF section, determining which files
|
|
are deleted, moved or unchanged. If a file is moved or unchanged,
|
|
it is added to memdb. After the INF section is completely scanned,
|
|
the memdb structure is processed, causing additional INF sections
|
|
to be generated if any changes were made to the file paths.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
StfLine - Specifies the STF line that has a CopySection or RemoveSection
|
|
command.
|
|
|
|
InfSection - Specifies the INF section that lists files to be processed
|
|
|
|
InstallDestDir - Specifies destination directory specified by STF line
|
|
|
|
Return Value:
|
|
|
|
TRUE if success, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOL DeletedOrMoved = FALSE;
|
|
PSTFINFLINE InfLine;
|
|
PCTSTR * InfFields;
|
|
UINT Fields;
|
|
TCHAR FileName[MAX_TCHAR_PATH];
|
|
TCHAR FullPath[MAX_TCHAR_PATH * 2];
|
|
MEMDB_ENUM e;
|
|
CONVERTPATH_RC rc;
|
|
BOOL FirstSectionDone;
|
|
BOOL CreatedFlag;
|
|
PSTFINFSECTION NewInfSection;
|
|
PSTFINFLINE SrcInfLine;
|
|
PTSTR UpdatedData;
|
|
TCHAR DirPart[MAX_TCHAR_PATH];
|
|
PCTSTR FilePart;
|
|
PTSTR p, q;
|
|
PTABLEENTRY DataEntry;
|
|
PCTSTR *Array;
|
|
BOOL result = FALSE;
|
|
|
|
//
|
|
// Step one: scan all files in the corresponding INF section,
|
|
// moving them to memdb.
|
|
//
|
|
|
|
InfLine = StfGetFirstLineInSectionStr (TablePtr, InfSection);
|
|
if (!InfLine) {
|
|
//
|
|
// Workaround: sometimes the people who write the STFs embed all kinds of
|
|
// quotes in a section name.
|
|
//
|
|
Array = ParseCommaList (TablePtr, InfSection);
|
|
if (Array) {
|
|
if (Array[0]) {
|
|
InfLine = StfGetFirstLineInSectionStr (TablePtr, Array[0]);
|
|
}
|
|
FreeCommaList (TablePtr, Array);
|
|
}
|
|
}
|
|
|
|
if (!InfLine) {
|
|
MYASSERT(InfSection);
|
|
DEBUGMSG ((DBG_STF, "STF file references section %s that does not exist", InfSection));
|
|
return TRUE;
|
|
}
|
|
|
|
__try {
|
|
do {
|
|
//
|
|
// Parse the INF line into fields
|
|
//
|
|
|
|
InfFields = ParseCommaList (TablePtr, InfLine->Data);
|
|
if (!InfFields) {
|
|
MYASSERT(InfLine->Data);
|
|
DEBUGMSG ((DBG_WARNING, "INF file has non-parsable data", InfLine->Data));
|
|
} else {
|
|
for (Fields = 0 ; InfFields[Fields] ; Fields++) {
|
|
}
|
|
|
|
if (Fields < 19) {
|
|
MYASSERT(InfLine->Data);
|
|
DEBUGMSG ((DBG_WARNING, "INF file line %s has less than 19 fields", InfLine->Data));
|
|
} else {
|
|
//
|
|
// Get the file name from this INF line (field #2)
|
|
//
|
|
|
|
pGetFileNameFromInfField (FileName, InfFields[1]);
|
|
|
|
if (TcharCount (InstallDestDir) + 1 + TcharCount (FileName) < ARRAYSIZE(FullPath)) {
|
|
StringCopy (FullPath, InstallDestDir);
|
|
StringCat (AppendPathWack (FullPath), FileName);
|
|
|
|
rc = ConvertWin9xPath (FullPath);
|
|
|
|
if (rc != CONVERTPATH_NOT_REMAPPED) {
|
|
DeletedOrMoved = TRUE;
|
|
}
|
|
|
|
if (rc != CONVERTPATH_DELETED) {
|
|
//
|
|
// Add this file to memdb
|
|
//
|
|
|
|
if (!MemDbSetValueEx (
|
|
MEMDB_CATEGORY_STF_TEMP,
|
|
FullPath,
|
|
NULL,
|
|
NULL,
|
|
(DWORD) InfLine,
|
|
NULL
|
|
)) {
|
|
LOG ((LOG_ERROR, "STF: MemDbSetValueEx failed"));
|
|
__leave;
|
|
}
|
|
}
|
|
} else {
|
|
LOG ((LOG_WARNING, "STF has path of %s\\%s that is too long", InstallDestDir, FileName));
|
|
}
|
|
}
|
|
|
|
FreeCommaList (TablePtr, InfFields);
|
|
}
|
|
|
|
InfLine = StfGetNextLineInSection (InfLine);
|
|
} while (InfLine);
|
|
|
|
if (!DeletedOrMoved) {
|
|
//
|
|
// No changes necessary
|
|
//
|
|
result = TRUE;
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// Now write out each unique directory to the INF. Update
|
|
// the STF line to point to the first new INF section.
|
|
//
|
|
|
|
FirstSectionDone = FALSE;
|
|
|
|
if (MemDbGetValueEx (&e, MEMDB_CATEGORY_STF_TEMP, NULL, NULL)) {
|
|
do {
|
|
//
|
|
// Name gives full path of new file location.
|
|
// Value points to INF line that is to be copied.
|
|
//
|
|
|
|
NewInfSection = pGetNewInfSection (TablePtr, e.szName, &CreatedFlag);
|
|
if (!NewInfSection) {
|
|
LOG ((LOG_ERROR, "Process Section Command failed because Get New Inf Section failed"));
|
|
__leave;
|
|
}
|
|
|
|
SrcInfLine = (PSTFINFLINE) e.dwValue;
|
|
|
|
_tcssafecpy (DirPart, e.szName, MAX_TCHAR_PATH);
|
|
FilePart = GetFileNameFromPath (DirPart);
|
|
MYASSERT (FilePart && FilePart > DirPart);
|
|
*_tcsdec2 (DirPart, FilePart) = 0;
|
|
|
|
//
|
|
// File name may have changed. If so, specify the file name in the
|
|
// angle brackets.
|
|
//
|
|
|
|
UpdatedData = DuplicatePathString (
|
|
SrcInfLine->Data,
|
|
SizeOfString (FilePart) + 2 * sizeof (TCHAR)
|
|
);
|
|
|
|
p = _tcschr (SrcInfLine->Data, TEXT(','));
|
|
MYASSERT(p);
|
|
p = _tcsinc (p);
|
|
q = _tcschr (p, TEXT(','));
|
|
MYASSERT(q);
|
|
p = _tcschr (p, TEXT('<'));
|
|
if (!p || p > q) {
|
|
p = q;
|
|
}
|
|
|
|
StringCopyAB (UpdatedData, SrcInfLine->Data, q);
|
|
wsprintf (_tcschr (UpdatedData, 0), TEXT("<%s>"), FilePart);
|
|
StringCat (UpdatedData, q);
|
|
|
|
DEBUGMSG ((DBG_STF, "INF changed from %s to %s", SrcInfLine->Data, UpdatedData));
|
|
StfAddInfLineToTable (TablePtr, NewInfSection, SrcInfLine->Key, UpdatedData, SrcInfLine->LineFlags);
|
|
|
|
//
|
|
// If first section, update STF line to use new section
|
|
//
|
|
|
|
if (!FirstSectionDone) {
|
|
DataEntry = GetTableEntry (TablePtr, StfLine, COLUMN_OBJECT_DATA, NULL);
|
|
if (!ReplaceTableEntryStr (TablePtr, DataEntry, pQuoteThis (NewInfSection->Name))) {
|
|
LOG ((LOG_ERROR, "Could not update table entry"));
|
|
__leave;
|
|
}
|
|
FirstSectionDone = TRUE;
|
|
}
|
|
|
|
//
|
|
// If not first section and CreateFlag is TRUE, create a new STF line
|
|
// and point it to new INF section.
|
|
//
|
|
|
|
else if (CreatedFlag) {
|
|
if (!pCreateNewStfLine (TablePtr, StfLine, pQuoteThis (NewInfSection->Name), DirPart)) {
|
|
LOG ((LOG_ERROR, "Could not create a new line"));
|
|
__leave;
|
|
}
|
|
}
|
|
} while (MemDbEnumNextValue (&e));
|
|
} else {
|
|
//
|
|
// All files were deleted, and this STF line is no longer needed.
|
|
//
|
|
|
|
DEBUGMSG ((DBG_STF, "STF Line %u is no longer needed", StfLine));
|
|
|
|
if (!pReplaceDirReferences (TablePtr, StfLine, InstallDestDir)) {
|
|
__leave;
|
|
}
|
|
|
|
if (!pDeleteStfLine (TablePtr, StfLine)) {
|
|
__leave;
|
|
}
|
|
}
|
|
|
|
result = TRUE;
|
|
}
|
|
__finally {
|
|
MemDbDeleteTree (MEMDB_CATEGORY_STF_TEMP);
|
|
MemDbDeleteTree (MEMDB_CATEGORY_STF_SECTIONS);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
VOID
|
|
pGenerateUniqueKeyName (
|
|
IN PSETUPTABLE TablePtr,
|
|
IN PSTFINFSECTION Section,
|
|
IN PCTSTR Root,
|
|
OUT PTSTR UniqueKey
|
|
)
|
|
{
|
|
UINT Sequencer = 0;
|
|
PTSTR p;
|
|
|
|
UniqueKey[0] = 0;
|
|
p = _tcsappend (UniqueKey, Root);
|
|
|
|
for (;;) {
|
|
Sequencer++;
|
|
wsprintf (p, TEXT("%03u"), Sequencer);
|
|
|
|
if (!StfFindLineInInfSection (TablePtr, Section, UniqueKey)) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
BOOL
|
|
pProcessLineCommand (
|
|
IN OUT PSETUPTABLE TablePtr,
|
|
IN UINT StfLine,
|
|
IN PCTSTR InfSection,
|
|
IN PCTSTR InfKey,
|
|
IN PCTSTR InstallDestDir
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pProcessLineCommand determinins if the file assoicated with the command
|
|
was deleted, moved or unchanged. If the file was deleted, the STF line
|
|
is deleted. If the file was moved, the STF line is adjusted. If the
|
|
file has no change, the routine does not modify the STF table.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
StfLine - Specifies the STF line that has a CopySection or RemoveSection
|
|
command.
|
|
|
|
InfSection - Specifies the INF section that lists the file to be processed
|
|
|
|
InfKey - Specifies the INF key in InfSection identifing the file
|
|
|
|
InstallDestDir - Specifies destination directory specified by STF line
|
|
|
|
Return Value:
|
|
|
|
TRUE if success, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
PSTFINFSECTION Section;
|
|
PSTFINFLINE InfLine;
|
|
PCTSTR *InfFields;
|
|
UINT Fields;
|
|
TCHAR FileName[MAX_TCHAR_PATH];
|
|
TCHAR FullPath[MAX_TCHAR_PATH * 2];
|
|
CONVERTPATH_RC rc;
|
|
TCHAR OrgDirPart[MAX_TCHAR_PATH];
|
|
PCTSTR OrgFilePart;
|
|
TCHAR DirPart[MAX_TCHAR_PATH];
|
|
PCTSTR FilePart;
|
|
PTABLEENTRY DataEntry;
|
|
PTABLEENTRY FileEntry;
|
|
PCTSTR *Array;
|
|
TCHAR NewKeyName[MAX_TCHAR_PATH];
|
|
PTSTR NewLine;
|
|
PTSTR p;
|
|
UINT Size;
|
|
PCTSTR OldField;
|
|
|
|
Section = StfFindInfSectionInTable (TablePtr, InfSection);
|
|
|
|
if (!Section) {
|
|
//
|
|
// Workaround: sometimes the people who write the STFs embed all kinds of
|
|
// quotes in a section name.
|
|
//
|
|
Array = ParseCommaList (TablePtr, InfSection);
|
|
if (Array) {
|
|
if (Array[0]) {
|
|
Section = StfFindInfSectionInTable (TablePtr, Array[0]);
|
|
}
|
|
|
|
FreeCommaList (TablePtr, Array);
|
|
}
|
|
}
|
|
|
|
if (!Section) {
|
|
MYASSERT(InfSection);
|
|
DEBUGMSG ((
|
|
DBG_STF,
|
|
"STF has reference to non-existent INF section ([%s])",
|
|
InfSection
|
|
));
|
|
return TRUE;
|
|
}
|
|
|
|
InfLine = StfFindLineInInfSection (TablePtr, Section, InfKey);
|
|
if (!InfLine) {
|
|
MYASSERT(InfSection && InfKey);
|
|
DEBUGMSG ((
|
|
DBG_STF,
|
|
"STF has reference to non-existent INF key ([%s], %s)",
|
|
InfSection,
|
|
InfKey
|
|
));
|
|
return TRUE;
|
|
}
|
|
|
|
//
|
|
// Build full path
|
|
//
|
|
|
|
InfFields = ParseCommaList (TablePtr, InfLine->Data);
|
|
|
|
__try {
|
|
if (!InfFields) {
|
|
MYASSERT(InfLine->Data);
|
|
DEBUGMSG ((DBG_WARNING, "INF file has non-parsable data", InfLine->Data));
|
|
|
|
#pragma prefast(suppress:242, "Don't care about perf of try/finally here")
|
|
return TRUE;
|
|
}
|
|
|
|
for (Fields = 0 ; InfFields[Fields] ; Fields++) {
|
|
/* empty */
|
|
}
|
|
|
|
if (Fields < 19) {
|
|
MYASSERT(InfLine->Data);
|
|
DEBUGMSG ((DBG_WARNING, "INF file line %s has less than 19 fields", InfLine->Data));
|
|
|
|
#pragma prefast(suppress:242, "Don't care about perf of try/finally here")
|
|
return TRUE;
|
|
}
|
|
|
|
//
|
|
// Get the file name from this INF line (field #2)
|
|
//
|
|
|
|
pGetFileNameFromInfField (FileName, InfFields[1]);
|
|
|
|
if (TcharCount (InstallDestDir) + 1 + TcharCount (FileName) < ARRAYSIZE (FullPath)) {
|
|
StringCopy (FullPath, InstallDestDir);
|
|
StringCopy (AppendPathWack (FullPath), FileName);
|
|
} else {
|
|
LOG ((LOG_WARNING, "INF file line %s has long path %s\\%s", InfLine->Data, InstallDestDir, FileName));
|
|
|
|
#pragma prefast(suppress:242, "Don't care about perf of try/finally here")
|
|
return TRUE;
|
|
}
|
|
}
|
|
__finally {
|
|
FreeCommaList (TablePtr, InfFields);
|
|
}
|
|
|
|
//
|
|
// Determine mapping
|
|
//
|
|
|
|
_tcssafecpy (OrgDirPart, FullPath, MAX_TCHAR_PATH);
|
|
OrgFilePart = GetFileNameFromPath (OrgDirPart);
|
|
if (OrgFilePart <= OrgDirPart) {
|
|
// File probably wasn't installed
|
|
return TRUE;
|
|
}
|
|
|
|
*_tcsdec2 (OrgDirPart, OrgFilePart) = 0;
|
|
|
|
rc = ConvertWin9xPath (FullPath);
|
|
|
|
_tcssafecpy (DirPart, FullPath, MAX_TCHAR_PATH);
|
|
FilePart = GetFileNameFromPath (DirPart);
|
|
MYASSERT (FilePart && FilePart > DirPart);
|
|
*_tcsdec2 (DirPart, FilePart) = 0;
|
|
|
|
//
|
|
// Deleted? Delete the STF line.
|
|
//
|
|
|
|
if (rc == CONVERTPATH_DELETED) {
|
|
DEBUGMSG ((DBG_STF, "STF Line %u is no longer needed", StfLine));
|
|
|
|
if (!pReplaceDirReferences (TablePtr, StfLine, InstallDestDir)) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (!pDeleteStfLine (TablePtr, StfLine)) {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Moved? Update the STF line.
|
|
//
|
|
|
|
else if (rc == CONVERTPATH_REMAPPED) {
|
|
//
|
|
// Has the file name changed? If so, point it to the new location.
|
|
//
|
|
|
|
if (!StringIMatch (OrgFilePart, FilePart)) {
|
|
//
|
|
// Update INI file by duplicating the INI line
|
|
//
|
|
|
|
// Generate a unique key name
|
|
pGenerateUniqueKeyName (TablePtr, Section, TEXT("WIN9XUPG_"), NewKeyName);
|
|
|
|
// Compute size needed
|
|
Size = 0;
|
|
for (Fields = 0 ; InfFields[Fields] ; Fields++) {
|
|
if (Fields != 1) {
|
|
Size += ByteCount (InfFields[Fields]);
|
|
Size += sizeof (TCHAR);
|
|
} else {
|
|
Size += ByteCount (FilePart);
|
|
}
|
|
}
|
|
|
|
// Generate the INF line
|
|
NewLine = AllocText (Size);
|
|
if (NewLine) {
|
|
p = NewLine;
|
|
*p = 0;
|
|
|
|
for (Fields = 0 ; InfFields[Fields] ; Fields++) {
|
|
if (Fields) {
|
|
p = _tcsappend (p, TEXT(","));
|
|
}
|
|
|
|
if (Fields == 1) {
|
|
p = _tcsappend (p, FilePart);
|
|
} else {
|
|
p = _tcsappend (p, InfFields[Fields]);
|
|
}
|
|
}
|
|
|
|
// Write the new line
|
|
StfAddInfLineToTable (
|
|
TablePtr,
|
|
Section,
|
|
NewKeyName,
|
|
NewLine,
|
|
LINEFLAG_KEY_QUOTED
|
|
);
|
|
|
|
FreeText (NewLine);
|
|
|
|
// Update the STF
|
|
FileEntry = GetTableEntry (TablePtr, StfLine, COLUMN_OBJECT_DATA, NULL);
|
|
MYASSERT (FileEntry);
|
|
|
|
OldField = GetTableEntryStr (TablePtr, FileEntry);
|
|
NewLine = AllocText (
|
|
ByteCount (FilePart) +
|
|
ByteCount (OldField) +
|
|
ByteCount (InfSection)
|
|
);
|
|
|
|
StringCopy (NewLine, OldField);
|
|
p = _tcschr (NewLine, TEXT(':'));
|
|
if (!p) {
|
|
p = NewLine;
|
|
} else {
|
|
p = _tcsinc (p);
|
|
MYASSERT (*p == TEXT(' '));
|
|
p = _tcsinc (p);
|
|
}
|
|
|
|
*p = 0;
|
|
p = _tcsappend (p, InfSection);
|
|
p = _tcsappend (p, TEXT(", "));
|
|
p = _tcsappend (p, NewKeyName);
|
|
|
|
// ignore memory failure, it will be picked up below
|
|
ReplaceTableEntryStr (TablePtr, FileEntry, NewLine);
|
|
|
|
FreeText (NewLine);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Store the directory change in the STF table.
|
|
//
|
|
|
|
DataEntry = GetTableEntry (TablePtr, StfLine, COLUMN_INSTALL_DESTDIR, NULL);
|
|
AppendWack (DirPart);
|
|
|
|
if (!ReplaceTableEntryStr (TablePtr, DataEntry, DirPart)) {
|
|
LOG ((
|
|
LOG_ERROR,
|
|
"Could not update table entry for single command"
|
|
));
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
PSTFINFSECTION
|
|
pGetNewInfSection (
|
|
IN PSETUPTABLE TablePtr,
|
|
IN PCTSTR FileSpec,
|
|
OUT PBOOL CreatedFlag
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pGetNewInfSection determines if a section already exists for the specified
|
|
file specification, and if so it returns the pointer to the existing
|
|
section. If the section does not exist, it creates a new section (making
|
|
sure there are no other sections with the same name), and returns a
|
|
pointer to the new section.
|
|
|
|
This routine is used by the code that splits one section into several.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
FileSpec - Specifies the full file path of the file being processed
|
|
|
|
CreatedFlag - Receives TRUE if a new section had to be created
|
|
|
|
Return Value:
|
|
|
|
A pointer to the INF section in which the file should be added to, or
|
|
NULL if an error occurred.
|
|
|
|
--*/
|
|
|
|
{
|
|
TCHAR DirName[MAX_TCHAR_PATH];
|
|
TCHAR Node[MEMDB_MAX];
|
|
DWORD SectionNum;
|
|
PTSTR p;
|
|
TCHAR SectionName[64];
|
|
static DWORD SectionSeq = 0;
|
|
|
|
*CreatedFlag = FALSE;
|
|
|
|
//
|
|
// See if section already exists, and if it does, return the section
|
|
// pointer.
|
|
//
|
|
|
|
_tcssafecpy (DirName, FileSpec, MAX_TCHAR_PATH);
|
|
p = _tcsrchr (DirName, TEXT('\\'));
|
|
if (p) {
|
|
*p = 0;
|
|
}
|
|
|
|
MemDbBuildKey (Node, MEMDB_CATEGORY_STF_SECTIONS, DirName, NULL, NULL);
|
|
|
|
if (MemDbGetValue (Node, &SectionNum)) {
|
|
wsprintf (SectionName, S_SECTIONNAME_SPRINTF, SectionNum);
|
|
return StfFindInfSectionInTable (TablePtr, SectionName);
|
|
}
|
|
|
|
//
|
|
// The section does not exist. Find an unused section, write the
|
|
// reference to memdb and return the section pointer.
|
|
//
|
|
|
|
while (TRUE) {
|
|
SectionSeq++;
|
|
wsprintf (SectionName, S_SECTIONNAME_SPRINTF, SectionSeq);
|
|
if (!StfFindInfSectionInTable (TablePtr, SectionName)) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
*CreatedFlag = TRUE;
|
|
MemDbSetValue (Node, SectionSeq);
|
|
return StfAddInfSectionToTable (TablePtr, SectionName);
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
pGetFileNameFromInfField (
|
|
OUT PTSTR FileName,
|
|
IN PCTSTR InfField
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pGetFileNameFromInfField extracts a long file name, enclosed between
|
|
angle-brackets. According to the STF spec, the syntax is shortname<longname>.
|
|
This routine returns longname.
|
|
|
|
Arguments:
|
|
|
|
FileName - Supplies a MAX_TCHAR_PATH buffer that receives the long file name.
|
|
|
|
InfField - Specifies the text from the INF field conforming to the
|
|
shortname<longname> syntax.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
PTSTR LongName;
|
|
PCTSTR p;
|
|
|
|
LongName = _tcschr (InfField, TEXT('<'));
|
|
if (LongName) {
|
|
_tcssafecpy (FileName, _tcsinc (LongName), MAX_TCHAR_PATH);
|
|
LongName = _tcschr (FileName, TEXT('>'));
|
|
if (LongName) {
|
|
*LongName = 0;
|
|
}
|
|
} else {
|
|
p = _tcsrchr (InfField, TEXT('\\'));
|
|
if (!p) {
|
|
p = InfField;
|
|
} else {
|
|
p = _tcsinc (p);
|
|
}
|
|
|
|
_tcssafecpy (FileName, p, MAX_TCHAR_PATH);
|
|
}
|
|
}
|
|
|
|
|
|
BOOL
|
|
pDeleteStfLine (
|
|
IN OUT PSETUPTABLE TablePtr,
|
|
IN UINT StfLine
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pDeleteStfLine removes an STF line from the table. It first checks to
|
|
see if a destination directory is specified, and if one is, it moves it
|
|
to the next line, unless the next line also has a destination directory.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
StfLine - Specifies the STF line to delete
|
|
|
|
Return Value:
|
|
|
|
TRUE if success, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
PTABLEENTRY TitleEntry;
|
|
PTABLEENTRY DataEntry;
|
|
BOOL b;
|
|
|
|
//
|
|
// We simply replace the command with CreateIniLine, which is harmless
|
|
//
|
|
|
|
TitleEntry = GetTableEntry (TablePtr, StfLine, COLUMN_COMMAND, NULL);
|
|
DataEntry = GetTableEntry (TablePtr, StfLine, COLUMN_OBJECT_DATA, NULL);
|
|
|
|
if (!TitleEntry || !DataEntry) {
|
|
MYASSERT (FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
b = ReplaceTableEntryStr (TablePtr, TitleEntry, TEXT("CreateIniLine"));
|
|
|
|
if (b) {
|
|
b = ReplaceTableEntryStr (
|
|
TablePtr,
|
|
DataEntry,
|
|
TEXT("\"WIN.INI\", \"Old Win9x Setting\", \"DummyKey\", \"unused\"")
|
|
);
|
|
}
|
|
|
|
return b;
|
|
|
|
|
|
#if 0
|
|
PTABLEENTRY NextLineEntry;
|
|
PCTSTR InstallDestDir;
|
|
PCTSTR NextInstallDestDir;
|
|
|
|
__try {
|
|
//
|
|
// Test for last line
|
|
//
|
|
|
|
if (StfLine + 1 >= TablePtr->LineCount) {
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// Obtain StfLine dest dir (column 10)
|
|
//
|
|
|
|
if (!GetTableEntry (TablePtr, StfLine, COLUMN_DEST_DIR, &InstallDestDir)) {
|
|
//
|
|
// StfLine is not valid (unexpected)
|
|
//
|
|
|
|
DEBUGMSG ((DBG_STF, "Line %u does not have column 10", StfLine));
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// If no dest dir, do not modify the next line
|
|
//
|
|
|
|
if (!InstallDestDir || !InstallDestDir[0]) {
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// Obtain next line's dest dir (column 10)
|
|
//
|
|
|
|
NextLineEntry = GetTableEntry (TablePtr, StfLine + 1, COLUMN_DEST_DIR, &NextInstallDestDir);
|
|
if (!NextLineEntry) {
|
|
//
|
|
// Next StfLine is not valid (unexpected)
|
|
//
|
|
|
|
DEBUGMSG ((DBG_WHOOPS, "pDeleteStfLine: Next line %u does not have column 10", StfLine+1));
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// If next line's dest dir is not empty, do not modify the line
|
|
//
|
|
|
|
if (NextInstallDestDir && NextInstallDestDir[0]) {
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// Now set InstallDestDir on NextLineEntry line
|
|
//
|
|
|
|
if (!ReplaceTableEntryStr (TablePtr, NextLineEntry, InstallDestDir)) {
|
|
DEBUGMSG ((
|
|
DBG_ERROR,
|
|
"Cannot replace a destination dir in STF file. "
|
|
"Line=%u, InstallDestDir=%s",
|
|
StfLine + 1,
|
|
InstallDestDir
|
|
));
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
__finally {
|
|
}
|
|
|
|
return DeleteLineInTable (TablePtr, StfLine);
|
|
#endif
|
|
}
|
|
|
|
|
|
BOOL
|
|
pReplaceDirReferences (
|
|
IN OUT PSETUPTABLE TablePtr,
|
|
IN UINT StfLine,
|
|
IN PCTSTR DirSpec
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pReplaceDirReferences scans column 14, looking for all references
|
|
to StfLine, and replaces the reference with DirSpec.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
StfLine - Specifies the STF line to substitute
|
|
|
|
DirSpec - Specifies the effective directory for the STF line.
|
|
This directory is used if the STF line has to be
|
|
deleted.
|
|
|
|
Return Value:
|
|
|
|
TRUE if success, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT Line, Count;
|
|
PTABLEENTRY InstallDestDirEntry;
|
|
PCTSTR InstallDestDir;
|
|
TCHAR NumStr[32];
|
|
UINT NumStrLen;
|
|
PCTSTR AfterPercentNum;
|
|
CHARTYPE c;
|
|
PCTSTR NewInstallDestDir;
|
|
|
|
NumStrLen = wsprintf (NumStr, TEXT("%%%u"), StfLine);
|
|
Count = TablePtr->LineCount;
|
|
|
|
for (Line = 0 ; Line < Count ; Line++) {
|
|
InstallDestDirEntry = GetTableEntry (TablePtr, Line, COLUMN_DEST_DIR, &InstallDestDir);
|
|
if (!InstallDestDirEntry) {
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// Does InstallDestDir have %<n> (where <n> equals StfLine)?
|
|
//
|
|
|
|
if (StringIMatchTcharCount (InstallDestDir, NumStr, NumStrLen)) {
|
|
//
|
|
// The next character must be a wack or nul
|
|
//
|
|
|
|
AfterPercentNum = InstallDestDir + NumStrLen;
|
|
|
|
c = (CHARTYPE)_tcsnextc (AfterPercentNum);
|
|
if (c == 0 || c == TEXT('\\')) {
|
|
//
|
|
// Create new dest dir
|
|
//
|
|
|
|
if (c) {
|
|
NewInstallDestDir = JoinPaths (DirSpec, _tcsinc (AfterPercentNum));
|
|
} else {
|
|
NewInstallDestDir = DuplicatePathString (DirSpec, 0);
|
|
}
|
|
|
|
__try {
|
|
if (!ReplaceTableEntryStr (TablePtr, InstallDestDirEntry, NewInstallDestDir)) {
|
|
LOG ((
|
|
LOG_ERROR,
|
|
"Cannot replace a destination dir in STF file. "
|
|
"Line=%u, NewInstallDestDir=%s",
|
|
Line,
|
|
NewInstallDestDir
|
|
));
|
|
|
|
#pragma prefast(suppress:242, "Don't care about perf of try/finally here")
|
|
return FALSE;
|
|
}
|
|
ELSE_DEBUGMSG ((
|
|
DBG_STF,
|
|
"Line %u: Dest dir %s replaced with %s",
|
|
Line,
|
|
InstallDestDir,
|
|
NewInstallDestDir
|
|
));
|
|
}
|
|
__finally {
|
|
FreePathString (NewInstallDestDir);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
BOOL
|
|
pCreateNewStfLine (
|
|
IN OUT PSETUPTABLE TablePtr,
|
|
IN UINT StfLine,
|
|
IN PCTSTR ObjectData,
|
|
IN PCTSTR InstallDestDir
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pCreateNewStfLine inserts a new line immediately following
|
|
StfLine, using the maximum object number. It copies the
|
|
STF line specified and modifies the ObjectData (column 5)
|
|
and InstallDestDir (column 14).
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the setup table file to process
|
|
|
|
StfLine - Specifies the prototype STF line
|
|
|
|
ObjectData - Specifies the replacement for the ObjectData
|
|
column
|
|
|
|
InstallDestDir - Specifies the replacement for the DestDir
|
|
column
|
|
|
|
Return Value:
|
|
|
|
TRUE if success, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT NewLine;
|
|
UINT NewObj;
|
|
PTABLEENTRY CopyEntry, NewEntry;
|
|
UINT Col;
|
|
TCHAR Buf[32], ReplaceBuf[32];
|
|
|
|
//
|
|
// Copy StfLine to StfLine+1, updating fields as necessary
|
|
//
|
|
|
|
NewLine = StfLine + 1;
|
|
TablePtr->MaxObj++;
|
|
NewObj = TablePtr->MaxObj;
|
|
|
|
if (!InsertEmptyLineInTable (TablePtr, NewLine)) {
|
|
LOG ((LOG_ERROR, "Unable to insert new line in STF table"));
|
|
return FALSE;
|
|
}
|
|
|
|
Col = 0;
|
|
while (TRUE) {
|
|
CopyEntry = GetTableEntry (TablePtr, StfLine, Col, NULL);
|
|
if (!CopyEntry) {
|
|
break;
|
|
}
|
|
|
|
if (!AppendTableEntry (TablePtr, NewLine, CopyEntry)) {
|
|
LOG ((LOG_ERROR, "Unable to append all entries to line"));
|
|
return FALSE;
|
|
}
|
|
|
|
NewEntry = GetTableEntry (TablePtr, NewLine, Col, NULL);
|
|
MYASSERT(NewEntry);
|
|
if (!NewEntry) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (Col == 0) {
|
|
wsprintf (Buf, TEXT("%u"), NewObj);
|
|
|
|
if (!ReplaceTableEntryStr (TablePtr, NewEntry, Buf)) {
|
|
LOG ((LOG_ERROR, "Unable to replace ObjID on line"));
|
|
return FALSE;
|
|
}
|
|
} else if (Col == COLUMN_OBJECT_DATA) {
|
|
if (!ReplaceTableEntryStr (TablePtr, NewEntry, ObjectData)) {
|
|
LOG ((LOG_ERROR, "Unable to replace ObjectData on line"));
|
|
return FALSE;
|
|
}
|
|
} else if (Col == COLUMN_INSTALL_DESTDIR) {
|
|
if (!ReplaceTableEntryStr (TablePtr, NewEntry, InstallDestDir)) {
|
|
LOG ((LOG_ERROR, "Unable to replace ObjectData on line"));
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
Col++;
|
|
}
|
|
|
|
//
|
|
// Find all lines with references to StfLine and add in NewLine
|
|
//
|
|
|
|
wsprintf (Buf, TEXT("%u"), StfLine);
|
|
wsprintf (ReplaceBuf, TEXT("%u %u"), StfLine, NewLine);
|
|
|
|
return pSearchAndReplaceObjectRefs (TablePtr, Buf, ReplaceBuf);
|
|
}
|
|
|
|
|
|
BOOL
|
|
pSearchAndReplaceObjectRefs (
|
|
IN OUT PSETUPTABLE TablePtr,
|
|
IN PCTSTR SrcStr,
|
|
IN PCTSTR DestStr
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pSearchAndReplaceObjectRefs scans column 5 of the setup table,
|
|
looking for any occurance of SrcStr and replacing it with
|
|
DestStr.
|
|
|
|
Arguments:
|
|
|
|
TablePtr - Specifies the STF table to process
|
|
|
|
SrcStr - Specifies the string to locate and replace
|
|
|
|
DestStr - Specifies the replacement string
|
|
|
|
Return Value:
|
|
|
|
TRUE if the STF file was converted, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT Line;
|
|
UINT Count;
|
|
PTABLEENTRY Entry;
|
|
PCTSTR LineString;
|
|
PCTSTR UpdatedString;
|
|
|
|
Count = TablePtr->LineCount;
|
|
|
|
for (Line = 0 ; Line < Count ; Line++) {
|
|
Entry = GetTableEntry (TablePtr, Line, COLUMN_OBJECT_DATA, &LineString);
|
|
if (!Entry || !LineString || !LineString[0]) {
|
|
continue;
|
|
}
|
|
|
|
UpdatedString = StringSearchAndReplace (LineString, SrcStr, DestStr);
|
|
|
|
if (UpdatedString) {
|
|
__try {
|
|
if (!ReplaceTableEntryStr (TablePtr, Entry, UpdatedString)) {
|
|
LOG ((LOG_ERROR, "Unable to replace text on line"));
|
|
|
|
#pragma prefast(suppress:242, "Don't care about perf of try/finally here")
|
|
return FALSE;
|
|
}
|
|
}
|
|
__finally {
|
|
FreePathString (UpdatedString);
|
|
}
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
pProcessSetupTableFile (
|
|
IN PCTSTR StfFileSpec
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pProcessSetupTableFile performs all processing on the file specified.
|
|
Here are the steps involved in converting an STF file:
|
|
|
|
- Determine the associated INF
|
|
- Prepare the SETUPTABLE structure
|
|
- Scan the table for file-based actions
|
|
- Convert file paths used by the actions
|
|
- Convert group references when STF lines are split
|
|
- Write the modifications to disk
|
|
- Replace the original INF and STF with the new versions
|
|
|
|
Arguments:
|
|
|
|
StfFileSpec - Specifies the full file path to the STF file needing processing.
|
|
The associated INF must be in the same directory as the STF
|
|
file referencing it.
|
|
|
|
Return Value:
|
|
|
|
TRUE if the STF file was converted, FALSE if failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
SETUPTABLE Table;
|
|
DWORD StfAttribs, InfAttribs;
|
|
BOOL b;
|
|
TCHAR SourceStf[MAX_TCHAR_PATH];
|
|
TCHAR DestStf[MAX_TCHAR_PATH];
|
|
TCHAR SourceInf[MAX_TCHAR_PATH];
|
|
TCHAR DestInf[MAX_TCHAR_PATH];
|
|
|
|
if (!CreateSetupTable (StfFileSpec, &Table)) {
|
|
DEBUGMSG ((DBG_STF, "ProcessSetupTableFile: Error parsing file %s.", StfFileSpec));
|
|
return FALSE;
|
|
}
|
|
|
|
__try {
|
|
//
|
|
// Process the table
|
|
//
|
|
|
|
if (!pProcessSetupTable (&Table)) {
|
|
DEBUGMSG ((DBG_STF, "ProcessSetupTableFile: Error processing table for %s.", StfFileSpec));
|
|
|
|
#pragma prefast(suppress:242, "Don't care about perf of try/finally here")
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Write changes to temporary files
|
|
//
|
|
|
|
if (!WriteSetupTable (&Table)) {
|
|
LOG ((LOG_ERROR, "Cannot write setup table for %s.", StfFileSpec));
|
|
|
|
#pragma prefast(suppress:242, "Don't care about perf of try/finally here")
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Copy paths before we destroy the table struct
|
|
//
|
|
|
|
_tcssafecpy (SourceStf, Table.SourceStfFileSpec, MAX_TCHAR_PATH);
|
|
_tcssafecpy (DestStf, Table.DestStfFileSpec, MAX_TCHAR_PATH);
|
|
_tcssafecpy (SourceInf, Table.SourceInfFileSpec, MAX_TCHAR_PATH);
|
|
if (Table.DestInfFileSpec) {
|
|
_tcssafecpy (DestInf, Table.DestInfFileSpec, MAX_TCHAR_PATH);
|
|
} else {
|
|
*DestInf = 0;
|
|
}
|
|
}
|
|
__finally {
|
|
DestroySetupTable (&Table);
|
|
}
|
|
|
|
|
|
//
|
|
// Replace the original files with temporary files
|
|
//
|
|
|
|
StfAttribs = GetFileAttributes (SourceStf);
|
|
if (StfAttribs != 0xffffffff) {
|
|
LONG rc;
|
|
|
|
SetFileAttributes (SourceStf, FILE_ATTRIBUTE_NORMAL);
|
|
DeleteFile (SourceStf);
|
|
|
|
rc = GetLastError();
|
|
|
|
b = OurMoveFile (DestStf, SourceStf);
|
|
if (!b) {
|
|
return FALSE;
|
|
}
|
|
|
|
SetFileAttributes (SourceStf, StfAttribs);
|
|
}
|
|
|
|
InfAttribs = GetFileAttributes (SourceInf);
|
|
if (InfAttribs != 0xffffffff && *DestInf) {
|
|
SetFileAttributes (SourceInf, FILE_ATTRIBUTE_NORMAL);
|
|
DeleteFile (SourceInf);
|
|
|
|
b = OurMoveFile (DestInf, SourceInf);
|
|
if (!b) {
|
|
return FALSE;
|
|
}
|
|
|
|
SetFileAttributes (SourceInf, InfAttribs);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|