664 lines
18 KiB
C++
Raw Normal View History

2001-01-01 00:00:00 +01:00
//+------------------------------------------------------------------
//
// Project: Windows NT4 DS Client Setup Wizard
//
// Purpose: Installs the Windows NT4 DS Client Files
//
// File: doinst.cpp
//
// History: Aug. 1998 Zeyong Xu Created
// Jan 2000 Jeff Jones (JeffJon) Modified
// - changed to be an NT setup
//
//------------------------------------------------------------------
#include <windows.h>
#include <setupapi.h>
#include <advpub.h>
#include "resource.h"
#include "dscsetup.h"
#include "wizard.h"
#include "doinst.h"
extern SInstallVariables g_sInstVar;
// do installation
DWORD DoInstallation(HWND hWnd)
{
if(g_sInstVar.m_nSetupResult == SETUP_SUCCESS)
{
// set the fake progressbar for DCOM and WAB nstall
g_sInstVar.m_uTimerID = SetTimer(hWnd,
1,
1000, // 1 seconds
Timer1Proc);
// do the custom action of NTLMv2
if(!DoEncSChannel())
g_sInstVar.m_nSetupResult = SETUP_ERROR;
// stop the fake progressbar
if(g_sInstVar.m_uTimerID)
KillTimer(hWnd, g_sInstVar.m_uTimerID);
// install adsi
if (!LaunchProcess(STR_INSTALL_ADSI))
{
g_sInstVar.m_nSetupResult = SETUP_ERROR;
}
// install dsclient
if(g_sInstVar.m_nSetupResult == SETUP_SUCCESS)
g_sInstVar.m_nSetupResult = LaunchINFInstall(hWnd);
}
return g_sInstVar.m_nSetupResult;
}
VOID CALLBACK Timer1Proc(HWND hwnd, // handle of window for timer messages
UINT uMsg, // WM_TIMER message
UINT idEvent, // timer identifier
DWORD dwTime) // current system time
{
static int nCount = 0;
if(nCount > 100)
nCount = 100;
// set the fake progressbar
SendMessage (g_sInstVar.m_hProgress, PBM_SETPOS, (WPARAM) nCount, 0);
nCount ++;
}
// This routine will do an installation based on those settings
// using the setupapi.dll
INT LaunchINFInstall( HWND hWnd )
{
TCHAR szInfFileName[MAX_PATH + 1];
TCHAR szInstallSection[MAX_TITLE];
BOOL bResult = FALSE;
// Context for my call back routine
HSPFILEQ hFileQueue;
HINF hInf;
PVOID pDefaultContext;
//
// Get inf handle
// must know where the inf is located
// SetupOpenInfFile will only look in windows\inf by default
//
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API. m_szSourcePath may
// not be NULL terminated and/or it may be larger than the size allocated
// for szInfFileName
lstrcpy(szInfFileName, g_sInstVar.m_szSourcePath);
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API. m_szSourcePath + STR_DSCLIENT_INF
// may be larger than szInfFileName
lstrcat(szInfFileName, STR_DSCLIENT_INF);
hInf = SetupOpenInfFile(szInfFileName, // If path,needs full path, else looks in %windir%\inf
NULL, // Inf Type, matches Class in [Version] section SetupClass=SAMPLE
INF_STYLE_WIN4, // or INF_STYLE_OLDNT
NULL); // Line where error occurs if inf is has a problem
if (hInf == INVALID_HANDLE_VALUE)
return SETUP_ERROR;
//
// Create a Setup file queue and initialize the default Setup
// queue callback routine.
//
hFileQueue = SetupOpenFileQueue();
if(hFileQueue == INVALID_HANDLE_VALUE)
{
SetupCloseInfFile(hInf);
return SETUP_ERROR;
}
// using SetupInitDefaultQueueCallback.
SendMessage (g_sInstVar.m_hProgress, PBM_SETPOS, (WPARAM) 0, 0);
pDefaultContext = SetupInitDefaultQueueCallbackEx(hWnd, // HWND of owner window
NULL, // HWND of alternate progress dialog which receives
0, // Message sent to above window indicating a progress message
0, // DWORD Reserved
NULL); // PVOID Reserved
if(!pDefaultContext)
{
// Close the queue and the inf file and return
SetupCloseFileQueue(hFileQueue);
SetupCloseInfFile(hInf);
return SETUP_ERROR;
}
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API. At minimum should
// use an n version so as not to exceed szInstallSection. szInstallSection
// was not initialized with zeros
lstrcpy (szInstallSection, STR_INSTALL_SECTIONNT4);
//
// Queue file operations and commit the queue.
//
bResult = SetupInstallFilesFromInfSection(hInf, // HINF that has the directory ids set above
NULL, // layout.inf if you have one, this a convient
hFileQueue, // Queue to add files to
szInstallSection, // SectionName,
g_sInstVar.m_szSourcePath, // Path where the source files are located
SP_COPY_NEWER );
//
// All the files for each component are now in one queue
// now we commit it to start the copy ui, this way the
// user has one long copy progress dialog--and for a big install
// can go get the cup of coffee
if(bResult)
bResult = SetupCommitFileQueue(hWnd, // Owner
hFileQueue, // Queue with the file list
QueueCallbackProc, // This is our handler, it calls the default for us
pDefaultContext); // Pointer to resources allocated with SetupInitDefaultQueueCallback/Ex
if (!bResult || (g_sInstVar.m_nSetupResult == SETUP_CANCEL))
{
SetupTermDefaultQueueCallback(pDefaultContext);
SetupCloseFileQueue(hFileQueue);
SetupCloseInfFile(hInf);
if(g_sInstVar.m_nSetupResult == SETUP_CANCEL)
return SETUP_CANCEL;
else
return SETUP_ERROR;
}
//
// NOTE: you can do the entire install
// for a section with this api but in this case
// we build the file list conditionally and
// do only out ProductInstall section for registy stuff
// Also using SPINST_FILES will do the files
// as above but only one section at a time
// so the progress bar would keep completing and starting over
// SPINST_ALL does files, registry and inis
//
bResult = SetupInstallFromInfSection(hWnd,
hInf,
szInstallSection,
SPINST_INIFILES | SPINST_REGISTRY,
HKEY_LOCAL_MACHINE,
NULL, //m_szSourcePath, // Path where the source files are located
0, //SP_COPY_NEWER,
NULL, //(PSP_FILE_CALLBACK) QueueCallbackProc,
NULL, //&MyInstallData,
NULL,
NULL);
//
// We're done so free the context, close the queue,
// and release the inf handle
//
SetupTermDefaultQueueCallback(pDefaultContext);
SetupCloseFileQueue(hFileQueue);
SetupCloseInfFile(hInf);
if (g_sInstVar.m_bSysDlls)
{
//
// register OCX file
//
if(!RegisterOCX())
{
return SETUP_ERROR;
}
}
//
// The custom registry action after dsclient.inf by Chandana Surlu
//
DoDsclientReg();
SendMessage (g_sInstVar.m_hProgress, PBM_SETPOS, (WPARAM) 100, 0);
InstallFinish(FALSE);
return SETUP_SUCCESS;
}
/*---------------------------------------------------------------------*/
/*---------------------------------------------------------------------*/
UINT CALLBACK QueueCallbackProc(PVOID pDefaultContext,
UINT Notification,
UINT_PTR Param1,
UINT_PTR Param2)
{
static INT snFilesCopied;
// synchronizing user cancel
// REVIEWED-2002/03/12-JeffJon-We want to allow the exception to propogate
// out.
EnterCriticalSection(&g_sInstVar.m_oCriticalSection);
LeaveCriticalSection(&g_sInstVar.m_oCriticalSection);
//instaniate dialog first time
if (g_sInstVar.m_nSetupResult == SETUP_CANCEL)
{
SetLastError (ERROR_CANCELLED);
return FILEOP_ABORT;
}
switch (Notification)
{
case SPFILENOTIFY_STARTQUEUE:
case SPFILENOTIFY_ENDQUEUE:
return FILEOP_DOIT;
case SPFILENOTIFY_STARTCOPY:
// update file name item
SetWindowText(g_sInstVar.m_hFileNameItem,
((PFILEPATHS) Param1)->Target);
break;
case SPFILENOTIFY_ENDCOPY:
snFilesCopied++;
// update dialog file progress with message
if ((snFilesCopied + 1)>= NUM_FILES_TOTAL)
{
SendMessage (g_sInstVar.m_hProgress,
PBM_SETPOS,
(WPARAM) 100,
0);
}
else
{
SendMessage (g_sInstVar.m_hProgress,
PBM_SETPOS,
(WPARAM) ((float)snFilesCopied /
(float)NUM_FILES_TOTAL * 100),
0);
}
break;
default:
break;
}
return SetupDefaultQueueCallback(pDefaultContext,
Notification,
Param1,
Param2);
}
VOID InstallFinish(BOOL nShow)
{
// ISSUE-2002/03/12-JeffJon-Should call WinExec with
// the full path to the exe
if(nShow)
WinExec("grpconv -o", SW_SHOWNORMAL);
else
WinExec("grpconv -o", SW_HIDE);
}
// launch Inf file to install this component
BOOL LaunchProcess(LPTSTR lpCommandLine)
{
BOOL bResult = FALSE;
STARTUPINFO si;
PROCESS_INFORMATION pi;
// its console window will be invisible to the user.
ZeroMemory(&pi,sizeof(PROCESS_INFORMATION));
ZeroMemory(&si,sizeof(STARTUPINFO));
si.cb = sizeof (STARTUPINFO);
si.dwFlags = STARTF_USESHOWWINDOW;
si.wShowWindow = SW_HIDE; // HideWindow
// ISSUE-2002/03/12-JeffJon-Should call CreateProcess with
// the full path to the exe
if(CreateProcess( NULL,
lpCommandLine,
NULL,
NULL,
FALSE,
0,
NULL,
NULL,
&si,
&pi ) )
{
// wait to finish the runing setup process
WaitForSingleObject(pi.hProcess,INFINITE);
// close process handle
if (pi.hProcess && pi.hProcess != INVALID_HANDLE_VALUE)
{
CloseHandle (pi.hProcess) ;
}
if (pi.hThread && pi.hThread != INVALID_HANDLE_VALUE)
{
CloseHandle (pi.hThread) ;
}
bResult = TRUE;
}
return bResult;
}
// register OCX file
BOOL RegisterOCX()
{
TCHAR szSystem[MAX_PATH + 1];
TCHAR szTemp[MAX_PATH + 1];
TCHAR szCmdline[MAX_PATH + 1];
BOOL bSuccess = TRUE;
if(!GetSystemDirectory(szSystem, MAX_PATH))
return FALSE;
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API.
// Should consider using the strsafe inline APIs
wsprintf(szTemp,
TEXT("%s%s%s"),
szSystem,
STR_REGISTER_REGSVR32_S_EXE,
szSystem);
//
// REVIEW_JEFFJON : we are not going to register it here
// Instead we are going to set the RunOnce regkey
// to register the dlls on reboot
//
if (g_sInstVar.m_bWabInst)
{
// register dsfolder.dll
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API.
// Should consider using the strsafe inline APIs
wsprintf(szCmdline,
TEXT("%s%s %s%s %s%s %s%s %s%s"),
szTemp,
STR_REGISTER_DSFOLDER_DLL,
szSystem,
STR_REGISTER_DSUIEXT_DLL,
szSystem,
STR_REGISTER_DSQUERY_DLL,
szSystem,
STR_REGISTER_CMNQUERY_DLL,
szSystem,
STR_REGISTER_DSPROP_DLL);
ULONG WinError = 0;
HKEY RunOnceKey = NULL;
ULONG Size = 0;
ULONG Type = REG_SZ;
DWORD dwDisp = 0;
// open reg key
WinError = RegCreateKeyEx(HKEY_LOCAL_MACHINE,
RUNONCE_KEY,
0,
NULL,
REG_OPTION_NON_VOLATILE,
KEY_ALL_ACCESS,
NULL,
&RunOnceKey,
&dwDisp);
if (WinError == ERROR_SUCCESS)
{
UINT BufferSize = strlen(szCmdline);
BufferSize++;
WinError = RegSetValueEx( RunOnceKey,
REG_DSUI_VALUE,
0,
Type,
(PUCHAR)szCmdline,
BufferSize);
if (WinError != ERROR_SUCCESS && bSuccess)
{
bSuccess = FALSE;
}
//
// Run wabinst.exe
//
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API.
// Should consider using the strsafe inline APIs
wsprintf(szCmdline,
TEXT("%s%s"),
szSystem,
STR_RUN_WABINST_EXE);
if (!LaunchProcess(szCmdline))
{
bSuccess = FALSE;
}
}
else
{
bSuccess = FALSE;
}
if (RunOnceKey)
{
RegCloseKey(RunOnceKey);
}
}
return bSuccess;
}
// The custom registry action after dsclient.inf by Chandana Surlu
VOID DoDsclientReg()
{
ULONG WinError = 0;
HKEY ProvidersKey = NULL;
ULONG Size = 0;
ULONG Type = REG_SZ;
ULONG BufferSize = 0;
LPSTR StringToBeWritten = NULL;
DWORD dwDisp;
BOOL bSuccess = FALSE;
// open reg key
WinError = RegCreateKeyEx(HKEY_LOCAL_MACHINE,
SECURITY_PROVIDERS_KEY,
0,
NULL,
REG_OPTION_NON_VOLATILE,
KEY_ALL_ACCESS,
NULL,
&ProvidersKey,
&dwDisp);
if (WinError != ERROR_SUCCESS)
{
if (WinError == ERROR_FILE_NOT_FOUND)
{
BufferSize = sizeof(NEGOTIAT);
StringToBeWritten= (LPSTR) LocalAlloc(0,BufferSize);
if (StringToBeWritten)
{
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API.
// Should consider using the strsafe inline APIs
strcpy (StringToBeWritten, NEGOTIAT);
bSuccess = TRUE;
}
}
}
else
{
WinError = RegQueryValueEx(ProvidersKey,
SECURITY_PROVIDERS_VALUE,
0,
&Type,
NULL,
&Size);
if ( WinError == ERROR_SUCCESS)
{
BufferSize = Size + sizeof(COMMA_BLANK) + sizeof(NEGOTIAT);
StringToBeWritten= (LPSTR) LocalAlloc(0,BufferSize);
if (StringToBeWritten)
{
WinError = RegQueryValueEx(ProvidersKey,
SECURITY_PROVIDERS_VALUE,
0,
&Type,
(PUCHAR) StringToBeWritten,
&Size);
if ( WinError == ERROR_SUCCESS)
{
if (NULL == strstr(StringToBeWritten, NEGOTIAT))
{
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API.
// Should consider using the strsafe inline APIs
strcat (StringToBeWritten, COMMA_BLANK);
strcat (StringToBeWritten, NEGOTIAT);
bSuccess = TRUE;
}
}
}
}
else if (WinError == ERROR_FILE_NOT_FOUND)
{
BufferSize = sizeof(NEGOTIAT) + sizeof(CHAR);
StringToBeWritten= (LPSTR) LocalAlloc(0,BufferSize);
if (StringToBeWritten)
{
// ISSUE-2002/03/12-JeffJon-Bad use of dangerous API.
// Should consider using the strsafe inline APIs
strcpy (StringToBeWritten, NEGOTIAT);
Type = REG_SZ;
bSuccess = TRUE;
}
}
}
if(bSuccess)
{
BufferSize = strlen(StringToBeWritten);
BufferSize++;
WinError = RegSetValueEx( ProvidersKey,
SECURITY_PROVIDERS_VALUE,
0,
Type,
(PUCHAR)StringToBeWritten,
BufferSize);
}
if (ProvidersKey)
{
RegCloseKey(ProvidersKey);
}
if (StringToBeWritten)
{
LocalFree(StringToBeWritten);
}
}
// The NTLMv2 custom action before dsclient.inf installation.
// Calling encrypted schannel installer to create dynamically a 128 bit secur32.dll
// to replace the old 56 bit secur32.dll.
BOOL DoEncSChannel()
{
FPGETENCSCHANNEL fpEncSChannel;
HINSTANCE hInst;
BYTE* pFileData;
DWORD dwSize = 0;
HANDLE hFile;
DWORD dwWritten;
BOOL bRet;
// load "instsec.dll"
// ISSUE-2002/03/12-JeffJon-Should call LoadLibrary with
// the full path to the library since we are running on NT4
hInst = LoadLibrary(STR_INSTSEC_DLL);
if(!hInst)
return TRUE;
// get the pointer of function "GetEncSChannel"
fpEncSChannel = (FPGETENCSCHANNEL) GetProcAddress(hInst, STR_GETENCSCHANNEL);
// calling GetEncSChannel to get the file data
if( !fpEncSChannel ||
fpEncSChannel(&pFileData, &dwSize) == FALSE ||
dwSize == 0)
{
FreeLibrary( hInst );
return TRUE;
}
// create file - "secur32.dll"
// REVIEWED-2002/03/12-JeffJon-File permissions are the
// minimum required
hFile = CreateFile(
STR_SECUR32_DLL, // pointer to name of the file "secur32.dll"
GENERIC_WRITE, // access (read-write) mode
0, // share mode
NULL, // pointer to security attributes
CREATE_ALWAYS, // how to create
FILE_ATTRIBUTE_NORMAL, // file attributes
NULL // handle to file with attributes to copy
);
if(hFile == INVALID_HANDLE_VALUE)
{
VirtualFree(pFileData, 0, MEM_RELEASE);
FreeLibrary( hInst );
return FALSE;
}
// write the file data to file "secur32.dll"
bRet = WriteFile(
hFile, // handle to file to write to
pFileData, // pointer to data to write to file
dwSize, // number of bytes to write
&dwWritten, // pointer to number of bytes written
NULL // pointer to structure for overlapped I/O
);
if(bRet && dwSize != dwWritten)
bRet = FALSE;
// clean memory
VirtualFree(pFileData, 0, MEM_RELEASE);
CloseHandle( hFile );
FreeLibrary( hInst );
return bRet;
}