2020-09-30 16:53:49 +02:00

650 lines
18 KiB
C++

//+------------------------------------------------------------------
//
// Project: Windows NT4 DS Client Setup Wizard
//
// Purpose: Installs the Windows NT4 DS Client Files
//
// File: dscsetup.cpp
//
// History: March 1998 Zeyong Xu Created
// Jan 2000 Jeff Jones (JeffJon) Modified
// - changed to be an NT setup
//
//------------------------------------------------------------------
#include <windows.h>
#include <prsht.h>
#include <setupapi.h>
#include <tchar.h>
#include <stdlib.h>
#include "resource.h"
#include "dscsetup.h"
#include "wizard.h"
#include "doinst.h"
SInstallVariables g_sInstVar;
// DllMain Entry
BOOL APIENTRY DllMain( HINSTANCE hInstance,
DWORD ul_reason_for_call,
LPVOID lpReserved
)
{
switch (ul_reason_for_call)
{
case DLL_PROCESS_ATTACH:
case DLL_THREAD_ATTACH:
case DLL_THREAD_DETACH:
case DLL_PROCESS_DETACH:
break;
}
return TRUE;
}
// This is an exported function.
DWORD WINAPI DoDscSetup(LPCSTR lpCmdLine)
{
// initialize the variables of installation
InitVariables();
ParseCmdline(const_cast<PSTR>(lpCmdLine));
//
// Go through setup if we are installing with anything but
// the /a flag
//
if (g_sInstVar.m_bSysDlls || g_sInstVar.m_bWabInst)
{
// create objects
if(!CreateObjects())
return SETUP_ERROR;
// Launch setup wizard
if(!DSCSetupWizard())
{
TCHAR szMessage[MAX_MESSAGE + 1];
TCHAR szTitle[MAX_TITLE + 1];
LoadString(g_sInstVar.m_hInstance,
IDS_ERROR_WIZARD,
szMessage,
MAX_MESSAGE);
LoadString(g_sInstVar.m_hInstance,
IDS_ERROR_TITLE,
szTitle,
MAX_TITLE);
// display a error message - Failure to load Setup Wizard
MessageBox(NULL,
szMessage, // address of text in message box
szTitle, // address of title of message box
MB_OK | MB_TOPMOST | MB_ICONERROR); // style of message box
g_sInstVar.m_nSetupResult = SETUP_ERROR;
}
// destroy objects
DestroyObjects();
if(g_sInstVar.m_nSetupResult == SETUP_SUCCESS &&
!g_sInstVar.m_bQuietMode
#ifdef MERRILL_LYNCH
&& !g_sInstVar.m_bNoReboot
#endif
)
{
// prompt reboot
SetupPromptReboot(NULL, // optional, handle to a file queue
NULL, // parent window of this dialog box
FALSE); // optional, do not prompt user);
}
else if(g_sInstVar.m_nSetupResult == SETUP_SUCCESS &&
g_sInstVar.m_bQuietMode
#ifdef MERRILL_LYNCH
&& !g_sInstVar.m_bNoReboot
#endif
)
{
HANDLE htoken = INVALID_HANDLE_VALUE;
do
{
// twiddle our process privileges to enable SE_SHUTDOWN_NAME
HRESULT hr = OpenProcessToken(GetCurrentProcess(),
TOKEN_ADJUST_PRIVILEGES,
&htoken);
if (FAILED(hr))
{
break;
}
LUID luid;
memset(&luid, 0, sizeof(luid));
hr = LookupPrivilegeValue(0, SE_SHUTDOWN_NAME, &luid);
if (FAILED(hr))
{
break;
}
TOKEN_PRIVILEGES privs;
memset(&privs, 0, sizeof(privs));
privs.PrivilegeCount = 1;
privs.Privileges[0].Luid = luid;
privs.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
hr = AdjustTokenPrivileges(htoken, 0, &privs, 0, 0, 0);
if (FAILED(hr))
{
break;
}
hr = ExitWindowsEx(EWX_REBOOT, 0);
if (FAILED(hr))
{
break;
}
}
while (0);
if (htoken != INVALID_HANDLE_VALUE)
{
CloseHandle(htoken);
}
}
}
else
{
//
// Setup was run with the /a flag. This means we
// don't want to show the UI and let the adsix86.inf
// handle the install for us
//
if (!LaunchProcess(STR_INSTALL_ADSIWREMOVE))
{
g_sInstVar.m_nSetupResult = SETUP_ERROR;
}
}
return g_sInstVar.m_nSetupResult;
}
VOID ParseCmdline(LPSTR lpCmdLine)
{
PCTSTR ptszTok = _tcstok(lpCmdLine, _T(" "));
do
{
if (ptszTok != NULL)
{
if (_tcsicmp(ptszTok, _T("/q")) == 0)
{
g_sInstVar.m_bQuietMode = TRUE;
}
if (_tcsicmp(ptszTok, _T("/a")) == 0)
{
g_sInstVar.m_bWabInst = FALSE;
g_sInstVar.m_bSysDlls = FALSE;
}
if (_tcsicmp(ptszTok, _T("/d")) == 0)
{
g_sInstVar.m_bWabInst = FALSE;
}
#ifdef MERRILL_LYNCH
if (_tcsicmp(ptszTok, _T("/n")) == 0)
{
g_sInstVar.m_bNoReboot = TRUE;
}
#endif
}
ptszTok = _tcstok(NULL, _T(" "));
} while (ptszTok != NULL);
}
// initialize the variables of installation
VOID InitVariables()
{
g_sInstVar.m_hInstance = GetModuleHandle(STR_DLL_NAME);
g_sInstVar.m_hInstallThread = NULL;
g_sInstVar.m_uTimerID = 0;
g_sInstVar.m_hBigBoldFont = NULL;
g_sInstVar.m_hProgress = NULL;
g_sInstVar.m_hFileNameItem = NULL;
g_sInstVar.m_bDCOMInstalled = FALSE;
g_sInstVar.m_bQuietMode = FALSE;
g_sInstVar.m_bWabInst = TRUE;
g_sInstVar.m_bSysDlls = TRUE;
#ifdef MERRILL_LYNCH
g_sInstVar.m_bNoReboot = FALSE;
#endif
g_sInstVar.m_nSetupResult = SETUP_SUCCESS;
// get source path
GetModuleFileName(g_sInstVar.m_hInstance,
g_sInstVar.m_szSourcePath,
MAX_PATH);
*(_tcsrchr(g_sInstVar.m_szSourcePath, CHAR_BACKSLASH) + 1) = TEXT('\0'); // Strip setup.exe off path
}
// start setup wizard
BOOL DSCSetupWizard()
{
PROPSHEETHEADER psh;
PROPSHEETPAGE psPage[SIZE_WIZARD_PAGE];
int i = 0;
//
// Setup the Welcome page
//
i=0;
ZeroMemory(&psPage[i],sizeof(PROPSHEETPAGE));
psPage[i].dwSize = sizeof(PROPSHEETPAGE);
psPage[i].dwFlags = PSP_USETITLE | PSP_HIDEHEADER;
psPage[i].hInstance = g_sInstVar.m_hInstance;
psPage[i].pszTemplate = MAKEINTRESOURCE(IDD_WELCOME);
psPage[i].pfnDlgProc = (DLGPROC) WelcomeDialogProc;
psPage[i].lParam = (LPARAM) 0;
psPage[i].pszTitle = MAKEINTRESOURCE(IDS_WIZARD_TITLE);
/* ntbug#337931: remove license page
//
// Setup the License Page
//
i++;
ZeroMemory(&psPage[i],sizeof(PROPSHEETPAGE));
psPage[i].dwSize = sizeof(PROPSHEETPAGE);
psPage[i].dwFlags = PSP_USETITLE |
PSP_USEHEADERTITLE |
PSP_USEHEADERSUBTITLE;
psPage[i].hInstance = g_sInstVar.m_hInstance;
psPage[i].pszTemplate = MAKEINTRESOURCE(IDD_LICENSE);
psPage[i].pfnDlgProc = (DLGPROC) LicenseDialogProc;
psPage[i].lParam = (LPARAM) 0;
psPage[i].pszTitle = MAKEINTRESOURCE(IDS_WIZARD_TITLE);
psPage[i].pszHeaderTitle = MAKEINTRESOURCE(IDS_HEADERTITLE_LICENSE);
psPage[i].pszHeaderSubTitle = MAKEINTRESOURCE(IDS_HEADERSUBTITLE_LICENSE);
*/
//
// Setup the Select Page
//
i++;
ZeroMemory(&psPage[i],sizeof(PROPSHEETPAGE));
psPage[i].dwSize = sizeof(PROPSHEETPAGE);
psPage[i].dwFlags = PSP_USETITLE |
PSP_USEHEADERTITLE |
PSP_USEHEADERSUBTITLE;
psPage[i].hInstance = g_sInstVar.m_hInstance;
psPage[i].pszTemplate = MAKEINTRESOURCE(IDD_CONFIRM);
psPage[i].pfnDlgProc = (DLGPROC) ConfirmDialogProc;
psPage[i].lParam = (LPARAM) 0;
psPage[i].pszTitle = MAKEINTRESOURCE(IDS_WIZARD_TITLE);
psPage[i].pszHeaderTitle = MAKEINTRESOURCE(IDS_HEADERTITLE_CONFIRM);
psPage[i].pszHeaderSubTitle = MAKEINTRESOURCE(IDS_HEADERSUBTITLE_CONFIRM);
//
// Setup the Confirm Page
//
i++;
ZeroMemory(&psPage[i],sizeof(PROPSHEETPAGE));
psPage[i].dwSize = sizeof(PROPSHEETPAGE);
psPage[i].dwFlags = PSP_USETITLE |
PSP_USEHEADERTITLE |
PSP_USEHEADERSUBTITLE;
psPage[i].hInstance = g_sInstVar.m_hInstance;
psPage[i].pszTemplate = MAKEINTRESOURCE(IDD_INSTALL);
psPage[i].pfnDlgProc = (DLGPROC) InstallDialogProc;
psPage[i].lParam = (LPARAM) 0;
psPage[i].pszTitle = MAKEINTRESOURCE(IDS_WIZARD_TITLE);
psPage[i].pszHeaderTitle = MAKEINTRESOURCE(IDS_HEADERTITLE_INSTALL);
psPage[i].pszHeaderSubTitle = MAKEINTRESOURCE(IDS_HEADERSUBTITLE_INSTALL);
//
// Setup the Completion page
//
i++;
ZeroMemory(&psPage[i],sizeof(PROPSHEETPAGE));
psPage[i].dwSize = sizeof(PROPSHEETPAGE);
psPage[i].dwFlags = PSP_USETITLE | PSP_HIDEHEADER;
psPage[i].hInstance = g_sInstVar.m_hInstance;
psPage[i].pszTemplate = MAKEINTRESOURCE(IDD_COMPLETION);
psPage[i].pfnDlgProc = (DLGPROC) CompletionDialogProc;
psPage[i].lParam = (LPARAM) 0;
psPage[i].pszTitle = MAKEINTRESOURCE(IDS_WIZARD_TITLE);
//
// Setup the wizard
//
ZeroMemory(&psh,sizeof(PROPSHEETHEADER));
psh.dwSize = sizeof(PROPSHEETHEADER);
// Windows 98 with 16 color display mode crashes when PSH_STRETCHWATERMARK flag is on.
psh.dwFlags = PSH_USEICONID |
PSH_PROPSHEETPAGE |
PSH_WIZARD97 |
PSH_WATERMARK |
PSH_HEADER; // | PSH_STRETCHWATERMARK;
psh.pszIcon = MAKEINTRESOURCE(IDI_ICON_APP);
psh.hInstance = g_sInstVar.m_hInstance;
psh.pszCaption = MAKEINTRESOURCE(IDS_WIZARD_TITLE);;
psh.nStartPage = 0;
psh.nPages = SIZE_WIZARD_PAGE;
psh.ppsp = (LPCPROPSHEETPAGE) &psPage;
//
// Run the wizard
//
if(g_sInstVar.m_bQuietMode)
{
if(!CheckDiskSpace())
return FALSE;
psh.nStartPage = 2;
}
else
psh.nStartPage = 0;
if( PropertySheet(&psh) < 0 ) // failure to load wizard
{
return FALSE;
}
//
// Because SetWindowLongPtr(hWnd, DWL_MSGRESULT, IDD_COMPLETION)
// doesn't work on Win95 (when users click Cancel, the wizard can't
// be routed to the Completion page), I added the following code to
// open the Completion page
//
if(!g_sInstVar.m_bQuietMode)
{
psh.nStartPage = 3;
if( PropertySheet(&psh) < 0 ) // failure to load wizard
{
return FALSE;
}
}
return TRUE;
}
// check for DCOM installed
void CheckDCOMInstalled()
{
HKEY hSubKey;
// check if IE 4.0 has been installed
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_CLASSES_ROOT,
STR_DCOM_REGKEY,
0,
KEY_READ,
&hSubKey) )
{
g_sInstVar.m_bDCOMInstalled = TRUE;
RegCloseKey(hSubKey);
}
}
// gets Disk Free space
DWORD64 SetupGetDiskFreeSpace()
{
DWORD dwSectorsPerCluster;
DWORD dwBytesPerSector;
DWORD dwNumberOfFreeClusters;
DWORD dwTotalNumberOfClusters;
DWORD64 d64FreeSpace = 0;
TCHAR szPathName[MAX_PATH + 1]; // address of root path
if(GetSystemDirectory(szPathName, // address of buffer for system directory
MAX_PATH)) // size of directory buffer);
{
if ( szPathName[1] == TEXT(':'))
{
// this is a drive letter
// assume it is of for d:backslash
szPathName[3] = TEXT('\0');
//get free space, GetDiskFreeSpaceEx() don't support in older Win95
if (GetDiskFreeSpace(szPathName, // address of root path
&dwSectorsPerCluster, // address of sectors per cluster
&dwBytesPerSector, // address of bytes per sector
&dwNumberOfFreeClusters, // address of number of free clusters
&dwTotalNumberOfClusters)) // address of total number of clusters
{
// calc total size
d64FreeSpace = DWORD64(dwSectorsPerCluster)
* dwBytesPerSector
* dwNumberOfFreeClusters;
}
}
}
return d64FreeSpace;
}
// check if DSClient has been installed
BOOL CheckDSClientInstalled()
{
HKEY hSubKey;
BOOL bResult = FALSE;
// open reg key of DS Client
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE,
STR_DSCLIENT_REGKEY,
0,
KEY_ALL_ACCESS,
&hSubKey) )
{
bResult = TRUE;
RegCloseKey(hSubKey);
}
return bResult;
}
/* ntbug#337931: remove license page
// load "License Agreement" text from license file
BOOL LoadLicenseFile(HWND hDlg)
{
BOOL bReturn = FALSE;
TCHAR szTitle[MAX_TITLE + 1];
TCHAR szTempBuffer[MAX_MESSAGE + 1];
TCHAR szLicenseFile[MAX_PATH + 1];
TCHAR szReturnTextBuffer[MAX_MESSAGE + 1];
LPTSTR lpszLicenseText = NULL;
HANDLE hFile;
DWORD dwNumberOfBytesRead, dwFileSize;
//
// Determine where we are installing from
// and specific the license file there
//
lstrcpy(szLicenseFile, g_sInstVar.m_szSourcePath);
lstrcat(szLicenseFile, STR_LICENSEFILE);
// Open License file
hFile = CreateFile(szLicenseFile, // pointer to name of the file
GENERIC_READ, // access (read-write) mode
FILE_SHARE_READ, // share mode
NULL, // pointer to security descriptor
OPEN_EXISTING, // how to create
FILE_ATTRIBUTE_NORMAL, // file attributes
NULL); // handle to file with attributes to copy
if(INVALID_HANDLE_VALUE != hFile)
{
// Read License file into string
// setup memory, get file size in bytes
dwFileSize = GetFileSize (hFile, NULL) ;
if (dwFileSize != 0xFFFFFFFF)
{
// this program is for Win98/95, it will work in MBCS not UNICODE
// this code is for ANSI US version, license.txt file uses the single byte character set(ANSI)
// if doing locolization, license.txt file should use the double byte character set(DBCS/MBCS)
lpszLicenseText = (LPTSTR) calloc (dwFileSize + sizeof(TCHAR),
sizeof(BYTE));
}
if(lpszLicenseText)
{
//read file
if (ReadFile(hFile, // handle of file to read
lpszLicenseText, // address of buffer that receives data
dwFileSize, // number of bytes to read
&dwNumberOfBytesRead, // address of number of bytes read
NULL)) // address of structure for data
{
// display license on dialog
SetDlgItemText(hDlg, IDC_LICENSE_TEXT, lpszLicenseText);
bReturn = TRUE;
}
// so free the memory
free(lpszLicenseText);
}
//close file handle
CloseHandle(hFile);
}
if(!bReturn)
{
// load string
LoadString(g_sInstVar.m_hInstance,
IDS_ERROR_TITLE,
szTitle,
MAX_TITLE);
LoadString(g_sInstVar.m_hInstance,
IDS_ERROR_LICENSEFILE,
szTempBuffer,
MAX_MESSAGE);
wsprintf(szReturnTextBuffer,
TEXT("%s %s"),
szTempBuffer,
szLicenseFile);
MessageBox(hDlg,
szReturnTextBuffer,
szTitle,
MB_OK | MB_TOPMOST | MB_ICONERROR);
}
return bReturn;
}
*/
// check disk space
BOOL CheckDiskSpace()
{
BOOL bResult = TRUE;
TCHAR szString[MAX_MESSAGE + MAX_TITLE + 1];
TCHAR szTitle[MAX_TITLE + 1];
TCHAR szMessage[MAX_MESSAGE + 1];
if(SIZE_TOTAL*MB_TO_BYTE > SetupGetDiskFreeSpace())
{
// load string
LoadString(g_sInstVar.m_hInstance,
IDS_ERROR_NODISKSPACE,
szMessage,
MAX_MESSAGE);
LoadString(g_sInstVar.m_hInstance,
IDS_ERROR_TITLE,
szTitle,
MAX_TITLE);
wsprintf(szString,
TEXT("%s %d MB."),
szMessage,
SIZE_TOTAL);
MessageBox(NULL,
szString,
szTitle,
MB_OK | MB_TOPMOST | MB_ICONERROR);
bResult = FALSE;
}
return bResult;
}
// create objects
BOOL CreateObjects()
{
try
{
// initialize the synchronizing object
InitializeCriticalSection(&g_sInstVar.m_oCriticalSection);
}
catch(...)
{
return FALSE;
}
// create a 12 pt big font
CreateBigFont();
return TRUE;
}
// destroy objects
VOID DestroyObjects()
{
// wait to finish the runing setup process
if(g_sInstVar.m_hInstallThread)
{
// wait the installation thread to finish
WaitForSingleObject(g_sInstVar.m_hInstallThread,INFINITE);
CloseHandle(g_sInstVar.m_hInstallThread);
}
// delete the synchronizing object
DeleteCriticalSection(&g_sInstVar.m_oCriticalSection);
//Frees up the space used by loading the fonts
if( g_sInstVar.m_hBigBoldFont )
{
DeleteObject( g_sInstVar.m_hBigBoldFont );
}
}
//create a big font for dialog title
VOID CreateBigFont()
{
NONCLIENTMETRICS ncm;
LOGFONT BigBoldLogFont;
HDC hdc;
// Create the fonts we need based on the dialog font
ZeroMemory(&ncm,sizeof(NONCLIENTMETRICS));
ncm.cbSize = sizeof(NONCLIENTMETRICS);
SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, &ncm, 0);
BigBoldLogFont = ncm.lfMessageFont;
BigBoldLogFont.lfWeight = FW_BOLD;
hdc = GetDC(NULL);
if( hdc )
{
BigBoldLogFont.lfHeight = 0 -
(GetDeviceCaps(hdc,LOGPIXELSY) *
SIZE_TITLE_FONT /
72);
g_sInstVar.m_hBigBoldFont = CreateFontIndirect(&BigBoldLogFont);
ReleaseDC(NULL,hdc);
}
}