Windows2003-3790/inetcore/setup/iexpress/wextract/utils.c
2020-09-30 16:53:55 +02:00

1117 lines
33 KiB
C

//***************************************************************************
//* Copyright (c) Microsoft Corporation 1995. All rights reserved. *
//***************************************************************************
//* *
//* UTILS.C - Win32 Based Cabinet File Self-extractor and installer utils. *
//* *
//***************************************************************************
//* *
//* Originally written by Jeff Webber. *
//* *
//***************************************************************************
//***************************************************************************
//* INCLUDE FILES *
//***************************************************************************
#include "pch.h"
#pragma hdrstop
#include "wextract.h"
#include "regstr.h"
#include "global.h"
#include <commctrl.h>
static TCHAR szRegRunOnceKey[] = "Software\\Microsoft\\Windows\\CurrentVersion\\RunOnce";
static TCHAR szNT4XDelayUntilReboot[] = "System\\CurrentControlSet\\Control\\Session Manager";
static TCHAR szNT4XPendingValue[] = "PendingFileRenameOperations";
static TCHAR szNT3XDelayUntilReboot[] = "System\\CurrentControlSet\\Control\\Session Manager\\FileRenameOperations";
static TCHAR szRegValNameTemplate[] = "wextract_cleanup%d";
static TCHAR szRegValTemplate[] = "%s /D:%s";
static TCHAR szRegValAdvpackTemplate[] = "rundll32.exe %sadvpack.dll,DelNodeRunDLL32 \"%s\"";
static TCHAR szBATCommand[] = "Command.com /c %s";
// store the RunOnce Clean-up reg keyname for this instance
//
TCHAR g_szRegValName[SMALL_BUF_LEN] = { 0 };
BOOL g_bConvertRunOnce = FALSE;
//***************************************************************************
//* Functions *
//***************************************************************************
typedef HRESULT (*CHECKTOKENMEMBERSHIP)(HANDLE TokenHandle, PSID SidToCheck, PBOOL IsMember);
BOOL CheckToken(BOOL *pfIsAdmin)
{
BOOL bNewNT5check = FALSE;
HINSTANCE hAdvapi32 = NULL;
CHECKTOKENMEMBERSHIP pf;
PSID AdministratorsGroup;
SID_IDENTIFIER_AUTHORITY NtAuthority = SECURITY_NT_AUTHORITY;
hAdvapi32 = LoadLibrary("advapi32.dll");
if (hAdvapi32)
{
pf = (CHECKTOKENMEMBERSHIP)GetProcAddress(hAdvapi32, "CheckTokenMembership");
if (pf)
{
bNewNT5check = TRUE;
*pfIsAdmin = FALSE;
if(AllocateAndInitializeSid( &NtAuthority, 2, SECURITY_BUILTIN_DOMAIN_RID,
DOMAIN_ALIAS_RID_ADMINS, 0, 0, 0, 0, 0, 0, &AdministratorsGroup) )
{
pf(NULL, AdministratorsGroup, pfIsAdmin);
FreeSid(AdministratorsGroup);
}
}
FreeLibrary(hAdvapi32);
}
return bNewNT5check;
}
// IsNTAdmin();
// Returns true if our process has admin priviliges.
// Returns false otherwise.
BOOL IsNTAdmin()
{
static int fIsAdmin = 2;
HANDLE hAccessToken;
PTOKEN_GROUPS ptgGroups;
DWORD dwReqSize;
UINT i;
SID_IDENTIFIER_AUTHORITY NtAuthority = SECURITY_NT_AUTHORITY;
PSID AdministratorsGroup;
BOOL bRet;
//
// If we have cached a value, return the cached value. Note I never
// set the cached value to false as I want to retry each time in
// case a previous failure was just a temp. problem (ie net access down)
//
bRet = FALSE;
ptgGroups = NULL;
if( fIsAdmin != 2 )
return (BOOL)fIsAdmin;
if (!CheckToken(&bRet))
{
if(!OpenProcessToken( GetCurrentProcess(), TOKEN_QUERY, &hAccessToken ) )
return FALSE;
// See how big of a buffer we need for the token information
if(!GetTokenInformation( hAccessToken, TokenGroups, NULL, 0, &dwReqSize))
{
// GetTokenInfo should the buffer size we need - Alloc a buffer
if(GetLastError() == ERROR_INSUFFICIENT_BUFFER)
ptgGroups = (PTOKEN_GROUPS) LocalAlloc(LMEM_FIXED, dwReqSize);
}
// ptgGroups could be NULL for a coupla reasons here:
// 1. The alloc above failed
// 2. GetTokenInformation actually managed to succeed the first time (possible?)
// 3. GetTokenInfo failed for a reason other than insufficient buffer
// Any of these seem justification for bailing.
// So, make sure it isn't null, then get the token info
if(ptgGroups && GetTokenInformation(hAccessToken, TokenGroups, ptgGroups, dwReqSize, &dwReqSize))
{
if(AllocateAndInitializeSid( &NtAuthority, 2, SECURITY_BUILTIN_DOMAIN_RID,
DOMAIN_ALIAS_RID_ADMINS, 0, 0, 0, 0, 0, 0, &AdministratorsGroup) )
{
// Search thru all the groups this process belongs to looking for the
// Admistrators Group.
for( i=0; i < ptgGroups->GroupCount; i++ )
{
if( EqualSid(ptgGroups->Groups[i].Sid, AdministratorsGroup) )
{
// Yea! This guy looks like an admin
fIsAdmin = TRUE;
bRet = TRUE;
break;
}
}
FreeSid(AdministratorsGroup);
}
}
if(ptgGroups)
LocalFree(ptgGroups);
// BUGBUG: Close handle here? doc's aren't clear whether this is needed.
CloseHandle(hAccessToken);
}
else if (bRet)
fIsAdmin = TRUE;
return bRet;
}
//**************************************************************************
//
// WarningDlgProc()
//
// Dialog proc for handling a continue/Exit dialog.
//
//**************************************************************************
INT_PTR CALLBACK WarningDlgProc( HWND hwnd, UINT msg,WPARAM wParam, LPARAM lParam)
{
char szMsg[MAX_STRING];
switch( msg )
{
case WM_INITDIALOG:
CenterWindow( hwnd, GetDesktopWindow() );
*szMsg = 0;
LoadString(g_hInst, (UINT)lParam, szMsg, sizeof(szMsg));
SetDlgItemText(hwnd, IDC_WARN_TEXT, szMsg);
MessageBeep((UINT)-1);
return( TRUE ); // Let default control be chosen.
case WM_COMMAND:
switch( wParam )
{
case IDC_EXIT:
case IDC_CONTINUE:
EndDialog( hwnd, wParam );
break;
default:
return FALSE;
}
return TRUE;
default:
break;
}
return( FALSE ); // Let default dialog processing do all.
}
// returns start of next field (or null if null), sets start to begining of the first field,
// with fields separated by separaters and nulls first separater after the first field
TCHAR* ExtractField( TCHAR **pstart, TCHAR * separaters)
{
LPTSTR start = *pstart;
int x = 0;
while(ANSIStrChr(separaters, *start)) {
if(*start == 0)
return(NULL);
start++;
}
*pstart = start;
while(!ANSIStrChr(separaters, start[x]) && (start[x] != 0))
x++;
if(start[x] == 0)
return(start + x);
start[x] = 0;
return(start + x + 1);
}
BOOL AnalyzeCmd( LPTSTR szOrigiCommand, LPTSTR *lplpCommand, BOOL *pfInfCmd )
{
TCHAR szTmp[MAX_PATH];
TCHAR szINFFile[MAX_PATH];
LPTSTR szNextField, szCurrField, szExt;
UINT secLength;
LPTSTR lpTempCmd, pszINFEngine;
lstrcpy( szTmp, szOrigiCommand );
// check if the command is LFN name
if ( szTmp[0] == '"' )
{
szCurrField = &szTmp[1];
szNextField = ExtractField( &szCurrField, "\"" );
}
else
{
szCurrField = szTmp;
szNextField = ExtractField( &szCurrField, " " );
}
if ( !IsRootPath( szCurrField ) )
{
// BUGBUG: when IsRootPath Failed, we did not check if the givn
// szCurrField is valid name or not. If it is not valid, the result
// of the AddPath will produce invalid file path. The error will come out at
// either SETUP engine or CreateProcess
//
lstrcpy( szINFFile, g_Sess.achDestDir );
AddPath( szINFFile, szCurrField );
}
else
lstrcpy( szINFFile, szCurrField );
// check if this is a INF file command
if ( ((szExt = ANSIStrRChr( szCurrField, '.' )) != NULL) && !lstrcmpi( szExt, ".INF" ) )
{
// check to see if this valid command
if ( !FileExists( szINFFile ) )
{
ErrorMsg1Param( NULL, IDS_ERR_FILENOTEXIST, szINFFile );
return FALSE;
}
// check if there is INF section install, and get the sec start point
szCurrField = szNextField;
szNextField = ExtractField( &szCurrField, "[" ); // skip things between .INF and [ section beginning
secLength = lstrlen( achDefaultSection );
if ( szNextField )
{
// in the case of: .INF<single-blank>[abc]
// the szNextField is "" while in the case of: .INF<multiple-blanks>[abc]
// szNextField points to "abc]". Therefore, the conditional pointer switch added here
//
if ( *szNextField )
{
szCurrField = szNextField;
}
szNextField = ExtractField( &szCurrField, "]" ); // get INF InstallSection name
if ( *szCurrField )
{
secLength = lstrlen( szCurrField );
}
}
lpTempCmd = (LPSTR) LocalAlloc( LPTR, 512);
if ( ! lpTempCmd )
{
ErrorMsg( NULL, IDS_ERR_NO_MEMORY );
return FALSE;
}
// store INF name for reboot checking use
g_uInfRebootOn = GetPrivateProfileInt( *szCurrField ? szCurrField : achDefaultSection, "Reboot", 0, szINFFile );
*pfInfCmd = TRUE; // no RunOnce entry needed
// check if we need use Advanced INF dll handling
if ( GetPrivateProfileString( SEC_VERSION, KEY_ADVINF, "", lpTempCmd, 8, szINFFile )
> 0 )
{
g_Sess.uExtractOpt |= EXTRACTOPT_ADVDLL;
// re-use the buf here
lstrcpy( szOrigiCommand, *szCurrField ? szCurrField : achDefaultSection );
lstrcpy( lpTempCmd, szINFFile );
}
else
{
g_Sess.uExtractOpt &= ~(EXTRACTOPT_ADVDLL);
if (g_wOSVer == _OSVER_WIN9X)
{
pszINFEngine = "setupx.dll";
GetShortPathName( szINFFile, szINFFile, sizeof(szINFFile) );
}
else
pszINFEngine = "setupapi.dll";
wsprintf( lpTempCmd, achSETUPDLL, pszINFEngine,
*szCurrField ? szCurrField : achDefaultSection, szINFFile );
}
}
else if ( ((szExt = ANSIStrRChr( szCurrField, '.' )) != NULL) && !lstrcmpi( szExt, ".BAT" ) )
{
lpTempCmd = (LPSTR) LocalAlloc( LPTR, lstrlen( szBATCommand ) + lstrlen( szINFFile ) + 8 );
if ( ! lpTempCmd )
{
ErrorMsg( NULL, IDS_ERR_NO_MEMORY );
return FALSE;
}
wsprintf( lpTempCmd, szBATCommand, szINFFile );
}
else
{
// assume EXE command
// you are here, the szINFFile contains the command with fully qualified pathname enterred either
// by User or appended by wextract.exe to Temp extracting file location.
DWORD dwAttr;
CHAR szCmd[2*MAX_STRING];
lpTempCmd = (LPSTR) LocalAlloc( LPTR, 2*MAX_STRING ); // 1K buf
if ( ! lpTempCmd )
{
ErrorMsg( NULL, IDS_ERR_NO_MEMORY );
return FALSE;
}
dwAttr = GetFileAttributes( szINFFile );
if ( (dwAttr == -1) || (dwAttr & FILE_ATTRIBUTE_DIRECTORY) )
{
// file is not found as it IS. Run it as it WAS!
// IS and WAS may be the same if user enterred fully qaulified name. CreateProcess will buff it.
lstrcpy( szCmd, szOrigiCommand );
}
else
{
// found it. Run it as it IS. Need to append switches if there is any.
lstrcpy( szCmd, szINFFile );
if ( szNextField && *szNextField )
{
lstrcat( szCmd, " " );
lstrcat( szCmd, szNextField );
}
}
// replace the #D with the directory this module is loaded or
// #E with the fullpath of the running EXE
ExpandCmdParams( szCmd, lpTempCmd );
}
*lplpCommand = lpTempCmd;
return TRUE;
}
void DisplayHelp()
{
MsgBox1Param( NULL, IDS_HELPMSG, "", MB_ICONINFORMATION, MB_OK );
}
DWORD CheckReboot( VOID )
{
DWORD dwReturn = 0xFFFFFFFF;
if ( !g_uInfRebootOn )
{
if (NeedReboot(g_dwRebootCheck, g_wOSVer))
dwReturn = EWX_REBOOT;
}
else
dwReturn = EWX_REBOOT; // reboot = 1 in inf file
return dwReturn;
}
// NT reboot
//
BOOL MyNTReboot()
{
HANDLE hToken;
TOKEN_PRIVILEGES tkp;
// get a token from this process
if ( !OpenProcessToken( GetCurrentProcess(),
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &hToken ) )
{
ErrorMsg( NULL, IDS_ERR_OPENPROCTK );
return FALSE;
}
// get the LUID for the shutdown privilege
LookupPrivilegeValue( NULL, SE_SHUTDOWN_NAME, &tkp.Privileges[0].Luid );
tkp.PrivilegeCount = 1;
tkp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
//get the shutdown privilege for this proces
if ( !AdjustTokenPrivileges( hToken, FALSE, &tkp, 0, (PTOKEN_PRIVILEGES)NULL, 0 ) )
{
ErrorMsg( NULL, IDS_ERR_ADJTKPRIV );
return FALSE;
}
// shutdown the system and force all applications to close
if (!ExitWindowsEx( EWX_REBOOT, 0 ) )
{
ErrorMsg( NULL, IDS_ERR_EXITWINEX );
return FALSE;
}
return TRUE;
}
// Display a dialog asking the user to restart Windows, with a button that
// will do it for them if possible.
//
void MyRestartDialog( DWORD dwRebootMode )
{
UINT id = IDCANCEL;
DWORD dwReturn;
// only if you checked and REBOOT_YES is true, you are here
if ( dwRebootMode & REBOOT_ALWAYS )
{
dwReturn = EWX_REBOOT;
}
else
{
dwReturn = CheckReboot();
}
if ( dwReturn == EWX_REBOOT )
{
if ( dwRebootMode & REBOOT_SILENT )
id = IDYES;
else
{
id = MsgBox1Param( NULL, IDS_RESTARTYESNO, "", MB_ICONINFORMATION, MB_YESNO );
}
if ( id == IDYES )
{
if ( dwReturn == EWX_REBOOT )
{
if ( g_wOSVer == _OSVER_WIN9X )
{
// By default (all platforms), we assume powerdown is possible
id = ExitWindowsEx( EWX_REBOOT, 0 );
}
else
{
MyNTReboot();
}
}
}
}
return;
}
// CleanRegRunOnce()
//
void CleanRegRunOnce()
{
HKEY hKey;
if ( g_szRegValName[0] == 0 )
{
return;
}
if ( RegOpenKeyEx( HKEY_LOCAL_MACHINE, szRegRunOnceKey, 0, KEY_WRITE, &hKey) == ERROR_SUCCESS )
{
RegDeleteValue( hKey, g_szRegValName );
RegCloseKey( hKey );
}
return;
}
void AddRegRunOnce()
{
HKEY hKey;
DWORD dwDisposition;
LPSTR szRegEntry;
TCHAR szBuf[MAX_PATH] = "";
TCHAR szAdvpack[MAX_PATH] = "";
int i;
DWORD dwTmp;
HANDLE hSetupLibrary;
BOOL fUseAdvpack = FALSE;
// prepare backup registry
if ( RegCreateKeyEx(HKEY_LOCAL_MACHINE, szRegRunOnceKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_READ | KEY_WRITE, NULL, &hKey, &dwDisposition) != ERROR_SUCCESS)
{
// reg problem, but not block the process
return;
}
// Check if key already exists -- if so, go with next number.
//
for (i=0; i<200; i++)
{
wsprintf( g_szRegValName, szRegValNameTemplate, i );
if ( RegQueryValueEx( hKey, g_szRegValName, 0, NULL, NULL, &dwTmp ) != ERROR_SUCCESS )
{
// g_szRegValName now has the key name we need for this instance
break;
}
}
if ( i == 200 )
{
// something is wrong, there are at lease 200 RunOnce enteries in the Registry
// bail out, don't add any more
RegCloseKey( hKey );
g_szRegValName[0] = 0;
return;
}
// check if ADVPACK in the system dir exports DelNodeRunDLL32;
// if so, use szRegValAdvpackTemplate, otherwise, use szRegValTemplate
GetSystemDirectory(szAdvpack, sizeof(szAdvpack));
AddPath(szAdvpack, ADVPACKDLL);
if ((hSetupLibrary = LoadLibrary(szAdvpack)) != NULL)
{
fUseAdvpack = GetProcAddress(hSetupLibrary, "DelNodeRunDLL32") != NULL;
FreeLibrary(hSetupLibrary);
}
if (fUseAdvpack)
{
if (GetSystemDirectory(szBuf, sizeof(szBuf)))
AddPath(szBuf, "");
}
else
{
// get current EXE filename
//
if ( !GetModuleFileName( g_hInst, szBuf, (DWORD)sizeof(szBuf) ) )
{
RegCloseKey( hKey );
return;
}
}
// alloc mem for store reg entry values
//
szRegEntry = (LPSTR) LocalAlloc( LPTR, lstrlen(szBuf) + lstrlen(g_Sess.achDestDir) + SMALL_BUF_LEN );
if ( !szRegEntry )
{
ErrorMsg( NULL, IDS_ERR_NO_MEMORY );
RegCloseKey( hKey );
return;
}
g_bConvertRunOnce = !fUseAdvpack;
wsprintf(szRegEntry, fUseAdvpack ? szRegValAdvpackTemplate : szRegValTemplate, szBuf, g_Sess.achDestDir);
RegSetValueEx( hKey, g_szRegValName, 0, REG_SZ, (CONST BYTE*)szRegEntry, lstrlen(szRegEntry)+1);
RegCloseKey(hKey);
LocalFree( szRegEntry );
return;
}
// Change the RunOnce entry that cleans up extracted files to use ADVPACK instead of wextract
void ConvertRegRunOnce()
{
if (*g_szRegValName)
{
HKEY hKey;
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, szRegRunOnceKey, 0, KEY_READ | KEY_WRITE, &hKey) == ERROR_SUCCESS)
{
TCHAR szRegEntry[2 * MAX_PATH + sizeof(szRegValAdvpackTemplate)];
DWORD dwSize = sizeof(szRegEntry);
// read the old value data that uses wextract and get the extracted files dir
if (RegQueryValueEx(hKey, g_szRegValName, NULL, NULL, (LPBYTE) szRegEntry, &dwSize) == ERROR_SUCCESS)
{
TCHAR szSysDir[MAX_PATH] = "";
if (GetSystemDirectory(szSysDir, sizeof(szSysDir)))
AddPath(szSysDir, "");
wsprintf(szRegEntry, szRegValAdvpackTemplate, szSysDir, g_Sess.achDestDir);
RegSetValueEx(hKey, g_szRegValName, 0, REG_SZ, (CONST BYTE *) szRegEntry, lstrlen(szRegEntry) + 1);
}
RegCloseKey(hKey);
}
}
return;
}
void DeleteMyDir( LPSTR lpDir )
{
char szFile[MAX_PATH];
WIN32_FIND_DATA fileData;
HANDLE hFindFile;
if ( lpDir == NULL || *lpDir == '\0' )
return;
lstrcpy( szFile, lpDir );
lstrcat( szFile, "*" );
hFindFile = FindFirstFile( szFile, &fileData );
if ( hFindFile == INVALID_HANDLE_VALUE )
return;
do
{
lstrcpy( szFile, lpDir );
if ( fileData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY )
{
if ( lstrcmp( fileData.cFileName, "." ) == 0 ||
lstrcmp( fileData.cFileName, ".." ) == 0 )
continue;
// delete the sub-dir
lstrcat( szFile, fileData.cFileName );
AddPath( szFile, "");
DeleteMyDir( szFile );
}
else
{
// delete the file
lstrcat( szFile, fileData.cFileName );
SetFileAttributes( szFile, FILE_ATTRIBUTE_NORMAL );
DeleteFile( szFile );
}
} while ( FindNextFile( hFindFile, &fileData ) );
FindClose( hFindFile );
RemoveDirectory( lpDir );
}
#if 0
//==================================================================
// AddPath()
//
void AddPath(LPSTR szPath, LPCSTR szName )
{
LPSTR szTmp;
// Find end of the string
szTmp = szPath + lstrlen(szPath);
// If no trailing backslash then add one
if ( szTmp > szPath && *(AnsiPrev( szPath, szTmp )) != '\\' )
*(szTmp++) = '\\';
// Add new name to existing path string
while ( *szName == ' ' ) szName++;
lstrcpy( szTmp, szName );
}
#endif
//---------------------------------------------------------------------------
// Returns TRUE if the given string is a UNC path.
//
// check if a path is a root path
//
// returns:
// TRUE for "X:\..." "\\foo\asdf\..."
// FALSE for others
BOOL IsRootPath(LPCSTR pPath)
{
if ( !pPath || (lstrlen(pPath) < 3) )
{
return FALSE;
}
// BUGBUG: this just smell like UNC, possible invalid UNC. If so,
// user will get error when later create process
if ( ( (*(pPath+1) == ':') && (*(pPath+2) == '\\') ) || // "X:\" case
( (*pPath == '\\') && (*(pPath + 1) == '\\' ) ) ) // UNC \\.... case
return TRUE;
else
return FALSE;
}
// BUGBUG:BUGBUG:BUGBUG:BUGBUG
// The code below is duplicated in advpack.dll. If you do changed/fixes to this code
// make sure to also change the code in advpack.dll
// Returns the size of wininit.ini in the windows directory.
// 0 if not found
DWORD GetWininitSize()
{
TCHAR szPath[MAX_PATH];
HFILE hFile;
DWORD dwSize = (DWORD)0;
if ( GetWindowsDirectory( szPath, MAX_PATH ) )
{
AddPath( szPath, "wininit.ini" );
// Make sure all changes have been saved to disk for accurate size reading
WritePrivateProfileString(NULL, NULL, NULL, szPath);
if ((hFile = _lopen(szPath, OF_READ|OF_SHARE_DENY_NONE)) != HFILE_ERROR)
{
dwSize = _llseek(hFile, 0L, FILE_END);
_lclose(hFile);
}
}
return dwSize;
}
// Returns the size of the value lpcszValue under lpcszRegKey
// 0 if the registry key or the value were not found
DWORD GetRegValueSize(LPCSTR lpcszRegKey, LPCSTR lpcszValue)
{
HKEY hKey;
DWORD dwValueSize = (DWORD)0;
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, lpcszRegKey, 0, KEY_READ, &hKey) == ERROR_SUCCESS)
{
if (RegQueryValueEx(hKey, lpcszValue, NULL, NULL, NULL,&dwValueSize) != ERROR_SUCCESS)
dwValueSize = (DWORD)0;
RegCloseKey(hKey);
}
return dwValueSize;
}
// Returns the number of Values in the key
// 0 if the registry key was not found or RegQueryInfoKey failed
DWORD GetNumberOfValues(LPCSTR lpcszRegKey)
{
HKEY hKey;
DWORD dwValueSize = (DWORD)0;
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, lpcszRegKey, 0, KEY_READ, &hKey) == ERROR_SUCCESS)
{
if (RegQueryInfoKey(hKey,
NULL, NULL, NULL, NULL, NULL, NULL,
&dwValueSize,
NULL, NULL, NULL, NULL) != ERROR_SUCCESS)
dwValueSize = (DWORD)0;
RegCloseKey(hKey);
}
return dwValueSize;
}
// Returns the rebootcheck value depending on the OS we get passed in.
DWORD NeedRebootInit(WORD wOSVer)
{
DWORD dwReturn = (DWORD)0;
switch (wOSVer)
{
case _OSVER_WIN9X:
dwReturn = GetWininitSize();
break;
case _OSVER_WINNT40:
case _OSVER_WINNT50:
dwReturn = GetRegValueSize(szNT4XDelayUntilReboot, szNT4XPendingValue);
break;
case _OSVER_WINNT3X:
dwReturn = GetNumberOfValues(szNT3XDelayUntilReboot);
break;
}
return dwReturn;
}
// Checks the passed in reboot check value against the current value.
// If they are different, we need to reboot.
// The reboot check value is dependend on the OS
BOOL NeedReboot(DWORD dwRebootCheck, WORD wOSVer)
{
return (dwRebootCheck != NeedRebootInit(wOSVer));
}
// check if Dir does not exist, create one.
//
BOOL IfNotExistCreateDir( LPTSTR lpDir )
{
DWORD attr;
if ( (attr = GetFileAttributes( lpDir )) == -1 )
{
return ( CreateDirectory( lpDir, NULL ) );
}
return (attr & FILE_ATTRIBUTE_DIRECTORY);
}
// check if the given dir is on Windows Drive
//
BOOL IsWindowsDrive( LPTSTR szPath )
{
TCHAR szWin[MAX_PATH];
if ( !GetWindowsDirectory( szWin, MAX_PATH ) )
{
ErrorMsg( NULL, IDS_ERR_GET_WIN_DIR );
ASSERT( FALSE );
}
return ( *szPath == szWin[0] );
}
PSTR MyULtoA( ULONG ulParam, PSTR pszOut )
{
wsprintf( pszOut, "%lu", ulParam );
return pszOut;
}
// display diskspace checking Error message
// it always return FALSE except that User answer YES on msgbox
//
BOOL DiskSpaceErrMsg( UINT msgType, ULONG ulExtractNeeded, DWORD dwInstNeeded, LPTSTR lpDrv )
{
TCHAR szSize[10];
BOOL bRet = FALSE;
// all the cases except one are returning FALSE, so we set Error code here
g_dwExitCode = ERROR_DISK_FULL;
if ( msgType == MSG_REQDSK_ERROR )
{
ErrorMsg1Param( NULL, IDS_ERR_NO_SPACE_ERR, MyULtoA((ulExtractNeeded+dwInstNeeded), szSize) );
}
else if ( msgType == MSG_REQDSK_RETRYCANCEL )
{
if ( MsgBox1Param( NULL, IDS_ERR_NO_SPACE_BOTH, MyULtoA( (ulExtractNeeded+dwInstNeeded), szSize),
MB_ICONQUESTION, MB_RETRYCANCEL|MB_DEFBUTTON1 ) == IDRETRY )
bRet = TRUE;
else
bRet = FALSE;
}
else if ( msgType == MSG_REQDSK_WARN )
{
// in /Q mode: MsgBox2Param return MB_OK which is not IDYES, so we fail the process.
//
if ( MsgBox2Param( NULL, IDS_ERR_NO_SPACE_INST, MyULtoA(dwInstNeeded, szSize), lpDrv,
MB_ICONINFORMATION, MB_YESNO | MB_DEFBUTTON2 ) == IDYES )
{
bRet = TRUE;
g_dwExitCode = S_OK;
}
}
//else ( msgType == MSG_REQDSK_NONE ) do nothing
return bRet;
}
BOOL GetFileTobeChecked( LPSTR szPath, int iSize, LPCSTR szNameStr )
{
char ch;
BOOL bComplete = FALSE;
szPath[0] = 0;
if ( *szNameStr == '#' )
{
ch = (CHAR)CharUpper((PSTR)*(++szNameStr));
szNameStr = CharNext( CharNext( szNameStr ) );
switch ( ch )
{
case 'S':
GetSystemDirectory( szPath, iSize );
break;
case 'W':
GetWindowsDirectory( szPath, iSize );
break;
case 'A':
default:
{
// look into reg AppPath
char szSubKey[MAX_PATH];
DWORD dwSize = sizeof( szSubKey );
HKEY hKey;
DWORD dwType;
lstrcpy( szSubKey, REGSTR_PATH_APPPATHS );
AddPath( szSubKey, szNameStr );
if ( RegOpenKeyEx( HKEY_LOCAL_MACHINE, szSubKey, 0, KEY_READ, &hKey ) == ERROR_SUCCESS )
{
if ( RegQueryValueEx(hKey, "", NULL, &dwType, (LPBYTE)szPath, &dwSize) == ERROR_SUCCESS )
{
if ((dwType == REG_EXPAND_SZ) &&
(ExpandEnvironmentStrings(szPath, szSubKey, sizeof(szSubKey))))
{
lstrcpy(szPath, szSubKey);
bComplete = TRUE;
}
else if (dwType == REG_SZ)
bComplete = TRUE;
}
RegCloseKey( hKey );
}
}
break;
}
}
else
GetSystemDirectory( szPath, iSize );
if ( !bComplete )
AddPath( szPath, szNameStr );
return TRUE;
}
BOOL CheckFileVersion( PTARGETVERINFO ptargetVers, LPSTR szPath, int isize, int *pidx )
{
unsigned uiSize;
DWORD dwVerInfoSize;
DWORD dwHandle;
VS_FIXEDFILEINFO * lpVSFixedFileInfo;
void FAR *lpBuffer;
HGLOBAL hgbl = NULL;
BOOL bRet = FALSE;
int ifrAnswer[2], itoAnswer[2], i, j;
PVERCHECK pfileV;
for ( i=0; i< (int)(ptargetVers->dwNumFiles); i++ )
{
pfileV = (PVERCHECK)( ptargetVers->szBuf + ptargetVers->dwFileOffs + i*sizeof(VERCHECK) );
if ( !GetFileTobeChecked( szPath, isize, (ptargetVers->szBuf + pfileV->dwNameOffs) ) )
goto EXIT;
dwVerInfoSize = GetFileVersionInfoSize(szPath, &dwHandle);
if (dwVerInfoSize)
{
// Alloc the memory for the version stamping
hgbl = GlobalAlloc(GHND, dwVerInfoSize);
if (hgbl == NULL)
goto EXIT;
lpBuffer = GlobalLock(hgbl);
if (lpBuffer == NULL)
goto EXIT;
// Read version stamping info
if (GetFileVersionInfo(szPath, dwHandle, dwVerInfoSize, lpBuffer))
{
// Get the value for Translation
if ( VerQueryValue(lpBuffer, "\\", (void FAR*FAR*)&lpVSFixedFileInfo, &uiSize) && (uiSize) )
{
for ( j=0; j<2; j++ )
{
ifrAnswer[j] = CompareVersion( lpVSFixedFileInfo->dwFileVersionMS, lpVSFixedFileInfo->dwFileVersionLS,
pfileV->vr[j].frVer.dwMV, pfileV->vr[j].frVer.dwLV );
itoAnswer[j] = CompareVersion( lpVSFixedFileInfo->dwFileVersionMS, lpVSFixedFileInfo->dwFileVersionLS,
pfileV->vr[j].toVer.dwMV, pfileV->vr[j].toVer.dwLV );
}
if ( (ifrAnswer[0] >= 0 && itoAnswer[0] <= 0) || (ifrAnswer[1] >= 0 && itoAnswer[1] <= 0) )
;
else
{
GlobalUnlock(hgbl);
goto EXIT;
}
}
}
GlobalUnlock(hgbl);
}
else
{
// file not exist case, if author specify install 1st ranges from version 0 to 0. Then do it!
if ( pfileV->vr[0].frVer.dwMV || pfileV->vr[0].frVer.dwLV )
{
goto EXIT;
}
}
}
bRet = TRUE;
EXIT:
*pidx = i;
if ( hgbl )
GlobalFree( hgbl );
return bRet;
}
UINT GetMsgboxFlag( DWORD dwFlag )
{
UINT uButton;
if ( dwFlag & VERCHK_YESNO )
uButton = MB_YESNO | MB_DEFBUTTON2;
else if ( dwFlag & VERCHK_OKCANCEL )
uButton = MB_OKCANCEL | MB_DEFBUTTON2;
else
uButton = MB_OK;
return uButton;
}
int CompareVersion(DWORD dwMS1, DWORD dwLS1, DWORD dwMS2, DWORD dwLS2)
{
if (dwMS1 < dwMS2)
return -1 ;
if (dwMS1 > dwMS2)
return 1 ;
if (dwLS1 < dwLS2)
return -1 ;
if (dwLS1 > dwLS2)
return 1 ;
return 0 ;
}
void ExpandCmdParams( PCSTR pszInParam, PSTR pszOutParam )
{
CHAR szModulePath[MAX_PATH];
LPSTR pszTmp;
*pszOutParam = '\0';
if ( !pszInParam || !*pszInParam )
return;
// get Module path
GetModuleFileName( g_hInst, szModulePath, (DWORD)sizeof(szModulePath) );
while ( *pszInParam != '\0' )
{
if (IsDBCSLeadByte(*pszInParam))
{
*pszOutParam = *pszInParam;
*(pszOutParam+1) = *(pszInParam+1);
}
else
*pszOutParam = *pszInParam;
if ( *pszInParam == '#' )
{
pszInParam = CharNext(pszInParam);
if ( (CHAR)CharUpper((PSTR)*pszInParam) == 'D' )
{
GetParentDir( szModulePath );
pszTmp = CharPrev(szModulePath, &szModulePath[lstrlen(szModulePath)]);
if (pszTmp && (*pszTmp == '\\'))
*pszTmp = '\0';
lstrcpy( pszOutParam, szModulePath );
pszOutParam += lstrlen( szModulePath );
}
else if ( (CHAR)CharUpper((PSTR)*pszInParam) == 'E' )
{
lstrcpy( pszOutParam, szModulePath );
pszOutParam += lstrlen( szModulePath );
}
else if ( *pszInParam == '#' )
pszOutParam = CharNext( pszOutParam );
}
else
pszOutParam = CharNext( pszOutParam );
pszInParam = CharNext(pszInParam);
}
*pszOutParam = '\0';
}