/*++ Copyright (c) 1998 Microsoft Corporation Module Name: winnt32.c Abstract: Stub loader for WinNT Setup program files. Author: Revisions: Ovidiu Temereanca (ovidiut) 09-Dec-1998 --*/ #include #include #include #include #include "winnt32.h" #include "winnt32p.h" #define STRSAFE_NO_DEPRECATE #include #define ARRAYSIZE(x) (sizeof((x)) / sizeof((x)[0])) #if defined(_AMD64_) || defined(_X86_) #if defined(_AMD64) #include "amd64\download.h" #else #include "i386\download.h" #endif #define INTERNAL_WINNT32_DIR TEXT("\\WINNT32") #define MAX_RETRY_INTERVAL_SECONDS 3600L #endif #define MAX_UPGCHK_ELAPSED_SECONDS (30 * 60) #define S_CHKSUM_FILE TEXT("DOSNET.INF") #define MAKEULONGLONG(low,high) ((ULONGLONG)(((DWORD)(low)) | ((ULONGLONG)((DWORD)(high))) << 32)) #define ALLOC_TEXT(chars) ((PTSTR)HeapAlloc (GetProcessHeap (), 0, ((chars) + 1) * sizeof (TCHAR))) #define FREE(p) HeapFree (GetProcessHeap (), 0, p) #define CHARS(string) (sizeof (string) / sizeof ((string)[0]) - 1) #define pFindEOS(String) pFindChar(String, 0) PTSTR FindLastWack ( IN PTSTR String ) /*++ Routine Description: FindLastWack returns a pointer to the last backslash character in the String Arguments: String - Specifies the string Return Value: The position of the last '\\' in the string or NULL if not found. --*/ { PTSTR p; PTSTR LastChar = NULL; for(p = String; *p; p = CharNext(p)) { if(*p == TEXT('\\')) { // the char '\' is never a lead byte LastChar = p; } } return LastChar; } PTSTR DuplicateText ( IN PCTSTR Text ) /*++ Routine Description: DuplicateText allocates memory and then copies a source string into that memory. Caller is responsible for freeing that memory. Arguments: Text - Specifies the source text Return Value: A pointer to the duplicate string; NULL if not enough memory. --*/ { PTSTR Dup; Dup = ALLOC_TEXT(lstrlen (Text)); if (Dup) { lstrcpy (Dup, Text); } return Dup; } PTSTR pFindChar ( IN PTSTR String, IN UINT Char ) /*++ Routine Description: pFindChar returns a pointer to the first occurence of the Char in the String Arguments: String - Specifies the string Char - Specifies the char to look for; can be null Return Value: A pointer to the first occurence of the char in this string or NULL if not found --*/ { while (*String) { if ((UINT)*String == Char) { return String; } String = CharNext (String); } return Char ? NULL : String; } VOID ConcatenatePaths ( IN PTSTR LeadingPath, IN PCTSTR TrailingPath ) /*++ Routine Description: ConcatenatePaths concatenates the two given paths, taking care to insert only one backslash between them. The resulting path is stored in LeadingPath. Arguments: LeadingPath - Specifies the leading path TrailingPath - Specifies the trailing path Return Value: none --*/ { PTSTR p; // // check for "\" at the end of leading dir // p = FindLastWack (LeadingPath); if (!p) { p = pFindEOS (LeadingPath); #pragma prefast(suppress:11, p is never NULL because 0 is ALWAYS found as the string terminator) *p++ = TEXT('\\'); } else { if (*(p + 1) == 0) { p++; } else { p = pFindEOS (p); *p++ = TEXT('\\'); } } // // check for "\" at the beginning of trailing dir // if (*TrailingPath == TEXT('\\')) { TrailingPath++; } lstrcpy (p, TrailingPath); } BOOL GetFileVersion ( IN PCTSTR FilePath, OUT PDWORD FileVersionMS, OPTIONAL OUT PDWORD FileVersionLS OPTIONAL ) { DWORD dwLength, dwTemp; LPVOID lpData; VS_FIXEDFILEINFO *VsInfo; UINT DataLength; BOOL b = FALSE; if (GetFileAttributes (FilePath) != (DWORD)-1) { if (dwLength = GetFileVersionInfoSize ((PTSTR)FilePath, &dwTemp)) { if (lpData = LocalAlloc (LPTR, dwLength)) { if (GetFileVersionInfo ((PTSTR)FilePath, 0, dwLength, lpData)) { if (VerQueryValue (lpData, TEXT("\\"), &VsInfo, &DataLength)) { if (FileVersionMS) { *FileVersionMS = VsInfo->dwFileVersionMS; } if (FileVersionLS) { *FileVersionLS = VsInfo->dwFileVersionLS; } b = TRUE; } } LocalFree (lpData); } } } return b; } #if defined(_AMD64_) || defined(_X86_) BOOL pReRun ( IN PCTSTR StartDir, IN PCTSTR WackExeName, IN PCTSTR CmdLineArguments, IN PCTSTR DefSourcesDir OPTIONAL ) /*++ Routine Description: pReRun tries to launch a instance of this exe from a local drive, specifing an additional command line parameter (/S:). Arguments: StartDir - Specifies the starting directory from where the instance will be launched WackExeName - Specifies the file name only of the EXE to launch, preceded by a backslash CmdLineArguments - Specifies the command line arguments initially supplied DefSourcesDir - Specifies the default directory containing instalation files Return Value: TRUE if the launch was successful --*/ { PTSTR CmdLine; INT Chars; STARTUPINFO StartupInfo; PROCESS_INFORMATION pi; BOOL b = FALSE; DWORD rc; Chars = lstrlen (StartDir) + lstrlen (WackExeName) + CHARS(" ") + lstrlen (CmdLineArguments); if (DefSourcesDir) { Chars += CHARS(" /S:") + lstrlen (DefSourcesDir); } CmdLine = ALLOC_TEXT(Chars); if (!CmdLine) { SetLastError (ERROR_NOT_ENOUGH_MEMORY); return FALSE; } lstrcpy (CmdLine, StartDir); lstrcat (CmdLine, WackExeName); lstrcat (CmdLine, TEXT(" ")); lstrcat (CmdLine, CmdLineArguments); if (DefSourcesDir) { lstrcat (CmdLine, TEXT(" /S:")); lstrcat (CmdLine, DefSourcesDir); } ZeroMemory(&StartupInfo, sizeof(StartupInfo)); StartupInfo.cb = sizeof(StartupInfo); b = CreateProcess ( NULL, CmdLine, NULL, NULL, FALSE, NORMAL_PRIORITY_CLASS, NULL, StartDir, &StartupInfo, &pi ); rc = GetLastError (); FREE (CmdLine); SetLastError (rc); return b; } BOOL pCleanup ( VOID ) /*++ Routine Description: pCleanup deletes all locally installed files and marks current running instance for deletion the next time system will reboot. Arguments: none Return Value: TRUE if the operation completed successfully; the machine will need to reboot before actual complete delete will take place. --*/ { TCHAR RunningInstancePath[MAX_PATH]; TCHAR Buffer[MAX_PATH]; BOOL b; DWORD StartingTime; PCTSTR p; if (!GetModuleFileName (NULL, RunningInstancePath, MAX_PATH)) { return FALSE; } // // wait until WINNT32\WINNT32.EXE file can be deleted // or the retry interval of time elapses // if (!GetWindowsDirectory (Buffer, MAX_PATH)) { return FALSE; } if (FAILED(StringCbCat(Buffer, sizeof(Buffer), INTERNAL_WINNT32_DIR))) { return FALSE; } p = FindLastWack ((PTSTR)RunningInstancePath); if (!p) { return FALSE; } if (FAILED(StringCbCat (Buffer, sizeof(Buffer), p))) { return FALSE; } StartingTime = GetTickCount (); while (GetFileAttributes (Buffer) != (DWORD)-1) { // // try to delete it // if (DeleteNode (Buffer)) { break; } // // give up if time elapses // if (GetTickCount () - StartingTime > 1000L * MAX_RETRY_INTERVAL_SECONDS) { break; } // // nothing useful to do; let the other processes run // Sleep (0); } // // wait until WINNT32\SETUPLOG.EXE file can be deleted // or the retry interval of time elapses // if (!GetWindowsDirectory (Buffer, MAX_PATH)) { return FALSE; } StringCbCat (Buffer, sizeof(Buffer), INTERNAL_WINNT32_DIR); if (FAILED(StringCbCat (Buffer, sizeof(Buffer), TEXT("\\SETUPLOG.EXE")))) { return FALSE; } StartingTime = GetTickCount (); while (GetFileAttributes (Buffer) != (DWORD)-1) { if (DeleteNode (Buffer)) { break; } if (GetTickCount () - StartingTime > 1000L * MAX_RETRY_INTERVAL_SECONDS) { break; } Sleep (0); } if (!GetWindowsDirectory (Buffer, MAX_PATH)) { return FALSE; } if (FAILED(StringCbCat (Buffer, sizeof(Buffer), INTERNAL_WINNT32_DIR))) { return FALSE; } b = DeleteNode (Buffer); if (!GetWindowsDirectory (Buffer, MAX_PATH)) { return FALSE; } if (FAILED(StringCbCat (Buffer, sizeof(Buffer), TEXT("\\WININIT.INI")))) { return FALSE; } return WritePrivateProfileString (TEXT("rename"), TEXT("NUL"), RunningInstancePath, Buffer) && b; } BOOL pShouldDownloadToLocalDisk ( IN PTSTR Path ) /*++ Routine Description: pShouldDownloadToLocalDisk returns TRUE if winnt32 files should be downloaded to a local disk first (like in the case of sources on a remote disk or on a CD) Arguments: Path - Specifies the path Return Value: TRUE if the specified path is on an untrusted media --*/ { TCHAR ch; BOOL Remote = TRUE; UINT type; if (Path[1] == TEXT(':') && Path[2] == TEXT('\\')) { ch = Path[3]; Path[3] = 0; type = GetDriveType (Path); Remote = (type == DRIVE_REMOTE) || (type == DRIVE_CDROM); Path[3] = ch; } return Remote; } VOID pCenterWindowOnDesktop ( HWND WndToCenter ) /*++ Routine Description: Centers a dialog relative to the 'work area' of the desktop. Arguments: WndToCenter - window handle of dialog to center Return Value: None. --*/ { RECT rcFrame, rcWindow; LONG x, y, w, h; POINT point; HWND Desktop = GetDesktopWindow (); point.x = point.y = 0; ClientToScreen(Desktop, &point); GetWindowRect(WndToCenter, &rcWindow); GetClientRect(Desktop, &rcFrame); w = rcWindow.right - rcWindow.left + 1; h = rcWindow.bottom - rcWindow.top + 1; x = point.x + ((rcFrame.right - rcFrame.left + 1 - w) / 2); y = point.y + ((rcFrame.bottom - rcFrame.top + 1 - h) / 2); // // Get the work area for the current desktop (i.e., the area that // the tray doesn't occupy). // if(!SystemParametersInfo (SPI_GETWORKAREA, 0, (PVOID)&rcFrame, 0)) { // // For some reason SPI failed, so use the full screen. // rcFrame.top = rcFrame.left = 0; rcFrame.right = GetSystemMetrics(SM_CXSCREEN); rcFrame.bottom = GetSystemMetrics(SM_CYSCREEN); } if(x + w > rcFrame.right) { x = rcFrame.right - w; } else if(x < rcFrame.left) { x = rcFrame.left; } if(y + h > rcFrame.bottom) { y = rcFrame.bottom - h; } else if(y < rcFrame.top) { y = rcFrame.top; } MoveWindow(WndToCenter, x, y, w, h, FALSE); } BOOL CALLBACK DlgProc ( HWND Dlg, UINT Msg, WPARAM wParam, LPARAM lParam ) /*++ Routine Description: This is the callback procedure for the dialog displayed while components are copied from the network Arguments: Dlg - Specifies the dialog window handle Msg - Specifies the message wParam - Specifies the first param lParam - Specifies the second param Return Value: Depends on the specific message. --*/ { static HANDLE Bitmap = NULL; static HCURSOR Cursor = NULL; RECT rect; HWND Text; BITMAP bm; INT i; switch (Msg) { case WM_INITDIALOG: Cursor = SetCursor (LoadCursor (NULL, IDC_WAIT)); ShowCursor (TRUE); Bitmap = LoadBitmap (GetModuleHandle (NULL), MAKEINTRESOURCE(IDB_INIT_WIN2000)); if (Bitmap) { if (GetObject (Bitmap, sizeof (bm), &bm)) { GetClientRect (Dlg, &rect); rect.right = bm.bmWidth; AdjustWindowRect (&rect, GetWindowLong (Dlg, GWL_STYLE), FALSE); SetWindowPos ( Dlg, NULL, 0, 0, rect.right - rect.left, rect.bottom - rect.top, SWP_NOMOVE | SWP_NOZORDER); } SendDlgItemMessage(Dlg, IDC_BITMAP, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)Bitmap); } GetClientRect (Dlg, &rect); i = rect.right - rect.left; Text = GetDlgItem (Dlg, IDC_TEXT); if (GetWindowRect (Text, &rect)) { i = (i - (rect.right - rect.left)) / 2; ScreenToClient (Dlg, (LPPOINT)&rect); SetWindowPos ( Text, NULL, i, rect.top, 0, 0, SWP_NOSIZE | SWP_NOZORDER); } pCenterWindowOnDesktop (Dlg); return TRUE; case WM_DESTROY: ShowCursor (FALSE); if (Cursor) { SetCursor (Cursor); Cursor = NULL; } if (Bitmap) { DeleteObject (Bitmap); Bitmap = NULL; } } return FALSE; } #endif INT pStringICompareCharCount ( IN PCTSTR String1, IN PCTSTR String2, IN DWORD CharCount ) /*++ Routine Description: This routine behaves like _tcsnicmp. Arguments: String1 - Specifies the first string String2 - Specifies the second string CharCount - Specifies the number of chars to compare at most Return Value: 0 if strings are equal; -1 if first string is lesser; 1 if first is greater --*/ { TCHAR ch1, ch2; if (!CharCount) { return 0; } while (*String1) { ch1 = (TCHAR)CharUpper ((LPTSTR)*String1); ch2 = (TCHAR)CharUpper ((LPTSTR)*String2); if (ch1 - ch2) { return ch1 - ch2; } CharCount--; if (!CharCount) { return 0; } String1 = CharNext (String1); String2 = CharNext (String2); } return -(*String2); } VOID pParseCmdLine ( IN PTSTR CmdStart, OUT PTSTR* ArgValues, OUT PTSTR pStr, OUT INT *NumArgs, OUT INT *NumBytes ) /* Routine Description: pParseCmdLine parses the command line and sets up the ArgValues array. On entry, CmdStart should point to the command line, ArgValues should point to memory for the ArgValues array, pStr points to memory to place the text of the arguments. If these are NULL, then no storing (only counting) is done. On exit, *NumArgs has the number of arguments (plus one for a final NULL argument), and *NumBytes has the number of bytes used in the buffer pointed to by args. Arguments: CmdStart - Specifies the command line having the form: ArgValues - Receives the arguments array; NULL means don't build array pStr - Receives the argument text; NULL means don't store text NumArgs - Receives the number of ArgValues entries created NumBytes - Receives the number of bytes used in buffer Return Value: none */ { PTSTR p; TCHAR c; INT inquote; /* 1 = inside quotes */ INT copychar; /* 1 = copy char to *args */ WORD numslash; /* num of backslashes seen */ *NumBytes = 0; *NumArgs = 1; /* the program name at least */ /* first scan the program name, copy it, and count the bytes */ p = CmdStart; if (ArgValues) *ArgValues++ = pStr; /* A quoted program name is handled here. The handling is much simpler than for other arguments. Basically, whatever lies between the leading double-quote and next one, or a terminal null character is simply accepted. Fancier handling is not required because the program name must be a legal NTFS/HPFS file name. Note that the double-quote characters are not copied, nor do they contribute to NumBytes. */ if (*p == TEXT('\"')) { /* scan from just past the first double-quote through the next double-quote, or up to a null, whichever comes first */ while ((*(++p) != TEXT('\"')) && (*p != TEXT('\0'))) { *NumBytes += sizeof(TCHAR); if (pStr) *pStr++ = *p; } /* append the terminating null */ *NumBytes += sizeof(TCHAR); if (pStr) *pStr++ = TEXT('\0'); /* if we stopped on a double-quote (usual case), skip over it */ if (*p == TEXT('\"')) p++; } else { /* Not a quoted program name */ do { *NumBytes += sizeof(TCHAR); if (pStr) *pStr++ = *p; c = *p++; } while (c > TEXT(' ')); if (c == TEXT('\0')) { p--; } else { if (pStr) *(pStr - 1) = TEXT('\0'); } } inquote = 0; /* loop on each argument */ for ( ; ; ) { if (*p) { while (*p == TEXT(' ') || *p == TEXT('\t')) ++p; } if (*p == TEXT('\0')) break; /* end of args */ /* scan an argument */ if (ArgValues) *ArgValues++ = pStr; /* store ptr to arg */ ++*NumArgs; /* loop through scanning one argument */ for ( ; ; ) { copychar = 1; /* Rules: 2N backslashes + " ==> N backslashes and begin/end quote 2N+1 backslashes + " ==> N backslashes + literal " N backslashes ==> N backslashes */ numslash = 0; while (*p == TEXT('\\')) { /* count number of backslashes for use below */ ++p; ++numslash; } if (*p == TEXT('\"')) { /* if 2N backslashes before, start/end quote, otherwise copy literally */ if (numslash % 2 == 0) { if (inquote) if (p[1] == TEXT('\"')) p++; /* Double quote inside quoted string */ else /* skip first quote char and copy second */ copychar = 0; else copychar = 0; /* don't copy quote */ inquote = !inquote; } numslash /= 2; /* divide numslash by two */ } /* copy slashes */ while (numslash--) { if (pStr) *pStr++ = TEXT('\\'); *NumBytes += sizeof(TCHAR); } /* if at end of arg, break loop */ if (*p == TEXT('\0') || (!inquote && (*p == TEXT(' ') || *p == TEXT('\t')))) break; /* copy character into argument */ if (copychar) { if (pStr) *pStr++ = *p; *NumBytes += sizeof(TCHAR); } ++p; } /* null-terminate the argument */ if (pStr) *pStr++ = TEXT('\0'); /* terminate string */ *NumBytes += sizeof(TCHAR); } } PTSTR* pCommandLineToArgv ( OUT INT* NumArgs ) /*++ Routine Description: pCommandLineToArgv tokens the command line in an array of arguments created on the heap. The number of entries in this array of args is stored in *NumArgs. The caller is responsible for freeing this array. Arguments: NumArgs - Receives the number of arguments in the array that is returned Return Value: An array of pointer to individual arguments specified on the command line --*/ { PTSTR CommandLine; TCHAR ModuleName[MAX_PATH]; PTSTR Start; INT Size; PTSTR* Args; CommandLine = GetCommandLine(); GetModuleFileName (NULL, ModuleName, MAX_PATH); // // If there's no command line at all (won't happen from cmd.exe, but // possibly another program), then we use pgmname as the command line // to parse, so that ArgValues[0] is initialized to the program name // Start = *CommandLine ? CommandLine : ModuleName; // // Find out how much space is needed to store args, // allocate space for ArgValues[] vector and strings, // and store args and ArgValues ptrs in block we allocate // pParseCmdLine (Start, NULL, NULL, NumArgs, &Size); Args = (PTSTR*) LocalAlloc ( LMEM_FIXED | LMEM_ZEROINIT, ((*NumArgs + 1) * sizeof(PTSTR)) + Size ); if (!Args) { return NULL; } pParseCmdLine (Start, Args, (PTSTR)(Args + *NumArgs), NumArgs, &Size); return Args; } VOID GetCmdLineArgs ( IN PCTSTR CommandLine, OUT BOOL* Cleanup, OUT BOOL* NoDownload, OUT PCTSTR* UnattendPrefix, OUT PCTSTR* UnattendFileName, OUT BOOL* DisableDynamicUpdates, OUT PCTSTR* DynamicUpdatesShare, OUT PCTSTR* RestartAnswerFile, OUT BOOL* LocalWinnt32, OUT BOOL* CheckUpgradeOnly, OUT PTSTR RemainingArgs ) /*++ Routine Description: GetCmdLineArgs retrieves download-specific commands from the specified command line and stores them in supplied buffers. Arguments: CommandLine - Specifies the command line to interpret Cleanup - Receives a bool indicating if a cleanup option was specified NoDownload - Receives a bool indicating if a no-download option was specified UnattendPrefix - Receives a pointer to the unattend command-line option, as specified by the user (including the terminating column) or NULL if not specified; caller is responsible for freeing the memory UnattendFileName - Receives a pointer to the unattended file name or NULL if not specified; caller is responsible for freeing the memory DisableDynamicUpdates - Receives a bool set if DU is to be disabled DynamicUpdatesShare - Receives a pointer to the dynamic updates share; caller is responsible for freeing the memory RestartAnswerFile - Receives a pointer to the /Restart: answer file LocalWinnt32 - Receives a bool indicating if a winnt32 runs from a local disk (after an automatic download) CheckUpgradeOnly - Receives a bool indicating if winnt32 runs in CheckUpgradeOnly mode RemainingArgs - Receives all remaining arguments not related to the download operation Return Value: none --*/ { INT ArgCount; PTSTR *ArgValues, *CrtArg; PTSTR CurrentArg, p; BOOL PassOn; *Cleanup = FALSE; *NoDownload = FALSE; *UnattendPrefix = NULL; *UnattendFileName = NULL; *DisableDynamicUpdates = FALSE; *DynamicUpdatesShare = NULL; *RemainingArgs = 0; *LocalWinnt32 = FALSE; *CheckUpgradeOnly = FALSE; *RestartAnswerFile = NULL; CrtArg = ArgValues = pCommandLineToArgv (&ArgCount); // // Skip program name. We should always get back ArgCount as at least 1, // but be robust anyway. // if (ArgCount) { ArgCount--; CrtArg++; } while (ArgCount--) { CurrentArg = *CrtArg++; PassOn = TRUE; if ((*CurrentArg == TEXT('/')) || (*CurrentArg == TEXT('-'))) { if (lstrcmpi (CurrentArg + 1, TEXT("LOCAL")) == 0) { *LocalWinnt32 = TRUE; PassOn = FALSE; } else if (lstrcmpi (CurrentArg + 1, TEXT("CLEANUP")) == 0) { *Cleanup = TRUE; PassOn = FALSE; } else if (lstrcmpi (CurrentArg + 1, TEXT("NODOWNLOAD")) == 0) { *NoDownload = TRUE; PassOn = FALSE; } else if (lstrcmpi (CurrentArg + 1, TEXT("CHECKUPGRADEONLY")) == 0) { *CheckUpgradeOnly = TRUE; } else if (pStringICompareCharCount (CurrentArg + 1, TEXT("UNATTEND"), 8) == 0) { p = pFindChar (CurrentArg + 1 + 8, TEXT(':')); if (p && *(p + 1)) { p++; *UnattendFileName = DuplicateText (p); *p = 0; *UnattendPrefix = DuplicateText (CurrentArg); PassOn = FALSE; } } else if (pStringICompareCharCount (CurrentArg + 1, TEXT("UNATTENDED"), 10) == 0) { p = pFindChar (CurrentArg + 1 + 10, TEXT(':')); if (p && *(p + 1)) { p++; *UnattendFileName = DuplicateText (p); *p = 0; *UnattendPrefix = DuplicateText (CurrentArg); PassOn = FALSE; } } else if (lstrcmpi (CurrentArg + 1, WINNT_U_DYNAMICUPDATESDISABLE) == 0) { *DisableDynamicUpdates = TRUE; } else if (pStringICompareCharCount (CurrentArg + 1, WINNT_U_DYNAMICUPDATESHARE, sizeof (WINNT_U_DYNAMICUPDATESHARE_A) - 1) == 0 && CurrentArg[sizeof (WINNT_U_DYNAMICUPDATESHARE_A)] == TEXT(':')) { *DynamicUpdatesShare = DuplicateText (CurrentArg + 1 + sizeof (WINNT_U_DYNAMICUPDATESHARE_A)); } else if (pStringICompareCharCount (CurrentArg + 1, TEXT("RESTART:"), 8) == 0) { *RestartAnswerFile = DuplicateText (CurrentArg + 1 + 8); } } if (PassOn) { if (*RemainingArgs) { lstrcat(RemainingArgs, TEXT(" ")); } lstrcat(RemainingArgs, CurrentArg); } } LocalFree ((HLOCAL) ArgValues); } BOOL DoesDirExist ( IN PCTSTR Path ) { WIN32_FIND_DATA fd; TCHAR test[MAX_PATH]; HANDLE h; BOOL b = FALSE; HRESULT hr; if (Path) { if (FAILED(StringCbCopy(test, sizeof(test), Path))) { return FALSE; } if (FAILED(StringCbCat(test, sizeof(test), TEXT("\\*")))) { return FALSE; } h = FindFirstFile (test, &fd); if (h != INVALID_HANDLE_VALUE) { FindClose (h); b = TRUE; } } return b; } ULONGLONG SystemTimeToFileTime64 ( IN PSYSTEMTIME SystemTime ) { FILETIME ft; ULARGE_INTEGER result; SystemTimeToFileTime (SystemTime, &ft); result.LowPart = ft.dwLowDateTime; result.HighPart = ft.dwHighDateTime; return result.QuadPart; } BOOL pComputeChecksum ( IN PCTSTR FileName, OUT PDWORD Chksum ) { DWORD chksum, size, dwords, bytes; HANDLE hFile, hMap = NULL; PVOID viewBase = NULL; PDWORD base, limit; PBYTE base2; DWORD rc; BOOL b = FALSE; hFile = CreateFile( FileName, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL ); if(hFile == INVALID_HANDLE_VALUE) { return FALSE; } __try { size = GetFileSize (hFile, NULL); if (size == (DWORD)-1) { __leave; } hMap = CreateFileMapping ( hFile, NULL, PAGE_READONLY, 0, size, NULL ); if (!hMap) { __leave; } viewBase = MapViewOfFile (hMap, FILE_MAP_READ, 0, 0, size); if (!viewBase) { __leave; } dwords = size / sizeof (DWORD); base = (PDWORD)viewBase; limit = base + dwords; chksum = 0; while (base < limit) { chksum += *base; base++; } bytes = size % sizeof (DWORD); base2 = (PBYTE)base; while (bytes) { chksum += *base2; base2++; bytes--; } b = TRUE; } __finally { if (!b) { rc = GetLastError (); } if (viewBase) { UnmapViewOfFile (viewBase); } if (hMap) { CloseHandle (hMap); } CloseHandle (hFile); if (!b) { SetLastError (rc); } } if (b) { *Chksum = chksum; } return b; } BOOL pGetFiletimeStamps ( IN PCTSTR FileName, OUT PFILETIME CreationTime, OUT PFILETIME LastWriteTime ) { WIN32_FIND_DATA fd; HANDLE h; h = FindFirstFile (FileName, &fd); if (h == INVALID_HANDLE_VALUE) { return FALSE; } FindClose (h); *CreationTime = fd.ftCreationTime; *LastWriteTime = fd.ftLastWriteTime; return TRUE; } PTSTR pGetRecentDUShare ( IN DWORD MaxElapsedSeconds ) { SYSTEMTIME lastDownload, currentTime; ULONGLONG lastDownloadIn100Ns, currentTimeIn100Ns; ULONGLONG difference; DWORD rc, size, type; HKEY key = NULL; BOOL b = FALSE; PTSTR duShare = NULL; TCHAR filePath[MAX_PATH]; PTSTR p; FILETIME ftCreationTime; FILETIME ftLastWriteTime; ULONGLONG data[2], storedData[2]; DWORD chksum, storedChksum; if (!GetModuleFileName (NULL, filePath, MAX_PATH)) { return NULL; } p = FindLastWack (filePath); if (!p) { return NULL; } p++; //now, p points after the wack. //Note that p cannot be greater than (filePath+MAX_PATH), since //we used GetModuleFileName() with MAX_PATH to put a string in the filePath buffer, //and FindLastWack() cannot go beyond the end of the string buffer. if (FAILED(StringCchCopy(p, (filePath+ARRAYSIZE(filePath)) - p, S_CHKSUM_FILE))) { return NULL; } rc = RegOpenKeyEx ( HKEY_LOCAL_MACHINE, TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Setup\\Winnt32\\5.1\\DUShare"), 0, KEY_READ, &key ); if (rc == ERROR_SUCCESS) { size = sizeof (lastDownload); rc = RegQueryValueEx ( key, TEXT("LastDownloadTime"), NULL, &type, (PBYTE) (&lastDownload), &size ); } if (rc == ERROR_SUCCESS && type == REG_BINARY && size == sizeof (lastDownload)) { // // Compare current time to report time // GetSystemTime (¤tTime); lastDownloadIn100Ns = SystemTimeToFileTime64 (&lastDownload); currentTimeIn100Ns = SystemTimeToFileTime64 (¤tTime); if (currentTimeIn100Ns > lastDownloadIn100Ns) { // // Compute difference in seconds // difference = currentTimeIn100Ns - lastDownloadIn100Ns; difference /= (10 * 1000 * 1000); if (difference < MaxElapsedSeconds) { b = TRUE; } } } if (b) { rc = RegQueryValueEx ( key, TEXT(""), NULL, &type, NULL, &size ); if (rc == ERROR_SUCCESS && type == REG_SZ && size > 0) { duShare = ALLOC_TEXT (size / sizeof (TCHAR)); if (duShare) { rc = RegQueryValueEx ( key, TEXT(""), NULL, NULL, (LPBYTE)duShare, &size ); if (rc != ERROR_SUCCESS || !DoesDirExist (duShare)) { FREE (duShare); duShare = NULL; } } } } if (duShare) { b = FALSE; if (pGetFiletimeStamps (filePath, &ftCreationTime, &ftLastWriteTime)) { rc = RegQueryValueEx ( key, TEXT("TimeStamp"), 0, &type, (LPBYTE)storedData, &size ); if (rc == ERROR_SUCCESS && type == REG_BINARY) { data[0] = ((ULONGLONG)ftCreationTime.dwHighDateTime << 32) | (ULONGLONG)ftCreationTime.dwLowDateTime; data[1] = ((ULONGLONG)ftLastWriteTime.dwHighDateTime << 32 ) | (ULONGLONG)ftLastWriteTime.dwLowDateTime; if (data[0] == storedData[0] && data[1] == storedData[1]) { b = TRUE; } } } if (b) { b = FALSE; if (pComputeChecksum (filePath, &chksum)) { rc = RegQueryValueEx ( key, TEXT("Checksum"), NULL, &type, (LPBYTE)&storedChksum, &size ); if (rc == ERROR_SUCCESS && type == REG_DWORD && storedChksum == chksum) { b = TRUE; } } } if (!b) { FREE (duShare); duShare = NULL; } } if (key) { RegCloseKey (key); } return duShare; } void _stdcall ModuleEntry( VOID ) /*++ Routine Description: ModuleEntry is the stub program that loads Windows 2000 Setup DLLs. Arguments: none Return Value: none. ExitProcess will set the process' exit code. --*/ { TCHAR RunningInstancePath[MAX_PATH]; TCHAR Temp[MAX_PATH]; TCHAR Text1[MAX_PATH+sizeof("msvcrt.dll")]; TCHAR Text2[MAX_PATH+MAX_PATH]; TCHAR Text3[MAX_PATH]; TCHAR *WackExeName, *p; TCHAR winnt32DllPath[MAX_PATH+MAX_PATH]; HMODULE WinNT32; BOOL Downloaded; DWORD d; BOOL b; HWND Dlg = NULL; HANDLE WinNT32Stub = NULL; PWINNT32 winnt32; HKEY key; DWORD type; PCTSTR moduleName; PSTR restartCmdLine = NULL; PTSTR RemainingArgs, NewCmdLine, UnattendPrefix, UnattendFileName; PTSTR DynamicUpdatesShare; BOOL Cleanup, NoDownload, DisableDynamicUpdates, LocalWinnt32, CheckUpgradeOnly; PTSTR RestartAnswerFile; UINT CmdLineLen; PTSTR FileName; PCTSTR ExtraFiles[2]; TCHAR cdFilePath[MAX_PATH+sizeof("win9xupg\\msvcrt.dll")]; PTSTR duShare = NULL; UINT tcharsNeeded; HRESULT hr; #if defined(_X86_) TCHAR DownloadDest[MAX_PATH] = TEXT(""); TCHAR DefSourcesDir[MAX_PATH]; BOOL IsWin9x; // // Check OS version. Disallow Win32s and NT < 4.00 // d = GetVersion(); if((d & 0xff) < 4) { if (LoadString (GetModuleHandle (NULL), IDS_VERERROR, Text1, sizeof(Text1)/sizeof(Text1[0])) && LoadString (GetModuleHandle (NULL), IDS_APPNAME, Text2, sizeof(Text2)/sizeof(Text2[0]))) { MessageBox (NULL, Text1, Text2, MB_ICONERROR | MB_OK | MB_SYSTEMMODAL); } ExitProcess (ERROR_OLD_WIN_VERSION); } IsWin9x = (d & 0x80000000) != 0; #else #define IsWin9x ((BOOL)FALSE) #endif // // get this instance's path // if (!GetModuleFileName(NULL, RunningInstancePath, MAX_PATH)) { ExitProcess (GetLastError ()); } WackExeName = FindLastWack (RunningInstancePath); if (!WackExeName) { // shut up PREfix. This case should never happen. ExitProcess (ERROR_BAD_PATHNAME); } // // Ansi version on Win95. Unicode on NT. // moduleName = IsWin9x ? TEXT("WINNT32A.DLL") : TEXT("WINNT32U.DLL"); winnt32DllPath[0] = 0; // // get command line options // allocate a bigger buffer for safety // RemainingArgs = ALLOC_TEXT(lstrlen(GetCommandLine()) * 2); if (!RemainingArgs) { ExitProcess (GetLastError ()); } GetCmdLineArgs ( GetCommandLine (), &Cleanup, &NoDownload, &UnattendPrefix, &UnattendFileName, &DisableDynamicUpdates, &DynamicUpdatesShare, &RestartAnswerFile, &LocalWinnt32, &CheckUpgradeOnly, RemainingArgs ); #if defined(_AMD64_) || defined(_X86_) if (Cleanup) { pCleanup (); ExitProcess (0); } #if defined(_X86_) if (IsWin9x) { WinNT32Stub = CreateEvent (NULL, FALSE, FALSE, TEXT("_WinNT32_Stub_")); if (!WinNT32Stub) { ExitProcess (GetLastError ()); } b = (GetLastError() == ERROR_SUCCESS); if (!NoDownload && !DynamicUpdatesShare && pShouldDownloadToLocalDisk (RunningInstancePath)) { Dlg = CreateDialog (GetModuleHandle(NULL), MAKEINTRESOURCE(IDD_SETUPINIT), NULL, DlgProc); GetWindowsDirectory (DownloadDest, MAX_PATH); StringCbCat (DownloadDest, sizeof(DownloadDest), INTERNAL_WINNT32_DIR); *WackExeName = 0; if (UnattendFileName && GetFullPathName (UnattendFileName, MAX_PATH, Temp, &FileName) && lstrcmpi (UnattendFileName, Temp) ) { ExtraFiles[0] = Temp; ExtraFiles[1] = NULL; } else { ExtraFiles[0] = NULL; FileName = UnattendFileName; } Downloaded = DownloadProgramFiles ( RunningInstancePath, DownloadDest, ExtraFiles ); *WackExeName = TEXT('\\'); if (Downloaded) { // // get default sources dir // //this lstrcpy is ok, since both buffers are of size MAX_PATH lstrcpy (DefSourcesDir, RunningInstancePath); *FindLastWack (DefSourcesDir) = 0; p = FindLastWack (DefSourcesDir); if (p && lstrcmpi(p, INTERNAL_WINNT32_DIR) == 0) { *p = 0; } if (FileName) { CmdLineLen = lstrlen (RemainingArgs); if (CmdLineLen > 0) { // // count the space between args // CmdLineLen += CHARS(" "); } CmdLineLen += lstrlen (UnattendPrefix); CmdLineLen += lstrlen (FileName); NewCmdLine = ALLOC_TEXT(CmdLineLen); if (NewCmdLine) { if (*RemainingArgs) { lstrcpy (NewCmdLine, RemainingArgs); lstrcat (NewCmdLine, TEXT(" ")); } else { *NewCmdLine = 0; } lstrcat (NewCmdLine, UnattendPrefix); lstrcat (NewCmdLine, FileName); FREE (RemainingArgs); RemainingArgs = NewCmdLine; NewCmdLine = NULL; } } // // append /LOCAL to the new processes command line // to let it know it's running from a local share // NewCmdLine = ALLOC_TEXT(lstrlen (RemainingArgs) + sizeof(TEXT(" /LOCAL"))/sizeof(TCHAR) + 1); if (NewCmdLine) { //we have pre-calculated the size of the NewCmdLine buffer wsprintf (NewCmdLine, TEXT("%s /%s"), RemainingArgs, TEXT("LOCAL")); if (pReRun (DownloadDest, WackExeName, NewCmdLine, DefSourcesDir)) { // // the new process will do it; this one will just die // but after the signal that the Setup Wizard is on // anyway, if something goes very wrong, // don't wait more than 10 sec. // this should be enough for the wizard to appear // (or any error message box) on any machine that installs W2K // WaitForSingleObject (WinNT32Stub, 10000); CloseHandle (WinNT32Stub); if (Dlg) { DestroyWindow (Dlg); } d = 0; } else { d = GetLastError (); } } else { d = ERROR_NOT_ENOUGH_MEMORY; } ExitProcess (d); } } if (!Dlg && WinNT32Stub) { CloseHandle (WinNT32Stub); WinNT32Stub = NULL; } } #endif // defined(_X86_) #endif // defined(_AMD64_) || defined(_X86_) if (RemainingArgs) { FREE(RemainingArgs); RemainingArgs = NULL; } if (UnattendPrefix) { FREE(UnattendPrefix); UnattendPrefix = NULL; } if (!DisableDynamicUpdates && !DynamicUpdatesShare) { PCTSTR af = NULL; if (RestartAnswerFile) { af = RestartAnswerFile; } else if (UnattendFileName) { if (GetFullPathName (UnattendFileName, MAX_PATH, Temp, &FileName)) { af = Temp; } } // // get the path from this answer file // if (af) { GetPrivateProfileString ( WINNT_UNATTENDED, WINNT_U_DYNAMICUPDATESDISABLE, TEXT(""), Text2, MAX_PATH, af ); DisableDynamicUpdates = !lstrcmpi (Text2, WINNT_A_YES); if (!DisableDynamicUpdates) { if (GetPrivateProfileString ( WINNT_UNATTENDED, WINNT_U_DYNAMICUPDATESHARE, TEXT(""), Text2, MAX_PATH, af )) { DynamicUpdatesShare = DuplicateText (Text2); } } } } if (UnattendFileName) { FREE(UnattendFileName); UnattendFileName = NULL; } b = FALSE; if (!CheckUpgradeOnly && !DisableDynamicUpdates && !DynamicUpdatesShare) { DynamicUpdatesShare = pGetRecentDUShare (MAX_UPGCHK_ELAPSED_SECONDS); if (DynamicUpdatesShare) { b = TRUE; } } d = ERROR_SUCCESS; if (!DisableDynamicUpdates && DynamicUpdatesShare) { DWORD regFileVersionMS, regFileVersionLS; DWORD cdFileVersionMS, cdFileVersionLS; // // check if there is a replacement module newer than the CD version // if (GetFileAttributes (DynamicUpdatesShare) == (DWORD)-1) { if (!b) { d = GetLastError (); if (LoadString (GetModuleHandle (NULL), IDS_APPNAME, Text3, sizeof(Text3)/sizeof(Text3[0])) && LoadString (GetModuleHandle (NULL), IDS_PATHERROR, Text1, sizeof(Text1)/sizeof(Text1[0]))) { //Note: Text2 is 2*MAX_PATH, so it is large enough to hold the message defined in the //resource string table, and one string of max size MAX_PATH. wsprintf (Text2, Text1, DynamicUpdatesShare); MessageBox (NULL, Text2, Text3, MB_ICONERROR | MB_OK | MB_SYSTEMMODAL); } } } else { //Note: DynamicUpdatesShare comes from the registry, so we have to treat it with caution StringCbCopy(Text2, sizeof(Text2), DynamicUpdatesShare); StringCbCat(Text2, sizeof(Text2), TEXT("\\WINNT32\\")); hr = StringCbCat(Text2, sizeof(Text2), moduleName); if (FAILED(hr)) { ExitProcess(ERROR_BUFFER_OVERFLOW); } if (GetFileAttributes (Text2) != (DWORD)-1 && GetFileVersion (Text2, ®FileVersionMS, ®FileVersionLS)) { tcharsNeeded = min(MAX_PATH, (INT)(WackExeName - RunningInstancePath + 2)); StringCchCopy(cdFilePath, tcharsNeeded, RunningInstancePath); hr = StringCchCat(cdFilePath, MAX_PATH, moduleName); if (FAILED(hr)) { ExitProcess(ERROR_BUFFER_OVERFLOW); } if (GetFileVersion (cdFilePath, &cdFileVersionMS, &cdFileVersionLS)) { if (MAKEULONGLONG(regFileVersionLS, regFileVersionMS) > MAKEULONGLONG(cdFileVersionLS, cdFileVersionMS)) { //This lstrcpy is ok, since the buffers are the same size lstrcpy (winnt32DllPath, Text2); } } } } FREE (DynamicUpdatesShare); DynamicUpdatesShare = NULL; } if (d == ERROR_SUCCESS) { #if defined(_X86_) // // before attempting to load the main module, make sure msvcrt.dll is present in the system dir // if (!GetSystemDirectory (Text1, MAX_PATH)) { ExitProcess (GetLastError ()); } //This is ok, since Text1 is MAX_PATH + 32 TCHARs long ConcatenatePaths (Text1, TEXT("msvcrt.dll")); d = GetFileAttributes (Text1); if (d == (DWORD)-1) { // // no local msvcrt.dll; copy the private file from CD // tcharsNeeded = min(MAX_PATH, (INT)(WackExeName - RunningInstancePath + 2)); StringCchCopy(cdFilePath, tcharsNeeded, RunningInstancePath); //this is ok, becuase cdFilePath is MAX_PATH + 32 TCHARs long ConcatenatePaths (cdFilePath, TEXT("win9xupg\\msvcrt.dll")); if (!CopyFile (cdFilePath, Text1, TRUE)) { ExitProcess (GetLastError ()); } } else if (d & FILE_ATTRIBUTE_DIRECTORY) { ExitProcess (ERROR_DIRECTORY); } #endif *WackExeName = 0; if (!winnt32DllPath[0]) { //The next two string operations are safe, since winnt32DllPath is 2*MAX_PATH TCHARs long lstrcpy (winnt32DllPath, RunningInstancePath); ConcatenatePaths (winnt32DllPath, moduleName); } b = FALSE; WinNT32 = LoadLibrary (winnt32DllPath); if(WinNT32) { winnt32 = (PWINNT32) GetProcAddress(WinNT32, "winnt32"); if (winnt32) { d = (*winnt32) (LocalWinnt32 ? RunningInstancePath : NULL, Dlg, WinNT32Stub, &restartCmdLine); b = TRUE; } FreeLibrary (WinNT32); } if (!b) { d = GetLastError (); if (LoadString (GetModuleHandle (NULL), IDS_APPNAME, Text3, sizeof(Text3)/sizeof(TCHAR)) && LoadString (GetModuleHandle (NULL), IDS_DLLERROR, Text1, sizeof(Text1)/sizeof(TCHAR))) { //This is safe, since Text2 is 2*MAX_PATH TCHARs long wsprintf (Text2, Text1, winnt32DllPath); MessageBox (NULL, Text2, Text3, MB_ICONERROR | MB_OK | MB_SYSTEMMODAL); } } } // // remove downloaded files // #ifdef _X86_ if (IsWin9x) { // // check if our local directory exists and if so delete it // if (LocalWinnt32 && GetFileAttributes (RunningInstancePath) != (DWORD)-1) { // // copy Winnt32.Exe to temp dir and rerun it from there with /CLEANUP option // // This is ok, since both buffers are of size MAX_PATH lstrcpy (DefSourcesDir, RunningInstancePath); CmdLineLen = GetTempPath (MAX_PATH, DownloadDest); if (!CmdLineLen) { // // an error occured; copy it to %windir% instead // GetWindowsDirectory (DownloadDest, MAX_PATH); } // // make sure temp path doesn't end in backslash // p = FindLastWack (DownloadDest); if (p && *(p + 1) == 0) { *p = 0; } *WackExeName = TEXT('\\'); if (CopyNode (DefSourcesDir, DownloadDest, WackExeName, FALSE, TRUE)) { if (!pReRun (DownloadDest, WackExeName, TEXT("/CLEANUP"), NULL)) { StringCbCatA (DownloadDest, sizeof(DownloadDest), WackExeName); DeleteNode (DownloadDest); } } } } #endif if (d == ERROR_SUCCESS) { // // check if a restart request was made // if (restartCmdLine) { STARTUPINFOA startupInfo; PROCESS_INFORMATION pi; ZeroMemory(&startupInfo, sizeof(startupInfo)); startupInfo.cb = sizeof(startupInfo); if (!CreateProcessA ( NULL, restartCmdLine, NULL, NULL, FALSE, NORMAL_PRIORITY_CLASS, NULL, NULL, &startupInfo, &pi )) { d = GetLastError (); } FREE (restartCmdLine); } } ExitProcess(d); }