/*++ Copyright (c) 1996 Microsoft Corporation Module Name: links.c Abstract: This source file implements the Win95 side of LNK and PIF processing Author: Calin Negreanu (calinn) 09-Feb-1998 Revision History: calinn 23-Sep-1998 Redesigned several pieces --*/ #include "pch.h" #include "migappp.h" #include "migdbp.h" POOLHANDLE g_LinksPool = NULL; INT g_LinkStubSequencer = 0; typedef struct _LINK_STRUCT { PCTSTR ReportEntry; PCTSTR Category; PCTSTR Context; PCTSTR Object; PCTSTR LinkName; PCTSTR LinkNameNoPath; PMIGDB_CONTEXT MigDbContext; } LINK_STRUCT, *PLINK_STRUCT; BOOL InitLinkAnnounce ( VOID ) { // // Create PoolMem for keeping all structures during this phase // g_LinksPool = PoolMemInitNamedPool ("Links Pool"); return TRUE; } BOOL DoneLinkAnnounce ( VOID ) { // Write LinkStub max sequencer data MemDbSetValue (MEMDB_CATEGORY_LINKSTUB_MAXSEQUENCE, g_LinkStubSequencer); // // Free Links Pool. // if (g_LinksPool != NULL) { PoolMemDestroyPool (g_LinksPool); g_LinksPool = NULL; } return TRUE; } BOOL SaveLinkFiles ( IN PFILE_HELPER_PARAMS Params ) { PCTSTR Ext; if (Params->Handled) { return TRUE; } Ext = GetFileExtensionFromPath (Params->FullFileSpec); // Save LNK and PIF filenames to memdb to enumerate later if (Ext && (StringIMatch (Ext, TEXT("LNK")) || StringIMatch (Ext, TEXT("PIF")))) { MemDbSetValueEx ( MEMDB_CATEGORY_SHORTCUTS, Params->FullFileSpec, NULL, NULL, 0, NULL ); } return TRUE; } VOID RemoveLinkFromSystem ( IN LPCTSTR LinkPath ) { // // Remove any move or copy operation specified for the link, then // mark it for deletion. // RemoveOperationsFromPath (LinkPath, ALL_DEST_CHANGE_OPERATIONS); MarkFileForDelete (LinkPath); } // // Function to send instruction to MemDb to edit a shell link or pif file. // It checks to see whether the link involved has been touched yet by any // MemDb operation. It modifies the target path, if in a relocating directory, // to one of the relocated copies. // VOID pAddLinkEditToMemDb ( IN PCTSTR LinkPath, IN PCTSTR NewTarget, IN PCTSTR NewArgs, IN PCTSTR NewWorkDir, IN PCTSTR NewIconPath, IN INT NewIconNr, IN PLNK_EXTRA_DATA ExtraData, OPTIONAL IN BOOL ForceToShowNormal ) { UINT sequencer; TCHAR tmpStr [20]; sequencer = AddOperationToPath (LinkPath, OPERATION_LINK_EDIT); if (sequencer == INVALID_OFFSET) { DEBUGMSG ((DBG_ERROR, "Cannot set OPERATION_LINK_EDIT on %s", LinkPath)); return; } if (NewTarget) { AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, NewTarget, MEMDB_CATEGORY_LINKEDIT_TARGET); } if (NewArgs) { AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, NewArgs, MEMDB_CATEGORY_LINKEDIT_ARGS); } if (NewWorkDir) { AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, NewWorkDir, MEMDB_CATEGORY_LINKEDIT_WORKDIR); } if (NewIconPath) { AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, NewIconPath, MEMDB_CATEGORY_LINKEDIT_ICONPATH); } if (NewIconPath) { _itoa (NewIconNr, tmpStr, 16); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_ICONNUMBER); } if (ForceToShowNormal) { AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, TEXT("1"), MEMDB_CATEGORY_LINKEDIT_SHOWNORMAL); } if (ExtraData) { _itoa (ExtraData->FullScreen, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_FULLSCREEN); _itoa (ExtraData->xSize, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_XSIZE); _itoa (ExtraData->ySize, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_YSIZE); _itoa (ExtraData->QuickEdit, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_QUICKEDIT); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, ExtraData->FontName, MEMDB_CATEGORY_LINKEDIT_FONTNAME); _itoa (ExtraData->xFontSize, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_XFONTSIZE); _itoa (ExtraData->yFontSize, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_YFONTSIZE); _itoa (ExtraData->FontWeight, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_FONTWEIGHT); _itoa (ExtraData->FontFamily, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_FONTFAMILY); _itoa (ExtraData->CurrentCodePage, tmpStr, 10); AddPropertyToPathEx (sequencer, OPERATION_LINK_EDIT, tmpStr, MEMDB_CATEGORY_LINKEDIT_CODEPAGE); } MYASSERT (IsFileMarkedForOperation (LinkPath, OPERATION_LINK_EDIT)); } // // Function to send instruction to MemDb to save some data about a link that's going to be edited. // We do that to be able to restore this link later using lnkstub.exe // UINT pAddLinkStubToMemDb ( IN PCTSTR LinkPath, IN PCTSTR OldTarget, IN PCTSTR OldArgs, IN PCTSTR OldWorkDir, IN PCTSTR OldIconPath, IN INT OldIconNr, IN DWORD OldShowMode, IN DWORD Announcement, IN DWORD Availability ) { UINT sequencer; TCHAR tmpStr [20]; MEMDB_ENUM e, e1; TCHAR key [MEMDB_MAX]; MYASSERT (OldTarget || OldWorkDir || OldIconPath || OldIconNr); sequencer = AddOperationToPath (LinkPath, OPERATION_LINK_STUB); if (sequencer == INVALID_OFFSET) { DEBUGMSG ((DBG_ERROR, "Cannot set OPERATION_LINK_STUB on %s", LinkPath)); return 0; } g_LinkStubSequencer++; if (OldTarget) { AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, OldTarget, MEMDB_CATEGORY_LINKSTUB_TARGET); } if (OldArgs) { AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, OldArgs, MEMDB_CATEGORY_LINKSTUB_ARGS); } if (OldWorkDir) { AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, OldWorkDir, MEMDB_CATEGORY_LINKSTUB_WORKDIR); } if (OldIconPath) { AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, OldIconPath, MEMDB_CATEGORY_LINKSTUB_ICONPATH); } if (OldIconPath) { _itoa (OldIconNr, tmpStr, 16); AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, tmpStr, MEMDB_CATEGORY_LINKSTUB_ICONNUMBER); } _itoa (OldShowMode, tmpStr, 16); AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, tmpStr, MEMDB_CATEGORY_LINKSTUB_SHOWMODE); _itoa (g_LinkStubSequencer, tmpStr, 16); AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, tmpStr, MEMDB_CATEGORY_LINKSTUB_SEQUENCER); _itoa (Announcement, tmpStr, 16); AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, tmpStr, MEMDB_CATEGORY_LINKSTUB_ANNOUNCEMENT); _itoa (Availability, tmpStr, 16); AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, tmpStr, MEMDB_CATEGORY_LINKSTUB_REPORTAVAIL); MemDbBuildKey (key, MEMDB_CATEGORY_REQFILES_MAIN, OldTarget, TEXT("*"), NULL); if (MemDbEnumFirstValue (&e, key, MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY)) { do { MemDbBuildKey (key, MEMDB_CATEGORY_REQFILES_ADDNL, e.szName, TEXT("*"), NULL); if (MemDbEnumFirstValue (&e1, key, MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY)) { AddPropertyToPathEx (sequencer, OPERATION_LINK_STUB, e1.szName, MEMDB_CATEGORY_LINKSTUB_REQFILE); } } while (MemDbEnumNextValue (&e)); } MYASSERT (IsFileMarkedForOperation (LinkPath, OPERATION_LINK_STUB)); return g_LinkStubSequencer; } BOOL pReportEntry ( IN PCTSTR ReportEntry, IN PCTSTR Category, IN PCTSTR Message, IN PCTSTR Context, IN PCTSTR Object ) { PCTSTR component; component = JoinPaths (ReportEntry, Category); MsgMgr_ContextMsg_Add (Context, component, Message); MsgMgr_LinkObjectWithContext (Context, Object); FreePathString (component); return TRUE; } PTSTR GetLastDirFromPath ( IN PCTSTR FileName ) { PTSTR result = NULL; PTSTR temp = NULL; PTSTR ptr; temp = DuplicatePathString (FileName, 0); __try { ptr = (PTSTR)GetFileNameFromPath (temp); if (ptr == temp) { __leave; } ptr = _tcsdec (temp, ptr); if (!ptr) { __leave; } *ptr = 0; ptr = (PTSTR)GetFileNameFromPath (temp); if (ptr == temp) { __leave; } result = DuplicatePathString (ptr, 0); } __finally { FreePathString (temp); } return result; } PTSTR GetDriveFromPath ( IN PCTSTR FileName ) { PTSTR result; PTSTR ptr; result = DuplicatePathString (FileName, 0); ptr = _tcschr (result, TEXT(':')); if (!ptr) { FreePathString (result); result = NULL; } else { *ptr = 0; } return result; } #define MAX_PRIORITY 0xFFFF BOOL HandleDeferredAnnounce ( IN PCTSTR LinkName, IN PCTSTR ModuleName, IN BOOL DosApp ) { TCHAR key [MEMDB_MAX]; PMIGDB_CONTEXT migDbContext; DWORD actType; PLINK_STRUCT linkStruct; PCTSTR reportEntry = NULL; DWORD priority; PCTSTR newLinkName = NULL; PCTSTR linkName = NULL; PCTSTR extPtr; MEMDB_ENUM eNicePaths; DWORD messageId = 0; PTSTR pattern = NULL; PTSTR category = NULL; PTSTR tempParse = NULL; PTSTR lastDir; PTSTR drive; DWORD oldValue; DWORD oldPrior; PTSTR argArray[3]; PCTSTR p; PTSTR q; BOOL reportEntryIsResource = TRUE; PCTSTR temp1, temp2; MYASSERT(ModuleName); MemDbBuildKey (key, MEMDB_CATEGORY_DEFERREDANNOUNCE, ModuleName, NULL, NULL); if (!MemDbGetValueAndFlags (key, (PDWORD)(&migDbContext), &actType)) { actType = ACT_UNKNOWN; migDbContext = NULL; } // // we need to set the following variables: // - ReportEntry - is going to be either "Software Incompatible with NT", // "Software with minor problems" or // "Software that require reinstallation" // - Category - is one of the following: - Localized section name // - link name (with friendly addition) // - Unlocalized section name // - Message - this is in migdb context // // - Object - this is module name // linkStruct = (PLINK_STRUCT) PoolMemGetMemory (g_LinksPool, sizeof (LINK_STRUCT)); ZeroMemory (linkStruct, sizeof (LINK_STRUCT)); linkStruct->MigDbContext = migDbContext; linkStruct->Object = PoolMemDuplicateString (g_LinksPool, ModuleName); switch (actType) { case ACT_REINSTALL: #if 0 if ((linkStruct->MigDbContext) && (linkStruct->MigDbContext->Message) ) { reportEntry = GetStringResource (MSG_MINOR_PROBLEM_ROOT); } else { reportEntry = GetStringResource (MSG_REINSTALL_ROOT); } #endif temp1 = GetStringResource (MSG_REINSTALL_ROOT); if (!temp1) { break; } temp2 = GetStringResource ( linkStruct->MigDbContext && linkStruct->MigDbContext->Message ? MSG_REINSTALL_DETAIL_SUBGROUP : MSG_REINSTALL_LIST_SUBGROUP ); if (!temp2) { break; } reportEntry = JoinPaths (temp1, temp2); reportEntryIsResource = FALSE; FreeStringResource (temp1); FreeStringResource (temp2); break; case ACT_REINSTALL_BLOCK: temp1 = GetStringResource (MSG_BLOCKING_ITEMS_ROOT); if (!temp1) { break; } temp2 = GetStringResource (MSG_REINSTALL_BLOCK_ROOT); if (!temp2) { break; } reportEntry = JoinPaths (temp1, temp2); reportEntryIsResource = FALSE; FreeStringResource (temp1); FreeStringResource (temp2); break; case ACT_MINORPROBLEMS: reportEntry = GetStringResource (MSG_MINOR_PROBLEM_ROOT); break; case ACT_INCOMPATIBLE: case ACT_INC_NOBADAPPS: case ACT_INC_IHVUTIL: case ACT_INC_PREINSTUTIL: case ACT_INC_SIMILAROSFUNC: if (DosApp && (*g_Boot16 != BOOT16_NO)) { reportEntry = GetStringResource (MSG_DOS_DESIGNED_ROOT); } else { temp1 = GetStringResource (MSG_INCOMPATIBLE_ROOT); switch (actType) { case ACT_INC_SIMILAROSFUNC: temp2 = GetStringResource (MSG_INCOMPATIBLE_UTIL_SIMILAR_FEATURE_SUBGROUP); break; case ACT_INC_PREINSTUTIL: temp2 = GetStringResource (MSG_INCOMPATIBLE_PREINSTALLED_UTIL_SUBGROUP); break; case ACT_INC_IHVUTIL: temp2 = GetStringResource (MSG_INCOMPATIBLE_HW_UTIL_SUBGROUP); break; default: temp2 = GetStringResource ( linkStruct->MigDbContext && linkStruct->MigDbContext->Message ? MSG_INCOMPATIBLE_DETAIL_SUBGROUP: MSG_TOTALLY_INCOMPATIBLE_SUBGROUP ); break; } MYASSERT (temp1 && temp2); reportEntry = JoinPaths (temp1, temp2); reportEntryIsResource = FALSE; FreeStringResource (temp1); FreeStringResource (temp2); } break; case ACT_INC_SAFETY: MYASSERT (LinkName); temp1 = GetStringResource (MSG_INCOMPATIBLE_ROOT); temp2 = GetStringResource (MSG_REMOVED_FOR_SAFETY_SUBGROUP); MYASSERT (temp1 && temp2); reportEntry = JoinPaths (temp1, temp2); reportEntryIsResource = FALSE; FreeStringResource (temp1); FreeStringResource (temp2); newLinkName = JoinPaths (S_RUNKEYFOLDER, GetFileNameFromPath (LinkName)); break; case ACT_UNKNOWN: reportEntry = GetStringResource (MSG_UNKNOWN_ROOT); break; default: LOG((LOG_ERROR, "Unknown action for deferred announcement.")); return FALSE; } if (!newLinkName) { newLinkName = LinkName; } if (reportEntry != NULL) { linkStruct->ReportEntry = PoolMemDuplicateString (g_LinksPool, reportEntry); if (reportEntryIsResource) { FreeStringResource (reportEntry); } else { FreePathString (reportEntry); } } linkStruct->LinkName = newLinkName?PoolMemDuplicateString (g_LinksPool, newLinkName):NULL; // // all we need to set now is the category // // if we have a migdb context with a Localized name section // if ((migDbContext != NULL) && (migDbContext->SectLocalizedName != NULL) ) { linkStruct->Context = PoolMemDuplicateString (g_LinksPool, migDbContext->SectLocalizedName); linkStruct->Category = PoolMemDuplicateString (g_LinksPool, migDbContext->SectLocalizedName); priority = 0; } else { linkStruct->Context = PoolMemDuplicateString (g_LinksPool, newLinkName?newLinkName:ModuleName); if (newLinkName == NULL) { MYASSERT (migDbContext); if (migDbContext->SectName) { linkStruct->Category = PoolMemDuplicateString (g_LinksPool, migDbContext->SectName); } else { linkStruct->Category = NULL; } priority = 0; } else { linkName = GetFileNameFromPath (newLinkName); extPtr = GetFileExtensionFromPath (linkName); if (extPtr != NULL) { extPtr = _tcsdec (linkName, extPtr); } if (extPtr == NULL) { extPtr = GetEndOfString (linkName); } messageId = 0; priority = MAX_PRIORITY; if (MemDbEnumFirstValue (&eNicePaths, MEMDB_CATEGORY_NICE_PATHS"\\*", MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY)) { do { pattern = JoinPaths (eNicePaths.szName, "\\*"); if (IsPatternMatch (pattern, newLinkName)) { if (priority > eNicePaths.UserFlags) { messageId = eNicePaths.dwValue; priority = eNicePaths.UserFlags; } } FreePathString (pattern); } while (MemDbEnumNextValue (&eNicePaths)); } category = AllocText ((PBYTE) extPtr - (PBYTE) linkName + sizeof (TCHAR)); p = linkName; q = category; while (p < extPtr) { if (_tcsnextc (p) == TEXT(' ')) { do { p++; } while (_tcsnextc (p) == TEXT(' ')); if (q > category && *p) { *q++ = TEXT(' '); } } else if (IsLeadByte (p)) { *q++ = *p++; *q++ = *p++; } else { *q++ = *p++; } } *q = 0; if (messageId == 0) { lastDir = GetLastDirFromPath (newLinkName); drive = GetDriveFromPath (newLinkName); if (drive != NULL) { drive[0] = (TCHAR)toupper (drive[0]); if (lastDir != NULL) { argArray [0] = category; argArray [1] = lastDir; argArray [2] = drive; tempParse = (PTSTR)ParseMessageID (MSG_NICE_PATH_DRIVE_AND_FOLDER, argArray); } else { argArray [0] = category; argArray [1] = drive; tempParse = (PTSTR)ParseMessageID (MSG_NICE_PATH_DRIVE, argArray); } } else { if (lastDir != NULL) { argArray [0] = category; argArray [1] = lastDir; tempParse = (PTSTR)ParseMessageID (MSG_NICE_PATH_FOLDER, argArray); } else { argArray [0] = category; tempParse = (PTSTR)ParseMessageID (MSG_NICE_PATH_LINK, argArray); } } linkStruct->Category = PoolMemDuplicateString (g_LinksPool, tempParse); FreeStringResourcePtrA (&tempParse); priority = MAX_PRIORITY; } else { tempParse = (PTSTR)ParseMessageID (messageId, &category); StringCopy (category, tempParse); linkStruct->Category = PoolMemDuplicateString (g_LinksPool, tempParse); FreeStringResourcePtrA (&tempParse); } FreeText (category); } } linkStruct->LinkNameNoPath = linkName?PoolMemDuplicateString (g_LinksPool, linkName):linkStruct->Context; MemDbBuildKey ( key, MEMDB_CATEGORY_REPORT_LINKS, linkStruct->ReportEntry, linkName?linkName:linkStruct->Context, ModuleName); if ((!MemDbGetValueAndFlags (key, &oldValue, &oldPrior)) || (oldPrior > priority) ) { MemDbSetValueAndFlags (key, (DWORD)linkStruct, priority, 0); } if (newLinkName != LinkName) { FreePathString (newLinkName); } return TRUE; } BOOL pIsGUIDLauncherApproved ( IN PCTSTR FileName ) { INFCONTEXT context; MYASSERT (g_Win95UpgInf != INVALID_HANDLE_VALUE); return (SetupFindFirstLine (g_Win95UpgInf, S_APPROVED_GUID_LAUNCHER, FileName, &context)); } #define GUID_LEN (sizeof ("{00000000-0000-0000-0000-000000000000}") - 1) #define GUID_DASH_1 (sizeof ("{00000000") - 1) #define GUID_DASH_2 (sizeof ("{00000000-0000") - 1) #define GUID_DASH_3 (sizeof ("{00000000-0000-0000") - 1) #define GUID_DASH_4 (sizeof ("{00000000-0000-0000-0000") - 1) BOOL pSendCmdLineGuidsToMemdb ( IN PCTSTR File, IN PCTSTR Target, IN PCTSTR Arguments ) /*++ Routine Description: pSendCmdLineGuidsToMemdb saves any GUIDs contained in a command line to memdb, along with the file name. Later, OLEREG resolves the GUIDs and deletes the file if a GUID is incompatible. Arguments: File - Specifies the file to delete if the command line arguments contain an invalid GUID. Target - Specifies the Target (needs to be one of the approved targets for the LNK file to go away in an incompatible case). Arguments - Specifies a command line that may contain one or more GUIDs in the {a-b-c-d-e} format. Return value: TRUE - the operation was successful FALSE - the operation failed --*/ { LPCTSTR p, q; DWORD Offset; BOOL b; static DWORD Seq = 0; TCHAR TextSeq[16]; TCHAR Guid[GUID_LEN + 1]; PCTSTR namePtr; namePtr = GetFileNameFromPath (Target); if (namePtr && pIsGUIDLauncherApproved (namePtr)) { p = _tcschr (Arguments, TEXT('{')); while (p) { q = _tcschr (p, TEXT('}')); if (q && ((q - p) == (GUID_LEN - 1))) { if (p[GUID_DASH_1] == TEXT('-') && p[GUID_DASH_2] == TEXT('-') && p[GUID_DASH_3] == TEXT('-') && p[GUID_DASH_4] == TEXT('-') ) { // // Extract the GUID // q = _tcsinc (q); StringCopyAB (Guid, p, q); // // Add the file name // b = MemDbSetValueEx ( MEMDB_CATEGORY_LINK_STRINGS, File, NULL, NULL, 0, &Offset ); if (b) { // // Now add an entry for the GUID // Seq++; wsprintf (TextSeq, TEXT("%u"), Seq); b = MemDbSetValueEx ( MEMDB_CATEGORY_LINK_GUIDS, Guid, TextSeq, NULL, Offset, NULL ); } if (!b) { LOG ((LOG_ERROR, "Failed to store command line guids.")); } } } p = _tcschr (p + 1, TEXT('{')); } } return TRUE; } BOOL pIsFileInStartup ( IN PCTSTR FileName ) { TCHAR key [MEMDB_MAX]; MemDbBuildKey (key, MEMDB_CATEGORY_SF_STARTUP, FileName, NULL, NULL); return (MemDbGetPatternValue (key, NULL)); } BOOL pProcessShortcut ( IN PCTSTR FileName, IN IShellLink *ShellLink, IN IPersistFile *PersistFile ) { TCHAR shortcutTarget [MEMDB_MAX]; TCHAR shortcutArgs [MEMDB_MAX]; TCHAR shortcutWorkDir [MEMDB_MAX]; TCHAR shortcutIconPath [MEMDB_MAX]; PTSTR shortcutNewTarget = NULL; PTSTR shortcutNewArgs = NULL; PTSTR shortcutNewIconPath = NULL; PTSTR shortcutNewWorkDir = NULL; PTSTR commandPath = NULL; PTSTR fullPath = NULL; PCTSTR extPtr; INT shortcutIcon; INT newShortcutIcon; DWORD shortcutShowMode; WORD shortcutHotKey; DWORD fileStatus; BOOL msDosMode; BOOL dosApp; DWORD attrib; LNK_EXTRA_DATA ExtraData; INT lnkIdx; TCHAR lnkIdxStr [10]; BOOL toBeModified = FALSE; BOOL ConvertedLnk = FALSE; DWORD announcement; DWORD availability; __try { fileStatus = GetFileStatusOnNt (FileName); if (((fileStatus & FILESTATUS_DELETED ) == FILESTATUS_DELETED ) || ((fileStatus & FILESTATUS_REPLACED) == FILESTATUS_REPLACED) ) { __leave; } if (!ExtractShortcutInfo ( shortcutTarget, shortcutArgs, shortcutWorkDir, shortcutIconPath, &shortcutIcon, &shortcutHotKey, &dosApp, &msDosMode, &shortcutShowMode, &ExtraData, FileName, ShellLink, PersistFile )) { __leave; } if (msDosMode) { // // we want to modify this PIF file so it doesn't have MSDOS mode set // we will only add it to the modify list. The NT side will know what // to do when a PIF is marked for beeing modify // toBeModified = TRUE; } if (IsFileMarkedForAnnounce (shortcutTarget)) { announcement = GetFileAnnouncement (shortcutTarget); if (g_ConfigOptions.ShowAllReport || ((announcement != ACT_INC_IHVUTIL) && (announcement != ACT_INC_PREINSTUTIL) && (announcement != ACT_INC_SIMILAROSFUNC) ) ) { HandleDeferredAnnounce (FileName, shortcutTarget, dosApp); } } fileStatus = GetFileStatusOnNt (shortcutTarget); if ((fileStatus & FILESTATUS_DELETED) == FILESTATUS_DELETED) { if (IsFileMarkedForAnnounce (shortcutTarget)) { if (!pIsFileInStartup (FileName)) { if (!g_ConfigOptions.KeepBadLinks) { RemoveLinkFromSystem (FileName); } else { // we only care about LNK files if (StringIMatch (GetFileExtensionFromPath (FileName), TEXT("LNK"))) { // let's see what kind of announcement we have here. // We want to leave the LNK as is if the app was announced // using MigDb. However, if the app was announced using // dynamic checking (module checking) then we want to point // this shortcut to our stub EXE announcement = GetFileAnnouncement (shortcutTarget); if ((announcement == ACT_INC_NOBADAPPS) || (announcement == ACT_REINSTALL) || (announcement == ACT_REINSTALL_BLOCK) || (announcement == ACT_INC_IHVUTIL) || (announcement == ACT_INC_PREINSTUTIL) || (announcement == ACT_INC_SIMILAROSFUNC) ) { // // This is the case when we want to redirect this LNK to point // to our lnk stub. Extract will fail if the icon is known-good. // In that case, keep using the target icon. // if (ExtractIconIntoDatFile ( (*shortcutIconPath)?shortcutIconPath:shortcutTarget, shortcutIcon, &g_IconContext, &newShortcutIcon )) { shortcutNewIconPath = JoinPaths (g_System32Dir, TEXT("migicons.exe")); shortcutIcon = newShortcutIcon; } else { shortcutNewIconPath = GetPathStringOnNt ( (*shortcutIconPath) ? shortcutIconPath : shortcutTarget ); } availability = g_ConfigOptions.ShowAllReport || ((announcement != ACT_INC_IHVUTIL) && (announcement != ACT_INC_PREINSTUTIL) && (announcement != ACT_INC_SIMILAROSFUNC) ); lnkIdx = pAddLinkStubToMemDb ( FileName, shortcutTarget, shortcutArgs, shortcutWorkDir, shortcutNewIconPath, shortcutIcon + 1, // Add 1 because lnkstub.exe is one-based, but we are zero based shortcutShowMode, announcement, availability ); wsprintf (lnkIdxStr, TEXT("%d"), lnkIdx); shortcutNewTarget = JoinPaths (g_System32Dir, S_LNKSTUB_EXE); shortcutNewArgs = DuplicatePathString (lnkIdxStr, 0); pAddLinkEditToMemDb ( FileName, shortcutNewTarget, shortcutNewArgs, shortcutNewWorkDir, shortcutNewIconPath, shortcutIcon, // don't add one -- shortcuts are zero based NULL, TRUE ); } } else { RemoveLinkFromSystem (FileName); } } } else { // // This is a startup item // RemoveLinkFromSystem (FileName); } } else { RemoveLinkFromSystem (FileName); } __leave; } if ((fileStatus & FILESTATUS_REPLACED) != FILESTATUS_REPLACED) { // // this target is not replaced by a migration DLL or by NT. We need // to know if this is a "known good" target. If not, we will announce // this link as beeing "unknown" // if (!IsFileMarkedAsKnownGood (shortcutTarget)) { fullPath = JoinPaths (shortcutWorkDir, shortcutTarget); if (!IsFileMarkedAsKnownGood (fullPath)) { extPtr = GetFileExtensionFromPath (shortcutTarget); if (extPtr) { if (StringIMatch (extPtr, TEXT("EXE"))) { // // This one statement controls our // "unknown" category. We have the // ability to list the things we don't // recognize. // // It is currently "off". // //HandleDeferredAnnounce (FileName, shortcutTarget, dosApp); } } } FreePathString (fullPath); } } // // If this LNK points to a target that will change, back up the // original LNK, because we might change it. // if (fileStatus & ALL_CHANGE_OPERATIONS) { MarkFileForBackup (FileName); } // // If target points to an OLE object, remove any links to incompatible OLE objects // pSendCmdLineGuidsToMemdb (FileName, shortcutTarget, shortcutArgs); //all we try to do now is to see if this lnk or pif file is going to be edited //on NT side. That is if target or icon should change. shortcutNewTarget = GetPathStringOnNt (shortcutTarget); if (!StringIMatch (shortcutNewTarget, shortcutTarget)) { toBeModified = TRUE; // // special case for COMMAND.COM // if (shortcutArgs [0] == 0) { commandPath = JoinPaths (g_System32Dir, S_COMMAND_COM); if (StringIMatch (commandPath, shortcutNewTarget)) { if (msDosMode) { // // remove MS-DOS mode PIF files that point to command.com // RemoveLinkFromSystem (FileName); // // If msdosmode was on, we need to determine how we are going to handle // boot16. We will turn on boot16 mode if: // (a) The .pif points to something besides command.com // (b) The .pif is in a shell folder. // // Note that the check for b simply entails seeing if the PIF file has // OPERATION_FILE_MOVE_SHELL_FOLDER associated with it. // // if (msDosMode && *g_Boot16 == BOOT16_AUTOMATIC) { if (!StringIMatch(GetFileNameFromPath (shortcutNewTarget?shortcutNewTarget:shortcutTarget), S_COMMAND_COM) || IsFileMarkedForOperation (FileName, OPERATION_FILE_MOVE_SHELL_FOLDER)) { *g_Boot16 = BOOT16_YES; } } __leave; } else { ConvertedLnk = TRUE; FreePathString (shortcutNewTarget); shortcutNewTarget = JoinPaths (g_System32Dir, S_CMD_EXE); } } FreePathString (commandPath); shortcutNewArgs = NULL; } else { shortcutNewArgs = DuplicatePathString (shortcutArgs, 0); } } else { FreePathString (shortcutNewTarget); shortcutNewTarget = NULL; } // // If msdosmode was on, we need to determine how we are going to handle // boot16. We will turn on boot16 mode if: // (a) The .pif points to something besides command.com // (b) The .pif is in a shell folder. // // Note that the check for b simply entails seeing if the PIF file has // OPERATION_FILE_MOVE_SHELL_FOLDER associated with it. // // if (msDosMode && *g_Boot16 == BOOT16_AUTOMATIC) { if (!StringIMatch(GetFileNameFromPath (shortcutNewTarget?shortcutNewTarget:shortcutTarget), S_COMMAND_COM) || IsFileMarkedForOperation (FileName, OPERATION_FILE_MOVE_SHELL_FOLDER)) { *g_Boot16 = BOOT16_YES; } } // // If the link points to a directory, see that the directory survives on NT. // Potentially this directory can be cleaned up if it's in a shell folder and // becomes empty after our ObsoleteLinks check // attrib = QuietGetFileAttributes (shortcutTarget); if ((attrib != INVALID_ATTRIBUTES) && (attrib & FILE_ATTRIBUTE_DIRECTORY) ){ MarkDirectoryAsPreserved (shortcutNewTarget?shortcutNewTarget:shortcutTarget); } //OK, so much with target, let's see what's with the work dir shortcutNewWorkDir = GetPathStringOnNt (shortcutWorkDir); if (!StringIMatch (shortcutNewWorkDir, shortcutWorkDir)) { toBeModified = TRUE; } else { FreePathString (shortcutNewWorkDir); shortcutNewWorkDir = NULL; } // // If the working dir for this link is a directory, see that the directory survives on NT. // Potentially this directory can be cleaned up if it's in a shell folder and // becomes empty after our ObsoleteLinks check // attrib = QuietGetFileAttributes (shortcutWorkDir); if ((attrib != INVALID_ATTRIBUTES) && (attrib & FILE_ATTRIBUTE_DIRECTORY) ){ MarkDirectoryAsPreserved (shortcutNewWorkDir?shortcutNewWorkDir:shortcutWorkDir); } //OK, so much with workdir, let's see what's with icon fileStatus = GetFileStatusOnNt (shortcutIconPath); if ((fileStatus & FILESTATUS_DELETED) || ((fileStatus & FILESTATUS_REPLACED) && (fileStatus & FILESTATUS_NTINSTALLED)) || (IsFileMarkedForOperation (shortcutIconPath, OPERATION_FILE_MOVE_SHELL_FOLDER)) ) { // // Our icon will go away, because our file is getting deleted or // replaced. Let's try to preserve it. Extract will fail only if // the icon is known-good. // if (ExtractIconIntoDatFile ( shortcutIconPath, shortcutIcon, &g_IconContext, &newShortcutIcon )) { shortcutNewIconPath = JoinPaths (g_System32Dir, TEXT("migicons.exe")); shortcutIcon = newShortcutIcon; toBeModified = TRUE; } } if (!shortcutNewIconPath) { shortcutNewIconPath = GetPathStringOnNt (shortcutIconPath); if (!StringIMatch (shortcutNewIconPath, shortcutIconPath)) { toBeModified = TRUE; } else { FreePathString (shortcutNewIconPath); shortcutNewIconPath = NULL; } } if (toBeModified) { if (ConvertedLnk) { // // Set this for modifying PIF to LNK // pAddLinkEditToMemDb ( FileName, shortcutNewTarget?shortcutNewTarget:shortcutTarget, shortcutNewArgs?shortcutNewArgs:shortcutArgs, shortcutNewWorkDir?shortcutNewWorkDir:shortcutWorkDir, shortcutNewIconPath?shortcutNewIconPath:shortcutIconPath, shortcutIcon, &ExtraData, FALSE ); } else { pAddLinkEditToMemDb ( FileName, shortcutNewTarget, shortcutNewArgs, shortcutNewWorkDir, shortcutNewIconPath, shortcutIcon, NULL, FALSE ); } } } __finally { if (shortcutNewWorkDir != NULL) { FreePathString (shortcutNewWorkDir); } if (shortcutNewIconPath != NULL) { FreePathString (shortcutNewIconPath); } if (shortcutNewArgs != NULL) { FreePathString (shortcutNewArgs); } if (shortcutNewTarget != NULL) { FreePathString (shortcutNewTarget); } } return TRUE; } PCTSTR pBuildNewCategory ( IN PCTSTR LinkName, IN PCTSTR Category, IN UINT Levels ) { PCTSTR *levPtrs = NULL; PCTSTR wackPtr = NULL; PCTSTR result = NULL; PCTSTR resultTmp = NULL; UINT index = 0; UINT indexLnk = 0; MYASSERT (Levels); levPtrs = (PCTSTR *) PoolMemGetMemory (g_LinksPool, (Levels + 1) * sizeof (PCTSTR)); wackPtr = LinkName; while (wackPtr) { levPtrs[index] = wackPtr; wackPtr = _tcschr (wackPtr, TEXT('\\')); if (wackPtr) { wackPtr = _tcsinc (wackPtr); index ++; if (index > Levels) { index = 0; } } } indexLnk = index; if (index == Levels) { index = 0; } else { index ++; } resultTmp = StringSearchAndReplace (levPtrs [index], levPtrs [indexLnk], Category); if (resultTmp) { result = StringSearchAndReplace (resultTmp, TEXT("\\"), TEXT("->")); } else { result = NULL; } FreePathString (resultTmp); PoolMemReleaseMemory (g_LinksPool, (PVOID) levPtrs); return result; } VOID pGatherInfoFromDefaultPif ( VOID ) { PCTSTR defaultPifPath = NULL; TCHAR tmpStr [20]; TCHAR pifTarget [MEMDB_MAX]; TCHAR pifArgs [MEMDB_MAX]; TCHAR pifWorkDir [MEMDB_MAX]; TCHAR pifIconPath [MEMDB_MAX]; INT pifIcon; BOOL pifMsDosMode; LNK_EXTRA_DATA pifExtraData; defaultPifPath = JoinPaths (g_WinDir, S_COMMAND_PIF); if (ExtractPifInfo ( pifTarget, pifArgs, pifWorkDir, pifIconPath, &pifIcon, &pifMsDosMode, &pifExtraData, defaultPifPath )) { _itoa (pifExtraData.FullScreen, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_FULLSCREEN, tmpStr, NULL, 0, NULL); _itoa (pifExtraData.xSize, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_XSIZE, tmpStr, NULL, 0, NULL); _itoa (pifExtraData.ySize, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_YSIZE, tmpStr, NULL, 0, NULL); _itoa (pifExtraData.QuickEdit, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_QUICKEDIT, tmpStr, NULL, 0, NULL); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_FONTNAME, pifExtraData.FontName, NULL, 0, NULL); _itoa (pifExtraData.xFontSize, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_XFONTSIZE, tmpStr, NULL, 0, NULL); _itoa (pifExtraData.yFontSize, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_YFONTSIZE, tmpStr, NULL, 0, NULL); _itoa (pifExtraData.FontWeight, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_FONTWEIGHT, tmpStr, NULL, 0, NULL); _itoa (pifExtraData.FontFamily, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_FONTFAMILY, tmpStr, NULL, 0, NULL); _itoa (pifExtraData.CurrentCodePage, tmpStr, 10); MemDbSetValueEx (MEMDB_CATEGORY_DEFAULT_PIF, MEMDB_CATEGORY_LINKEDIT_CODEPAGE, tmpStr, NULL, 0, NULL); } FreePathString (defaultPifPath); } BOOL pProcessLinks ( VOID ) { MEMDB_ENUM enumItems; MEMDB_ENUM enumDups; TCHAR pattern[MEMDB_MAX]; IShellLink *shellLink; IPersistFile *persistFile; PLINK_STRUCT linkStruct, linkDup; BOOL resolved; PCTSTR newCategory = NULL; PCTSTR dupCategory = NULL; UINT levels = 0; DWORD count = 0; MYASSERT (g_LinksPool); if (InitCOMLink (&shellLink, &persistFile)) { wsprintf (pattern, TEXT("%s\\*"), MEMDB_CATEGORY_SHORTCUTS); if (MemDbEnumFirstValue ( &enumItems, pattern, MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY )) { do { if (!SafeModeActionCrashed (SAFEMODEID_LNK9X, enumItems.szName)) { SafeModeRegisterAction(SAFEMODEID_LNK9X, enumItems.szName); if (!pProcessShortcut (enumItems.szName, shellLink, persistFile)) { LOG((LOG_ERROR, "Error processing shortcut %s", enumItems.szName)); } count++; if (!(count % 4)) { TickProgressBar (); } SafeModeUnregisterAction(); } } while (MemDbEnumNextValue (&enumItems)); } FreeCOMLink (&shellLink, &persistFile); } if (MemDbEnumFirstValue (&enumItems, MEMDB_CATEGORY_REPORT_LINKS"\\*", MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY)) { do { newCategory = NULL; levels = 0; linkStruct = (PLINK_STRUCT)enumItems.dwValue; if (linkStruct->LinkName) { resolved = !(StringIMatch (linkStruct->LinkNameNoPath, GetFileNameFromPath (linkStruct->LinkName))); } else { resolved = TRUE; } while (!resolved) { resolved = TRUE; MemDbBuildKey ( pattern, MEMDB_CATEGORY_REPORT_LINKS, TEXT("*"), linkStruct->LinkNameNoPath, TEXT("*") ); if (MemDbEnumFirstValue (&enumDups, pattern, MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY)) { do { linkDup = (PLINK_STRUCT)enumDups.dwValue; if ((enumItems.Offset != enumDups.Offset) && (enumItems.UserFlags == enumDups.UserFlags) && (StringIMatch (linkStruct->Category, linkDup->Category)) ) { if (newCategory) { dupCategory = pBuildNewCategory (linkDup->LinkName, linkDup->Category, levels); if (!dupCategory) { MYASSERT (FALSE); continue; } if (!StringIMatch (dupCategory, newCategory)) { FreePathString (dupCategory); continue; } FreePathString (newCategory); } levels++; newCategory = pBuildNewCategory (linkStruct->LinkName, linkStruct->Category, levels); resolved = FALSE; break; } } while (MemDbEnumNextValue (&enumDups)); } } pReportEntry ( linkStruct->ReportEntry, newCategory?newCategory:linkStruct->Category, linkStruct->MigDbContext?linkStruct->MigDbContext->Message:NULL, linkStruct->Context, linkStruct->Object ); if (newCategory) { newCategory = NULL; } } while (MemDbEnumNextValue (&enumItems)); } TickProgressBar (); // gather default command prompt attributes pGatherInfoFromDefaultPif (); DoneLinkAnnounce (); // // Delete MemDb tree used for this phase // MemDbDeleteTree (MEMDB_CATEGORY_REPORT_LINKS); return TRUE; } DWORD ProcessLinks ( IN DWORD Request ) { switch (Request) { case REQUEST_QUERYTICKS: return TICKS_PROCESS_LINKS; case REQUEST_RUN: if (!pProcessLinks ()) { return GetLastError (); } else { return ERROR_SUCCESS; } default: DEBUGMSG ((DBG_ERROR, "Bad parameter in ProcessLinks")); } return 0; } BOOL pProcessCPLs ( VOID ) { CHAR pattern[MEMDB_MAX]; MEMDB_ENUM enumItems; DWORD announcement; PMIGDB_CONTEXT context; MemDbBuildKey (pattern, MEMDB_CATEGORY_CPLS, TEXT("*"), NULL, NULL); if (MemDbEnumFirstValue (&enumItems, pattern, MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY)) { do { if ((IsFileMarkedForAnnounce (enumItems.szName)) && (IsDisplayableCPL (enumItems.szName)) ) { announcement = GetFileAnnouncement (enumItems.szName); context = (PMIGDB_CONTEXT) GetFileAnnouncementContext (enumItems.szName); ReportControlPanelApplet ( enumItems.szName, context, announcement ); } } while (MemDbEnumNextValue (&enumItems)); } return TRUE; } DWORD ProcessCPLs ( IN DWORD Request ) { switch (Request) { case REQUEST_QUERYTICKS: return TICKS_PROCESS_CPLS; case REQUEST_RUN: if (!pProcessCPLs ()) { return GetLastError (); } else { return ERROR_SUCCESS; } default: DEBUGMSG ((DBG_ERROR, "Bad parameter in ProcessCPLs")); } return 0; }