xref: /reactos/dll/win32/syssetup/install.c (revision 3b800165)
1 /*
2  * COPYRIGHT:         See COPYING in the top level directory
3  * PROJECT:           ReactOS system libraries
4  * PURPOSE:           System setup
5  * FILE:              dll/win32/syssetup/install.c
6  * PROGRAMER:         Eric Kohl
7  */
8 
9 /* INCLUDES *****************************************************************/
10 
11 #include "precomp.h"
12 
13 #define COBJMACROS
14 
15 #include <io.h>
16 #include <wincon.h>
17 #include <winnls.h>
18 #include <winsvc.h>
19 #include <userenv.h>
20 #include <shlobj.h>
21 #include <shlwapi.h>
22 #include <shobjidl.h>
23 #include <rpcproxy.h>
24 #include <ndk/cmfuncs.h>
25 
26 #define NDEBUG
27 #include <debug.h>
28 
29 //DWORD WINAPI
30 //CMP_WaitNoPendingInstallEvents(DWORD dwTimeout);
31 
32 DWORD WINAPI
33 SetupStartService(LPCWSTR lpServiceName, BOOL bWait);
34 
35 /* GLOBALS ******************************************************************/
36 
37 HINF hSysSetupInf = INVALID_HANDLE_VALUE;
38 ADMIN_INFO AdminInfo;
39 
40 typedef struct _DLG_DATA
41 {
42     HBITMAP hLogoBitmap;
43     HBITMAP hBarBitmap;
44     HWND hWndBarCtrl;
45     DWORD BarCounter;
46     DWORD BarWidth;
47     DWORD BarHeight;
48 } DLG_DATA, *PDLG_DATA;
49 
50 /* FUNCTIONS ****************************************************************/
51 
52 static VOID
FatalError(char * pszFmt,...)53 FatalError(char *pszFmt,...)
54 {
55     char szBuffer[512];
56     va_list ap;
57 
58     va_start(ap, pszFmt);
59     vsprintf(szBuffer, pszFmt, ap);
60     va_end(ap);
61 
62     LogItem(NULL, L"Failed");
63 
64     strcat(szBuffer, "\nRebooting now!");
65     MessageBoxA(NULL,
66                 szBuffer,
67                 "ReactOS Setup",
68                 MB_OK);
69 }
70 
71 static HRESULT
CreateShellLink(LPCWSTR pszLinkPath,LPCWSTR pszCmd,LPCWSTR pszArg,LPCWSTR pszDir,LPCWSTR pszIconPath,INT iIconNr,LPCWSTR pszComment)72 CreateShellLink(
73     LPCWSTR pszLinkPath,
74     LPCWSTR pszCmd,
75     LPCWSTR pszArg,
76     LPCWSTR pszDir,
77     LPCWSTR pszIconPath,
78     INT iIconNr,
79     LPCWSTR pszComment)
80 {
81     IShellLinkW *psl;
82     IPersistFile *ppf;
83 
84     HRESULT hr = CoCreateInstance(&CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, &IID_IShellLink, (LPVOID*)&psl);
85 
86     if (SUCCEEDED(hr))
87     {
88         hr = IShellLinkW_SetPath(psl, pszCmd);
89 
90         if (pszArg)
91             hr = IShellLinkW_SetArguments(psl, pszArg);
92 
93         if (pszDir)
94             hr = IShellLinkW_SetWorkingDirectory(psl, pszDir);
95 
96         if (pszIconPath)
97             hr = IShellLinkW_SetIconLocation(psl, pszIconPath, iIconNr);
98 
99         if (pszComment)
100             hr = IShellLinkW_SetDescription(psl, pszComment);
101 
102         hr = IShellLinkW_QueryInterface(psl, &IID_IPersistFile, (LPVOID*)&ppf);
103 
104         if (SUCCEEDED(hr))
105         {
106             hr = IPersistFile_Save(ppf, pszLinkPath, TRUE);
107             IPersistFile_Release(ppf);
108         }
109 
110         IShellLinkW_Release(psl);
111     }
112 
113     return hr;
114 }
115 
116 
117 static BOOL
CreateShortcut(LPCWSTR pszFolder,LPCWSTR pszName,LPCWSTR pszCommand,LPCWSTR pszDescription,INT iIconNr,LPCWSTR pszWorkingDir,LPCWSTR pszArgs)118 CreateShortcut(
119     LPCWSTR pszFolder,
120     LPCWSTR pszName,
121     LPCWSTR pszCommand,
122     LPCWSTR pszDescription,
123     INT iIconNr,
124     LPCWSTR pszWorkingDir,
125     LPCWSTR pszArgs)
126 {
127     DWORD dwLen;
128     LPWSTR Ptr;
129     LPWSTR lpFilePart;
130     WCHAR szPath[MAX_PATH];
131     WCHAR szWorkingDirBuf[MAX_PATH];
132 
133     /* If no working directory is provided, try to compute a default one */
134     if (pszWorkingDir == NULL || pszWorkingDir[0] == L'\0')
135     {
136         if (ExpandEnvironmentStringsW(pszCommand, szPath, ARRAYSIZE(szPath)) == 0)
137             wcscpy(szPath, pszCommand);
138 
139         dwLen = GetFullPathNameW(szPath,
140                                  ARRAYSIZE(szWorkingDirBuf),
141                                  szWorkingDirBuf,
142                                  &lpFilePart);
143         if (dwLen != 0 && dwLen <= ARRAYSIZE(szWorkingDirBuf))
144         {
145             /* Since those should only be called with (.exe) files,
146                lpFilePart has not to be NULL */
147             ASSERT(lpFilePart != NULL);
148 
149             /* We're only interested in the path. Cut the file name off.
150                Also remove the trailing backslash unless the working directory
151                is only going to be a drive, i.e. C:\ */
152             *(lpFilePart--) = L'\0';
153             if (!(lpFilePart - szWorkingDirBuf == 2 &&
154                   szWorkingDirBuf[1] == L':' && szWorkingDirBuf[2] == L'\\'))
155             {
156                 *lpFilePart = L'\0';
157             }
158             pszWorkingDir = szWorkingDirBuf;
159         }
160     }
161 
162     /* If we failed to compute a working directory, just do not use one */
163     if (pszWorkingDir && pszWorkingDir[0] == L'\0')
164         pszWorkingDir = NULL;
165 
166     /* Build the shortcut file name */
167     wcscpy(szPath, pszFolder);
168     Ptr = PathAddBackslash(szPath);
169     wcscpy(Ptr, pszName);
170 
171     /* Create the shortcut */
172     return SUCCEEDED(CreateShellLink(szPath,
173                                      pszCommand,
174                                      pszArgs,
175                                      pszWorkingDir,
176                                      /* Special value to indicate no icon */
177                                      (iIconNr != -1 ? pszCommand : NULL),
178                                      iIconNr,
179                                      pszDescription));
180 }
181 
182 
CreateShortcutsFromSection(HINF hinf,LPWSTR pszSection,LPCWSTR pszFolder)183 static BOOL CreateShortcutsFromSection(HINF hinf, LPWSTR pszSection, LPCWSTR pszFolder)
184 {
185     INFCONTEXT Context;
186     DWORD dwFieldCount;
187     INT iIconNr;
188     WCHAR szCommand[MAX_PATH];
189     WCHAR szName[MAX_PATH];
190     WCHAR szDescription[MAX_PATH];
191     WCHAR szDirectory[MAX_PATH];
192     WCHAR szArgs[MAX_PATH];
193 
194     if (!SetupFindFirstLine(hinf, pszSection, NULL, &Context))
195         return FALSE;
196 
197     do
198     {
199         dwFieldCount = SetupGetFieldCount(&Context);
200         if (dwFieldCount < 3)
201             continue;
202 
203         if (!SetupGetStringFieldW(&Context, 1, szCommand, ARRAYSIZE(szCommand), NULL))
204             continue;
205 
206         if (!SetupGetStringFieldW(&Context, 2, szName, ARRAYSIZE(szName), NULL))
207             continue;
208 
209         if (!SetupGetStringFieldW(&Context, 3, szDescription, ARRAYSIZE(szDescription), NULL))
210             continue;
211 
212         if (dwFieldCount < 4 || !SetupGetIntField(&Context, 4, &iIconNr))
213             iIconNr = -1; /* Special value to indicate no icon */
214 
215         if (dwFieldCount < 5 || !SetupGetStringFieldW(&Context, 5, szDirectory, ARRAYSIZE(szDirectory), NULL))
216             szDirectory[0] = L'\0';
217 
218         if (dwFieldCount < 6 || !SetupGetStringFieldW(&Context, 6, szArgs, ARRAYSIZE(szArgs), NULL))
219             szArgs[0] = L'\0';
220 
221         wcscat(szName, L".lnk");
222 
223         CreateShortcut(pszFolder, szName, szCommand, szDescription, iIconNr, szDirectory, szArgs);
224 
225     } while (SetupFindNextLine(&Context, &Context));
226 
227     return TRUE;
228 }
229 
CreateShortcuts(HINF hinf,LPCWSTR szSection)230 static BOOL CreateShortcuts(HINF hinf, LPCWSTR szSection)
231 {
232     INFCONTEXT Context;
233     WCHAR szPath[MAX_PATH];
234     WCHAR szFolder[MAX_PATH];
235     WCHAR szFolderSection[MAX_PATH];
236     INT csidl;
237 
238     CoInitialize(NULL);
239 
240     if (!SetupFindFirstLine(hinf, szSection, NULL, &Context))
241         return FALSE;
242 
243     do
244     {
245         if (SetupGetFieldCount(&Context) < 2)
246             continue;
247 
248         if (!SetupGetStringFieldW(&Context, 0, szFolderSection, ARRAYSIZE(szFolderSection), NULL))
249             continue;
250 
251         if (!SetupGetIntField(&Context, 1, &csidl))
252             continue;
253 
254         if (!SetupGetStringFieldW(&Context, 2, szFolder, ARRAYSIZE(szFolder), NULL))
255             continue;
256 
257         if (FAILED(SHGetFolderPathAndSubDirW(NULL, csidl|CSIDL_FLAG_CREATE, (HANDLE)-1, SHGFP_TYPE_DEFAULT, szFolder, szPath)))
258             continue;
259 
260         CreateShortcutsFromSection(hinf, szFolderSection, szPath);
261 
262     } while (SetupFindNextLine(&Context, &Context));
263 
264     CoUninitialize();
265 
266     return TRUE;
267 }
268 
269 static VOID
CreateTempDir(IN LPCWSTR VarName)270 CreateTempDir(
271     IN LPCWSTR VarName)
272 {
273     WCHAR szTempDir[MAX_PATH];
274     WCHAR szBuffer[MAX_PATH];
275     DWORD dwLength;
276     HKEY hKey;
277 
278     if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
279                       L"SYSTEM\\CurrentControlSet\\Control\\Session Manager\\Environment",
280                       0,
281                       KEY_QUERY_VALUE,
282                       &hKey) != ERROR_SUCCESS)
283     {
284         FatalError("Error: %lu\n", GetLastError());
285         return;
286     }
287 
288     /* Get temp dir */
289     dwLength = sizeof(szBuffer);
290     if (RegQueryValueExW(hKey,
291                          VarName,
292                          NULL,
293                          NULL,
294                          (LPBYTE)szBuffer,
295                          &dwLength) != ERROR_SUCCESS)
296     {
297         FatalError("Error: %lu\n", GetLastError());
298         goto cleanup;
299     }
300 
301     /* Expand it */
302     if (!ExpandEnvironmentStringsW(szBuffer, szTempDir, ARRAYSIZE(szTempDir)))
303     {
304         FatalError("Error: %lu\n", GetLastError());
305         goto cleanup;
306     }
307 
308     /* Create profiles directory */
309     if (!CreateDirectoryW(szTempDir, NULL))
310     {
311         if (GetLastError() != ERROR_ALREADY_EXISTS)
312         {
313             FatalError("Error: %lu\n", GetLastError());
314             goto cleanup;
315         }
316     }
317 
318 cleanup:
319     RegCloseKey(hKey);
320 }
321 
322 static BOOL
InstallSysSetupInfDevices(VOID)323 InstallSysSetupInfDevices(VOID)
324 {
325     INFCONTEXT InfContext;
326     WCHAR szLineBuffer[256];
327     DWORD dwLineLength;
328 
329     if (!SetupFindFirstLineW(hSysSetupInf,
330                             L"DeviceInfsToInstall",
331                             NULL,
332                             &InfContext))
333     {
334         return FALSE;
335     }
336 
337     do
338     {
339         if (!SetupGetStringFieldW(&InfContext,
340                                   0,
341                                   szLineBuffer,
342                                   ARRAYSIZE(szLineBuffer),
343                                   &dwLineLength))
344         {
345             return FALSE;
346         }
347 
348         if (!SetupDiInstallClassW(NULL, szLineBuffer, DI_QUIETINSTALL, NULL))
349         {
350             return FALSE;
351         }
352     }
353     while (SetupFindNextLine(&InfContext, &InfContext));
354 
355     return TRUE;
356 }
357 
358 static BOOL
InstallSysSetupInfComponents(VOID)359 InstallSysSetupInfComponents(VOID)
360 {
361     INFCONTEXT InfContext;
362     WCHAR szNameBuffer[256];
363     WCHAR szSectionBuffer[256];
364     HINF hComponentInf = INVALID_HANDLE_VALUE;
365 
366     if (!SetupFindFirstLineW(hSysSetupInf,
367                              L"Infs.Always",
368                              NULL,
369                              &InfContext))
370     {
371         DPRINT("No Inf.Always section found\n");
372     }
373     else
374     {
375         do
376         {
377             if (!SetupGetStringFieldW(&InfContext,
378                                       1, // Get the component name
379                                       szNameBuffer,
380                                       ARRAYSIZE(szNameBuffer),
381                                       NULL))
382             {
383                 FatalError("Error while trying to get component name\n");
384                 return FALSE;
385             }
386 
387             if (!SetupGetStringFieldW(&InfContext,
388                                       2, // Get the component install section
389                                       szSectionBuffer,
390                                       ARRAYSIZE(szSectionBuffer),
391                                       NULL))
392             {
393                 FatalError("Error while trying to get component install section\n");
394                 return FALSE;
395             }
396 
397             DPRINT("Trying to execute install section '%S' from '%S'\n", szSectionBuffer, szNameBuffer);
398 
399             hComponentInf = SetupOpenInfFileW(szNameBuffer,
400                                               NULL,
401                                               INF_STYLE_WIN4,
402                                               NULL);
403 
404             if (hComponentInf == INVALID_HANDLE_VALUE)
405             {
406                 FatalError("SetupOpenInfFileW() failed to open '%S' (Error: %lu)\n", szNameBuffer, GetLastError());
407                 return FALSE;
408             }
409 
410             if (!SetupInstallFromInfSectionW(NULL,
411                                              hComponentInf,
412                                              szSectionBuffer,
413                                              SPINST_ALL,
414                                              NULL,
415                                              NULL,
416                                              SP_COPY_NEWER,
417                                              SetupDefaultQueueCallbackW,
418                                              NULL,
419                                              NULL,
420                                              NULL))
421            {
422                 FatalError("Error while trying to install : %S (Error: %lu)\n", szNameBuffer, GetLastError());
423                 SetupCloseInfFile(hComponentInf);
424                 return FALSE;
425            }
426 
427            SetupCloseInfFile(hComponentInf);
428         }
429         while (SetupFindNextLine(&InfContext, &InfContext));
430     }
431 
432     return TRUE;
433 }
434 
435 
436 
437 BOOL
RegisterTypeLibraries(HINF hinf,LPCWSTR szSection)438 RegisterTypeLibraries(HINF hinf, LPCWSTR szSection)
439 {
440     INFCONTEXT InfContext;
441     BOOL res;
442     WCHAR szName[MAX_PATH];
443     WCHAR szPath[MAX_PATH];
444     INT csidl;
445     LPWSTR p;
446     HMODULE hmod;
447     HRESULT hret;
448 
449     /* Begin iterating the entries in the inf section */
450     res = SetupFindFirstLine(hinf, szSection, NULL, &InfContext);
451     if (!res) return FALSE;
452 
453     do
454     {
455         /* Get the name of the current type library */
456         if (!SetupGetStringFieldW(&InfContext, 1, szName, ARRAYSIZE(szName), NULL))
457         {
458             FatalError("SetupGetStringFieldW failed\n");
459             continue;
460         }
461 
462         if (!SetupGetIntField(&InfContext, 2, &csidl))
463             csidl = CSIDL_SYSTEM;
464 
465         hret = SHGetFolderPathW(NULL, csidl, NULL, 0, szPath);
466         if (FAILED(hret))
467         {
468             FatalError("SHGetFolderPathW failed hret=0x%lx\n", hret);
469             continue;
470         }
471 
472         p = PathAddBackslash(szPath);
473         wcscpy(p, szName);
474 
475         hmod = LoadLibraryW(szPath);
476         if (hmod == NULL)
477         {
478             FatalError("LoadLibraryW failed\n");
479             continue;
480         }
481 
482         __wine_register_resources(hmod);
483 
484     } while (SetupFindNextLine(&InfContext, &InfContext));
485 
486     return TRUE;
487 }
488 
489 static BOOL
EnableUserModePnpManager(VOID)490 EnableUserModePnpManager(VOID)
491 {
492     SC_HANDLE hSCManager = NULL;
493     SC_HANDLE hService = NULL;
494     SERVICE_STATUS_PROCESS ServiceStatus;
495     BOOL bRet = FALSE;
496     DWORD BytesNeeded, WaitTime;
497 
498     hSCManager = OpenSCManagerW(NULL, NULL, SC_MANAGER_ENUMERATE_SERVICE);
499     if (hSCManager == NULL)
500     {
501         DPRINT1("Unable to open the service control manager.\n");
502         DPRINT1("Last Error %d\n", GetLastError());
503         goto cleanup;
504     }
505 
506     hService = OpenServiceW(hSCManager,
507                             L"PlugPlay",
508                             SERVICE_CHANGE_CONFIG | SERVICE_START | SERVICE_QUERY_STATUS);
509     if (hService == NULL)
510     {
511         DPRINT1("Unable to open PlugPlay service\n");
512         goto cleanup;
513     }
514 
515     bRet = ChangeServiceConfigW(hService,
516                                 SERVICE_NO_CHANGE,
517                                 SERVICE_AUTO_START,
518                                 SERVICE_NO_CHANGE,
519                                 NULL, NULL, NULL,
520                                 NULL, NULL, NULL, NULL);
521     if (!bRet)
522     {
523         DPRINT1("Unable to change the service configuration\n");
524         goto cleanup;
525     }
526 
527     bRet = StartServiceW(hService, 0, NULL);
528     if (!bRet && (GetLastError() != ERROR_SERVICE_ALREADY_RUNNING))
529     {
530         DPRINT1("Unable to start service\n");
531         goto cleanup;
532     }
533 
534     while (TRUE)
535     {
536         bRet = QueryServiceStatusEx(hService,
537                                     SC_STATUS_PROCESS_INFO,
538                                     (LPBYTE)&ServiceStatus,
539                                     sizeof(ServiceStatus),
540                                     &BytesNeeded);
541         if (!bRet)
542         {
543             DPRINT1("QueryServiceStatusEx() failed for PlugPlay service (error 0x%x)\n", GetLastError());
544             goto cleanup;
545         }
546 
547         if (ServiceStatus.dwCurrentState != SERVICE_START_PENDING)
548             break;
549 
550         WaitTime = ServiceStatus.dwWaitHint / 10;
551         if (WaitTime < 1000) WaitTime = 1000;
552         else if (WaitTime > 10000) WaitTime = 10000;
553         Sleep(WaitTime);
554     };
555 
556     if (ServiceStatus.dwCurrentState != SERVICE_RUNNING)
557     {
558         bRet = FALSE;
559         DPRINT1("Failed to start PlugPlay service\n");
560         goto cleanup;
561     }
562 
563     bRet = TRUE;
564 
565 cleanup:
566     if (hService != NULL)
567         CloseServiceHandle(hService);
568     if (hSCManager != NULL)
569         CloseServiceHandle(hSCManager);
570     return bRet;
571 }
572 
573 static VOID
AdjustStatusMessageWindow(HWND hwndDlg,PDLG_DATA pDlgData)574 AdjustStatusMessageWindow(HWND hwndDlg, PDLG_DATA pDlgData)
575 {
576     INT xOld, yOld, cxOld, cyOld;
577     INT xNew, yNew, cxNew, cyNew;
578     INT cxLabel, cyLabel, dyLabel;
579     RECT rc, rcBar, rcLabel, rcWnd;
580     BITMAP bmLogo, bmBar;
581     DWORD style, exstyle;
582     HWND hwndLogo = GetDlgItem(hwndDlg, IDC_ROSLOGO);
583     HWND hwndBar = GetDlgItem(hwndDlg, IDC_BAR);
584     HWND hwndLabel = GetDlgItem(hwndDlg, IDC_STATUSLABEL);
585 
586     /* This adjustment is for CJK only */
587     switch (PRIMARYLANGID(GetUserDefaultLangID()))
588     {
589         case LANG_CHINESE:
590         case LANG_JAPANESE:
591         case LANG_KOREAN:
592             break;
593 
594         default:
595             return;
596     }
597 
598     if (!GetObjectW(pDlgData->hLogoBitmap, sizeof(BITMAP), &bmLogo) ||
599         !GetObjectW(pDlgData->hBarBitmap, sizeof(BITMAP), &bmBar))
600     {
601         return;
602     }
603 
604     GetWindowRect(hwndBar, &rcBar);
605     MapWindowPoints(NULL, hwndDlg, (LPPOINT)&rcBar, 2);
606     dyLabel = bmLogo.bmHeight - rcBar.top;
607 
608     GetWindowRect(hwndLabel, &rcLabel);
609     MapWindowPoints(NULL, hwndDlg, (LPPOINT)&rcLabel, 2);
610     cxLabel = rcLabel.right - rcLabel.left;
611     cyLabel = rcLabel.bottom - rcLabel.top;
612 
613     MoveWindow(hwndLogo, 0, 0, bmLogo.bmWidth, bmLogo.bmHeight, TRUE);
614     MoveWindow(hwndBar, 0, bmLogo.bmHeight, bmLogo.bmWidth, bmBar.bmHeight, TRUE);
615     MoveWindow(hwndLabel, rcLabel.left, rcLabel.top + dyLabel, cxLabel, cyLabel, TRUE);
616 
617     GetWindowRect(hwndDlg, &rcWnd);
618     xOld = rcWnd.left;
619     yOld = rcWnd.top;
620     cxOld = rcWnd.right - rcWnd.left;
621     cyOld = rcWnd.bottom - rcWnd.top;
622 
623     GetClientRect(hwndDlg, &rc);
624     SetRect(&rc, 0, 0, bmLogo.bmWidth, rc.bottom - rc.top); /* new client size */
625 
626     style = (DWORD)GetWindowLongPtrW(hwndDlg, GWL_STYLE);
627     exstyle = (DWORD)GetWindowLongPtrW(hwndDlg, GWL_EXSTYLE);
628     AdjustWindowRectEx(&rc, style, FALSE, exstyle);
629 
630     cxNew = rc.right - rc.left;
631     cyNew = (rc.bottom - rc.top) + dyLabel;
632     xNew = xOld - (cxNew - cxOld) / 2;
633     yNew = yOld - (cyNew - cyOld) / 2;
634     MoveWindow(hwndDlg, xNew, yNew, cxNew, cyNew, TRUE);
635 }
636 
637 static INT_PTR CALLBACK
StatusMessageWindowProc(IN HWND hwndDlg,IN UINT uMsg,IN WPARAM wParam,IN LPARAM lParam)638 StatusMessageWindowProc(
639     IN HWND hwndDlg,
640     IN UINT uMsg,
641     IN WPARAM wParam,
642     IN LPARAM lParam)
643 {
644     PDLG_DATA pDlgData;
645     UNREFERENCED_PARAMETER(wParam);
646 
647     pDlgData = (PDLG_DATA)GetWindowLongPtrW(hwndDlg, GWLP_USERDATA);
648 
649     /* pDlgData is required for each case except WM_INITDIALOG */
650     if (uMsg != WM_INITDIALOG && pDlgData == NULL) return FALSE;
651 
652     switch (uMsg)
653     {
654         case WM_INITDIALOG:
655         {
656             BITMAP bm;
657             WCHAR szMsg[256];
658 
659             /* Allocate pDlgData */
660             pDlgData = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*pDlgData));
661             if (pDlgData)
662             {
663                 /* Set pDlgData to GWLP_USERDATA, so we can get it for new messages */
664                 SetWindowLongPtrW(hwndDlg, GWLP_USERDATA, (LONG_PTR)pDlgData);
665 
666                 /* Load bitmaps */
667                 pDlgData->hLogoBitmap = LoadImageW(hDllInstance,
668                                                     MAKEINTRESOURCEW(IDB_REACTOS), IMAGE_BITMAP,
669                                                     0, 0, LR_DEFAULTCOLOR);
670 
671                 pDlgData->hBarBitmap = LoadImageW(hDllInstance, MAKEINTRESOURCEW(IDB_LINE),
672                                                 IMAGE_BITMAP, 0, 0, LR_DEFAULTCOLOR);
673                 GetObject(pDlgData->hBarBitmap, sizeof(bm), &bm);
674                 pDlgData->BarWidth = bm.bmWidth;
675                 pDlgData->BarHeight = bm.bmHeight;
676 
677                 if (pDlgData->hLogoBitmap && pDlgData->hBarBitmap)
678                 {
679                     if (SetTimer(hwndDlg, IDT_BAR, 20, NULL) == 0)
680                     {
681                         DPRINT1("SetTimer(IDT_BAR) failed: %lu\n", GetLastError());
682                     }
683 
684                     /* Get the animation bar control */
685                     pDlgData->hWndBarCtrl = GetDlgItem(hwndDlg, IDC_BAR);
686                 }
687             }
688 
689             /* Get and set status text */
690             if (!LoadStringW(hDllInstance, IDS_STATUS_INSTALL_DEV, szMsg, ARRAYSIZE(szMsg)))
691                 return FALSE;
692             SetDlgItemTextW(hwndDlg, IDC_STATUSLABEL, szMsg);
693 
694             AdjustStatusMessageWindow(hwndDlg, pDlgData);
695             return TRUE;
696         }
697 
698         case WM_TIMER:
699         {
700             if (pDlgData->hBarBitmap)
701             {
702                 /*
703                  * Default rotation bar image width is 413 (same as logo)
704                  * We can divide 413 by 7 without remainder
705                  */
706                 pDlgData->BarCounter = (pDlgData->BarCounter + 7) % pDlgData->BarWidth;
707                 InvalidateRect(pDlgData->hWndBarCtrl, NULL, FALSE);
708                 UpdateWindow(pDlgData->hWndBarCtrl);
709             }
710             return TRUE;
711         }
712 
713         case WM_DRAWITEM:
714         {
715             LPDRAWITEMSTRUCT lpDis = (LPDRAWITEMSTRUCT)lParam;
716 
717             if (lpDis->CtlID != IDC_BAR)
718             {
719                 return FALSE;
720             }
721 
722             if (pDlgData->hBarBitmap)
723             {
724                 HDC hdcMem;
725                 HGDIOBJ hOld;
726                 DWORD off = pDlgData->BarCounter;
727                 DWORD iw = pDlgData->BarWidth;
728                 DWORD ih = pDlgData->BarHeight;
729 
730                 hdcMem = CreateCompatibleDC(lpDis->hDC);
731                 hOld = SelectObject(hdcMem, pDlgData->hBarBitmap);
732                 BitBlt(lpDis->hDC, off, 0, iw - off, ih, hdcMem, 0, 0, SRCCOPY);
733                 BitBlt(lpDis->hDC, 0, 0, off, ih, hdcMem, iw - off, 0, SRCCOPY);
734                 SelectObject(hdcMem, hOld);
735                 DeleteDC(hdcMem);
736                 return TRUE;
737             }
738             return FALSE;
739         }
740 
741         case WM_DESTROY:
742         {
743             if (pDlgData->hBarBitmap)
744             {
745                 KillTimer(hwndDlg, IDT_BAR);
746             }
747 
748             DeleteObject(pDlgData->hLogoBitmap);
749             DeleteObject(pDlgData->hBarBitmap);
750             HeapFree(GetProcessHeap(), 0, pDlgData);
751             return TRUE;
752         }
753     }
754     return FALSE;
755 }
756 
757 static DWORD WINAPI
ShowStatusMessageThread(IN LPVOID lpParameter)758 ShowStatusMessageThread(
759     IN LPVOID lpParameter)
760 {
761     HWND hWnd;
762     MSG Msg;
763     UNREFERENCED_PARAMETER(lpParameter);
764 
765     hWnd = CreateDialogParam(hDllInstance,
766                              MAKEINTRESOURCE(IDD_STATUSWINDOW_DLG),
767                              GetDesktopWindow(),
768                              StatusMessageWindowProc,
769                              (LPARAM)NULL);
770     if (!hWnd)
771         return 0;
772 
773     ShowWindow(hWnd, SW_SHOW);
774 
775     /* Message loop for the Status window */
776     while (GetMessage(&Msg, NULL, 0, 0))
777     {
778         TranslateMessage(&Msg);
779         DispatchMessage(&Msg);
780     }
781 
782     EndDialog(hWnd, 0);
783 
784     return 0;
785 }
786 
787 static LONG
ReadRegSzKey(IN HKEY hKey,IN LPCWSTR pszKey,OUT LPWSTR * pValue)788 ReadRegSzKey(
789     IN HKEY hKey,
790     IN LPCWSTR pszKey,
791     OUT LPWSTR* pValue)
792 {
793     LONG rc;
794     DWORD dwType;
795     DWORD cbData = 0;
796     LPWSTR pwszValue;
797 
798     if (!pValue)
799         return ERROR_INVALID_PARAMETER;
800 
801     *pValue = NULL;
802     rc = RegQueryValueExW(hKey, pszKey, NULL, &dwType, NULL, &cbData);
803     if (rc != ERROR_SUCCESS)
804         return rc;
805     if (dwType != REG_SZ)
806         return ERROR_FILE_NOT_FOUND;
807     pwszValue = HeapAlloc(GetProcessHeap(), 0, cbData + sizeof(WCHAR));
808     if (!pwszValue)
809         return ERROR_NOT_ENOUGH_MEMORY;
810     rc = RegQueryValueExW(hKey, pszKey, NULL, NULL, (LPBYTE)pwszValue, &cbData);
811     if (rc != ERROR_SUCCESS)
812     {
813         HeapFree(GetProcessHeap(), 0, pwszValue);
814         return rc;
815     }
816     /* NULL-terminate the string */
817     pwszValue[cbData / sizeof(WCHAR)] = '\0';
818 
819     *pValue = pwszValue;
820     return ERROR_SUCCESS;
821 }
822 
823 static BOOL
IsConsoleBoot(VOID)824 IsConsoleBoot(VOID)
825 {
826     HKEY hControlKey = NULL;
827     LPWSTR pwszSystemStartOptions = NULL;
828     LPWSTR pwszCurrentOption, pwszNextOption; /* Pointers into SystemStartOptions */
829     BOOL bConsoleBoot = FALSE;
830     LONG rc;
831 
832     rc = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
833                        L"SYSTEM\\CurrentControlSet\\Control",
834                        0,
835                        KEY_QUERY_VALUE,
836                        &hControlKey);
837     if (rc != ERROR_SUCCESS)
838         goto cleanup;
839 
840     rc = ReadRegSzKey(hControlKey, L"SystemStartOptions", &pwszSystemStartOptions);
841     if (rc != ERROR_SUCCESS)
842         goto cleanup;
843 
844     /* Check for CONSOLE switch in SystemStartOptions */
845     pwszCurrentOption = pwszSystemStartOptions;
846     while (pwszCurrentOption)
847     {
848         pwszNextOption = wcschr(pwszCurrentOption, L' ');
849         if (pwszNextOption)
850             *pwszNextOption = L'\0';
851         if (_wcsicmp(pwszCurrentOption, L"CONSOLE") == 0)
852         {
853             DPRINT("Found %S. Switching to console boot\n", pwszCurrentOption);
854             bConsoleBoot = TRUE;
855             goto cleanup;
856         }
857         pwszCurrentOption = pwszNextOption ? pwszNextOption + 1 : NULL;
858     }
859 
860 cleanup:
861     if (hControlKey != NULL)
862         RegCloseKey(hControlKey);
863     if (pwszSystemStartOptions)
864         HeapFree(GetProcessHeap(), 0, pwszSystemStartOptions);
865     return bConsoleBoot;
866 }
867 
868 static BOOL
CommonInstall(VOID)869 CommonInstall(VOID)
870 {
871     HANDLE hThread = NULL;
872     BOOL bResult = FALSE;
873 
874     hSysSetupInf = SetupOpenInfFileW(L"syssetup.inf",
875                                      NULL,
876                                      INF_STYLE_WIN4,
877                                      NULL);
878     if (hSysSetupInf == INVALID_HANDLE_VALUE)
879     {
880         FatalError("SetupOpenInfFileW() failed to open 'syssetup.inf' (Error: %lu)\n", GetLastError());
881         return FALSE;
882     }
883 
884     if (!InstallSysSetupInfDevices())
885     {
886         FatalError("InstallSysSetupInfDevices() failed!\n");
887         goto Exit;
888     }
889 
890     if(!InstallSysSetupInfComponents())
891     {
892         FatalError("InstallSysSetupInfComponents() failed!\n");
893         goto Exit;
894     }
895 
896     if (!IsConsoleBoot())
897     {
898         hThread = CreateThread(NULL,
899                                0,
900                                ShowStatusMessageThread,
901                                NULL,
902                                0,
903                                NULL);
904     }
905 
906     if (!EnableUserModePnpManager())
907     {
908         FatalError("EnableUserModePnpManager() failed!\n");
909         goto Exit;
910     }
911 
912     if (CMP_WaitNoPendingInstallEvents(INFINITE) != WAIT_OBJECT_0)
913     {
914         FatalError("CMP_WaitNoPendingInstallEvents() failed!\n");
915         goto Exit;
916     }
917 
918     bResult = TRUE;
919 
920 Exit:
921 
922     if (bResult == FALSE)
923     {
924         SetupCloseInfFile(hSysSetupInf);
925     }
926 
927     if (hThread != NULL)
928     {
929         PostThreadMessage(GetThreadId(hThread), WM_QUIT, 0, 0);
930         WaitForSingleObject(hThread, INFINITE);
931         CloseHandle(hThread);
932     }
933 
934     return bResult;
935 }
936 
937 static
938 DWORD
InstallLiveCD(VOID)939 InstallLiveCD(VOID)
940 {
941     STARTUPINFOW StartupInfo;
942     PROCESS_INFORMATION ProcessInformation;
943     BOOL bRes;
944 
945     if (!CommonInstall())
946         goto error;
947 
948     /* Install the TCP/IP protocol driver */
949     bRes = InstallNetworkComponent(L"MS_TCPIP");
950     if (!bRes && GetLastError() != ERROR_FILE_NOT_FOUND)
951     {
952         DPRINT("InstallNetworkComponent() failed with error 0x%lx\n", GetLastError());
953     }
954     else
955     {
956         /* Start the TCP/IP protocol driver */
957         SetupStartService(L"Tcpip", FALSE);
958         SetupStartService(L"Dhcp", FALSE);
959         SetupStartService(L"Dnscache", FALSE);
960     }
961 
962     /* Register components */
963     _SEH2_TRY
964     {
965         if (!SetupInstallFromInfSectionW(NULL,
966                                          hSysSetupInf, L"RegistrationPhase2",
967                                          SPINST_ALL,
968                                          0, NULL, 0, NULL, NULL, NULL, NULL))
969         {
970             DPRINT1("SetupInstallFromInfSectionW failed!\n");
971         }
972 
973         RegisterTypeLibraries(hSysSetupInf, L"TypeLibraries");
974     }
975     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
976     {
977         DPRINT1("Catching exception\n");
978     }
979     _SEH2_END;
980 
981     SetupCloseInfFile(hSysSetupInf);
982 
983     /* Run the shell */
984     ZeroMemory(&StartupInfo, sizeof(StartupInfo));
985     StartupInfo.cb = sizeof(StartupInfo);
986     bRes = CreateProcessW(L"userinit.exe",
987                           NULL,
988                           NULL,
989                           NULL,
990                           FALSE,
991                           0,
992                           NULL,
993                           NULL,
994                           &StartupInfo,
995                           &ProcessInformation);
996     if (!bRes)
997         goto error;
998 
999     CloseHandle(ProcessInformation.hThread);
1000     CloseHandle(ProcessInformation.hProcess);
1001 
1002     return 0;
1003 
1004 error:
1005     MessageBoxW(
1006         NULL,
1007         L"Failed to load LiveCD! You can shutdown your computer, or press ENTER to reboot.",
1008         L"ReactOS LiveCD",
1009         MB_OK);
1010     return 0;
1011 }
1012 
1013 
1014 static BOOL
SetSetupType(DWORD dwSetupType)1015 SetSetupType(DWORD dwSetupType)
1016 {
1017     DWORD dwError;
1018     HKEY hKey;
1019 
1020     dwError = RegOpenKeyExW(
1021         HKEY_LOCAL_MACHINE,
1022         L"SYSTEM\\Setup",
1023         0,
1024         KEY_SET_VALUE,
1025         &hKey);
1026     if (dwError != ERROR_SUCCESS)
1027         return FALSE;
1028 
1029     dwError = RegSetValueExW(
1030         hKey,
1031         L"SetupType",
1032         0,
1033         REG_DWORD,
1034         (LPBYTE)&dwSetupType,
1035         sizeof(DWORD));
1036     RegCloseKey(hKey);
1037     if (dwError != ERROR_SUCCESS)
1038         return FALSE;
1039 
1040     return TRUE;
1041 }
1042 
1043 static DWORD CALLBACK
HotkeyThread(LPVOID Parameter)1044 HotkeyThread(LPVOID Parameter)
1045 {
1046     ATOM hotkey;
1047     MSG msg;
1048 
1049     DPRINT("HotkeyThread start\n");
1050 
1051     hotkey = GlobalAddAtomW(L"Setup Shift+F10 Hotkey");
1052     if (!RegisterHotKey(NULL, hotkey, MOD_SHIFT, VK_F10))
1053         DPRINT1("RegisterHotKey failed with %lu\n", GetLastError());
1054 
1055     while (GetMessageW(&msg, NULL, 0, 0))
1056     {
1057         if (msg.hwnd == NULL && msg.message == WM_HOTKEY && msg.wParam == hotkey)
1058         {
1059             WCHAR CmdLine[] = L"cmd.exe"; // CreateProcess can modify this buffer.
1060             STARTUPINFOW si = { sizeof(si) };
1061             PROCESS_INFORMATION pi;
1062 
1063             if (CreateProcessW(NULL,
1064                                CmdLine,
1065                                NULL,
1066                                NULL,
1067                                FALSE,
1068                                CREATE_NEW_CONSOLE,
1069                                NULL,
1070                                NULL,
1071                                &si,
1072                                &pi))
1073             {
1074                 CloseHandle(pi.hProcess);
1075                 CloseHandle(pi.hThread);
1076             }
1077             else
1078             {
1079                 DPRINT1("Failed to launch command prompt: %lu\n", GetLastError());
1080             }
1081         }
1082     }
1083 
1084     UnregisterHotKey(NULL, hotkey);
1085     GlobalDeleteAtom(hotkey);
1086 
1087     DPRINT("HotkeyThread terminate\n");
1088     return 0;
1089 }
1090 
1091 
1092 static
1093 BOOL
InitializeProgramFilesDir(VOID)1094 InitializeProgramFilesDir(VOID)
1095 {
1096     LONG Error;
1097     HKEY hKey;
1098     DWORD dwLength;
1099     WCHAR szProgramFilesDirPath[MAX_PATH];
1100     WCHAR szCommonFilesDirPath[MAX_PATH];
1101     WCHAR szBuffer[MAX_PATH];
1102 
1103     /* Load 'Program Files' location */
1104     if (!LoadStringW(hDllInstance,
1105                      IDS_PROGRAMFILES,
1106                      szBuffer,
1107                      ARRAYSIZE(szBuffer)))
1108     {
1109         DPRINT1("Error: %lu\n", GetLastError());
1110         return FALSE;
1111     }
1112 
1113     if (!LoadStringW(hDllInstance,
1114                      IDS_COMMONFILES,
1115                      szCommonFilesDirPath,
1116                      ARRAYSIZE(szCommonFilesDirPath)))
1117     {
1118         DPRINT1("Warning: %lu\n", GetLastError());
1119     }
1120 
1121     /* Expand it */
1122     if (!ExpandEnvironmentStringsW(szBuffer,
1123                                    szProgramFilesDirPath,
1124                                    ARRAYSIZE(szProgramFilesDirPath)))
1125     {
1126         DPRINT1("Error: %lu\n", GetLastError());
1127         return FALSE;
1128     }
1129 
1130     wcscpy(szBuffer, szProgramFilesDirPath);
1131     wcscat(szBuffer, L"\\");
1132     wcscat(szBuffer, szCommonFilesDirPath);
1133 
1134     if (!ExpandEnvironmentStringsW(szBuffer,
1135                                    szCommonFilesDirPath,
1136                                    ARRAYSIZE(szCommonFilesDirPath)))
1137     {
1138         DPRINT1("Warning: %lu\n", GetLastError());
1139     }
1140 
1141     /* Store it */
1142     Error = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
1143                           L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion",
1144                           0,
1145                           KEY_SET_VALUE,
1146                           &hKey);
1147     if (Error != ERROR_SUCCESS)
1148     {
1149         DPRINT1("Error: %lu\n", Error);
1150         return FALSE;
1151     }
1152 
1153     dwLength = (wcslen(szProgramFilesDirPath) + 1) * sizeof(WCHAR);
1154     Error = RegSetValueExW(hKey,
1155                            L"ProgramFilesDir",
1156                            0,
1157                            REG_SZ,
1158                            (LPBYTE)szProgramFilesDirPath,
1159                            dwLength);
1160     if (Error != ERROR_SUCCESS)
1161     {
1162         DPRINT1("Error: %lu\n", Error);
1163         RegCloseKey(hKey);
1164         return FALSE;
1165     }
1166 
1167     dwLength = (wcslen(szCommonFilesDirPath) + 1) * sizeof(WCHAR);
1168     Error = RegSetValueExW(hKey,
1169                            L"CommonFilesDir",
1170                            0,
1171                            REG_SZ,
1172                            (LPBYTE)szCommonFilesDirPath,
1173                            dwLength);
1174     if (Error != ERROR_SUCCESS)
1175     {
1176         DPRINT1("Warning: %lu\n", Error);
1177     }
1178 
1179     RegCloseKey(hKey);
1180 
1181     /* Create directory */
1182     // FIXME: Security!
1183     if (!CreateDirectoryW(szProgramFilesDirPath, NULL))
1184     {
1185         if (GetLastError() != ERROR_ALREADY_EXISTS)
1186         {
1187             DPRINT1("Error: %lu\n", GetLastError());
1188             return FALSE;
1189         }
1190     }
1191 
1192     /* Create directory */
1193     // FIXME: Security!
1194     if (!CreateDirectoryW(szCommonFilesDirPath, NULL))
1195     {
1196         if (GetLastError() != ERROR_ALREADY_EXISTS)
1197         {
1198             DPRINT1("Warning: %lu\n", GetLastError());
1199             // return FALSE;
1200         }
1201     }
1202 
1203     return TRUE;
1204 }
1205 
1206 
1207 static
1208 VOID
InitializeDefaultUserLocale(VOID)1209 InitializeDefaultUserLocale(VOID)
1210 {
1211     WCHAR szBuffer[80];
1212     PWSTR ptr;
1213     HKEY hLocaleKey;
1214     DWORD ret;
1215     DWORD dwSize;
1216     LCID lcid;
1217     INT i;
1218 
1219     struct {LCTYPE LCType; PWSTR pValue;} LocaleData[] = {
1220         /* Number */
1221         {LOCALE_SDECIMAL, L"sDecimal"},
1222         {LOCALE_STHOUSAND, L"sThousand"},
1223         {LOCALE_SNEGATIVESIGN, L"sNegativeSign"},
1224         {LOCALE_SPOSITIVESIGN, L"sPositiveSign"},
1225         {LOCALE_SGROUPING, L"sGrouping"},
1226         {LOCALE_SLIST, L"sList"},
1227         {LOCALE_SNATIVEDIGITS, L"sNativeDigits"},
1228         {LOCALE_INEGNUMBER, L"iNegNumber"},
1229         {LOCALE_IDIGITS, L"iDigits"},
1230         {LOCALE_ILZERO, L"iLZero"},
1231         {LOCALE_IMEASURE, L"iMeasure"},
1232         {LOCALE_IDIGITSUBSTITUTION, L"NumShape"},
1233 
1234         /* Currency */
1235         {LOCALE_SCURRENCY, L"sCurrency"},
1236         {LOCALE_SMONDECIMALSEP, L"sMonDecimalSep"},
1237         {LOCALE_SMONTHOUSANDSEP, L"sMonThousandSep"},
1238         {LOCALE_SMONGROUPING, L"sMonGrouping"},
1239         {LOCALE_ICURRENCY, L"iCurrency"},
1240         {LOCALE_INEGCURR, L"iNegCurr"},
1241         {LOCALE_ICURRDIGITS, L"iCurrDigits"},
1242 
1243         /* Time */
1244         {LOCALE_STIMEFORMAT, L"sTimeFormat"},
1245         {LOCALE_STIME, L"sTime"},
1246         {LOCALE_S1159, L"s1159"},
1247         {LOCALE_S2359, L"s2359"},
1248         {LOCALE_ITIME, L"iTime"},
1249         {LOCALE_ITIMEMARKPOSN, L"iTimePrefix"},
1250         {LOCALE_ITLZERO, L"iTLZero"},
1251 
1252         /* Date */
1253         {LOCALE_SLONGDATE, L"sLongDate"},
1254         {LOCALE_SSHORTDATE, L"sShortDate"},
1255         {LOCALE_SDATE, L"sDate"},
1256         {LOCALE_IFIRSTDAYOFWEEK, L"iFirstDayOfWeek"},
1257         {LOCALE_IFIRSTWEEKOFYEAR, L"iFirstWeekOfYear"},
1258         {LOCALE_IDATE, L"iDate"},
1259         {LOCALE_ICALENDARTYPE, L"iCalendarType"},
1260 
1261         /* Misc */
1262         {LOCALE_SCOUNTRY, L"sCountry"},
1263         {LOCALE_SABBREVLANGNAME, L"sLanguage"},
1264         {LOCALE_ICOUNTRY, L"iCountry"},
1265         {0, NULL}};
1266 
1267     ret = RegOpenKeyExW(HKEY_USERS,
1268                         L".DEFAULT\\Control Panel\\International",
1269                         0,
1270                         KEY_READ | KEY_WRITE,
1271                         &hLocaleKey);
1272     if (ret != ERROR_SUCCESS)
1273     {
1274         return;
1275     }
1276 
1277     dwSize = 9 * sizeof(WCHAR);
1278     ret = RegQueryValueExW(hLocaleKey,
1279                            L"Locale",
1280                            NULL,
1281                            NULL,
1282                            (PBYTE)szBuffer,
1283                            &dwSize);
1284     if (ret != ERROR_SUCCESS)
1285         goto done;
1286 
1287     lcid = (LCID)wcstoul(szBuffer, &ptr, 16);
1288     if (lcid == 0)
1289         goto done;
1290 
1291     i = 0;
1292     while (LocaleData[i].pValue != NULL)
1293     {
1294         if (GetLocaleInfoW(lcid,
1295                            LocaleData[i].LCType | LOCALE_NOUSEROVERRIDE,
1296                            szBuffer,
1297                            ARRAYSIZE(szBuffer)))
1298         {
1299             RegSetValueExW(hLocaleKey,
1300                            LocaleData[i].pValue,
1301                            0,
1302                            REG_SZ,
1303                            (PBYTE)szBuffer,
1304                            (wcslen(szBuffer) + 1) * sizeof(WCHAR));
1305         }
1306 
1307         i++;
1308     }
1309 
1310 done:
1311     RegCloseKey(hLocaleKey);
1312 }
1313 
1314 
1315 static
1316 DWORD
SaveDefaultUserHive(VOID)1317 SaveDefaultUserHive(VOID)
1318 {
1319     WCHAR szDefaultUserHive[MAX_PATH];
1320     HKEY hUserKey = NULL;
1321     DWORD cchSize;
1322     DWORD dwError;
1323 
1324     DPRINT("SaveDefaultUserHive()\n");
1325 
1326     cchSize = ARRAYSIZE(szDefaultUserHive);
1327     GetDefaultUserProfileDirectoryW(szDefaultUserHive, &cchSize);
1328 
1329     wcscat(szDefaultUserHive, L"\\ntuser.dat");
1330 
1331     dwError = RegOpenKeyExW(HKEY_USERS,
1332                             L".DEFAULT",
1333                             0,
1334                             KEY_READ,
1335                             &hUserKey);
1336     if (dwError != ERROR_SUCCESS)
1337     {
1338         DPRINT1("RegOpenKeyExW() failed (Error %lu)\n", dwError);
1339         return dwError;
1340     }
1341 
1342     pSetupEnablePrivilege(L"SeBackupPrivilege", TRUE);
1343 
1344     /* Save the Default hive */
1345     dwError = RegSaveKeyExW(hUserKey,
1346                             szDefaultUserHive,
1347                             NULL,
1348                             REG_STANDARD_FORMAT);
1349     if (dwError == ERROR_ALREADY_EXISTS)
1350     {
1351         WCHAR szBackupHive[MAX_PATH];
1352 
1353         /* Build the backup hive file name by replacing the extension */
1354         wcscpy(szBackupHive, szDefaultUserHive);
1355         wcscpy(&szBackupHive[wcslen(szBackupHive) - 4], L".bak");
1356 
1357         /* Back up the existing default user hive by renaming it, replacing any possible existing old backup */
1358         if (!MoveFileExW(szDefaultUserHive,
1359                          szBackupHive,
1360                          MOVEFILE_REPLACE_EXISTING))
1361         {
1362             dwError = GetLastError();
1363             DPRINT1("Failed to create a default-user hive backup '%S', MoveFileExW failed (Error %lu)\n",
1364                     szBackupHive, dwError);
1365         }
1366         else
1367         {
1368             /* The backup has been done, retry saving the Default hive */
1369             dwError = RegSaveKeyExW(hUserKey,
1370                                     szDefaultUserHive,
1371                                     NULL,
1372                                     REG_STANDARD_FORMAT);
1373         }
1374     }
1375     if (dwError != ERROR_SUCCESS)
1376     {
1377         DPRINT1("RegSaveKeyExW() failed (Error %lu)\n", dwError);
1378     }
1379 
1380     pSetupEnablePrivilege(L"SeBackupPrivilege", FALSE);
1381 
1382     RegCloseKey(hUserKey);
1383 
1384     return dwError;
1385 }
1386 
1387 
1388 static
1389 DWORD
InstallReactOS(VOID)1390 InstallReactOS(VOID)
1391 {
1392     WCHAR szBuffer[MAX_PATH];
1393     HANDLE token;
1394     TOKEN_PRIVILEGES privs;
1395     HKEY hKey;
1396     HINF hShortcutsInf;
1397     HANDLE hHotkeyThread;
1398     BOOL ret;
1399 
1400     InitializeSetupActionLog(FALSE);
1401     LogItem(NULL, L"Installing ReactOS");
1402 
1403     CreateTempDir(L"TEMP");
1404     CreateTempDir(L"TMP");
1405 
1406     if (!InitializeProgramFilesDir())
1407     {
1408         FatalError("InitializeProgramFilesDir() failed");
1409         return 0;
1410     }
1411 
1412     if (!InitializeProfiles())
1413     {
1414         FatalError("InitializeProfiles() failed");
1415         return 0;
1416     }
1417 
1418     InitializeDefaultUserLocale();
1419 
1420     if (GetWindowsDirectoryW(szBuffer, ARRAYSIZE(szBuffer)))
1421     {
1422         if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
1423                           L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion",
1424                           0,
1425                           KEY_WRITE,
1426                           &hKey) == ERROR_SUCCESS)
1427         {
1428             RegSetValueExW(hKey,
1429                            L"PathName",
1430                            0,
1431                            REG_SZ,
1432                            (LPBYTE)szBuffer,
1433                            (wcslen(szBuffer) + 1) * sizeof(WCHAR));
1434 
1435             RegSetValueExW(hKey,
1436                            L"SystemRoot",
1437                            0,
1438                            REG_SZ,
1439                            (LPBYTE)szBuffer,
1440                            (wcslen(szBuffer) + 1) * sizeof(WCHAR));
1441 
1442             RegCloseKey(hKey);
1443         }
1444 
1445         PathAddBackslash(szBuffer);
1446         wcscat(szBuffer, L"system");
1447         CreateDirectory(szBuffer, NULL);
1448     }
1449 
1450     if (SaveDefaultUserHive() != ERROR_SUCCESS)
1451     {
1452         FatalError("SaveDefaultUserHive() failed");
1453         return 0;
1454     }
1455 
1456     if (!CopySystemProfile(0))
1457     {
1458         FatalError("CopySystemProfile() failed");
1459         return 0;
1460     }
1461 
1462     hHotkeyThread = CreateThread(NULL, 0, HotkeyThread, NULL, 0, NULL);
1463 
1464     if (!CommonInstall())
1465         return 0;
1466 
1467     /* Install the TCP/IP protocol driver */
1468     ret = InstallNetworkComponent(L"MS_TCPIP");
1469     if (!ret && GetLastError() != ERROR_FILE_NOT_FOUND)
1470     {
1471         DPRINT("InstallNetworkComponent() failed with error 0x%lx\n", GetLastError());
1472     }
1473     else
1474     {
1475         /* Start the TCP/IP protocol driver */
1476         SetupStartService(L"Tcpip", FALSE);
1477         SetupStartService(L"Dhcp", FALSE);
1478         SetupStartService(L"Dnscache", FALSE);
1479     }
1480 
1481     InstallWizard();
1482 
1483     InstallSecurity();
1484 
1485     SetAutoAdminLogon();
1486 
1487     hShortcutsInf = SetupOpenInfFileW(L"shortcuts.inf",
1488                                       NULL,
1489                                       INF_STYLE_WIN4,
1490                                       NULL);
1491     if (hShortcutsInf == INVALID_HANDLE_VALUE)
1492     {
1493         FatalError("Failed to open shortcuts.inf");
1494         return 0;
1495     }
1496 
1497     if (!CreateShortcuts(hShortcutsInf, L"ShortcutFolders"))
1498     {
1499         FatalError("CreateShortcuts() failed");
1500         return 0;
1501     }
1502 
1503     SetupCloseInfFile(hShortcutsInf);
1504 
1505     hShortcutsInf = SetupOpenInfFileW(L"rosapps_shortcuts.inf",
1506                                        NULL,
1507                                        INF_STYLE_WIN4,
1508                                        NULL);
1509     if (hShortcutsInf != INVALID_HANDLE_VALUE)
1510     {
1511         if (!CreateShortcuts(hShortcutsInf, L"ShortcutFolders"))
1512         {
1513             FatalError("CreateShortcuts(rosapps) failed");
1514             return 0;
1515         }
1516         SetupCloseInfFile(hShortcutsInf);
1517     }
1518 
1519     SetupCloseInfFile(hSysSetupInf);
1520     SetSetupType(0);
1521 
1522     if (hHotkeyThread)
1523     {
1524         PostThreadMessage(GetThreadId(hHotkeyThread), WM_QUIT, 0, 0);
1525         CloseHandle(hHotkeyThread);
1526     }
1527 
1528     LogItem(NULL, L"Installing ReactOS done");
1529     TerminateSetupActionLog();
1530 
1531     if (AdminInfo.Name != NULL)
1532         RtlFreeHeap(RtlGetProcessHeap(), 0, AdminInfo.Name);
1533 
1534     if (AdminInfo.Domain != NULL)
1535         RtlFreeHeap(RtlGetProcessHeap(), 0, AdminInfo.Domain);
1536 
1537     if (AdminInfo.Password != NULL)
1538         RtlFreeHeap(RtlGetProcessHeap(), 0, AdminInfo.Password);
1539 
1540     /* Get shutdown privilege */
1541     if (!OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &token))
1542     {
1543         FatalError("OpenProcessToken() failed!");
1544         return 0;
1545     }
1546     if (!LookupPrivilegeValue(NULL,
1547                               SE_SHUTDOWN_NAME,
1548                               &privs.Privileges[0].Luid))
1549     {
1550         FatalError("LookupPrivilegeValue() failed!");
1551         return 0;
1552     }
1553     privs.PrivilegeCount = 1;
1554     privs.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
1555     if (AdjustTokenPrivileges(token,
1556                               FALSE,
1557                               &privs,
1558                               0,
1559                               (PTOKEN_PRIVILEGES)NULL,
1560                               NULL) == 0)
1561     {
1562         FatalError("AdjustTokenPrivileges() failed!");
1563         return 0;
1564     }
1565 
1566     ExitWindowsEx(EWX_REBOOT, 0);
1567     return 0;
1568 }
1569 
1570 
1571 /*
1572  * Standard Windows-compatible export, which dispatches
1573  * to either 'InstallReactOS' or 'InstallLiveCD'.
1574  */
1575 INT
1576 WINAPI
InstallWindowsNt(INT argc,WCHAR ** argv)1577 InstallWindowsNt(INT argc, WCHAR** argv)
1578 {
1579     INT i;
1580     PWSTR p;
1581 
1582     for (i = 0; i < argc; ++i)
1583     {
1584         p = argv[i];
1585         if (*p == L'-')
1586         {
1587             p++;
1588 
1589             // NOTE: On Windows, "mini" means "minimal UI", and can be used
1590             // in addition to "newsetup"; these options are not exclusive.
1591             if (_wcsicmp(p, L"newsetup") == 0)
1592                 return (INT)InstallReactOS();
1593             else if (_wcsicmp(p, L"mini") == 0)
1594                 return (INT)InstallLiveCD();
1595 
1596             /* Add support for other switches */
1597         }
1598     }
1599 
1600     return 0;
1601 }
1602 
1603 
1604 /*
1605  * @unimplemented
1606  */
1607 DWORD WINAPI
SetupChangeFontSize(IN HANDLE hWnd,IN LPCWSTR lpszFontSize)1608 SetupChangeFontSize(
1609     IN HANDLE hWnd,
1610     IN LPCWSTR lpszFontSize)
1611 {
1612     SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
1613     return FALSE;
1614 }
1615 
1616 /*
1617  * @unimplemented
1618  */
1619 DWORD WINAPI
SetupChangeLocaleEx(HWND hWnd,LCID Lcid,LPCWSTR lpSrcRootPath,char Unknown,DWORD dwUnused1,DWORD dwUnused2)1620 SetupChangeLocaleEx(HWND hWnd,
1621                     LCID Lcid,
1622                     LPCWSTR lpSrcRootPath,
1623                     char Unknown,
1624                     DWORD dwUnused1,
1625                     DWORD dwUnused2)
1626 {
1627     SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
1628     return FALSE;
1629 }
1630 
1631 /*
1632  * @implemented
1633  */
1634 DWORD WINAPI
SetupChangeLocale(HWND hWnd,LCID Lcid)1635 SetupChangeLocale(HWND hWnd, LCID Lcid)
1636 {
1637     return SetupChangeLocaleEx(hWnd, Lcid, NULL, 0, 0, 0);
1638 }
1639 
1640 
1641 DWORD
1642 WINAPI
SetupStartService(LPCWSTR lpServiceName,BOOL bWait)1643 SetupStartService(
1644     LPCWSTR lpServiceName,
1645     BOOL bWait)
1646 {
1647     SC_HANDLE hManager = NULL;
1648     SC_HANDLE hService = NULL;
1649     DWORD dwError = ERROR_SUCCESS;
1650 
1651     hManager = OpenSCManagerW(NULL,
1652                               NULL,
1653                               SC_MANAGER_ALL_ACCESS);
1654     if (hManager == NULL)
1655     {
1656         dwError = GetLastError();
1657         goto done;
1658     }
1659 
1660     hService = OpenServiceW(hManager,
1661                             lpServiceName,
1662                             SERVICE_START);
1663     if (hService == NULL)
1664     {
1665         dwError = GetLastError();
1666         goto done;
1667     }
1668 
1669     if (!StartService(hService, 0, NULL))
1670     {
1671         dwError = GetLastError();
1672         goto done;
1673     }
1674 
1675 done:
1676     if (hService != NULL)
1677         CloseServiceHandle(hService);
1678 
1679     if (hManager != NULL)
1680         CloseServiceHandle(hManager);
1681 
1682     return dwError;
1683 }
1684