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.
 
 
 
 
 
 

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;
}