//+------------------------------------------------------------------ // // 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 #include #include #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; }