xref: /reactos/ntoskrnl/ex/init.c (revision 8a978a17)
1 /*
2  * PROJECT:         ReactOS Kernel
3  * LICENSE:         GPL - See COPYING in the top level directory
4  * FILE:            ntoskrnl/ex/init.c
5  * PURPOSE:         Executive Initialization Code
6  * PROGRAMMERS:     Alex Ionescu (alex.ionescu@reactos.org)
7  *                  Eric Kohl
8  */
9 
10 /* INCLUDES ******************************************************************/
11 
12 #include <ntoskrnl.h>
13 #include <reactos/buildno.h>
14 #define NDEBUG
15 #include <debug.h>
16 
17 /* This is the size that we can expect from the win 2003 loader */
18 #define LOADER_PARAMETER_EXTENSION_MIN_SIZE \
19     RTL_SIZEOF_THROUGH_FIELD(LOADER_PARAMETER_EXTENSION, AcpiTableSize)
20 
21 /* Temporary hack */
22 CODE_SEG("INIT")
23 BOOLEAN
24 NTAPI
25 MmArmInitSystem(
26     IN ULONG Phase,
27     IN PLOADER_PARAMETER_BLOCK LoaderBlock
28 );
29 
30 typedef struct _INIT_BUFFER
31 {
32     WCHAR DebugBuffer[256];
33     CHAR VersionBuffer[256];
34     CHAR BootlogHeader[256];
35     CHAR VersionNumber[24];
36     RTL_USER_PROCESS_INFORMATION ProcessInfo;
37     WCHAR RegistryBuffer[256];
38 } INIT_BUFFER, *PINIT_BUFFER;
39 
40 /* DATA **********************************************************************/
41 
42 /* NT Version Info */
43 ULONG NtMajorVersion = VER_PRODUCTMAJORVERSION;
44 ULONG NtMinorVersion = VER_PRODUCTMINORVERSION;
45 #if DBG /* Checked Build */
46 ULONG NtBuildNumber = VER_PRODUCTBUILD | 0xC0000000;
47 #else   /* Free Build */
48 ULONG NtBuildNumber = VER_PRODUCTBUILD;
49 #endif
50 
51 /* NT System Info */
52 ULONG NtGlobalFlag = 0;
53 ULONG ExSuiteMask;
54 
55 /* Cm Version Info */
56 ULONG CmNtSpBuildNumber;
57 ULONG CmNtCSDVersion;
58 ULONG CmNtCSDReleaseType;
59 UNICODE_STRING CmVersionString;
60 UNICODE_STRING CmCSDVersionString;
61 
62 CHAR NtBuildLab[] = KERNEL_VERSION_BUILD_STR "."
63                     REACTOS_COMPILER_NAME "_" REACTOS_COMPILER_VERSION;
64 
65 /* Init flags and settings */
66 ULONG ExpInitializationPhase;
67 BOOLEAN ExpInTextModeSetup;
68 BOOLEAN IoRemoteBootClient;
69 ULONG InitSafeBootMode;
70 BOOLEAN InitIsWinPEMode, InitWinPEModeType;
71 
72 /* NT Boot Path */
73 UNICODE_STRING NtSystemRoot;
74 
75 /* NT Initial User Application */
76 WCHAR NtInitialUserProcessBuffer[128] = L"\\SystemRoot\\System32\\smss.exe";
77 ULONG NtInitialUserProcessBufferLength = sizeof(NtInitialUserProcessBuffer) -
78                                          sizeof(WCHAR);
79 ULONG NtInitialUserProcessBufferType = REG_SZ;
80 
81 /* Boot NLS information */
82 PVOID ExpNlsTableBase;
83 ULONG ExpAnsiCodePageDataOffset, ExpOemCodePageDataOffset;
84 ULONG ExpUnicodeCaseTableDataOffset;
85 NLSTABLEINFO ExpNlsTableInfo;
86 SIZE_T ExpNlsTableSize;
87 PVOID ExpNlsSectionPointer;
88 
89 /* CMOS Timer Sanity */
90 BOOLEAN ExCmosClockIsSane = TRUE;
91 BOOLEAN ExpRealTimeIsUniversal;
92 
93 /* FUNCTIONS ****************************************************************/
94 
95 CODE_SEG("INIT")
96 NTSTATUS
97 NTAPI
98 ExpCreateSystemRootLink(IN PLOADER_PARAMETER_BLOCK LoaderBlock)
99 {
100     UNICODE_STRING LinkName;
101     OBJECT_ATTRIBUTES ObjectAttributes;
102     HANDLE LinkHandle;
103     NTSTATUS Status;
104     ANSI_STRING AnsiName;
105     CHAR Buffer[256];
106     ANSI_STRING TargetString;
107     UNICODE_STRING TargetName;
108 
109     /* Initialize the ArcName tree */
110     RtlInitUnicodeString(&LinkName, L"\\ArcName");
111     InitializeObjectAttributes(&ObjectAttributes,
112                                &LinkName,
113                                OBJ_CASE_INSENSITIVE | OBJ_PERMANENT,
114                                NULL,
115                                SePublicDefaultUnrestrictedSd);
116 
117     /* Create it */
118     Status = NtCreateDirectoryObject(&LinkHandle,
119                                      DIRECTORY_ALL_ACCESS,
120                                      &ObjectAttributes);
121     if (!NT_SUCCESS(Status))
122     {
123         /* Failed */
124         KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED, Status, 1, 0, 0);
125     }
126 
127     /* Close the LinkHandle */
128     NtClose(LinkHandle);
129 
130     /* Initialize the Device tree */
131     RtlInitUnicodeString(&LinkName, L"\\Device");
132     InitializeObjectAttributes(&ObjectAttributes,
133                                &LinkName,
134                                OBJ_CASE_INSENSITIVE | OBJ_PERMANENT,
135                                NULL,
136                                SePublicDefaultUnrestrictedSd);
137 
138     /* Create it */
139     Status = NtCreateDirectoryObject(&LinkHandle,
140                                      DIRECTORY_ALL_ACCESS,
141                                      &ObjectAttributes);
142     if (!NT_SUCCESS(Status))
143     {
144         /* Failed */
145         KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED, Status, 2, 0, 0);
146     }
147 
148     /* Close the LinkHandle */
149     ObCloseHandle(LinkHandle, KernelMode);
150 
151     /* Create the system root symlink name */
152     RtlInitAnsiString(&AnsiName, "\\SystemRoot");
153     Status = RtlAnsiStringToUnicodeString(&LinkName, &AnsiName, TRUE);
154     if (!NT_SUCCESS(Status))
155     {
156         /* Failed */
157         KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED, Status, 3, 0, 0);
158     }
159 
160     /* Initialize the attributes for the link */
161     InitializeObjectAttributes(&ObjectAttributes,
162                                &LinkName,
163                                OBJ_CASE_INSENSITIVE | OBJ_PERMANENT,
164                                NULL,
165                                SePublicDefaultUnrestrictedSd);
166 
167     /* Build the ARC name */
168     sprintf(Buffer,
169             "\\ArcName\\%s%s",
170             LoaderBlock->ArcBootDeviceName,
171             LoaderBlock->NtBootPathName);
172     Buffer[strlen(Buffer) - 1] = ANSI_NULL;
173 
174     /* Convert it to Unicode */
175     RtlInitString(&TargetString, Buffer);
176     Status = RtlAnsiStringToUnicodeString(&TargetName,
177                                           &TargetString,
178                                           TRUE);
179     if (!NT_SUCCESS(Status))
180     {
181         /* We failed, bugcheck */
182         KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED, Status, 4, 0, 0);
183     }
184 
185     /* Create it */
186     Status = NtCreateSymbolicLinkObject(&LinkHandle,
187                                         SYMBOLIC_LINK_ALL_ACCESS,
188                                         &ObjectAttributes,
189                                         &TargetName);
190 
191     /* Free the strings */
192     RtlFreeUnicodeString(&LinkName);
193     RtlFreeUnicodeString(&TargetName);
194 
195     /* Check if creating the link failed */
196     if (!NT_SUCCESS(Status))
197     {
198         /* Failed */
199         KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED, Status, 5, 0, 0);
200     }
201 
202     /* Close the handle and return success */
203     ObCloseHandle(LinkHandle, KernelMode);
204     return STATUS_SUCCESS;
205 }
206 
207 CODE_SEG("INIT")
208 VOID
209 NTAPI
210 ExpInitNls(IN PLOADER_PARAMETER_BLOCK LoaderBlock)
211 {
212     LARGE_INTEGER SectionSize;
213     NTSTATUS Status;
214     HANDLE NlsSection;
215     PVOID SectionBase = NULL;
216     SIZE_T ViewSize = 0;
217     LARGE_INTEGER SectionOffset = {{0, 0}};
218     PLIST_ENTRY ListHead, NextEntry;
219     PMEMORY_ALLOCATION_DESCRIPTOR MdBlock;
220     ULONG NlsTablesEncountered = 0;
221     SIZE_T NlsTableSizes[3] = {0, 0, 0}; /* 3 NLS tables */
222 
223     /* Check if this is boot-time phase 0 initialization */
224     if (!ExpInitializationPhase)
225     {
226         /* Loop the memory descriptors */
227         ListHead = &LoaderBlock->MemoryDescriptorListHead;
228         NextEntry = ListHead->Flink;
229         while (NextEntry != ListHead)
230         {
231             /* Get the current block */
232             MdBlock = CONTAINING_RECORD(NextEntry,
233                                         MEMORY_ALLOCATION_DESCRIPTOR,
234                                         ListEntry);
235 
236             /* Check if this is an NLS block */
237             if (MdBlock->MemoryType == LoaderNlsData)
238             {
239                 /* Increase the table size */
240                 ExpNlsTableSize += MdBlock->PageCount * PAGE_SIZE;
241 
242                 /* FreeLdr-specific */
243                 NlsTableSizes[NlsTablesEncountered] = MdBlock->PageCount * PAGE_SIZE;
244                 NlsTablesEncountered++;
245                 ASSERT(NlsTablesEncountered < 4);
246             }
247 
248             /* Go to the next block */
249             NextEntry = MdBlock->ListEntry.Flink;
250         }
251 
252         /* Allocate the a new buffer since loader memory will be freed */
253         ExpNlsTableBase = ExAllocatePoolWithTag(NonPagedPool,
254                                                 ExpNlsTableSize,
255                                                 TAG_RTLI);
256         if (!ExpNlsTableBase) KeBugCheck(PHASE0_INITIALIZATION_FAILED);
257 
258         /* Copy the codepage data in its new location. */
259         if (NlsTablesEncountered == 1)
260         {
261             /* Ntldr-way boot process */
262             RtlCopyMemory(ExpNlsTableBase,
263                           LoaderBlock->NlsData->AnsiCodePageData,
264                           ExpNlsTableSize);
265         }
266         else
267         {
268             /*
269             * In NT, the memory blocks are contiguous, but in ReactOS they aren't,
270             * so unless someone fixes FreeLdr, we'll have to use this icky hack.
271             */
272             RtlCopyMemory(ExpNlsTableBase,
273                           LoaderBlock->NlsData->AnsiCodePageData,
274                           NlsTableSizes[0]);
275 
276             RtlCopyMemory((PVOID)((ULONG_PTR)ExpNlsTableBase + NlsTableSizes[0]),
277                           LoaderBlock->NlsData->OemCodePageData,
278                           NlsTableSizes[1]);
279 
280             RtlCopyMemory((PVOID)((ULONG_PTR)ExpNlsTableBase + NlsTableSizes[0] +
281                           NlsTableSizes[1]),
282                           LoaderBlock->NlsData->UnicodeCodePageData,
283                           NlsTableSizes[2]);
284             /* End of Hack */
285         }
286 
287         /* Initialize and reset the NLS TAbles */
288         RtlInitNlsTables((PVOID)((ULONG_PTR)ExpNlsTableBase +
289                                  ExpAnsiCodePageDataOffset),
290                          (PVOID)((ULONG_PTR)ExpNlsTableBase +
291                                  ExpOemCodePageDataOffset),
292                          (PVOID)((ULONG_PTR)ExpNlsTableBase +
293                                  ExpUnicodeCaseTableDataOffset),
294                          &ExpNlsTableInfo);
295         RtlResetRtlTranslations(&ExpNlsTableInfo);
296         return;
297     }
298 
299     /* Set the section size */
300     SectionSize.QuadPart = ExpNlsTableSize;
301 
302     /* Create the NLS Section */
303     Status = ZwCreateSection(&NlsSection,
304                              SECTION_ALL_ACCESS,
305                              NULL,
306                              &SectionSize,
307                              PAGE_READWRITE,
308                              SEC_COMMIT | 0x1,
309                              NULL);
310     if (!NT_SUCCESS(Status))
311     {
312         /* Failed */
313         KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 1, 0, 0);
314     }
315 
316     /* Get a pointer to the section */
317     Status = ObReferenceObjectByHandle(NlsSection,
318                                        SECTION_ALL_ACCESS,
319                                        MmSectionObjectType,
320                                        KernelMode,
321                                        &ExpNlsSectionPointer,
322                                        NULL);
323     ObCloseHandle(NlsSection, KernelMode);
324     if (!NT_SUCCESS(Status))
325     {
326         /* Failed */
327         KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 2, 0, 0);
328     }
329 
330     /* Map the NLS Section in system space */
331     Status = MmMapViewInSystemSpace(ExpNlsSectionPointer,
332                                     &SectionBase,
333                                     &ExpNlsTableSize);
334     if (!NT_SUCCESS(Status))
335     {
336         /* Failed */
337         KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 3, 0, 0);
338     }
339 
340     /* Copy the codepage data in its new location. */
341     ASSERT(SectionBase >= MmSystemRangeStart);
342     RtlCopyMemory(SectionBase, ExpNlsTableBase, ExpNlsTableSize);
343 
344     /* Free the previously allocated buffer and set the new location */
345     ExFreePoolWithTag(ExpNlsTableBase, TAG_RTLI);
346     ExpNlsTableBase = SectionBase;
347 
348     /* Initialize the NLS Tables */
349     RtlInitNlsTables((PVOID)((ULONG_PTR)ExpNlsTableBase +
350                              ExpAnsiCodePageDataOffset),
351                      (PVOID)((ULONG_PTR)ExpNlsTableBase +
352                              ExpOemCodePageDataOffset),
353                      (PVOID)((ULONG_PTR)ExpNlsTableBase +
354                              ExpUnicodeCaseTableDataOffset),
355                      &ExpNlsTableInfo);
356     RtlResetRtlTranslations(&ExpNlsTableInfo);
357 
358     /* Reset the base to 0 */
359     SectionBase = NULL;
360 
361     /* Map the section in the system process */
362     Status = MmMapViewOfSection(ExpNlsSectionPointer,
363                                 PsGetCurrentProcess(),
364                                 &SectionBase,
365                                 0L,
366                                 0L,
367                                 &SectionOffset,
368                                 &ViewSize,
369                                 ViewShare,
370                                 0L,
371                                 PAGE_READWRITE);
372     if (!NT_SUCCESS(Status))
373     {
374         /* Failed */
375         KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 5, 0, 0);
376     }
377 
378     /* Copy the table into the system process and set this as the base */
379     RtlCopyMemory(SectionBase, ExpNlsTableBase, ExpNlsTableSize);
380     ExpNlsTableBase = SectionBase;
381 }
382 
383 CODE_SEG("INIT")
384 VOID
385 NTAPI
386 ExpLoadInitialProcess(IN PINIT_BUFFER InitBuffer,
387                       OUT PRTL_USER_PROCESS_PARAMETERS *ProcessParameters,
388                       OUT PCHAR *ProcessEnvironment)
389 {
390     NTSTATUS Status;
391     SIZE_T Size;
392     PWSTR p;
393     UNICODE_STRING NullString = RTL_CONSTANT_STRING(L"");
394     UNICODE_STRING SmssName, Environment, SystemDriveString, DebugString;
395     PVOID EnvironmentPtr = NULL;
396     PRTL_USER_PROCESS_INFORMATION ProcessInformation;
397     PRTL_USER_PROCESS_PARAMETERS ProcessParams = NULL;
398 
399     NullString.Length = sizeof(WCHAR);
400 
401     /* Use the initial buffer, after the strings */
402     ProcessInformation = &InitBuffer->ProcessInfo;
403 
404     /* Allocate memory for the process parameters */
405     Size = sizeof(*ProcessParams) + ((MAX_WIN32_PATH * 6) * sizeof(WCHAR));
406     Status = ZwAllocateVirtualMemory(NtCurrentProcess(),
407                                      (PVOID*)&ProcessParams,
408                                      0,
409                                      &Size,
410                                      MEM_RESERVE | MEM_COMMIT,
411                                      PAGE_READWRITE);
412     if (!NT_SUCCESS(Status))
413     {
414         /* Failed, display error */
415         _snwprintf(InitBuffer->DebugBuffer,
416                    sizeof(InitBuffer->DebugBuffer)/sizeof(WCHAR),
417                    L"INIT: Unable to allocate Process Parameters. 0x%lx",
418                    Status);
419         RtlInitUnicodeString(&DebugString, InitBuffer->DebugBuffer);
420         ZwDisplayString(&DebugString);
421 
422         /* Bugcheck the system */
423         KeBugCheckEx(SESSION1_INITIALIZATION_FAILED, Status, 0, 0, 0);
424     }
425 
426     /* Setup the basic header, and give the process the low 1MB to itself */
427     ProcessParams->Length = (ULONG)Size;
428     ProcessParams->MaximumLength = (ULONG)Size;
429     ProcessParams->Flags = RTL_USER_PROCESS_PARAMETERS_NORMALIZED |
430                            RTL_USER_PROCESS_PARAMETERS_RESERVE_1MB;
431 
432     /* Allocate a page for the environment */
433     Size = PAGE_SIZE;
434     Status = ZwAllocateVirtualMemory(NtCurrentProcess(),
435                                      &EnvironmentPtr,
436                                      0,
437                                      &Size,
438                                      MEM_RESERVE | MEM_COMMIT,
439                                      PAGE_READWRITE);
440     if (!NT_SUCCESS(Status))
441     {
442         /* Failed, display error */
443         _snwprintf(InitBuffer->DebugBuffer,
444                    sizeof(InitBuffer->DebugBuffer)/sizeof(WCHAR),
445                    L"INIT: Unable to allocate Process Environment. 0x%lx",
446                    Status);
447         RtlInitUnicodeString(&DebugString, InitBuffer->DebugBuffer);
448         ZwDisplayString(&DebugString);
449 
450         /* Bugcheck the system */
451         KeBugCheckEx(SESSION2_INITIALIZATION_FAILED, Status, 0, 0, 0);
452     }
453 
454     /* Write the pointer */
455     ProcessParams->Environment = EnvironmentPtr;
456 
457     /* Make a buffer for the DOS path */
458     p = (PWSTR)(ProcessParams + 1);
459     ProcessParams->CurrentDirectory.DosPath.Buffer = p;
460     ProcessParams->CurrentDirectory.DosPath.MaximumLength = MAX_WIN32_PATH *
461                                                             sizeof(WCHAR);
462 
463     /* Copy the DOS path */
464     RtlCopyUnicodeString(&ProcessParams->CurrentDirectory.DosPath,
465                          &NtSystemRoot);
466 
467     /* Make a buffer for the DLL Path */
468     p = (PWSTR)((PCHAR)ProcessParams->CurrentDirectory.DosPath.Buffer +
469                 ProcessParams->CurrentDirectory.DosPath.MaximumLength);
470     ProcessParams->DllPath.Buffer = p;
471     ProcessParams->DllPath.MaximumLength = MAX_WIN32_PATH * sizeof(WCHAR);
472 
473     /* Copy the DLL path and append the system32 directory */
474     RtlCopyUnicodeString(&ProcessParams->DllPath,
475                          &ProcessParams->CurrentDirectory.DosPath);
476     RtlAppendUnicodeToString(&ProcessParams->DllPath, L"\\System32");
477 
478     /* Make a buffer for the image name */
479     p = (PWSTR)((PCHAR)ProcessParams->DllPath.Buffer +
480                 ProcessParams->DllPath.MaximumLength);
481     ProcessParams->ImagePathName.Buffer = p;
482     ProcessParams->ImagePathName.MaximumLength = MAX_WIN32_PATH * sizeof(WCHAR);
483 
484     /* Make sure the buffer is a valid string which within the given length */
485     if ((NtInitialUserProcessBufferType != REG_SZ) ||
486         ((NtInitialUserProcessBufferLength != MAXULONG) &&
487          ((NtInitialUserProcessBufferLength < sizeof(WCHAR)) ||
488           (NtInitialUserProcessBufferLength >
489            sizeof(NtInitialUserProcessBuffer) - sizeof(WCHAR)))))
490     {
491         /* Invalid initial process string, bugcheck */
492         KeBugCheckEx(SESSION2_INITIALIZATION_FAILED,
493                      STATUS_INVALID_PARAMETER,
494                      NtInitialUserProcessBufferType,
495                      NtInitialUserProcessBufferLength,
496                      sizeof(NtInitialUserProcessBuffer));
497     }
498 
499     /* Cut out anything after a space */
500     p = NtInitialUserProcessBuffer;
501     while ((*p) && (*p != L' ')) p++;
502 
503     /* Set the image path length */
504     ProcessParams->ImagePathName.Length =
505         (USHORT)((PCHAR)p - (PCHAR)NtInitialUserProcessBuffer);
506 
507     /* Copy the actual buffer */
508     RtlCopyMemory(ProcessParams->ImagePathName.Buffer,
509                   NtInitialUserProcessBuffer,
510                   ProcessParams->ImagePathName.Length);
511 
512     /* Null-terminate it */
513     ProcessParams->ImagePathName.Buffer[ProcessParams->ImagePathName.Length /
514                                         sizeof(WCHAR)] = UNICODE_NULL;
515 
516     /* Make a buffer for the command line */
517     p = (PWSTR)((PCHAR)ProcessParams->ImagePathName.Buffer +
518                 ProcessParams->ImagePathName.MaximumLength);
519     ProcessParams->CommandLine.Buffer = p;
520     ProcessParams->CommandLine.MaximumLength = MAX_WIN32_PATH * sizeof(WCHAR);
521 
522     /* Add the image name to the command line */
523     RtlAppendUnicodeToString(&ProcessParams->CommandLine,
524                              NtInitialUserProcessBuffer);
525 
526     /* Create the environment string */
527     RtlInitEmptyUnicodeString(&Environment,
528                               ProcessParams->Environment,
529                               (USHORT)Size);
530 
531     /* Append the DLL path to it */
532     RtlAppendUnicodeToString(&Environment, L"Path=");
533     RtlAppendUnicodeStringToString(&Environment, &ProcessParams->DllPath);
534     RtlAppendUnicodeStringToString(&Environment, &NullString);
535 
536     /* Create the system drive string */
537     SystemDriveString = NtSystemRoot;
538     SystemDriveString.Length = 2 * sizeof(WCHAR);
539 
540     /* Append it to the environment */
541     RtlAppendUnicodeToString(&Environment, L"SystemDrive=");
542     RtlAppendUnicodeStringToString(&Environment, &SystemDriveString);
543     RtlAppendUnicodeStringToString(&Environment, &NullString);
544 
545     /* Append the system root to the environment */
546     RtlAppendUnicodeToString(&Environment, L"SystemRoot=");
547     RtlAppendUnicodeStringToString(&Environment, &NtSystemRoot);
548     RtlAppendUnicodeStringToString(&Environment, &NullString);
549 
550     /* Prepare the prefetcher */
551     //CcPfBeginBootPhase(150);
552 
553     /* Create SMSS process */
554     SmssName = ProcessParams->ImagePathName;
555     Status = RtlCreateUserProcess(&SmssName,
556                                   OBJ_CASE_INSENSITIVE,
557                                   RtlDeNormalizeProcessParams(ProcessParams),
558                                   NULL,
559                                   NULL,
560                                   NULL,
561                                   FALSE,
562                                   NULL,
563                                   NULL,
564                                   ProcessInformation);
565     if (!NT_SUCCESS(Status))
566     {
567         /* Failed, display error */
568         _snwprintf(InitBuffer->DebugBuffer,
569                    sizeof(InitBuffer->DebugBuffer)/sizeof(WCHAR),
570                    L"INIT: Unable to create Session Manager. 0x%lx",
571                    Status);
572         RtlInitUnicodeString(&DebugString, InitBuffer->DebugBuffer);
573         ZwDisplayString(&DebugString);
574 
575         /* Bugcheck the system */
576         KeBugCheckEx(SESSION3_INITIALIZATION_FAILED, Status, 0, 0, 0);
577     }
578 
579     /* Resume the thread */
580     Status = ZwResumeThread(ProcessInformation->ThreadHandle, NULL);
581     if (!NT_SUCCESS(Status))
582     {
583         /* Failed, display error */
584         _snwprintf(InitBuffer->DebugBuffer,
585                    sizeof(InitBuffer->DebugBuffer)/sizeof(WCHAR),
586                    L"INIT: Unable to resume Session Manager. 0x%lx",
587                    Status);
588         RtlInitUnicodeString(&DebugString, InitBuffer->DebugBuffer);
589         ZwDisplayString(&DebugString);
590 
591         /* Bugcheck the system */
592         KeBugCheckEx(SESSION4_INITIALIZATION_FAILED, Status, 0, 0, 0);
593     }
594 
595     /* Return success */
596     *ProcessParameters = ProcessParams;
597     *ProcessEnvironment = EnvironmentPtr;
598 }
599 
600 CODE_SEG("INIT")
601 ULONG
602 NTAPI
603 ExComputeTickCountMultiplier(IN ULONG ClockIncrement)
604 {
605     ULONG MsRemainder = 0, MsIncrement;
606     ULONG IncrementRemainder;
607     ULONG i;
608 
609     /* Count the number of milliseconds for each clock interrupt */
610     MsIncrement = ClockIncrement / (10 * 1000);
611 
612     /* Count the remainder from the division above, with 24-bit precision */
613     IncrementRemainder = ClockIncrement - (MsIncrement * (10 * 1000));
614     for (i= 0; i < 24; i++)
615     {
616         /* Shift the remainders */
617         MsRemainder <<= 1;
618         IncrementRemainder <<= 1;
619 
620         /* Check if we've went past 1 ms */
621         if (IncrementRemainder >= (10 * 1000))
622         {
623             /* Increase the remainder by one, and substract from increment */
624             IncrementRemainder -= (10 * 1000);
625             MsRemainder |= 1;
626         }
627     }
628 
629     /* Return the increment */
630     return (MsIncrement << 24) | MsRemainder;
631 }
632 
633 CODE_SEG("INIT")
634 BOOLEAN
635 NTAPI
636 ExpInitSystemPhase0(VOID)
637 {
638     /* Initialize EXRESOURCE Support */
639     ExpResourceInitialization();
640 
641     /* Initialize the environment lock */
642     ExInitializeFastMutex(&ExpEnvironmentLock);
643 
644     /* Initialize the lookaside lists and locks */
645     ExpInitLookasideLists();
646 
647     /* Initialize the Firmware Table resource and listhead */
648     InitializeListHead(&ExpFirmwareTableProviderListHead);
649     ExInitializeResourceLite(&ExpFirmwareTableResource);
650 
651     /* Set the suite mask to maximum and return */
652     ExSuiteMask = 0xFFFFFFFF;
653     return TRUE;
654 }
655 
656 CODE_SEG("INIT")
657 BOOLEAN
658 NTAPI
659 ExpInitSystemPhase1(VOID)
660 {
661     /* Initialize worker threads */
662     ExpInitializeWorkerThreads();
663 
664     /* Initialize pushlocks */
665     ExpInitializePushLocks();
666 
667     /* Initialize events and event pairs */
668     if (ExpInitializeEventImplementation() == FALSE)
669     {
670         DPRINT1("Executive: Event initialization failed\n");
671         return FALSE;
672     }
673     if (ExpInitializeEventPairImplementation() == FALSE)
674     {
675         DPRINT1("Executive: Event Pair initialization failed\n");
676         return FALSE;
677     }
678 
679     /* Initialize mutants */
680     if (ExpInitializeMutantImplementation() == FALSE)
681     {
682         DPRINT1("Executive: Mutant initialization failed\n");
683         return FALSE;
684     }
685 
686     /* Initialize callbacks */
687     if (ExpInitializeCallbacks() == FALSE)
688     {
689         DPRINT1("Executive: Callback initialization failed\n");
690         return FALSE;
691     }
692 
693     /* Initialize semaphores */
694     if (ExpInitializeSemaphoreImplementation() == FALSE)
695     {
696         DPRINT1("Executive: Semaphore initialization failed\n");
697         return FALSE;
698     }
699 
700     /* Initialize timers */
701     if (ExpInitializeTimerImplementation() == FALSE)
702     {
703         DPRINT1("Executive: Timer initialization failed\n");
704         return FALSE;
705     }
706 
707     /* Initialize profiling */
708     if (ExpInitializeProfileImplementation() == FALSE)
709     {
710         DPRINT1("Executive: Profile initialization failed\n");
711         return FALSE;
712     }
713 
714     /* Initialize UUIDs */
715     if (ExpUuidInitialization() == FALSE)
716     {
717         DPRINT1("Executive: Uuid initialization failed\n");
718         return FALSE;
719     }
720 
721     /* Initialize keyed events */
722     if (ExpInitializeKeyedEventImplementation() == FALSE)
723     {
724         DPRINT1("Executive: Keyed event initialization failed\n");
725         return FALSE;
726     }
727 
728     /* Initialize Win32K */
729     if (ExpWin32kInit() == FALSE)
730     {
731         DPRINT1("Executive: Win32 initialization failed\n");
732         return FALSE;
733     }
734     return TRUE;
735 }
736 
737 CODE_SEG("INIT")
738 BOOLEAN
739 NTAPI
740 ExInitSystem(VOID)
741 {
742     /* Check the initialization phase */
743     switch (ExpInitializationPhase)
744     {
745         case 0:
746 
747             /* Do Phase 0 */
748             return ExpInitSystemPhase0();
749 
750         case 1:
751 
752             /* Do Phase 1 */
753             return ExpInitSystemPhase1();
754 
755         default:
756 
757             /* Don't know any other phase! Bugcheck! */
758             KeBugCheck(UNEXPECTED_INITIALIZATION_CALL);
759             return FALSE;
760     }
761 }
762 
763 CODE_SEG("INIT")
764 BOOLEAN
765 NTAPI
766 ExpIsLoaderValid(IN PLOADER_PARAMETER_BLOCK LoaderBlock)
767 {
768     PLOADER_PARAMETER_EXTENSION Extension;
769 
770     /* Get the loader extension */
771     Extension = LoaderBlock->Extension;
772 
773     /* Validate the size (Windows 2003 loader doesn't provide more) */
774     if (Extension->Size < LOADER_PARAMETER_EXTENSION_MIN_SIZE) return FALSE;
775 
776     /* Don't validate upper versions */
777     if (Extension->MajorVersion > VER_PRODUCTMAJORVERSION) return TRUE;
778 
779     /* Fail if this is NT 4 */
780     if (Extension->MajorVersion < VER_PRODUCTMAJORVERSION) return FALSE;
781 
782     /* Fail if this is XP */
783     if (Extension->MinorVersion < VER_PRODUCTMINORVERSION) return FALSE;
784 
785     /* This is 2003 or newer, approve it */
786     return TRUE;
787 }
788 
789 CODE_SEG("INIT")
790 VOID
791 NTAPI
792 ExpLoadBootSymbols(IN PLOADER_PARAMETER_BLOCK LoaderBlock)
793 {
794     ULONG i = 0;
795     PLIST_ENTRY NextEntry;
796     ULONG Count, Length;
797     PWCHAR Name;
798     PLDR_DATA_TABLE_ENTRY LdrEntry;
799     CHAR NameBuffer[256];
800     STRING SymbolString;
801     NTSTATUS Status;
802 
803     /* Loop the driver list */
804     NextEntry = LoaderBlock->LoadOrderListHead.Flink;
805     while (NextEntry != &LoaderBlock->LoadOrderListHead)
806     {
807         /* Skip the first two images */
808         if (i >= 2)
809         {
810             /* Get the entry */
811             LdrEntry = CONTAINING_RECORD(NextEntry,
812                                          LDR_DATA_TABLE_ENTRY,
813                                          InLoadOrderLinks);
814             if (LdrEntry->FullDllName.Buffer[0] == L'\\')
815             {
816                 /* We have a name, read its data */
817                 Name = LdrEntry->FullDllName.Buffer;
818                 Length = LdrEntry->FullDllName.Length / sizeof(WCHAR);
819 
820                 /* Check if our buffer can hold it */
821                 if (sizeof(NameBuffer) < Length + sizeof(ANSI_NULL))
822                 {
823                     /* It's too long */
824                     Status = STATUS_BUFFER_OVERFLOW;
825                 }
826                 else
827                 {
828                     /* Copy the name */
829                     Count = 0;
830                     do
831                     {
832                         /* Copy the character */
833                         NameBuffer[Count++] = (CHAR)*Name++;
834                     } while (Count < Length);
835 
836                     /* Null-terminate */
837                     NameBuffer[Count] = ANSI_NULL;
838                     Status = STATUS_SUCCESS;
839                 }
840             }
841             else
842             {
843                 /* Safely print the string into our buffer */
844                 Status = RtlStringCbPrintfA(NameBuffer,
845                                             sizeof(NameBuffer),
846                                             "%S\\System32\\Drivers\\%wZ",
847                                             &SharedUserData->NtSystemRoot[2],
848                                             &LdrEntry->BaseDllName);
849             }
850 
851             /* Check if the buffer was ok */
852             if (NT_SUCCESS(Status))
853             {
854                 /* Initialize the STRING for the debugger */
855                 RtlInitString(&SymbolString, NameBuffer);
856 
857                 /* Load the symbols */
858                 DbgLoadImageSymbols(&SymbolString,
859                                     LdrEntry->DllBase,
860                                     (ULONG_PTR)PsGetCurrentProcessId());
861             }
862         }
863 
864         /* Go to the next entry */
865         i++;
866         NextEntry = NextEntry->Flink;
867     }
868 }
869 
870 CODE_SEG("INIT")
871 VOID
872 NTAPI
873 ExBurnMemory(IN PLOADER_PARAMETER_BLOCK LoaderBlock,
874              IN ULONG_PTR PagesToDestroy,
875              IN TYPE_OF_MEMORY MemoryType)
876 {
877     PLIST_ENTRY ListEntry;
878     PMEMORY_ALLOCATION_DESCRIPTOR MemDescriptor;
879 
880     DPRINT1("Burn RAM amount: %lu pages\n", PagesToDestroy);
881 
882     /* Loop the memory descriptors, beginning at the end */
883     for (ListEntry = LoaderBlock->MemoryDescriptorListHead.Blink;
884          ListEntry != &LoaderBlock->MemoryDescriptorListHead;
885          ListEntry = ListEntry->Blink)
886     {
887         /* Get the memory descriptor structure */
888         MemDescriptor = CONTAINING_RECORD(ListEntry,
889                                           MEMORY_ALLOCATION_DESCRIPTOR,
890                                           ListEntry);
891 
892         /* Is memory free there or is it temporary? */
893         if (MemDescriptor->MemoryType == LoaderFree ||
894             MemDescriptor->MemoryType == LoaderFirmwareTemporary)
895         {
896             /* Check if the descriptor has more pages than we want */
897             if (MemDescriptor->PageCount > PagesToDestroy)
898             {
899                 /* Change block's page count, ntoskrnl doesn't care much */
900                 MemDescriptor->PageCount -= PagesToDestroy;
901                 break;
902             }
903             else
904             {
905                 /* Change block type */
906                 MemDescriptor->MemoryType = MemoryType;
907                 PagesToDestroy -= MemDescriptor->PageCount;
908 
909                 /* Check if we are done */
910                 if (PagesToDestroy == 0) break;
911             }
912         }
913     }
914 }
915 
916 CODE_SEG("INIT")
917 VOID
918 NTAPI
919 ExpInitializeExecutive(IN ULONG Cpu,
920                        IN PLOADER_PARAMETER_BLOCK LoaderBlock)
921 {
922     PNLS_DATA_BLOCK NlsData;
923     CHAR Buffer[256];
924     ANSI_STRING AnsiPath;
925     NTSTATUS Status;
926     PCHAR CommandLine, PerfMem;
927     ULONG PerfMemUsed;
928     PLDR_DATA_TABLE_ENTRY NtosEntry;
929     PMESSAGE_RESOURCE_ENTRY MsgEntry;
930     ANSI_STRING CSDString;
931     size_t Remaining = 0;
932     PCHAR RcEnd = NULL;
933     CHAR VersionBuffer[65];
934 
935     /* Validate Loader */
936     if (!ExpIsLoaderValid(LoaderBlock))
937     {
938         /* Invalid loader version */
939         KeBugCheckEx(MISMATCHED_HAL,
940                      3,
941                      LoaderBlock->Extension->Size,
942                      LoaderBlock->Extension->MajorVersion,
943                      LoaderBlock->Extension->MinorVersion);
944     }
945 
946     /* Initialize PRCB pool lookaside pointers */
947     ExInitPoolLookasidePointers();
948 
949     /* Check if this is an application CPU */
950     if (Cpu)
951     {
952         /* Then simply initialize it with HAL */
953         if (!HalInitSystem(ExpInitializationPhase, LoaderBlock))
954         {
955             /* Initialization failed */
956             KeBugCheck(HAL_INITIALIZATION_FAILED);
957         }
958 
959         /* We're done */
960         return;
961     }
962 
963     /* Assume no text-mode or remote boot */
964     ExpInTextModeSetup = FALSE;
965     IoRemoteBootClient = FALSE;
966 
967     /* Check if we have a setup loader block */
968     if (LoaderBlock->SetupLdrBlock)
969     {
970         /* Check if this is text-mode setup */
971         if (LoaderBlock->SetupLdrBlock->Flags & SETUPLDR_TEXT_MODE)
972             ExpInTextModeSetup = TRUE;
973 
974         /* Check if this is network boot */
975         if (LoaderBlock->SetupLdrBlock->Flags & SETUPLDR_REMOTE_BOOT)
976         {
977             /* Set variable */
978             IoRemoteBootClient = TRUE;
979 
980             /* Make sure we're actually booting off the network */
981             ASSERT(!_memicmp(LoaderBlock->ArcBootDeviceName, "net(0)", 6));
982         }
983     }
984 
985     /* Set phase to 0 */
986     ExpInitializationPhase = 0;
987 
988     /* Get boot command line */
989     CommandLine = LoaderBlock->LoadOptions;
990     if (CommandLine)
991     {
992         /* Upcase it for comparison and check if we're in performance mode */
993         _strupr(CommandLine);
994         PerfMem = strstr(CommandLine, "PERFMEM");
995         if (PerfMem)
996         {
997             /* Check if the user gave a number of bytes to use */
998             PerfMem = strstr(PerfMem, "=");
999             if (PerfMem)
1000             {
1001                 /* Read the number of pages we'll use */
1002                 PerfMemUsed = atol(PerfMem + 1) * (1024 * 1024 / PAGE_SIZE);
1003                 if (PerfMemUsed)
1004                 {
1005                     /* FIXME: TODO */
1006                     DPRINT1("BBT performance mode not yet supported."
1007                             "/PERFMEM option ignored.\n");
1008                 }
1009             }
1010         }
1011 
1012         /* Check if we're burning memory */
1013         PerfMem = strstr(CommandLine, "BURNMEMORY");
1014         if (PerfMem)
1015         {
1016             /* Check if the user gave a number of bytes to use */
1017             PerfMem = strstr(PerfMem, "=");
1018             if (PerfMem)
1019             {
1020                 /* Read the number of pages we'll use */
1021                 PerfMemUsed = atol(PerfMem + 1) * (1024 * 1024 / PAGE_SIZE);
1022                 if (PerfMemUsed) ExBurnMemory(LoaderBlock, PerfMemUsed, LoaderBad);
1023             }
1024         }
1025     }
1026 
1027     /* Setup NLS Base and offsets */
1028     NlsData = LoaderBlock->NlsData;
1029     ExpNlsTableBase = NlsData->AnsiCodePageData;
1030     ExpAnsiCodePageDataOffset = 0;
1031     ExpOemCodePageDataOffset = (ULONG)((ULONG_PTR)NlsData->OemCodePageData -
1032                                        (ULONG_PTR)NlsData->AnsiCodePageData);
1033     ExpUnicodeCaseTableDataOffset = (ULONG)((ULONG_PTR)NlsData->UnicodeCodePageData -
1034                                             (ULONG_PTR)NlsData->AnsiCodePageData);
1035 
1036     /* Initialize the NLS Tables */
1037     RtlInitNlsTables((PVOID)((ULONG_PTR)ExpNlsTableBase +
1038                              ExpAnsiCodePageDataOffset),
1039                      (PVOID)((ULONG_PTR)ExpNlsTableBase +
1040                              ExpOemCodePageDataOffset),
1041                      (PVOID)((ULONG_PTR)ExpNlsTableBase +
1042                              ExpUnicodeCaseTableDataOffset),
1043                      &ExpNlsTableInfo);
1044     RtlResetRtlTranslations(&ExpNlsTableInfo);
1045 
1046     /* Now initialize the HAL */
1047     if (!HalInitSystem(ExpInitializationPhase, LoaderBlock))
1048     {
1049         /* HAL failed to initialize, bugcheck */
1050         KeBugCheck(HAL_INITIALIZATION_FAILED);
1051     }
1052 
1053     /* Make sure interrupts are active now */
1054     _enable();
1055 
1056     /* Clear the crypto exponent */
1057     SharedUserData->CryptoExponent = 0;
1058 
1059     /* Set global flags for the checked build */
1060 #if DBG
1061     NtGlobalFlag |= FLG_ENABLE_CLOSE_EXCEPTIONS |
1062                     FLG_ENABLE_KDEBUG_SYMBOL_LOAD;
1063 #endif
1064 
1065     /* Setup NT System Root Path */
1066     sprintf(Buffer, "C:%s", LoaderBlock->NtBootPathName);
1067 
1068     /* Convert to ANSI_STRING and null-terminate it */
1069     RtlInitString(&AnsiPath, Buffer);
1070     Buffer[--AnsiPath.Length] = ANSI_NULL;
1071 
1072     /* Get the string from KUSER_SHARED_DATA's buffer */
1073     RtlInitEmptyUnicodeString(&NtSystemRoot,
1074                               SharedUserData->NtSystemRoot,
1075                               sizeof(SharedUserData->NtSystemRoot));
1076 
1077     /* Now fill it in */
1078     Status = RtlAnsiStringToUnicodeString(&NtSystemRoot, &AnsiPath, FALSE);
1079     if (!NT_SUCCESS(Status)) KeBugCheck(SESSION3_INITIALIZATION_FAILED);
1080 
1081     /* Setup bugcheck messages */
1082     KiInitializeBugCheck();
1083 
1084     /* Setup initial system settings */
1085     CmGetSystemControlValues(LoaderBlock->RegistryBase, CmControlVector);
1086 
1087     /* Set the Service Pack Number and add it to the CSD Version number if needed */
1088     CmNtSpBuildNumber = VER_PRODUCTBUILD_QFE;
1089     if (((CmNtCSDVersion & 0xFFFF0000) == 0) && (CmNtCSDReleaseType == 1))
1090     {
1091         CmNtCSDVersion |= (VER_PRODUCTBUILD_QFE << 16);
1092     }
1093 
1094     /* Add loaded CmNtGlobalFlag value */
1095     NtGlobalFlag |= CmNtGlobalFlag;
1096 
1097     /* Initialize the executive at phase 0 */
1098     if (!ExInitSystem()) KeBugCheck(PHASE0_INITIALIZATION_FAILED);
1099 
1100     /* Initialize the memory manager at phase 0 */
1101     if (!MmArmInitSystem(0, LoaderBlock)) KeBugCheck(PHASE0_INITIALIZATION_FAILED);
1102 
1103     /* Load boot symbols */
1104     ExpLoadBootSymbols(LoaderBlock);
1105 
1106     /* Check if we should break after symbol load */
1107     if (KdBreakAfterSymbolLoad) DbgBreakPointWithStatus(DBG_STATUS_CONTROL_C);
1108 
1109     /* Check if this loader is compatible with NT 5.2 */
1110     if (LoaderBlock->Extension->Size >= sizeof(LOADER_PARAMETER_EXTENSION))
1111     {
1112         /* Setup headless terminal settings */
1113         HeadlessInit(LoaderBlock);
1114     }
1115 
1116     /* Set system ranges */
1117 #ifdef _M_AMD64
1118     SharedUserData->Reserved1 = MM_HIGHEST_USER_ADDRESS_WOW64;
1119     SharedUserData->Reserved3 = MM_SYSTEM_RANGE_START_WOW64;
1120 #else
1121     SharedUserData->Reserved1 = (ULONG_PTR)MmHighestUserAddress;
1122     SharedUserData->Reserved3 = (ULONG_PTR)MmSystemRangeStart;
1123 #endif
1124 
1125     /* Make a copy of the NLS Tables */
1126     ExpInitNls(LoaderBlock);
1127 
1128     /* Get the kernel's load entry */
1129     NtosEntry = CONTAINING_RECORD(LoaderBlock->LoadOrderListHead.Flink,
1130                                   LDR_DATA_TABLE_ENTRY,
1131                                   InLoadOrderLinks);
1132 
1133     /* Check if this is a service pack */
1134     if (CmNtCSDVersion & 0xFFFF)
1135     {
1136         /* Get the service pack string */
1137         Status = RtlFindMessage(NtosEntry->DllBase,
1138                                 11,
1139                                 0,
1140                                 WINDOWS_NT_CSD_STRING,
1141                                 &MsgEntry);
1142         if (NT_SUCCESS(Status))
1143         {
1144             /* Setup the string */
1145             RtlInitAnsiString(&CSDString, (PCHAR)MsgEntry->Text);
1146 
1147             /* Remove trailing newline */
1148             while ((CSDString.Length > 0) &&
1149                    ((CSDString.Buffer[CSDString.Length - 1] == '\r') ||
1150                     (CSDString.Buffer[CSDString.Length - 1] == '\n')))
1151             {
1152                 /* Skip the trailing character */
1153                 CSDString.Length--;
1154             }
1155 
1156             /* Fill the buffer with version information */
1157             Status = RtlStringCbPrintfA(Buffer,
1158                                         sizeof(Buffer),
1159                                         "%Z %u%c",
1160                                         &CSDString,
1161                                         (CmNtCSDVersion & 0xFF00) >> 8,
1162                                         (CmNtCSDVersion & 0xFF) ?
1163                                         'A' + (CmNtCSDVersion & 0xFF) - 1 :
1164                                         ANSI_NULL);
1165         }
1166         else
1167         {
1168             /* Build default string */
1169             Status = RtlStringCbPrintfA(Buffer,
1170                                         sizeof(Buffer),
1171                                         "CSD %04x",
1172                                         CmNtCSDVersion);
1173         }
1174 
1175         /* Check for success */
1176         if (!NT_SUCCESS(Status))
1177         {
1178             /* Fail */
1179             KeBugCheckEx(PHASE0_INITIALIZATION_FAILED, Status, 0, 0, 0);
1180         }
1181     }
1182     else
1183     {
1184         /* Then this is a beta */
1185         Status = RtlStringCbCopyExA(Buffer,
1186                                     sizeof(Buffer),
1187                                     VER_PRODUCTBETA_STR,
1188                                     NULL,
1189                                     &Remaining,
1190                                     0);
1191         if (!NT_SUCCESS(Status))
1192         {
1193             /* Fail */
1194             KeBugCheckEx(PHASE0_INITIALIZATION_FAILED, Status, 0, 0, 0);
1195         }
1196 
1197         /* Update length */
1198         CmCSDVersionString.MaximumLength = sizeof(Buffer) - (USHORT)Remaining;
1199     }
1200 
1201     /* Check if we have an RC number */
1202     if ((CmNtCSDVersion & 0xFFFF0000) && (CmNtCSDReleaseType == 1))
1203     {
1204         /* Check if we have no version data yet */
1205         if (!(*Buffer))
1206         {
1207             /* Set defaults */
1208             Remaining = sizeof(Buffer);
1209             RcEnd = Buffer;
1210         }
1211         else
1212         {
1213             /* Add comma and space */
1214             Status = RtlStringCbCatExA(Buffer,
1215                                        sizeof(Buffer),
1216                                        ", ",
1217                                        &RcEnd,
1218                                        &Remaining,
1219                                        0);
1220             if (!NT_SUCCESS(Status))
1221             {
1222                 /* Fail */
1223                 KeBugCheckEx(PHASE0_INITIALIZATION_FAILED, Status, 0, 0, 0);
1224             }
1225         }
1226 
1227         /* Add the version format string */
1228         Status = RtlStringCbPrintfA(RcEnd,
1229                                     Remaining,
1230                                     "v.%u",
1231                                     (CmNtCSDVersion & 0xFFFF0000) >> 16);
1232         if (!NT_SUCCESS(Status))
1233         {
1234             /* Fail */
1235             KeBugCheckEx(PHASE0_INITIALIZATION_FAILED, Status, 0, 0, 0);
1236         }
1237     }
1238 
1239     /* Now setup the final string */
1240     RtlInitAnsiString(&CSDString, Buffer);
1241     Status = RtlAnsiStringToUnicodeString(&CmCSDVersionString,
1242                                           &CSDString,
1243                                           TRUE);
1244     if (!NT_SUCCESS(Status))
1245     {
1246         /* Fail */
1247         KeBugCheckEx(PHASE0_INITIALIZATION_FAILED, Status, 0, 0, 0);
1248     }
1249 
1250     /* Add our version */
1251     Status = RtlStringCbPrintfA(VersionBuffer,
1252                                 sizeof(VersionBuffer),
1253                                 "%u.%u",
1254                                 VER_PRODUCTMAJORVERSION,
1255                                 VER_PRODUCTMINORVERSION);
1256     if (!NT_SUCCESS(Status))
1257     {
1258         /* Fail */
1259         KeBugCheckEx(PHASE0_INITIALIZATION_FAILED, Status, 0, 0, 0);
1260     }
1261 
1262     /* Build the final version string */
1263     RtlCreateUnicodeStringFromAsciiz(&CmVersionString, VersionBuffer);
1264 
1265     /* Check if the user wants a kernel stack trace database */
1266     if (NtGlobalFlag & FLG_KERNEL_STACK_TRACE_DB)
1267     {
1268         /* FIXME: TODO */
1269         DPRINT1("Kernel-mode stack trace support not yet present."
1270                 "FLG_KERNEL_STACK_TRACE_DB flag ignored.\n");
1271     }
1272 
1273     /* Check if he wanted exception logging */
1274     if (NtGlobalFlag & FLG_ENABLE_EXCEPTION_LOGGING)
1275     {
1276         /* FIXME: TODO */
1277         DPRINT1("Kernel-mode exception logging support not yet present."
1278                 "FLG_ENABLE_EXCEPTION_LOGGING flag ignored.\n");
1279     }
1280 
1281     /* Initialize the Handle Table */
1282     ExpInitializeHandleTables();
1283 
1284 #if DBG
1285     /* On checked builds, allocate the system call count table */
1286     KeServiceDescriptorTable[0].Count =
1287         ExAllocatePoolWithTag(NonPagedPool,
1288                               KiServiceLimit * sizeof(ULONG),
1289                               'llaC');
1290 
1291     /* Use it for the shadow table too */
1292     KeServiceDescriptorTableShadow[0].Count = KeServiceDescriptorTable[0].Count;
1293 
1294     /* Make sure allocation succeeded */
1295     if (KeServiceDescriptorTable[0].Count)
1296     {
1297         /* Zero the call counts to 0 */
1298         RtlZeroMemory(KeServiceDescriptorTable[0].Count,
1299                       KiServiceLimit * sizeof(ULONG));
1300     }
1301 #endif
1302 
1303     /* Create the Basic Object Manager Types to allow new Object Types */
1304     if (!ObInitSystem()) KeBugCheck(OBJECT_INITIALIZATION_FAILED);
1305 
1306     /* Load basic Security for other Managers */
1307     if (!SeInitSystem()) KeBugCheck(SECURITY_INITIALIZATION_FAILED);
1308 
1309     /* Initialize the Process Manager */
1310     if (!PsInitSystem(LoaderBlock)) KeBugCheck(PROCESS_INITIALIZATION_FAILED);
1311 
1312     /* Initialize the PnP Manager */
1313     if (!PpInitSystem()) KeBugCheck(PP0_INITIALIZATION_FAILED);
1314 
1315     /* Initialize the User-Mode Debugging Subsystem */
1316     DbgkInitialize();
1317 
1318     /* Calculate the tick count multiplier */
1319     ExpTickCountMultiplier = ExComputeTickCountMultiplier(KeMaximumIncrement);
1320     SharedUserData->TickCountMultiplier = ExpTickCountMultiplier;
1321 
1322     /* Set the OS Version */
1323     SharedUserData->NtMajorVersion = NtMajorVersion;
1324     SharedUserData->NtMinorVersion = NtMinorVersion;
1325 
1326     /* Set the machine type */
1327     SharedUserData->ImageNumberLow = IMAGE_FILE_MACHINE_NATIVE;
1328     SharedUserData->ImageNumberHigh = IMAGE_FILE_MACHINE_NATIVE;
1329 }
1330 
1331 VOID
1332 NTAPI
1333 MmFreeLoaderBlock(IN PLOADER_PARAMETER_BLOCK LoaderBlock);
1334 
1335 CODE_SEG("INIT")
1336 VOID
1337 NTAPI
1338 Phase1InitializationDiscard(IN PVOID Context)
1339 {
1340     PLOADER_PARAMETER_BLOCK LoaderBlock = Context;
1341     NTSTATUS Status, MsgStatus;
1342     TIME_FIELDS TimeFields;
1343     LARGE_INTEGER SystemBootTime, UniversalBootTime, OldTime, Timeout;
1344     BOOLEAN SosEnabled, NoGuiBoot, ResetBias = FALSE, AlternateShell = FALSE;
1345     PLDR_DATA_TABLE_ENTRY NtosEntry;
1346     PMESSAGE_RESOURCE_ENTRY MsgEntry;
1347     PCHAR CommandLine, Y2KHackRequired, SafeBoot, Environment;
1348     PCHAR StringBuffer, EndBuffer, BeginBuffer, MpString = "";
1349     PINIT_BUFFER InitBuffer;
1350     ANSI_STRING TempString;
1351     ULONG LastTzBias, Length, YearHack = 0, Disposition, MessageCode = 0;
1352     SIZE_T Size;
1353     size_t Remaining;
1354     PRTL_USER_PROCESS_INFORMATION ProcessInfo;
1355     KEY_VALUE_PARTIAL_INFORMATION KeyPartialInfo;
1356     UNICODE_STRING KeyName;
1357     OBJECT_ATTRIBUTES ObjectAttributes;
1358     HANDLE KeyHandle, OptionHandle;
1359     PRTL_USER_PROCESS_PARAMETERS ProcessParameters = NULL;
1360 
1361     /* Allocate the initialization buffer */
1362     InitBuffer = ExAllocatePoolWithTag(NonPagedPool,
1363                                        sizeof(INIT_BUFFER),
1364                                        TAG_INIT);
1365     if (!InitBuffer)
1366     {
1367         /* Bugcheck */
1368         KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, STATUS_NO_MEMORY, 8, 0, 0);
1369     }
1370 
1371     /* Set to phase 1 */
1372     ExpInitializationPhase = 1;
1373 
1374     /* Set us at maximum priority */
1375     KeSetPriorityThread(KeGetCurrentThread(), HIGH_PRIORITY);
1376 
1377     /* Do Phase 1 HAL Initialization */
1378     if (!HalInitSystem(1, LoaderBlock)) KeBugCheck(HAL1_INITIALIZATION_FAILED);
1379 
1380     /* Get the command line and upcase it */
1381     CommandLine = (LoaderBlock->LoadOptions ? _strupr(LoaderBlock->LoadOptions) : NULL);
1382 
1383     /* Check if GUI Boot is enabled */
1384     NoGuiBoot = (CommandLine && strstr(CommandLine, "NOGUIBOOT") != NULL);
1385 
1386     /* Get the SOS setting */
1387     SosEnabled = (CommandLine && strstr(CommandLine, "SOS") != NULL);
1388 
1389     /* Setup the boot driver */
1390     InbvEnableBootDriver(!NoGuiBoot);
1391     InbvDriverInitialize(LoaderBlock, IDB_MAX_RESOURCE);
1392 
1393     /* Check if GUI boot is enabled */
1394     if (!NoGuiBoot)
1395     {
1396         /* It is, display the boot logo and enable printing strings */
1397         InbvEnableDisplayString(SosEnabled);
1398         DisplayBootBitmap(SosEnabled);
1399     }
1400     else
1401     {
1402         /* Release display ownership if not using GUI boot */
1403         InbvNotifyDisplayOwnershipLost(NULL);
1404 
1405         /* Don't allow boot-time strings */
1406         InbvEnableDisplayString(FALSE);
1407     }
1408 
1409     /* Check if this is LiveCD (WinPE) mode */
1410     if (CommandLine && strstr(CommandLine, "MININT") != NULL)
1411     {
1412         /* Setup WinPE Settings */
1413         InitIsWinPEMode = TRUE;
1414         InitWinPEModeType |= (strstr(CommandLine, "INRAM") != NULL) ? 0x80000000 : 0x00000001;
1415     }
1416 
1417     /* Get the kernel's load entry */
1418     NtosEntry = CONTAINING_RECORD(LoaderBlock->LoadOrderListHead.Flink,
1419                                   LDR_DATA_TABLE_ENTRY,
1420                                   InLoadOrderLinks);
1421 
1422     /* Find the banner message */
1423     MsgStatus = RtlFindMessage(NtosEntry->DllBase,
1424                                11,
1425                                0,
1426                                WINDOWS_NT_BANNER,
1427                                &MsgEntry);
1428 
1429     /* Setup defaults and check if we have a version string */
1430     StringBuffer = InitBuffer->VersionBuffer;
1431     BeginBuffer = StringBuffer;
1432     EndBuffer = StringBuffer;
1433     Remaining = sizeof(InitBuffer->VersionBuffer);
1434     if (CmCSDVersionString.Length)
1435     {
1436         /* Print the version string */
1437         Status = RtlStringCbPrintfExA(StringBuffer,
1438                                       Remaining,
1439                                       &EndBuffer,
1440                                       &Remaining,
1441                                       0,
1442                                       ": %wZ",
1443                                       &CmCSDVersionString);
1444         if (!NT_SUCCESS(Status))
1445         {
1446             /* Bugcheck */
1447             KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 7, 0, 0);
1448         }
1449     }
1450     else
1451     {
1452         /* No version */
1453         *EndBuffer = ANSI_NULL; /* Null-terminate the string */
1454     }
1455 
1456     /* Skip over the null-terminator to start a new string */
1457     ++EndBuffer;
1458     --Remaining;
1459 
1460     /* Build the version number */
1461     StringBuffer = InitBuffer->VersionNumber;
1462     Status = RtlStringCbPrintfA(StringBuffer,
1463                                 sizeof(InitBuffer->VersionNumber),
1464                                 "%u.%u",
1465                                 VER_PRODUCTMAJORVERSION,
1466                                 VER_PRODUCTMINORVERSION);
1467     if (!NT_SUCCESS(Status))
1468     {
1469         /* Bugcheck */
1470         KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 7, 0, 0);
1471     }
1472 
1473     /* Check if we had found a banner message */
1474     if (NT_SUCCESS(MsgStatus))
1475     {
1476         /* Create the banner message */
1477         /* ReactOS specific: Report ReactOS version, NtBuildLab information and reported NT kernel version */
1478         Status = RtlStringCbPrintfA(EndBuffer,
1479                                     Remaining,
1480                                     (PCHAR)MsgEntry->Text,
1481                                     KERNEL_VERSION_STR,
1482                                     NtBuildLab,
1483                                     StringBuffer,
1484                                     NtBuildNumber & 0xFFFF,
1485                                     BeginBuffer);
1486         if (!NT_SUCCESS(Status))
1487         {
1488             /* Bugcheck */
1489             KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 7, 0, 0);
1490         }
1491     }
1492     else
1493     {
1494         /* Use hard-coded banner message */
1495         Status = RtlStringCbCopyA(EndBuffer, Remaining, "REACTOS (R)\r\n");
1496         if (!NT_SUCCESS(Status))
1497         {
1498             /* Bugcheck */
1499             KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 7, 0, 0);
1500         }
1501     }
1502 
1503     /* Display the version string on-screen */
1504     InbvDisplayString(EndBuffer);
1505 
1506     /* Initialize Power Subsystem in Phase 0 */
1507     if (!PoInitSystem(0)) KeBugCheck(INTERNAL_POWER_ERROR);
1508 
1509     /* Check for Y2K hack */
1510     Y2KHackRequired = CommandLine ? strstr(CommandLine, "YEAR") : NULL;
1511     if (Y2KHackRequired) Y2KHackRequired = strstr(Y2KHackRequired, "=");
1512     if (Y2KHackRequired) YearHack = atol(Y2KHackRequired + 1);
1513 
1514     /* Query the clock */
1515     if ((ExCmosClockIsSane) && (HalQueryRealTimeClock(&TimeFields)))
1516     {
1517         /* Check if we're using the Y2K hack */
1518         if (Y2KHackRequired) TimeFields.Year = (CSHORT)YearHack;
1519 
1520         /* Convert to time fields */
1521         RtlTimeFieldsToTime(&TimeFields, &SystemBootTime);
1522         UniversalBootTime = SystemBootTime;
1523 
1524         /* Check if real time is GMT */
1525         if (!ExpRealTimeIsUniversal)
1526         {
1527             /* Check if we don't have a valid bias */
1528             if (ExpLastTimeZoneBias == MAXULONG)
1529             {
1530                 /* Reset */
1531                 ResetBias = TRUE;
1532                 ExpLastTimeZoneBias = ExpAltTimeZoneBias;
1533             }
1534 
1535             /* Calculate the bias in seconds */
1536             ExpTimeZoneBias.QuadPart = Int32x32To64(ExpLastTimeZoneBias * 60,
1537                                                     10000000);
1538 
1539             /* Set the boot time-zone bias */
1540             SharedUserData->TimeZoneBias.High2Time = ExpTimeZoneBias.HighPart;
1541             SharedUserData->TimeZoneBias.LowPart = ExpTimeZoneBias.LowPart;
1542             SharedUserData->TimeZoneBias.High1Time = ExpTimeZoneBias.HighPart;
1543 
1544             /* Convert the boot time to local time, and set it */
1545             UniversalBootTime.QuadPart = SystemBootTime.QuadPart +
1546                                          ExpTimeZoneBias.QuadPart;
1547         }
1548 
1549         /* Update the system time */
1550         KeSetSystemTime(&UniversalBootTime, &OldTime, FALSE, NULL);
1551 
1552         /* Do system callback */
1553         PoNotifySystemTimeSet();
1554 
1555         /* Remember this as the boot time */
1556         KeBootTime = UniversalBootTime;
1557         KeBootTimeBias = 0;
1558     }
1559 
1560     /* Initialize all processors */
1561     if (!HalAllProcessorsStarted()) KeBugCheck(HAL1_INITIALIZATION_FAILED);
1562 
1563 #ifdef CONFIG_SMP
1564     /* HACK: We should use RtlFindMessage and not only fallback to this */
1565     MpString = "MultiProcessor Kernel\r\n";
1566 #endif
1567 
1568     /* Setup the "MP" String */
1569     RtlInitAnsiString(&TempString, MpString);
1570 
1571     /* Make sure to remove the \r\n if we actually have a string */
1572     while ((TempString.Length > 0) &&
1573            ((TempString.Buffer[TempString.Length - 1] == '\r') ||
1574             (TempString.Buffer[TempString.Length - 1] == '\n')))
1575     {
1576         /* Skip the trailing character */
1577         TempString.Length--;
1578     }
1579 
1580     /* Get the information string from our resource file */
1581     MsgStatus = RtlFindMessage(NtosEntry->DllBase,
1582                                11,
1583                                0,
1584                                KeNumberProcessors > 1 ?
1585                                WINDOWS_NT_INFO_STRING_PLURAL :
1586                                WINDOWS_NT_INFO_STRING,
1587                                &MsgEntry);
1588 
1589     /* Get total RAM size, in MiB */
1590     /* Round size up. Assumed to better match actual physical RAM size */
1591     Size = ALIGN_UP_BY(MmNumberOfPhysicalPages * PAGE_SIZE, 1024 * 1024) / (1024 * 1024);
1592 
1593     /* Create the string */
1594     StringBuffer = InitBuffer->VersionBuffer;
1595     Status = RtlStringCbPrintfA(StringBuffer,
1596                                 sizeof(InitBuffer->VersionBuffer),
1597                                 NT_SUCCESS(MsgStatus) ?
1598                                 (PCHAR)MsgEntry->Text :
1599                                 "%u System Processor [%Iu MB Memory] %Z\r\n",
1600                                 KeNumberProcessors,
1601                                 Size,
1602                                 &TempString);
1603     if (!NT_SUCCESS(Status))
1604     {
1605         /* Bugcheck */
1606         KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 4, 0, 0);
1607     }
1608 
1609     /* Display RAM and CPU count */
1610     InbvDisplayString(StringBuffer);
1611 
1612     /* Update the progress bar */
1613     InbvUpdateProgressBar(5);
1614 
1615     /* Call OB initialization again */
1616     if (!ObInitSystem()) KeBugCheck(OBJECT1_INITIALIZATION_FAILED);
1617 
1618     /* Initialize Basic System Objects and Worker Threads */
1619     if (!ExInitSystem()) KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, 0, 0, 1, 0);
1620 
1621     /* Initialize the later stages of the kernel */
1622     if (!KeInitSystem()) KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, 0, 0, 2, 0);
1623 
1624     /* Call KD Providers at Phase 1 */
1625     if (!KdInitSystem(ExpInitializationPhase, KeLoaderBlock))
1626     {
1627         /* Failed, bugcheck */
1628         KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, 0, 0, 3, 0);
1629     }
1630 
1631     /* Initialize the SRM in Phase 1 */
1632     if (!SeInitSystem()) KeBugCheck(SECURITY1_INITIALIZATION_FAILED);
1633 
1634     /* Update the progress bar */
1635     InbvUpdateProgressBar(10);
1636 
1637     /* Create SystemRoot Link */
1638     Status = ExpCreateSystemRootLink(LoaderBlock);
1639     if (!NT_SUCCESS(Status))
1640     {
1641         /* Failed to create the system root link */
1642         KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED, Status, 0, 0, 0);
1643     }
1644 
1645     /* Set up Region Maps, Sections and the Paging File */
1646     if (!MmInitSystem(1, LoaderBlock)) KeBugCheck(MEMORY1_INITIALIZATION_FAILED);
1647 
1648     /* Create NLS section */
1649     ExpInitNls(LoaderBlock);
1650 
1651     /* Initialize Cache Views */
1652     if (!CcInitializeCacheManager()) KeBugCheck(CACHE_INITIALIZATION_FAILED);
1653 
1654     /* Initialize the Registry */
1655     if (!CmInitSystem1()) KeBugCheck(CONFIG_INITIALIZATION_FAILED);
1656 
1657     /* Initialize Prefetcher */
1658     CcPfInitializePrefetcher();
1659 
1660     /* Update progress bar */
1661     InbvUpdateProgressBar(15);
1662 
1663     /* Update timezone information */
1664     LastTzBias = ExpLastTimeZoneBias;
1665     ExRefreshTimeZoneInformation(&SystemBootTime);
1666 
1667     /* Check if we're resetting timezone data */
1668     if (ResetBias)
1669     {
1670         /* Convert the local time to system time */
1671         ExLocalTimeToSystemTime(&SystemBootTime, &UniversalBootTime);
1672         KeBootTime = UniversalBootTime;
1673         KeBootTimeBias = 0;
1674 
1675         /* Set the new time */
1676         KeSetSystemTime(&UniversalBootTime, &OldTime, FALSE, NULL);
1677     }
1678     else
1679     {
1680         /* Check if the timezone switched and update the time */
1681         if (LastTzBias != ExpLastTimeZoneBias) ZwSetSystemTime(NULL, NULL);
1682     }
1683 
1684     /* Initialize the File System Runtime Library */
1685     if (!FsRtlInitSystem()) KeBugCheck(FILE_INITIALIZATION_FAILED);
1686 
1687     /* Initialize range lists */
1688     RtlInitializeRangeListPackage();
1689 
1690     /* Report all resources used by HAL */
1691     HalReportResourceUsage();
1692 
1693     /* Call the debugger DLL */
1694     KdDebuggerInitialize1(LoaderBlock);
1695 
1696     /* Setup PnP Manager in phase 1 */
1697     if (!PpInitSystem()) KeBugCheck(PP1_INITIALIZATION_FAILED);
1698 
1699     /* Update progress bar */
1700     InbvUpdateProgressBar(20);
1701 
1702     /* Initialize LPC */
1703     if (!LpcInitSystem()) KeBugCheck(LPC_INITIALIZATION_FAILED);
1704 
1705     /* Make sure we have a command line */
1706     if (CommandLine)
1707     {
1708         /* Check if this is a safe mode boot */
1709         SafeBoot = strstr(CommandLine, "SAFEBOOT:");
1710         if (SafeBoot)
1711         {
1712             /* Check what kind of boot this is */
1713             SafeBoot += 9;
1714             if (!strncmp(SafeBoot, "MINIMAL", 7))
1715             {
1716                 /* Minimal mode */
1717                 InitSafeBootMode = 1;
1718                 SafeBoot += 7;
1719                 MessageCode = BOOTING_IN_SAFEMODE_MINIMAL;
1720             }
1721             else if (!strncmp(SafeBoot, "NETWORK", 7))
1722             {
1723                 /* With Networking */
1724                 InitSafeBootMode = 2;
1725                 SafeBoot += 7;
1726                 MessageCode = BOOTING_IN_SAFEMODE_NETWORK;
1727             }
1728             else if (!strncmp(SafeBoot, "DSREPAIR", 8))
1729             {
1730                 /* Domain Server Repair */
1731                 InitSafeBootMode = 3;
1732                 SafeBoot += 8;
1733                 MessageCode = BOOTING_IN_SAFEMODE_DSREPAIR;
1734 
1735             }
1736             else
1737             {
1738                 /* Invalid */
1739                 InitSafeBootMode = 0;
1740             }
1741 
1742             /* Check if there's any settings left */
1743             if (*SafeBoot)
1744             {
1745                 /* Check if an alternate shell was requested */
1746                 if (!strncmp(SafeBoot, "(ALTERNATESHELL)", 16))
1747                 {
1748                     /* Remember this for later */
1749                     AlternateShell = TRUE;
1750                 }
1751             }
1752 
1753             /* Find the message to print out */
1754             Status = RtlFindMessage(NtosEntry->DllBase,
1755                                     11,
1756                                     0,
1757                                     MessageCode,
1758                                     &MsgEntry);
1759             if (NT_SUCCESS(Status))
1760             {
1761                 /* Display it */
1762                 InbvDisplayString((PCHAR)MsgEntry->Text);
1763             }
1764         }
1765     }
1766 
1767     /* Make sure we have a command line */
1768     if (CommandLine)
1769     {
1770         /* Check if bootlogging is enabled */
1771         if (strstr(CommandLine, "BOOTLOG"))
1772         {
1773             /* Find the message to print out */
1774             Status = RtlFindMessage(NtosEntry->DllBase,
1775                                     11,
1776                                     0,
1777                                     BOOTLOG_ENABLED,
1778                                     &MsgEntry);
1779             if (NT_SUCCESS(Status))
1780             {
1781                 /* Display it */
1782                 InbvDisplayString((PCHAR)MsgEntry->Text);
1783             }
1784 
1785             /* Setup boot logging */
1786             //IopInitializeBootLogging(LoaderBlock, InitBuffer->BootlogHeader);
1787         }
1788     }
1789 
1790     /* Setup the Executive in Phase 2 */
1791     //ExInitSystemPhase2();
1792 
1793     /* Update progress bar */
1794     InbvUpdateProgressBar(25);
1795 
1796     /* No KD Time Slip is pending */
1797     KdpTimeSlipPending = 0;
1798 
1799     /* Initialize in-place execution support */
1800     XIPInit(LoaderBlock);
1801 
1802     /* Set maximum update to 75% */
1803     InbvSetProgressBarSubset(25, 75);
1804 
1805     /* Initialize the I/O Subsystem */
1806     if (!IoInitSystem(LoaderBlock)) KeBugCheck(IO1_INITIALIZATION_FAILED);
1807 
1808     /* Set maximum update to 100% */
1809     InbvSetProgressBarSubset(0, 100);
1810 
1811     /* Are we in safe mode? */
1812     if (InitSafeBootMode)
1813     {
1814         /* Open the safe boot key */
1815         RtlInitUnicodeString(&KeyName,
1816                              L"\\REGISTRY\\MACHINE\\SYSTEM\\CURRENTCONTROLSET"
1817                              L"\\CONTROL\\SAFEBOOT");
1818         InitializeObjectAttributes(&ObjectAttributes,
1819                                    &KeyName,
1820                                    OBJ_CASE_INSENSITIVE,
1821                                    NULL,
1822                                    NULL);
1823         Status = ZwOpenKey(&KeyHandle, KEY_ALL_ACCESS, &ObjectAttributes);
1824         if (NT_SUCCESS(Status))
1825         {
1826             /* First check if we have an alternate shell */
1827             if (AlternateShell)
1828             {
1829                 /* Make sure that the registry has one setup */
1830                 RtlInitUnicodeString(&KeyName, L"AlternateShell");
1831                 Status = NtQueryValueKey(KeyHandle,
1832                                          &KeyName,
1833                                          KeyValuePartialInformation,
1834                                          &KeyPartialInfo,
1835                                          sizeof(KeyPartialInfo),
1836                                          &Length);
1837                 if (!(NT_SUCCESS(Status) || Status == STATUS_BUFFER_OVERFLOW))
1838                 {
1839                     AlternateShell = FALSE;
1840                 }
1841             }
1842 
1843             /* Create the option key */
1844             RtlInitUnicodeString(&KeyName, L"Option");
1845             InitializeObjectAttributes(&ObjectAttributes,
1846                                        &KeyName,
1847                                        OBJ_CASE_INSENSITIVE,
1848                                        KeyHandle,
1849                                        NULL);
1850             Status = ZwCreateKey(&OptionHandle,
1851                                  KEY_ALL_ACCESS,
1852                                  &ObjectAttributes,
1853                                  0,
1854                                  NULL,
1855                                  REG_OPTION_VOLATILE,
1856                                  &Disposition);
1857             NtClose(KeyHandle);
1858 
1859             /* Check if the key create worked */
1860             if (NT_SUCCESS(Status))
1861             {
1862                 /* Write the safe boot type */
1863                 RtlInitUnicodeString(&KeyName, L"OptionValue");
1864                 NtSetValueKey(OptionHandle,
1865                               &KeyName,
1866                               0,
1867                               REG_DWORD,
1868                               &InitSafeBootMode,
1869                               sizeof(InitSafeBootMode));
1870 
1871                 /* Check if we have to use an alternate shell */
1872                 if (AlternateShell)
1873                 {
1874                     /* Remember this for later */
1875                     Disposition = TRUE;
1876                     RtlInitUnicodeString(&KeyName, L"UseAlternateShell");
1877                     NtSetValueKey(OptionHandle,
1878                                   &KeyName,
1879                                   0,
1880                                   REG_DWORD,
1881                                   &Disposition,
1882                                   sizeof(Disposition));
1883                 }
1884 
1885                 /* Close the options key handle */
1886                 NtClose(OptionHandle);
1887             }
1888         }
1889     }
1890 
1891     /* Are we in Win PE mode? */
1892     if (InitIsWinPEMode)
1893     {
1894         /* Open the safe control key */
1895         RtlInitUnicodeString(&KeyName,
1896                              L"\\REGISTRY\\MACHINE\\SYSTEM\\CURRENTCONTROLSET"
1897                              L"\\CONTROL");
1898         InitializeObjectAttributes(&ObjectAttributes,
1899                                    &KeyName,
1900                                    OBJ_CASE_INSENSITIVE,
1901                                    NULL,
1902                                    NULL);
1903         Status = ZwOpenKey(&KeyHandle, KEY_ALL_ACCESS, &ObjectAttributes);
1904         if (!NT_SUCCESS(Status))
1905         {
1906             /* Bugcheck */
1907             KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 6, 0, 0);
1908         }
1909 
1910         /* Create the MiniNT key */
1911         RtlInitUnicodeString(&KeyName, L"MiniNT");
1912         InitializeObjectAttributes(&ObjectAttributes,
1913                                    &KeyName,
1914                                    OBJ_CASE_INSENSITIVE,
1915                                    KeyHandle,
1916                                    NULL);
1917         Status = ZwCreateKey(&OptionHandle,
1918                              KEY_ALL_ACCESS,
1919                              &ObjectAttributes,
1920                              0,
1921                              NULL,
1922                              REG_OPTION_VOLATILE,
1923                              &Disposition);
1924         if (!NT_SUCCESS(Status))
1925         {
1926             /* Bugcheck */
1927             KeBugCheckEx(PHASE1_INITIALIZATION_FAILED, Status, 6, 0, 0);
1928         }
1929 
1930         /* Close the handles */
1931         NtClose(KeyHandle);
1932         NtClose(OptionHandle);
1933     }
1934 
1935     /* FIXME: This doesn't do anything for now */
1936     MmArmInitSystem(2, LoaderBlock);
1937 
1938     /* Update progress bar */
1939     InbvUpdateProgressBar(80);
1940 
1941     /* Initialize VDM support */
1942 #if defined(_M_IX86)
1943     KeI386VdmInitialize();
1944 #endif
1945 
1946     /* Initialize Power Subsystem in Phase 1*/
1947     if (!PoInitSystem(1)) KeBugCheck(INTERNAL_POWER_ERROR);
1948 
1949     /* Update progress bar */
1950     InbvUpdateProgressBar(90);
1951 
1952     /* Initialize the Process Manager at Phase 1 */
1953     if (!PsInitSystem(LoaderBlock)) KeBugCheck(PROCESS1_INITIALIZATION_FAILED);
1954 
1955     /* Make sure nobody touches the loader block again */
1956     if (LoaderBlock == KeLoaderBlock) KeLoaderBlock = NULL;
1957     MmFreeLoaderBlock(LoaderBlock);
1958     LoaderBlock = Context = NULL;
1959 
1960     /* Initialize the SRM in phase 1 */
1961     if (!SeRmInitPhase1()) KeBugCheck(PROCESS1_INITIALIZATION_FAILED);
1962 
1963     /* Update progress bar */
1964     InbvUpdateProgressBar(100);
1965 
1966     /* Clear the screen */
1967     if (InbvBootDriverInstalled) FinalizeBootLogo();
1968 
1969     /* Allow strings to be displayed */
1970     InbvEnableDisplayString(TRUE);
1971 
1972     /* Launch initial process */
1973     DPRINT("Free non-cache pages: %lx\n", MmAvailablePages + MiMemoryConsumers[MC_CACHE].PagesUsed);
1974     ProcessInfo = &InitBuffer->ProcessInfo;
1975     ExpLoadInitialProcess(InitBuffer, &ProcessParameters, &Environment);
1976 
1977     /* Wait 5 seconds for initial process to initialize */
1978     Timeout.QuadPart = Int32x32To64(5, -10000000);
1979     Status = ZwWaitForSingleObject(ProcessInfo->ProcessHandle, FALSE, &Timeout);
1980     if (Status == STATUS_SUCCESS)
1981     {
1982         /* Failed, display error */
1983         DPRINT1("INIT: Session Manager terminated.\n");
1984 
1985         /* Bugcheck the system if SMSS couldn't initialize */
1986         KeBugCheck(SESSION5_INITIALIZATION_FAILED);
1987     }
1988 
1989     /* Close process handles */
1990     ZwClose(ProcessInfo->ThreadHandle);
1991     ZwClose(ProcessInfo->ProcessHandle);
1992 
1993     /* Free the initial process environment */
1994     Size = 0;
1995     ZwFreeVirtualMemory(NtCurrentProcess(),
1996                         (PVOID*)&Environment,
1997                         &Size,
1998                         MEM_RELEASE);
1999 
2000     /* Free the initial process parameters */
2001     Size = 0;
2002     ZwFreeVirtualMemory(NtCurrentProcess(),
2003                         (PVOID*)&ProcessParameters,
2004                         &Size,
2005                         MEM_RELEASE);
2006 
2007     /* Increase init phase */
2008     ExpInitializationPhase++;
2009 
2010     /* Free the boot buffer */
2011     ExFreePoolWithTag(InitBuffer, TAG_INIT);
2012     DPRINT("Free non-cache pages: %lx\n", MmAvailablePages + MiMemoryConsumers[MC_CACHE].PagesUsed);
2013 }
2014 
2015 VOID
2016 NTAPI
2017 Phase1Initialization(IN PVOID Context)
2018 {
2019     /* Do the .INIT part of Phase 1 which we can free later */
2020     Phase1InitializationDiscard(Context);
2021 
2022     /* Jump into zero page thread */
2023     MmZeroPageThread();
2024 }
2025