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

3151 lines
91 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1989-1998 Microsoft Corporation
Module Name:
init.c
Abstract:
Main source file of the NTOS system initialization subcomponent.
Author:
Steve Wood (stevewo) 31-Mar-1989
Revision History:
--*/
#include "ntos.h"
#include "ntimage.h"
#include <zwapi.h>
#include <ntdddisk.h>
#include <kddll.h>
#include <setupblk.h>
#include <fsrtl.h>
#include <ntverp.h>
#include "stdlib.h"
#include "stdio.h"
#include <string.h>
#include <safeboot.h>
#include <inbv.h>
#include <hdlsblk.h>
#include <hdlsterm.h>
#include "anim.h"
#include "xip.h"
UNICODE_STRING NtSystemRoot;
VOID
ExpInitializeExecutive(
IN ULONG Number,
IN PLOADER_PARAMETER_BLOCK LoaderBlock
);
NTKERNELAPI
BOOLEAN
ExpRefreshTimeZoneInformation(
IN PLARGE_INTEGER CurrentUniversalTime
);
NTSTATUS
CreateSystemRootLink(
IN PLOADER_PARAMETER_BLOCK LoaderBlock
);
static USHORT
NameToOrdinal (
IN PSZ NameOfEntryPoint,
IN ULONG_PTR DllBase,
IN ULONG NumberOfNames,
IN PULONG NameTableBase,
IN PUSHORT NameOrdinalTableBase
);
NTSTATUS
LookupEntryPoint (
IN PVOID DllBase,
IN PSZ NameOfEntryPoint,
OUT PVOID *AddressOfEntryPoint
);
#if defined(_X86_)
VOID
KiInitializeInterruptTimers(
VOID
);
#endif
PFN_COUNT
ExBurnMemory(
IN PLOADER_PARAMETER_BLOCK LoaderBlock,
IN PFN_COUNT NumberOfPagesToBurn,
IN TYPE_OF_MEMORY MemoryTypeForRemovedPages,
IN PMEMORY_ALLOCATION_DESCRIPTOR NewMemoryDescriptor OPTIONAL
);
VOID
DisplayFilter(
PUCHAR *String
);
NTSTATUS
RtlInitializeStackTraceDataBase(
IN PVOID CommitBase,
IN SIZE_T CommitSize,
IN SIZE_T ReserveSize
);
BOOLEAN
ExpIsLoaderValid(
IN PLOADER_PARAMETER_BLOCK LoaderBlock
);
#ifdef ALLOC_PRAGMA
//
// The INIT section is not pageable during initialization, so these
// functions can be in INIT rather than in .text.
//
#pragma alloc_text(INIT,ExBurnMemory)
#pragma alloc_text(INIT,ExpInitializeExecutive)
#pragma alloc_text(INIT,Phase1Initialization)
#pragma alloc_text(INIT,CreateSystemRootLink)
#pragma alloc_text(INIT,LookupEntryPoint)
#pragma alloc_text(INIT,NameToOrdinal)
#pragma alloc_text(INIT,ExpIsLoaderValid)
#endif
//
// Define global static data used during initialization.
//
ULONG NtGlobalFlag;
extern PMESSAGE_RESOURCE_BLOCK KiBugCheckMessages;
extern UCHAR CmProcessorMismatch;
const ULONG NtMajorVersion = VER_PRODUCTMAJORVERSION;
const ULONG NtMinorVersion = VER_PRODUCTMINORVERSION;
#if DBG
ULONG NtBuildNumber = VER_PRODUCTBUILD | 0xC0000000;
#else
ULONG NtBuildNumber = VER_PRODUCTBUILD | 0xF0000000;
#endif
#if defined(__BUILDMACHINE__)
#if defined(__BUILDDATE__)
#define B2(w,x,y) "" #w "." #x "." #y
#define B1(w,x,y) B2(w, x, y)
#define BUILD_MACHINE_TAG B1(VER_PRODUCTBUILD, __BUILDMACHINE__, __BUILDDATE__)
#else
#define B2(w,x) "" #w "." #x
#define B1(w,x) B2(w,x)
#define BUILD_MACHINE_TAG B1(VER_PRODUCTBUILD, __BUILDMACHINE__)
#endif
#else
#define BUILD_MACHINE_TAG ""
#endif
const CHAR NtBuildLab[] = BUILD_MACHINE_TAG;
ULONG InitializationPhase;
extern BOOLEAN ShowProgressBar;
extern KiServiceLimit;
extern PMESSAGE_RESOURCE_DATA KiBugCodeMessages;
extern ULONG KdpTimeSlipPending;
extern BOOLEAN KdBreakAfterSymbolLoad;
extern CM_SYSTEM_CONTROL_VECTOR CmControlVector[];
ULONG CmNtCSDVersion;
ULONG CmBrand;
UNICODE_STRING CmVersionString;
UNICODE_STRING CmCSDVersionString;
ULONG InitSafeBootMode;
BOOLEAN InitIsWinPEMode = FALSE;
ULONG InitWinPEModeType = INIT_WINPEMODE_NONE;
WCHAR NtInitialUserProcessBuffer[128] = L"\\SystemRoot\\System32\\smss.exe";
ULONG NtInitialUserProcessBufferLength =
sizeof(NtInitialUserProcessBuffer) - sizeof(WCHAR);
ULONG NtInitialUserProcessBufferType = REG_SZ;
#if defined(_X86_)
extern ULONG KeNumprocSpecified;
#endif
typedef struct _EXLOCK {
KSPIN_LOCK SpinLock;
KIRQL Irql;
} EXLOCK, *PEXLOCK;
#ifdef ALLOC_PRAGMA
NTSTATUS
ExpInitializeLockRoutine(
PEXLOCK Lock
);
#pragma alloc_text(INIT,ExpInitializeLockRoutine)
#endif
BOOLEAN
ExpOkayToLockRoutine(
IN PEXLOCK Lock
)
{
return TRUE;
}
NTSTATUS
ExpInitializeLockRoutine(
PEXLOCK Lock
)
{
KeInitializeSpinLock(&Lock->SpinLock);
return STATUS_SUCCESS;
}
NTSTATUS
ExpAcquireLockRoutine(
PEXLOCK Lock
)
{
ExAcquireSpinLock(&Lock->SpinLock,&Lock->Irql);
return STATUS_SUCCESS;
}
NTSTATUS
ExpReleaseLockRoutine(
PEXLOCK Lock
)
{
ExReleaseSpinLock(&Lock->SpinLock,Lock->Irql);
return STATUS_SUCCESS;
}
#if 0
NTSTATUS
ExpDeleteLockRoutine(
PEXLOCK Lock
)
{
return STATUS_SUCCESS;
}
#endif //0
#ifdef ALLOC_DATA_PRAGMA
#pragma data_seg("INITDATA")
#endif
ULONG CmNtGlobalFlag = 0;
NLSTABLEINFO InitTableInfo;
ULONG InitNlsTableSize;
PVOID InitNlsTableBase;
PFN_COUNT BBTPagesToReserve;
#ifdef ALLOC_DATA_PRAGMA
#pragma data_seg("PAGEDATA")
#endif
PVOID InitNlsSectionPointer = NULL;
ULONG InitAnsiCodePageDataOffset = 0;
ULONG InitOemCodePageDataOffset = 0;
ULONG InitUnicodeCaseTableDataOffset = 0;
#ifdef ALLOC_DATA_PRAGMA
#pragma data_seg()
#endif
PVOID BBTBuffer;
MEMORY_ALLOCATION_DESCRIPTOR BBTMemoryDescriptor;
#define COLOR_BLACK 0
#define COLOR_BLUE 2
#define COLOR_DARKGRAY 4
#define COLOR_GRAY 9
#define COLOR_WHITE 15
extern BOOLEAN InbvBootDriverInstalled;
VOID
DisplayBootBitmap (
IN BOOLEAN DisplayOnScreen
)
/*++
Routine Description:
Draws the gui boot screen.
Arguments:
DisplayOnScreen - TRUE to dump text to the screen, FALSE otherwise.
Return Value:
None.
Environment:
This routine may be called more than once, and should not be marked INIT.
--*/
{
LARGE_INTEGER DueTime;
static BOOLEAN FirstCall = TRUE;
ROT_BAR_TYPE TempRotBarSelection = RB_UNSPECIFIED;
if (FirstCall == FALSE) {
//
// Disable current animation
//
InbvAcquireLock();
RotBarSelection = RB_UNSPECIFIED;
InbvReleaseLock();
}
ShowProgressBar = FALSE;
if (DisplayOnScreen) {
PUCHAR BitmapTop, BitmapBottom;
if (SharedUserData->NtProductType == NtProductWinNt) {
InbvSetTextColor(COLOR_WHITE);
InbvSolidColorFill(0, 0, 639, 479, 7); // background
InbvSolidColorFill(0, 421, 639, 479, 1); // bottom
BitmapTop = InbvGetResourceAddress(6);
BitmapBottom = InbvGetResourceAddress(7);
} else { // srv
InbvSetTextColor(14);
InbvSolidColorFill(0, 0, 639, 479, 6); // background
InbvSolidColorFill(0, 421, 639, 479, 1); // bottom
BitmapTop = InbvGetResourceAddress(14);
BitmapBottom = InbvGetResourceAddress(15);
}
TempRotBarSelection = RB_UNSPECIFIED;
InbvSetScrollRegion(32, 80, 631, 400);
if (BitmapTop && BitmapBottom) {
InbvBitBlt(BitmapBottom, 0, 419);
InbvBitBlt(BitmapTop, 0, 0);
}
} else {
PUCHAR BarBitmap = NULL;
PUCHAR TextBitmap = NULL;
PUCHAR Bitmap = NULL;
PUCHAR LogoBitmap = NULL;
InbvInstallDisplayStringFilter(DisplayFilter);
if (!InbvBootDriverInstalled) {
return;
}
Bitmap = InbvGetResourceAddress(1); // workstation bitmap
if (ExVerifySuite(EmbeddedNT)) { // embd and pro have the same bar, but different text
TextBitmap = InbvGetResourceAddress(12); // embedded edition title text
BarBitmap = InbvGetResourceAddress(8); // pro and embedded editions progress bar
}
else if (SharedUserData->NtProductType == NtProductWinNt) { // home or pro
if (ExVerifySuite(Personal)) { // home
BarBitmap = InbvGetResourceAddress(9); // home edition progress bar
TextBitmap = InbvGetResourceAddress(11); // home edition title text
}
else { // pro
BarBitmap = InbvGetResourceAddress(8); // pro and embedded editions progress bar
switch (CmBrand) {
case 1: // TabletPc
TextBitmap = InbvGetResourceAddress(17);
break;
case 2: // eHome Freestyle
TextBitmap = InbvGetResourceAddress(18);
break;
default: // Professional title text
TextBitmap = InbvGetResourceAddress(10);
}
}
}
else { // srv
BarBitmap = InbvGetResourceAddress(4); // srv edition progress bar
LogoBitmap = InbvGetResourceAddress(13); // srv edition logo and title
}
if (Bitmap) {
TempRotBarSelection = RB_SQUARE_CELLS;
}
//
// Set positions for scrolling bar.
//
if (Bitmap) {
InbvBitBlt(Bitmap, 0, 0);
//if (SharedUserData->NtProductType == NtProductServer) {
if (SharedUserData->NtProductType != NtProductWinNt) {
extern BOOLEAN ExpInTextModeSetup; // defined at base\ntos\ex\exinit.c
// Tweak the logo to make it neutral (e.g. remove "XP")
{
UCHAR sav_copyright[64];
InbvScreenToBufferBlt(sav_copyright, 413, 237, 7, 7, 8);
InbvSolidColorFill(418,230,454,256,0);
InbvBufferToScreenBlt(sav_copyright, 413, 237, 7, 7, 8);
}
// HACK: in case of "text mode setup" (ExpInTextModeSetup == TRUE)
// we can't determine the SKU so we displaying neutral bitmap
// without specific SKU title (e.g. just Windows) and server's progress bar
if (ExpInTextModeSetup) {
TextBitmap = NULL;
}
else {
// Overwrite the XP logo with .NET logo
if (LogoBitmap) {
InbvBitBlt(LogoBitmap, 180, 121);
}
}
}
}
if (TextBitmap) {
InbvBitBlt(TextBitmap, 220, 272);
}
if (BarBitmap) {
InbvBitBlt(BarBitmap, 0, 0);
}
}
InbvAcquireLock();
RotBarSelection = TempRotBarSelection;
InbvRotBarInit();
InbvReleaseLock();
if (FirstCall) {
//
// If we got here, we are showing the boot bitmap.
// Start a timer to support animation.
//
HANDLE ThreadHandle;
PsCreateSystemThread(&ThreadHandle,
0L,
NULL,
NULL,
NULL,
InbvRotateGuiBootDisplay,
NULL);
}
FirstCall = FALSE;
}
VOID
DisplayFilter(
IN OUT PUCHAR *String
)
/*++
Routine Description:
This routine monitors InbvDisplayString output. If it sees something
which needs to be displayed on the screen, it triggers the output screen.
Arguments:
String - Pointer to a string pointer.
Returns:
None.
Notes:
This routine will be called anytime a string is displayed via the
Inbv routines. It cannot be paged!
--*/
{
static const UCHAR EmptyString = 0;
static BOOLEAN NonDotHit = FALSE;
if ((NonDotHit == FALSE) && (strcmp(*String, ".") == 0)) {
*String = (PUCHAR)&EmptyString;
} else {
NonDotHit = TRUE;
InbvInstallDisplayStringFilter((INBV_DISPLAY_STRING_FILTER)NULL);
DisplayBootBitmap(TRUE);
}
}
PFN_COUNT
ExBurnMemory (
IN PLOADER_PARAMETER_BLOCK LoaderBlock,
IN PFN_COUNT NumberOfPagesToBurn,
IN TYPE_OF_MEMORY MemoryTypeForRemovedPages,
IN PMEMORY_ALLOCATION_DESCRIPTOR NewMemoryDescriptor OPTIONAL
)
/*++
Routine Description:
This routine removes memory from the system loader block thus simulating
a machine with less physical memory without having to physically remove it.
Arguments:
LoaderBlock - Supplies a pointer to the loader parameter block.
NumberOfPagesToBurn - Supplies the number of pages to burn.
MemoryTypeForRemovedPages - Supplies the type to mark into the loader block
for the burned pages.
NewMemoryDescriptor - If non-NULL, this supplies a pointer to a memory
block to be used if a split is needed.
Return Value:
Number of pages actually burned.
Environment:
Kernel mode.
--*/
{
PLIST_ENTRY ListHead;
PLIST_ENTRY NextEntry;
PFN_COUNT PagesRemaining;
PMEMORY_ALLOCATION_DESCRIPTOR MemoryDescriptor;
PagesRemaining = NumberOfPagesToBurn;
//
// Look backwards through physical memory to leave it like
// it otherwise would be. ie: that's the way most people add memory
// modules to their systems.
//
ListHead = &LoaderBlock->MemoryDescriptorListHead;
NextEntry = ListHead->Blink;
do {
MemoryDescriptor = CONTAINING_RECORD(NextEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
if ((MemoryDescriptor->MemoryType == LoaderFree ||
MemoryDescriptor->MemoryType == LoaderFirmwareTemporary) &&
MemoryDescriptor->PageCount != 0) {
if (MemoryDescriptor->PageCount > PagesRemaining) {
//
// This block has enough pages.
// Split it into two and mark it as requested.
//
MemoryDescriptor->PageCount = MemoryDescriptor->PageCount -
PagesRemaining;
if (ARGUMENT_PRESENT (NewMemoryDescriptor)) {
NewMemoryDescriptor->BasePage = MemoryDescriptor->BasePage +
MemoryDescriptor->PageCount;
NewMemoryDescriptor->PageCount = PagesRemaining;
NewMemoryDescriptor->MemoryType = MemoryTypeForRemovedPages;
InsertTailList (MemoryDescriptor->ListEntry.Blink,
&NewMemoryDescriptor->ListEntry);
}
PagesRemaining = 0;
break;
}
PagesRemaining -= MemoryDescriptor->PageCount;
MemoryDescriptor->MemoryType = MemoryTypeForRemovedPages;
}
NextEntry = NextEntry->Blink;
} while (NextEntry != ListHead);
return NumberOfPagesToBurn - PagesRemaining;
}
extern BOOLEAN ExpInTextModeSetup;
BOOLEAN
ExpIsLoaderValid(
IN PLOADER_PARAMETER_BLOCK LoaderBlock
)
{
ULONG majorVersion;
ULONG minorVersion;
CHAR versionBuffer[64];
PCHAR major;
PCHAR minor;
ULONG minSize;
//
// Make sure that we got loaded by a matching or newer loader.
// First do a size check to make sure we can even read the version fields.
//
minSize = FIELD_OFFSET(LOADER_PARAMETER_EXTENSION, MinorVersion) + RTL_FIELD_SIZE(LOADER_PARAMETER_EXTENSION, MinorVersion);
if (LoaderBlock->Extension->Size >= minSize) {
//
// Safe to do the version check.
//
major = strcpy(versionBuffer, VER_PRODUCTVERSION_STR);
minor = strchr(major, '.');
majorVersion = atoi(major);
if( minor != NULL ) {
*minor++ = '\0';
minorVersion = atoi(minor);
} else {
minorVersion = 0;
}
//
// Check the version.
//
if (LoaderBlock->Extension->MajorVersion > majorVersion ||
(LoaderBlock->Extension->MajorVersion == majorVersion &&
LoaderBlock->Extension->MinorVersion >= minorVersion)) {
return TRUE;
}
}
return FALSE;
}
VOID
ExpInitializeExecutive(
IN ULONG Number,
IN PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
This routine is called from the kernel initialization routine during
bootstrap to initialize the executive and all of its subcomponents.
Each subcomponent is potentially called twice to perform Phase 0, and
then Phase 1 initialization. During Phase 0 initialization, the only
activity that may be performed is the initialization of subcomponent
specific data. Phase 0 initialization is performed in the context of
the kernel start up routine with interrupts disabled. During Phase 1
initialization, the system is fully operational and subcomponents may
do any initialization that is necessary.
Arguments:
Number - Supplies the processor number currently initializing.
LoaderBlock - Supplies a pointer to a loader parameter block.
Return Value:
None.
--*/
{
PFN_COUNT PagesToBurn;
PCHAR Options;
PCHAR MemoryOption;
NTSTATUS Status;
PKLDR_DATA_TABLE_ENTRY DataTableEntry;
PMESSAGE_RESOURCE_ENTRY MessageEntry;
PLIST_ENTRY NextEntry;
ANSI_STRING AnsiString;
STRING NameString;
CHAR Buffer[ 256 ];
BOOLEAN BufferSizeOk;
ULONG ImageCount;
ULONG i;
ULONG_PTR ResourceIdPath[3];
PIMAGE_RESOURCE_DATA_ENTRY ResourceDataEntry;
PMESSAGE_RESOURCE_DATA MessageData;
CHAR VersionBuffer[ 64 ];
PCHAR s;
PLIST_ENTRY NextMd;
PMEMORY_ALLOCATION_DESCRIPTOR MemoryDescriptor;
if (!ExpIsLoaderValid(LoaderBlock)) {
KeBugCheckEx(MISMATCHED_HAL,
3,
LoaderBlock->Extension->Size,
LoaderBlock->Extension->MajorVersion,
LoaderBlock->Extension->MinorVersion
);
}
//
// Initialize PRCB pool lookaside pointers.
//
ExInitPoolLookasidePointers ();
if (Number == 0) {
//
// Determine whether this is textmode setup and whether this is a
// remote boot client.
//
ExpInTextModeSetup = FALSE;
IoRemoteBootClient = FALSE;
if (LoaderBlock->SetupLoaderBlock != NULL) {
if ((LoaderBlock->SetupLoaderBlock->Flags & SETUPBLK_FLAGS_IS_TEXTMODE) != 0) {
ExpInTextModeSetup = TRUE;
}
if ((LoaderBlock->SetupLoaderBlock->Flags & SETUPBLK_FLAGS_IS_REMOTE_BOOT) != 0) {
IoRemoteBootClient = TRUE;
ASSERT( _memicmp( LoaderBlock->ArcBootDeviceName, "net(0)", 6 ) == 0 );
}
}
#if defined(REMOTE_BOOT)
SharedUserData->SystemFlags = 0;
if (IoRemoteBootClient) {
SharedUserData->SystemFlags |= SYSTEM_FLAG_REMOTE_BOOT_CLIENT;
}
#endif // defined(REMOTE_BOOT)
//
// Indicate that we are in phase 0.
//
InitializationPhase = 0L;
Options = LoaderBlock->LoadOptions;
if (Options != NULL) {
//
// If in BBT mode, remove the requested amount of memory from the
// loader block and use it for BBT purposes instead.
//
_strupr(Options);
MemoryOption = strstr(Options, "PERFMEM");
if (MemoryOption != NULL) {
MemoryOption = strstr (MemoryOption,"=");
if (MemoryOption != NULL) {
PagesToBurn = (PFN_COUNT) atol (MemoryOption + 1);
//
// Convert MB to pages.
//
PagesToBurn *= ((1024 * 1024) / PAGE_SIZE);
if (PagesToBurn != 0) {
PERFINFO_INIT_TRACEFLAGS(Options, MemoryOption);
BBTPagesToReserve = ExBurnMemory (LoaderBlock,
PagesToBurn,
LoaderBBTMemory,
&BBTMemoryDescriptor);
}
}
}
//
// Burn memory - consume the amount of memory
// specified in the OS Load Options. This is used
// for testing reduced memory configurations.
//
MemoryOption = strstr(Options, "BURNMEMORY");
if (MemoryOption != NULL) {
MemoryOption = strstr(MemoryOption,"=");
if (MemoryOption != NULL ) {
PagesToBurn = (PFN_COUNT) atol (MemoryOption + 1);
//
// Convert MB to pages.
//
PagesToBurn *= ((1024 * 1024) / PAGE_SIZE);
if (PagesToBurn != 0) {
ExBurnMemory (LoaderBlock,
PagesToBurn,
LoaderBad,
NULL);
}
}
}
}
//
// Initialize the translation tables using the loader
// loaded tables.
//
InitNlsTableBase = LoaderBlock->NlsData->AnsiCodePageData;
InitAnsiCodePageDataOffset = 0;
InitOemCodePageDataOffset = (ULONG)((PUCHAR)LoaderBlock->NlsData->OemCodePageData - (PUCHAR)LoaderBlock->NlsData->AnsiCodePageData);
InitUnicodeCaseTableDataOffset = (ULONG)((PUCHAR)LoaderBlock->NlsData->UnicodeCaseTableData - (PUCHAR)LoaderBlock->NlsData->AnsiCodePageData);
RtlInitNlsTables(
(PVOID)((PUCHAR)InitNlsTableBase+InitAnsiCodePageDataOffset),
(PVOID)((PUCHAR)InitNlsTableBase+InitOemCodePageDataOffset),
(PVOID)((PUCHAR)InitNlsTableBase+InitUnicodeCaseTableDataOffset),
&InitTableInfo
);
RtlResetRtlTranslations(&InitTableInfo);
//
// Initialize the Hardware Architecture Layer (HAL).
//
if (HalInitSystem(InitializationPhase, LoaderBlock) == FALSE) {
KeBugCheck(HAL_INITIALIZATION_FAILED);
}
#if defined(_APIC_TPR_)
HalpIRQLToTPR = LoaderBlock->Extension->HalpIRQLToTPR;
HalpVectorToIRQL = LoaderBlock->Extension->HalpVectorToIRQL;
#endif
//
// Enable interrupts now that the HAL has initialized.
//
#if defined(_X86_)
_enable();
#endif
//
// Set the interrupt time forward so the Win32 tick count will wrap
// within one hour to make rollover errors show up in fewer than 49.7
// days.
//
#if DBG
KeAdjustInterruptTime((LONGLONG)(MAXULONG - (60 * 60 * 1000)) * 10 * 1000);
#endif
//
// Initialize the crypto exponent... Set to 0 when systems leave ms!
//
#ifdef TEST_BUILD_EXPONENT
#pragma message("WARNING: building kernel with TESTKEY enabled!")
#else
#define TEST_BUILD_EXPONENT 0
#endif
SharedUserData->CryptoExponent = TEST_BUILD_EXPONENT;
#if DBG
NtGlobalFlag |= FLG_ENABLE_CLOSE_EXCEPTIONS |
FLG_ENABLE_KDEBUG_SYMBOL_LOAD;
#endif
sprintf( Buffer, "C:%s", LoaderBlock->NtBootPathName );
RtlInitString( &AnsiString, Buffer );
Buffer[ --AnsiString.Length ] = '\0';
NtSystemRoot.Buffer = SharedUserData->NtSystemRoot;
NtSystemRoot.MaximumLength = sizeof( SharedUserData->NtSystemRoot );
NtSystemRoot.Length = 0;
Status = RtlAnsiStringToUnicodeString( &NtSystemRoot,
&AnsiString,
FALSE
);
if (!NT_SUCCESS( Status )) {
KeBugCheck(SESSION3_INITIALIZATION_FAILED);
}
//
// Find the address of BugCheck message block resource and put it
// in KiBugCodeMessages.
//
// WARNING: This code assumes that the KLDR_DATA_TABLE_ENTRY for
// ntoskrnl.exe is always the first in the loaded module list.
//
DataTableEntry = CONTAINING_RECORD(LoaderBlock->LoadOrderListHead.Flink,
KLDR_DATA_TABLE_ENTRY,
InLoadOrderLinks);
ResourceIdPath[0] = 11;
ResourceIdPath[1] = 1;
ResourceIdPath[2] = 0;
Status = LdrFindResource_U (DataTableEntry->DllBase,
ResourceIdPath,
3,
(VOID *) &ResourceDataEntry);
if (NT_SUCCESS(Status)) {
Status = LdrAccessResource (DataTableEntry->DllBase,
ResourceDataEntry,
&MessageData,
NULL);
if (NT_SUCCESS(Status)) {
KiBugCodeMessages = MessageData;
}
}
#if !defined(NT_UP)
//
// Verify that the kernel and HAL images are suitable for MP systems.
//
// N.B. Loading of kernel and HAL symbols now occurs in kdinit.
//
ImageCount = 0;
NextEntry = LoaderBlock->LoadOrderListHead.Flink;
while ((NextEntry != &LoaderBlock->LoadOrderListHead) && (ImageCount < 2)) {
DataTableEntry = CONTAINING_RECORD(NextEntry,
KLDR_DATA_TABLE_ENTRY,
InLoadOrderLinks);
ImageCount += 1;
if ( !MmVerifyImageIsOkForMpUse(DataTableEntry->DllBase) ) {
KeBugCheckEx(UP_DRIVER_ON_MP_SYSTEM,
(ULONG_PTR)DataTableEntry->DllBase,
0,
0,
0);
}
NextEntry = NextEntry->Flink;
}
#endif // !defined(NT_UP)
//
// Get system control values out of the registry.
//
CmGetSystemControlValues(LoaderBlock->RegistryBase, &CmControlVector[0]);
CmNtGlobalFlag &= FLG_VALID_BITS; // Toss bogus bits.
#ifdef VER_PRODUCTRCVERSION
if ((CmNtCSDVersion & 0xFFFF0000) == 0) {
CmNtCSDVersion |= VER_PRODUCTRCVERSION << 16;
}
#endif
NtGlobalFlag |= CmNtGlobalFlag;
#if !DBG
if (!(CmNtGlobalFlag & FLG_ENABLE_KDEBUG_SYMBOL_LOAD)) {
NtGlobalFlag &= ~FLG_ENABLE_KDEBUG_SYMBOL_LOAD;
}
#endif
//
// Initialize the ExResource package.
//
if (!ExInitSystem()) {
KeBugCheck(PHASE0_INITIALIZATION_FAILED);
}
//
// Get multinode configuration (if any).
//
KeNumaInitialize();
//
// Initialize memory management and the memory allocation pools.
//
MmInitSystem (0, LoaderBlock);
//
// Scan the loaded module list and load the driver image symbols.
//
ImageCount = 0;
NextEntry = LoaderBlock->LoadOrderListHead.Flink;
while (NextEntry != &LoaderBlock->LoadOrderListHead) {
BufferSizeOk = TRUE;
if (ImageCount >= 2) {
ULONG Count;
WCHAR *Filename;
ULONG Length;
//
// Get the address of the data table entry for the next component.
//
DataTableEntry = CONTAINING_RECORD(NextEntry,
KLDR_DATA_TABLE_ENTRY,
InLoadOrderLinks);
//
// Load the symbols via the kernel debugger
// for the next component.
//
if (DataTableEntry->FullDllName.Buffer[0] == L'\\') {
//
// Correct fullname already available
//
Filename = DataTableEntry->FullDllName.Buffer;
Length = DataTableEntry->FullDllName.Length / sizeof(WCHAR);
if (sizeof(Buffer) < Length + sizeof(ANSI_NULL)) {
//
// DllName too long.
//
BufferSizeOk = FALSE;
} else {
Count = 0;
do {
Buffer[Count++] = (CHAR)*Filename++;
} while (Count < Length);
Buffer[Count] = 0;
}
} else {
//
// Assume drivers
//
if (sizeof(Buffer) < 18 + NtSystemRoot.Length / sizeof(WCHAR) - 2
+ DataTableEntry->BaseDllName.Length / sizeof(WCHAR)
+ sizeof(ANSI_NULL)) {
//
// ignore the driver entry, it must have been corrupt.
//
BufferSizeOk = FALSE;
} else {
sprintf (Buffer, "%ws\\System32\\Drivers\\%wZ",
&SharedUserData->NtSystemRoot[2],
&DataTableEntry->BaseDllName);
}
}
if (BufferSizeOk) {
RtlInitString (&NameString, Buffer );
DbgLoadImageSymbols (&NameString,
DataTableEntry->DllBase,
(ULONG)-1);
#if !defined(NT_UP)
if (!MmVerifyImageIsOkForMpUse(DataTableEntry->DllBase)) {
KeBugCheckEx(UP_DRIVER_ON_MP_SYSTEM,(ULONG_PTR)DataTableEntry->DllBase,0,0,0);
}
#endif // NT_UP
}
}
ImageCount += 1;
NextEntry = NextEntry->Flink;
}
//
// If break after symbol load is specified, then break into the
// debugger.
//
if (KdBreakAfterSymbolLoad != FALSE) {
DbgBreakPointWithStatus(DBG_STATUS_CONTROL_C);
}
//
// Turn on the headless terminal now, if we are of a sufficiently
// new vintage of loader
//
if (LoaderBlock->Extension->Size >= sizeof (LOADER_PARAMETER_EXTENSION)) {
HeadlessInit(LoaderBlock);
}
//
// These fields are supported for legacy 3rd party 32-bit software
// only. New code should call NtQueryInformationSystem() to get them.
//
#if defined(_WIN64)
SharedUserData->Reserved1 = 0x7ffeffff; // 2gb HighestUserAddress
SharedUserData->Reserved3 = 0x80000000; // 2gb SystemRangeStart
#else
//
// Set the highest user address and the start of the system range in
// the shared memory block.
//
// N.B. This is not a constant value if the target system is an x86
// with 3gb of user virtual address space.
//
SharedUserData->Reserved1 = (ULONG)MM_HIGHEST_USER_ADDRESS;
SharedUserData->Reserved3 = (ULONG)MmSystemRangeStart;
#endif
//
// Snapshot the NLS tables into paged pool and then
// reset the translation tables.
//
// Walk through the memory descriptors and size the NLS data.
//
NextMd = LoaderBlock->MemoryDescriptorListHead.Flink;
while (NextMd != &LoaderBlock->MemoryDescriptorListHead) {
MemoryDescriptor = CONTAINING_RECORD(NextMd,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
if (MemoryDescriptor->MemoryType == LoaderNlsData) {
InitNlsTableSize += MemoryDescriptor->PageCount*PAGE_SIZE;
}
NextMd = MemoryDescriptor->ListEntry.Flink;
}
InitNlsTableBase = ExAllocatePoolWithTag (NonPagedPool,
InitNlsTableSize,
' slN');
if (InitNlsTableBase == NULL) {
KeBugCheck(PHASE0_INITIALIZATION_FAILED);
}
//
// Copy the NLS data into the dynamic buffer so that we can
// free the buffers allocated by the loader. The loader guarantees
// contiguous buffers and the base of all the tables is the ANSI
// code page data.
//
RtlCopyMemory (InitNlsTableBase,
LoaderBlock->NlsData->AnsiCodePageData,
InitNlsTableSize);
RtlInitNlsTables ((PVOID)((PUCHAR)InitNlsTableBase+InitAnsiCodePageDataOffset),
(PVOID)((PUCHAR)InitNlsTableBase+InitOemCodePageDataOffset),
(PVOID)((PUCHAR)InitNlsTableBase+InitUnicodeCaseTableDataOffset),
&InitTableInfo);
RtlResetRtlTranslations (&InitTableInfo);
//
// Determine System version information.
//
DataTableEntry = CONTAINING_RECORD(LoaderBlock->LoadOrderListHead.Flink,
KLDR_DATA_TABLE_ENTRY,
InLoadOrderLinks);
if (CmNtCSDVersion & 0xFFFF) {
Status = RtlFindMessage (DataTableEntry->DllBase, 11, 0,
WINDOWS_NT_CSD_STRING, &MessageEntry);
if (NT_SUCCESS( Status )) {
RtlInitAnsiString( &AnsiString, MessageEntry->Text );
AnsiString.Length -= 2;
sprintf( Buffer,
"%Z %u%c",
&AnsiString,
(CmNtCSDVersion & 0xFF00) >> 8,
(CmNtCSDVersion & 0xFF) ? 'A' + (CmNtCSDVersion & 0xFF) - 1 : '\0');
}
else {
sprintf( Buffer, "CSD %04x", CmNtCSDVersion );
}
}
else {
CmCSDVersionString.MaximumLength = (USHORT) sprintf( Buffer, VER_PRODUCTBETA_STR );
}
//
// High-order 16-bits of CSDVersion contain RC number. If non-zero
// display it after the Service Pack number.
//
if (CmNtCSDVersion & 0xFFFF0000) {
s = Buffer + strlen( Buffer );
if (s != Buffer) {
*s++ = ',';
*s++ = ' ';
}
Status = RtlFindMessage (DataTableEntry->DllBase, 11, 0,
WINDOWS_NT_RC_STRING, &MessageEntry);
if (NT_SUCCESS(Status)) {
RtlInitAnsiString( &AnsiString, MessageEntry->Text );
AnsiString.Length -= 2;
}
else {
RtlInitAnsiString( &AnsiString, "RC" );
}
s += sprintf( s,
"%Z %u",
&AnsiString,
(CmNtCSDVersion & 0xFF000000) >> 24
);
if (CmNtCSDVersion & 0x00FF0000) {
s += sprintf( s, ".%u", (CmNtCSDVersion & 0x00FF0000) >> 16 );
}
*s++ = '\0';
}
RtlInitAnsiString( &AnsiString, Buffer );
Status = RtlAnsiStringToUnicodeString( &CmCSDVersionString, &AnsiString, TRUE );
if (!NT_SUCCESS (Status)) {
KeBugCheckEx(PHASE0_INITIALIZATION_FAILED,Status,0,0,0);
}
sprintf( VersionBuffer, "%u.%u", NtMajorVersion, NtMinorVersion );
RtlCreateUnicodeStringFromAsciiz( &CmVersionString, VersionBuffer );
if (NtGlobalFlag & FLG_KERNEL_STACK_TRACE_DB) {
PVOID StackTraceDataBase;
ULONG StackTraceDataBaseLength;
NTSTATUS Status;
StackTraceDataBaseLength = 512 * 1024;
switch ( MmQuerySystemSize() ) {
case MmMediumSystem :
StackTraceDataBaseLength = 1024 * 1024;
break;
case MmLargeSystem :
StackTraceDataBaseLength = 2048 * 1024;
break;
}
StackTraceDataBase = ExAllocatePoolWithTag( NonPagedPool,
StackTraceDataBaseLength,
'catS');
if (StackTraceDataBase != NULL) {
KdPrint(( "INIT: Kernel mode stack back trace enabled "
"with %u KB buffer.\n", StackTraceDataBaseLength / 1024 ));
Status = RtlInitializeStackTraceDataBase (StackTraceDataBase,
StackTraceDataBaseLength,
StackTraceDataBaseLength);
} else {
Status = STATUS_NO_MEMORY;
}
if (!NT_SUCCESS( Status )) {
KdPrint(( "INIT: Unable to initialize stack trace data base - Status == %lx\n", Status ));
}
}
if (NtGlobalFlag & FLG_ENABLE_EXCEPTION_LOGGING) {
RtlInitializeExceptionLog(MAX_EXCEPTION_LOG);
}
ExInitializeHandleTablePackage();
#if DBG
//
// Allocate and zero the system service count table.
//
KeServiceDescriptorTable[0].Count =
(PULONG)ExAllocatePoolWithTag(NonPagedPool,
KiServiceLimit * sizeof(ULONG),
'llac');
KeServiceDescriptorTableShadow[0].Count = KeServiceDescriptorTable[0].Count;
if (KeServiceDescriptorTable[0].Count != NULL ) {
RtlZeroMemory((PVOID)KeServiceDescriptorTable[0].Count,
KiServiceLimit * sizeof(ULONG));
}
#endif
if (!ObInitSystem()) {
KeBugCheck(OBJECT_INITIALIZATION_FAILED);
}
if (!SeInitSystem()) {
KeBugCheck(SECURITY_INITIALIZATION_FAILED);
}
if (PsInitSystem(0, LoaderBlock) == FALSE) {
KeBugCheck(PROCESS_INITIALIZATION_FAILED);
}
if (!PpInitSystem()) {
KeBugCheck(PP0_INITIALIZATION_FAILED);
}
//
// Initialize debug system.
//
DbgkInitialize ();
//
// Compute the tick count multiplier that is used for computing the
// windows millisecond tick count and copy the resultant value to
// the memory that is shared between user and kernel mode.
//
ExpTickCountMultiplier = ExComputeTickCountMultiplier(KeMaximumIncrement);
SharedUserData->TickCountMultiplier = ExpTickCountMultiplier;
//
// Set the base os version into shared memory
//
SharedUserData->NtMajorVersion = NtMajorVersion;
SharedUserData->NtMinorVersion = NtMinorVersion;
//
// Set the supported image number range used to determine by the
// loader if a particular image can be executed on the host system.
// Eventually this will need to be dynamically computed. Also set
// the architecture specific feature bits.
//
#if defined(_AMD64_)
SharedUserData->ImageNumberLow = IMAGE_FILE_MACHINE_AMD64;
SharedUserData->ImageNumberHigh = IMAGE_FILE_MACHINE_AMD64;
#elif defined(_X86_)
SharedUserData->ImageNumberLow = IMAGE_FILE_MACHINE_I386;
SharedUserData->ImageNumberHigh = IMAGE_FILE_MACHINE_I386;
#elif defined(_IA64_)
SharedUserData->ImageNumberLow = IMAGE_FILE_MACHINE_IA64;
SharedUserData->ImageNumberHigh = IMAGE_FILE_MACHINE_IA64;
#else
#error "no target architecture"
#endif
}
else {
//
// Initialize the Hardware Architecture Layer (HAL).
//
if (HalInitSystem(InitializationPhase, LoaderBlock) == FALSE) {
KeBugCheck(HAL_INITIALIZATION_FAILED);
}
}
return;
}
VOID
xcpt4 (
VOID
);
VOID
Phase1Initialization(
IN PVOID Context
)
{
PCHAR s;
PLOADER_PARAMETER_BLOCK LoaderBlock;
PETHREAD Thread;
PKPRCB Prcb;
KPRIORITY Priority;
NTSTATUS Status;
UNICODE_STRING SessionManager;
PRTL_USER_PROCESS_PARAMETERS ProcessParameters;
PVOID Address;
PFN_COUNT MemorySize;
SIZE_T Size;
ULONG Index;
RTL_USER_PROCESS_INFORMATION ProcessInformation;
LARGE_INTEGER UniversalTime;
LARGE_INTEGER CmosTime;
LARGE_INTEGER OldTime;
TIME_FIELDS TimeFields;
UNICODE_STRING UnicodeDebugString;
ANSI_STRING AnsiDebugString;
UNICODE_STRING EnvString, NullString, UnicodeSystemDriveString;
CHAR DebugBuffer[256];
CHAR BootLogBuffer[256]; // must be the same size as DebugBuffer
PWSTR Src, Dst;
BOOLEAN ResetActiveTimeBias;
HANDLE NlsSection;
LARGE_INTEGER SectionSize;
LARGE_INTEGER SectionOffset;
PVOID SectionBase;
PVOID ViewBase;
ULONG CacheViewSize;
SIZE_T CapturedViewSize;
ULONG SavedViewSize;
LONG BootTimeZoneBias;
PKLDR_DATA_TABLE_ENTRY DataTableEntry;
CHAR VersionBuffer[24];
PMESSAGE_RESOURCE_ENTRY MessageEntry;
#ifndef NT_UP
PMESSAGE_RESOURCE_ENTRY MessageEntry1;
#endif
PCHAR MPKernelString;
PCHAR Options;
PCHAR YearOverrideOption, SafeModeOption, BootLogOption;
LONG CurrentYear = 0;
PSTR SafeBoot;
BOOLEAN UseAlternateShell = FALSE;
#if defined(REMOTE_BOOT)
BOOLEAN NetBootRequiresFormat = FALSE;
BOOLEAN NetBootDisconnected = FALSE;
CHAR NetBootHalName[MAX_HAL_NAME_LENGTH + 1];
UNICODE_STRING TmpUnicodeString;
#endif // defined(REMOTE_BOOT)
BOOLEAN NOGUIBOOT;
BOOLEAN SOS;
PVOID Environment;
//
// The following is a dummy reference to an inline function to force a
// reference to the function so it won't get discard before it can be
// exported.
//
KeAreApcsDisabled();
//
// Set the phase number and raise the priority of current thread to
// a high priority so it will not be preempted during initialization.
//
ResetActiveTimeBias = FALSE;
InitializationPhase = 1;
Thread = PsGetCurrentThread();
Priority = KeSetPriorityThread( &Thread->Tcb,MAXIMUM_PRIORITY - 1 );
LoaderBlock = (PLOADER_PARAMETER_BLOCK)Context;
//
// Put Phase 1 initialization calls here.
//
if (HalInitSystem(InitializationPhase, LoaderBlock) == FALSE) {
KeBugCheck(HAL1_INITIALIZATION_FAILED);
}
//
// Allow the boot video driver to behave differently based on the
// OsLoadOptions.
//
Options = LoaderBlock->LoadOptions ? _strupr(LoaderBlock->LoadOptions) : NULL;
if (Options) {
NOGUIBOOT = (BOOLEAN)(strstr(Options, "NOGUIBOOT") != NULL);
} else {
NOGUIBOOT = FALSE;
}
InbvEnableBootDriver((BOOLEAN)!NOGUIBOOT);
//
// There is now enough functionality for the system Boot Video
// Driver to run.
//
InbvDriverInitialize(LoaderBlock, 18);
if (NOGUIBOOT) {
//
// If the user specified the noguiboot switch we don't want to
// use the bootvid driver, so release display ownership.
//
InbvNotifyDisplayOwnershipLost(NULL);
}
if (Options) {
SOS = (BOOLEAN)(strstr(Options, "SOS") != NULL);
} else {
SOS = FALSE;
}
if (NOGUIBOOT) {
InbvEnableDisplayString(FALSE);
} else {
InbvEnableDisplayString(SOS);
DisplayBootBitmap(SOS);
}
//
// Check whether we are booting into WinPE
//
if (Options) {
if (strstr(Options, "MININT") != NULL) {
InitIsWinPEMode = TRUE;
if (strstr(Options, "INRAM") != NULL) {
InitWinPEModeType |= INIT_WINPEMODE_INRAM;
} else {
InitWinPEModeType |= INIT_WINPEMODE_REGULAR;
}
}
}
//
// Now that the HAL is available and memory management has sized
// memory, display the initial system banner containing the version number.
// Under normal circumstances, this is the first message displayed
// to the user by the OS.
//
DataTableEntry = CONTAINING_RECORD(LoaderBlock->LoadOrderListHead.Flink,
KLDR_DATA_TABLE_ENTRY,
InLoadOrderLinks);
Status = RtlFindMessage (DataTableEntry->DllBase,
11,
0,
WINDOWS_NT_BANNER,
&MessageEntry);
s = DebugBuffer;
if (CmCSDVersionString.Length != 0) {
s += sprintf( s, ": %wZ", &CmCSDVersionString );
}
*s++ = '\0';
sprintf( VersionBuffer, "%u.%u", NtMajorVersion, NtMinorVersion );
if (NT_SUCCESS(Status)) {
sprintf (s,
MessageEntry->Text,
VersionBuffer,
NtBuildNumber & 0xFFFF,
DebugBuffer);
} else {
//
// Could not find the WINDOWS_NT_BANNER message.
//
sprintf (s, "MICROSOFT (R) WINDOWS (TM)\n");
}
InbvDisplayString(s);
RtlCopyMemory (BootLogBuffer, DebugBuffer, sizeof(DebugBuffer));
//
// Initialize the Power subsystem.
//
if (!PoInitSystem(0)) {
KeBugCheck(INTERNAL_POWER_ERROR);
}
//
// The user may have put a /YEAR=2000 switch on
// the OSLOADOPTIONS line. This allows us to
// enforce a particular year on hardware that
// has a broken clock.
//
if (Options) {
YearOverrideOption = strstr(Options, "YEAR");
if (YearOverrideOption != NULL) {
YearOverrideOption = strstr(YearOverrideOption,"=");
}
if (YearOverrideOption != NULL) {
CurrentYear = atol(YearOverrideOption + 1);
}
}
//
// Initialize the system time and set the time the system was booted.
//
// N.B. This cannot be done until after the phase one initialization
// of the HAL Layer.
//
if (ExCmosClockIsSane
&& HalQueryRealTimeClock(&TimeFields)) {
//
// If appropriate, override the year.
//
if (YearOverrideOption) {
TimeFields.Year = (SHORT)CurrentYear;
}
RtlTimeFieldsToTime(&TimeFields, &CmosTime);
UniversalTime = CmosTime;
if ( !ExpRealTimeIsUniversal ) {
//
// If the system stores time in local time. This is converted to
// universal time before going any further
//
// If we have previously set the time through NT, then
// ExpLastTimeZoneBias should contain the timezone bias in effect
// when the clock was set. Otherwise, we will have to resort to
// our next best guess which would be the programmed bias stored in
// the registry
//
if ( ExpLastTimeZoneBias == -1 ) {
ResetActiveTimeBias = TRUE;
ExpLastTimeZoneBias = ExpAltTimeZoneBias;
}
ExpTimeZoneBias.QuadPart = Int32x32To64(
ExpLastTimeZoneBias*60, // Bias in seconds
10000000
);
SharedUserData->TimeZoneBias.High2Time = ExpTimeZoneBias.HighPart;
SharedUserData->TimeZoneBias.LowPart = ExpTimeZoneBias.LowPart;
SharedUserData->TimeZoneBias.High1Time = ExpTimeZoneBias.HighPart;
UniversalTime.QuadPart = CmosTime.QuadPart + ExpTimeZoneBias.QuadPart;
}
KeSetSystemTime(&UniversalTime, &OldTime, FALSE, NULL);
//
// Notify other components that the system time has been set
//
PoNotifySystemTimeSet();
KeBootTime = UniversalTime;
KeBootTimeBias = 0;
}
MPKernelString = "";
#ifndef NT_UP
//
// Enforce processor licensing.
//
if (KeLicensedProcessors) {
if (KeRegisteredProcessors > KeLicensedProcessors) {
KeRegisteredProcessors = KeLicensedProcessors;
}
}
if (Options) {
ULONG NewRegisteredProcessors;
PCHAR NumProcOption;
NumProcOption = strstr(Options, "NUMPROC");
if (NumProcOption != NULL) {
NumProcOption = strstr(NumProcOption,"=");
}
if (NumProcOption != NULL) {
NewRegisteredProcessors = atol(NumProcOption+1);
if (NewRegisteredProcessors < KeRegisteredProcessors) {
KeRegisteredProcessors = NewRegisteredProcessors;
}
#if defined(_X86_)
KeNumprocSpecified = NewRegisteredProcessors;
#endif
}
}
//
// If this is an MP build of the kernel start any other processors now
//
KeStartAllProcessors();
//
// Since starting processors has thrown off the system time, get it again
// from the RTC and set the system time again.
//
if (ExCmosClockIsSane
&& HalQueryRealTimeClock(&TimeFields)) {
if (YearOverrideOption) {
TimeFields.Year = (SHORT)CurrentYear;
}
RtlTimeFieldsToTime(&TimeFields, &CmosTime);
if ( !ExpRealTimeIsUniversal ) {
UniversalTime.QuadPart = CmosTime.QuadPart + ExpTimeZoneBias.QuadPart;
}
KeSetSystemTime(&UniversalTime, &OldTime, TRUE, NULL);
}
//
// Set the affinity of the system process and all of its threads to
// all processors in the host configuration.
//
KeSetAffinityProcess(KeGetCurrentThread()->ApcState.Process,
KeActiveProcessors);
Status = RtlFindMessage (DataTableEntry->DllBase, 11, 0,
WINDOWS_NT_MP_STRING, &MessageEntry1);
if (NT_SUCCESS( Status )) {
MPKernelString = MessageEntry1->Text;
}
else {
MPKernelString = "MultiProcessor Kernel\r\n";
}
#endif
//
// Signify to the HAL that all processors have been started and any
// post initialization should be performed.
//
if (!HalAllProcessorsStarted()) {
KeBugCheck(HAL1_INITIALIZATION_FAILED);
}
RtlInitAnsiString( &AnsiDebugString, MPKernelString );
if (AnsiDebugString.Length >= 2) {
AnsiDebugString.Length -= 2;
}
//
// Now that the processors have started, display number of processors
// and size of memory.
//
Status = RtlFindMessage( DataTableEntry->DllBase,
11,
0,
KeNumberProcessors > 1 ? WINDOWS_NT_INFO_STRING_PLURAL
: WINDOWS_NT_INFO_STRING,
&MessageEntry
);
MemorySize = 0;
for (Index=0; Index < MmPhysicalMemoryBlock->NumberOfRuns; Index++) {
MemorySize += (PFN_COUNT)MmPhysicalMemoryBlock->Run[Index].PageCount;
}
sprintf (DebugBuffer,
NT_SUCCESS(Status) ? MessageEntry->Text : "%u System Processor [%u MB Memory] %Z\n",
KeNumberProcessors,
(MemorySize + (1 << (20 - PAGE_SHIFT)) - 1) >> (20 - PAGE_SHIFT),
&AnsiDebugString);
InbvDisplayString(DebugBuffer);
InbvUpdateProgressBar(5);
#if defined(REMOTE_BOOT)
//
// Save any information from NetBoot for later.
//
if (IoRemoteBootClient) {
ULONG Flags;
ASSERT(LoaderBlock->SetupLoaderBlock != NULL);
Flags = LoaderBlock->SetupLoaderBlock->Flags;
NetBootDisconnected = (BOOLEAN)((Flags & SETUPBLK_FLAGS_DISCONNECTED) != 0);
NetBootRequiresFormat = (BOOLEAN)((Flags & SETUPBLK_FLAGS_FORMAT_NEEDED) != 0);
memcpy(NetBootHalName,
LoaderBlock->SetupLoaderBlock->NetBootHalName,
sizeof(NetBootHalName));
}
#endif // defined(REMOTE_BOOT)
//
// Initialize OB, EX, KE, and KD.
//
if (!ObInitSystem()) {
KeBugCheck(OBJECT1_INITIALIZATION_FAILED);
}
if (!ExInitSystem()) {
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,STATUS_UNSUCCESSFUL,0,1,0);
}
if (!KeInitSystem()) {
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,STATUS_UNSUCCESSFUL,0,2,0);
}
if (!KdInitSystem(InitializationPhase, NULL)) {
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,STATUS_UNSUCCESSFUL,0,3,0);
}
//
// SE expects directory and executive objects to be available, but
// must be before device drivers are initialized.
//
if (!SeInitSystem()) {
KeBugCheck(SECURITY1_INITIALIZATION_FAILED);
}
InbvUpdateProgressBar(10);
//
// Create the symbolic link to \SystemRoot.
//
Status = CreateSystemRootLink(LoaderBlock);
if ( !NT_SUCCESS(Status) ) {
KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED,Status,0,0,0);
}
if (MmInitSystem(1, LoaderBlock) == FALSE) {
KeBugCheck(MEMORY1_INITIALIZATION_FAILED);
}
//
// Snapshot the NLS tables into a page file backed section, and then
// reset the translation tables.
//
SectionSize.HighPart = 0;
SectionSize.LowPart = InitNlsTableSize;
Status = ZwCreateSection(
&NlsSection,
SECTION_ALL_ACCESS,
NULL,
&SectionSize,
PAGE_READWRITE,
SEC_COMMIT,
NULL
);
if (!NT_SUCCESS(Status)) {
KdPrint(("INIT: Nls Section Creation Failed %x\n",Status));
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,Status,1,0,0);
}
Status = ObReferenceObjectByHandle(
NlsSection,
SECTION_ALL_ACCESS,
MmSectionObjectType,
KernelMode,
&InitNlsSectionPointer,
NULL
);
ZwClose(NlsSection);
if ( !NT_SUCCESS(Status) ) {
KdPrint(("INIT: Nls Section Reference Failed %x\n",Status));
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,Status,2,0,0);
}
SectionBase = NULL;
CacheViewSize = SectionSize.LowPart;
SavedViewSize = CacheViewSize;
SectionSize.LowPart = 0;
Status = MmMapViewInSystemCache (InitNlsSectionPointer,
&SectionBase,
&SectionSize,
&CacheViewSize);
if (!NT_SUCCESS(Status)) {
KdPrint(("INIT: Map In System Cache Failed %x\n",Status));
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,Status,3,0,0);
}
//
// Copy the NLS data into the dynamic buffer so that we can
// free the buffers allocated by the loader. The loader guarantees
// contiguous buffers and the base of all the tables is the ANSI
// code page data.
//
RtlCopyMemory (SectionBase, InitNlsTableBase, InitNlsTableSize);
//
// Unmap the view to remove all pages from memory. This prevents
// these tables from consuming memory in the system cache while
// the system cache is underutilized during bootup.
//
MmUnmapViewInSystemCache (SectionBase, InitNlsSectionPointer, FALSE);
SectionBase = NULL;
//
// Map it back into the system cache, but now the pages will no
// longer be valid.
//
Status = MmMapViewInSystemCache(
InitNlsSectionPointer,
&SectionBase,
&SectionSize,
&SavedViewSize
);
if ( !NT_SUCCESS(Status) ) {
KdPrint(("INIT: Map In System Cache Failed %x\n",Status));
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,Status,4,0,0);
}
ExFreePool(InitNlsTableBase);
InitNlsTableBase = SectionBase;
RtlInitNlsTables(
(PVOID)((PUCHAR)InitNlsTableBase+InitAnsiCodePageDataOffset),
(PVOID)((PUCHAR)InitNlsTableBase+InitOemCodePageDataOffset),
(PVOID)((PUCHAR)InitNlsTableBase+InitUnicodeCaseTableDataOffset),
&InitTableInfo
);
RtlResetRtlTranslations(&InitTableInfo);
ViewBase = NULL;
SectionOffset.LowPart = 0;
SectionOffset.HighPart = 0;
CapturedViewSize = 0;
//
// Map the system dll into the user part of the address space
//
Status = MmMapViewOfSection (InitNlsSectionPointer,
PsGetCurrentProcess(),
&ViewBase,
0L,
0L,
&SectionOffset,
&CapturedViewSize,
ViewShare,
0L,
PAGE_READWRITE);
if (!NT_SUCCESS(Status)) {
KdPrint(("INIT: Map In User Portion Failed %x\n",Status));
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,Status,5,0,0);
}
RtlCopyMemory (ViewBase, InitNlsTableBase, InitNlsTableSize);
InitNlsTableBase = ViewBase;
//
// Initialize the cache manager.
//
if (!CcInitializeCacheManager()) {
KeBugCheck(CACHE_INITIALIZATION_FAILED);
}
//
// Config management (particularly the registry) gets initialized in
// two parts. Part 1 makes \REGISTRY\MACHINE\SYSTEM and
// \REGISTRY\MACHINE\HARDWARE available. These are needed to
// complete IO init.
//
if (!CmInitSystem1(LoaderBlock)) {
KeBugCheck(CONFIG_INITIALIZATION_FAILED);
}
//
// Initialize the prefetcher after registry is initialized so we can
// query the prefetching parameters.
//
CcPfInitializePrefetcher();
InbvUpdateProgressBar(15);
//
// Compute timezone bias and next cutover date.
//
BootTimeZoneBias = ExpLastTimeZoneBias;
ExpRefreshTimeZoneInformation(&CmosTime);
if (ResetActiveTimeBias) {
ExLocalTimeToSystemTime(&CmosTime,&UniversalTime);
KeBootTime = UniversalTime;
KeBootTimeBias = 0;
KeSetSystemTime(&UniversalTime, &OldTime, FALSE, NULL);
}
else {
//
// Check to see if a timezone switch occurred prior to boot...
//
if (BootTimeZoneBias != ExpLastTimeZoneBias) {
ZwSetSystemTime(NULL,NULL);
}
}
if (!FsRtlInitSystem()) {
KeBugCheck(FILE_INITIALIZATION_FAILED);
}
//
// Initialize the range list package - this must be before PNP
// initialization as PNP uses range lists.
//
RtlInitializeRangeListPackage();
HalReportResourceUsage();
KdDebuggerInitialize1(LoaderBlock);
//
// Perform phase1 initialization of the Plug and Play manager. This
// must be done before the I/O system initializes.
//
if (!PpInitSystem()) {
KeBugCheck(PP1_INITIALIZATION_FAILED);
}
InbvUpdateProgressBar(20);
//
// LPC needs to be initialized before the I/O system, since
// some drivers may create system threads that will terminate
// and cause LPC to be called.
//
if (!LpcInitSystem()) {
KeBugCheck(LPC_INITIALIZATION_FAILED);
}
//
// Check for the existence of the safeboot option.
//
if (Options) {
SafeBoot = strstr(Options,SAFEBOOT_LOAD_OPTION_A);
} else {
SafeBoot = FALSE;
}
if (SafeBoot) {
//
// Isolate the safeboot option.
//
SafeBoot += strlen(SAFEBOOT_LOAD_OPTION_A);
//
// Set the safeboot mode.
//
if (strncmp(SafeBoot,SAFEBOOT_MINIMAL_STR_A,strlen(SAFEBOOT_MINIMAL_STR_A))==0) {
InitSafeBootMode = SAFEBOOT_MINIMAL;
SafeBoot += strlen(SAFEBOOT_MINIMAL_STR_A);
} else if (strncmp(SafeBoot,SAFEBOOT_NETWORK_STR_A,strlen(SAFEBOOT_NETWORK_STR_A))==0) {
InitSafeBootMode = SAFEBOOT_NETWORK;
SafeBoot += strlen(SAFEBOOT_NETWORK_STR_A);
} else if (strncmp(SafeBoot,SAFEBOOT_DSREPAIR_STR_A,strlen(SAFEBOOT_DSREPAIR_STR_A))==0) {
InitSafeBootMode = SAFEBOOT_DSREPAIR;
SafeBoot += strlen(SAFEBOOT_DSREPAIR_STR_A);
} else {
InitSafeBootMode = 0;
}
if (*SafeBoot && strncmp(SafeBoot,SAFEBOOT_ALTERNATESHELL_STR_A,strlen(SAFEBOOT_ALTERNATESHELL_STR_A))==0) {
UseAlternateShell = TRUE;
}
if (InitSafeBootMode) {
PKLDR_DATA_TABLE_ENTRY DataTableEntry;
PMESSAGE_RESOURCE_ENTRY MessageEntry;
ULONG MsgId = 0;
DataTableEntry = CONTAINING_RECORD(LoaderBlock->LoadOrderListHead.Flink,
KLDR_DATA_TABLE_ENTRY,
InLoadOrderLinks);
switch (InitSafeBootMode) {
case SAFEBOOT_MINIMAL:
MsgId = BOOTING_IN_SAFEMODE_MINIMAL;
break;
case SAFEBOOT_NETWORK:
MsgId = BOOTING_IN_SAFEMODE_NETWORK;
break;
case SAFEBOOT_DSREPAIR:
MsgId = BOOTING_IN_SAFEMODE_DSREPAIR;
break;
}
Status = RtlFindMessage (DataTableEntry->DllBase, 11, 0, MsgId, &MessageEntry);
if (NT_SUCCESS( Status )) {
InbvDisplayString(MessageEntry->Text);
}
}
}
//
// Check for the existence of the bootlog option.
//
if (Options) {
BootLogOption = strstr(Options, "BOOTLOG");
} else {
BootLogOption = FALSE;
}
if (BootLogOption) {
Status = RtlFindMessage (DataTableEntry->DllBase, 11, 0, BOOTLOG_ENABLED, &MessageEntry);
if (NT_SUCCESS( Status )) {
InbvDisplayString(MessageEntry->Text);
}
IopInitializeBootLogging(LoaderBlock, BootLogBuffer);
}
//
// Now that system time is running, initialize more of the Executive.
//
ExInitSystemPhase2();
InbvUpdateProgressBar(25);
//
// Allow time slip notification changes.
//
KdpTimeSlipPending = 0;
//
// If we are running XIP, we have to initialize XIP before the I/O system calls xipdisk.sys
// This is defined to be nothing on platforms that do not support XIP.
//
XIPInit(LoaderBlock);
//
// Initialize the Io system.
//
// IoInitSystem updates progress bar updates from 25 to 75 %.
//
InbvSetProgressBarSubset(25, 75);
if (!IoInitSystem(LoaderBlock)) {
KeBugCheck(IO1_INITIALIZATION_FAILED);
}
//
// Clear progress bar subset, goes back to absolute mode.
//
InbvSetProgressBarSubset(0, 100);
//
// Set the registry value that indicates we've booted in safeboot mode.
//
if (InitSafeBootMode) {
HANDLE hSafeBoot,hOption;
UNICODE_STRING string;
OBJECT_ATTRIBUTES objectAttributes;
ULONG disposition;
UCHAR Buffer[FIELD_OFFSET(KEY_VALUE_PARTIAL_INFORMATION, Data) + 32];
ULONG length;
PKEY_VALUE_PARTIAL_INFORMATION keyValue;
InitializeObjectAttributes(
&objectAttributes,
&CmRegistryMachineSystemCurrentControlSetControlSafeBoot,
OBJ_CASE_INSENSITIVE,
NULL,
NULL
);
Status = ZwOpenKey(
&hSafeBoot,
KEY_ALL_ACCESS,
&objectAttributes
);
if (NT_SUCCESS(Status)) {
if (UseAlternateShell) {
RtlInitUnicodeString( &string, L"AlternateShell" );
keyValue = (PKEY_VALUE_PARTIAL_INFORMATION)Buffer;
RtlZeroMemory(Buffer, sizeof(Buffer));
Status = NtQueryValueKey(
hSafeBoot,
&string,
KeyValuePartialInformation,
keyValue,
sizeof(Buffer),
&length
);
if (!NT_SUCCESS(Status)) {
UseAlternateShell = FALSE;
}
}
RtlInitUnicodeString( &string, L"Option" );
InitializeObjectAttributes(
&objectAttributes,
&string,
OBJ_CASE_INSENSITIVE,
hSafeBoot,
NULL
);
Status = ZwCreateKey(
&hOption,
KEY_ALL_ACCESS,
&objectAttributes,
0,
NULL,
REG_OPTION_VOLATILE,
&disposition
);
NtClose(hSafeBoot);
if (NT_SUCCESS(Status)) {
RtlInitUnicodeString( &string, L"OptionValue" );
Status = NtSetValueKey(
hOption,
&string,
0,
REG_DWORD,
&InitSafeBootMode,
sizeof(ULONG)
);
if (UseAlternateShell) {
RtlInitUnicodeString( &string, L"UseAlternateShell" );
Index = 1;
Status = NtSetValueKey(
hOption,
&string,
0,
REG_DWORD,
&Index,
sizeof(ULONG)
);
}
NtClose(hOption);
}
}
}
//
// Create the Mini NT boot key, to indicate to the user mode
// programs that we are in Mini NT environment.
//
if (InitIsWinPEMode) {
WCHAR KeyName[256] = {0};
HANDLE hControl;
UNICODE_STRING String;
OBJECT_ATTRIBUTES ObjAttrs;
ULONG Disposition;
wcsncpy(KeyName, CmRegistryMachineSystemCurrentControlSet.Buffer,
CmRegistryMachineSystemCurrentControlSet.Length);
wcscat(KeyName, L"\\Control");
RtlInitUnicodeString(&String, KeyName);
InitializeObjectAttributes(
&ObjAttrs,
&String,
OBJ_CASE_INSENSITIVE,
NULL,
NULL
);
Status = ZwOpenKey(
&hControl,
KEY_ALL_ACCESS,
&ObjAttrs
);
if (NT_SUCCESS(Status)) {
HANDLE hMiniNT;
RtlInitUnicodeString(&String, L"MiniNT");
InitializeObjectAttributes(
&ObjAttrs,
&String,
OBJ_CASE_INSENSITIVE,
hControl,
NULL
);
Status = ZwCreateKey(
&hMiniNT,
KEY_ALL_ACCESS,
&ObjAttrs,
0,
NULL,
REG_OPTION_VOLATILE,
&Disposition
);
if (NT_SUCCESS(Status)) {
ZwClose(hMiniNT);
}
ZwClose(hControl);
}
//
// If we could not create the key, then bug check
// since we can't boot into mini NT anyway.
//
if (!NT_SUCCESS(Status)) {
KeBugCheckEx(PHASE1_INITIALIZATION_FAILED,Status,6,0,0);
}
}
//
// Begin paging the executive if desired.
//
MmInitSystem(2, LoaderBlock);
InbvUpdateProgressBar(80);
#if defined(_X86_)
//
// Initialize Vdm specific stuff
//
// Note: If this fails, Vdms may not be able to run, but it isn't
// necessary to bugcheck the system because of this.
//
KeI386VdmInitialize();
#if !defined(NT_UP)
//
// Now that the error log interface has been initialized, write
// an informational message if it was determined that the
// processors in the system are at differing revision levels.
//
if (CmProcessorMismatch != 0) {
PIO_ERROR_LOG_PACKET ErrLog;
ErrLog = IoAllocateGenericErrorLogEntry(ERROR_LOG_MAXIMUM_SIZE);
if (ErrLog) {
//
// Fill it in and write it out.
//
ErrLog->FinalStatus = STATUS_MP_PROCESSOR_MISMATCH;
ErrLog->ErrorCode = STATUS_MP_PROCESSOR_MISMATCH;
ErrLog->UniqueErrorValue = CmProcessorMismatch;
IoWriteErrorLogEntry(ErrLog);
}
}
#endif // !NT_UP
#endif // _X86_
if (!PoInitSystem(1)) {
KeBugCheck(INTERNAL_POWER_ERROR);
}
//
// Okay to call PsInitSystem now that \SystemRoot is defined so it can
// locate NTDLL.DLL and SMSS.EXE.
//
if (PsInitSystem(1, LoaderBlock) == FALSE) {
KeBugCheck(PROCESS1_INITIALIZATION_FAILED);
}
InbvUpdateProgressBar(85);
//
// Force KeBugCheck to look at PsLoadedModuleList now that it is setup.
//
if (LoaderBlock == KeLoaderBlock) {
KeLoaderBlock = NULL;
}
//
// Free loader block.
//
MmFreeLoaderBlock (LoaderBlock);
LoaderBlock = NULL;
Context = NULL;
//
// Perform Phase 1 Reference Monitor Initialization. This includes
// creating the Reference Monitor Command Server Thread, a permanent
// thread of the System Init process. That thread will create an LPC
// port called the Reference Monitor Command Port through which
// commands sent by the Local Security Authority Subsystem will be
// received. These commands (e.g. Enable Auditing) change the Reference
// Monitor State.
//
if (!SeRmInitPhase1()) {
KeBugCheck(REFMON_INITIALIZATION_FAILED);
}
InbvUpdateProgressBar(90);
//
// Set up process parameters for the Session Manager Subsystem.
//
// NOTE: Remote boot allocates an extra DOS_MAX_PATH_LENGTH number of
// WCHARs in order to hold command line arguments to smss.exe.
//
Size = sizeof( *ProcessParameters ) +
((DOS_MAX_PATH_LENGTH * 6) * sizeof( WCHAR ));
ProcessParameters = NULL;
Status = ZwAllocateVirtualMemory( NtCurrentProcess(),
(PVOID *)&ProcessParameters,
0,
&Size,
MEM_COMMIT,
PAGE_READWRITE
);
if (!NT_SUCCESS( Status )) {
#if DBG
sprintf(DebugBuffer,
"INIT: Unable to allocate Process Parameters. 0x%lx\n",
Status);
RtlInitAnsiString(&AnsiDebugString, DebugBuffer);
if (NT_SUCCESS(RtlAnsiStringToUnicodeString(&UnicodeDebugString,
&AnsiDebugString,
TRUE)) == FALSE) {
KeBugCheck(SESSION1_INITIALIZATION_FAILED);
}
ZwDisplayString(&UnicodeDebugString);
#endif // DBG
KeBugCheckEx(SESSION1_INITIALIZATION_FAILED,Status,0,0,0);
}
ProcessParameters->Length = (ULONG)Size;
ProcessParameters->MaximumLength = (ULONG)Size;
//
// Reserve the low 1 MB of address space in the session manager.
// Setup gets started using a replacement for the session manager
// and that process needs to be able to use the vga driver on x86,
// which uses int10 and thus requires the low 1 meg to be reserved
// in the process. The cost is so low that we just do this all the
// time, even when setup isn't running.
//
ProcessParameters->Flags = RTL_USER_PROC_PARAMS_NORMALIZED | RTL_USER_PROC_RESERVE_1MB;
Size = PAGE_SIZE;
Environment = NULL;
Status = ZwAllocateVirtualMemory( NtCurrentProcess(),
&Environment,
0,
&Size,
MEM_COMMIT,
PAGE_READWRITE
);
if (!NT_SUCCESS( Status )) {
#if DBG
sprintf(DebugBuffer,
"INIT: Unable to allocate Process Environment 0x%lx\n",
Status);
RtlInitAnsiString(&AnsiDebugString, DebugBuffer);
if (NT_SUCCESS(RtlAnsiStringToUnicodeString(&UnicodeDebugString,
&AnsiDebugString,
TRUE)) == FALSE) {
KeBugCheck(SESSION2_INITIALIZATION_FAILED);
}
ZwDisplayString(&UnicodeDebugString);
#endif // DBG
KeBugCheckEx(SESSION2_INITIALIZATION_FAILED,Status,0,0,0);
}
ProcessParameters->Environment = Environment;
Dst = (PWSTR)(ProcessParameters + 1);
ProcessParameters->CurrentDirectory.DosPath.Buffer = Dst;
ProcessParameters->CurrentDirectory.DosPath.MaximumLength = DOS_MAX_PATH_LENGTH * sizeof( WCHAR );
RtlCopyUnicodeString( &ProcessParameters->CurrentDirectory.DosPath,
&NtSystemRoot
);
Dst = (PWSTR)((PCHAR)ProcessParameters->CurrentDirectory.DosPath.Buffer +
ProcessParameters->CurrentDirectory.DosPath.MaximumLength
);
ProcessParameters->DllPath.Buffer = Dst;
ProcessParameters->DllPath.MaximumLength = DOS_MAX_PATH_LENGTH * sizeof( WCHAR );
RtlCopyUnicodeString( &ProcessParameters->DllPath,
&ProcessParameters->CurrentDirectory.DosPath
);
RtlAppendUnicodeToString( &ProcessParameters->DllPath, L"\\System32" );
Dst = (PWSTR)((PCHAR)ProcessParameters->DllPath.Buffer +
ProcessParameters->DllPath.MaximumLength
);
ProcessParameters->ImagePathName.Buffer = Dst;
ProcessParameters->ImagePathName.MaximumLength = DOS_MAX_PATH_LENGTH * sizeof( WCHAR );
if (NtInitialUserProcessBufferType != REG_SZ ||
(NtInitialUserProcessBufferLength != (ULONG)-1 &&
(NtInitialUserProcessBufferLength < sizeof(WCHAR) ||
NtInitialUserProcessBufferLength >
sizeof(NtInitialUserProcessBuffer) - sizeof(WCHAR)))) {
KeBugCheckEx(SESSION2_INITIALIZATION_FAILED,
STATUS_INVALID_PARAMETER,
NtInitialUserProcessBufferType,
NtInitialUserProcessBufferLength,
sizeof(NtInitialUserProcessBuffer));
}
// Executable names with spaces don't need to
// be supported so just find the first space and
// assume it terminates the process image name.
Src = NtInitialUserProcessBuffer;
while (*Src && *Src != L' ') {
Src++;
}
ProcessParameters->ImagePathName.Length =
(USHORT)((PUCHAR)Src - (PUCHAR)NtInitialUserProcessBuffer);
RtlCopyMemory(ProcessParameters->ImagePathName.Buffer,
NtInitialUserProcessBuffer,
ProcessParameters->ImagePathName.Length);
ProcessParameters->ImagePathName.Buffer[ProcessParameters->ImagePathName.Length / sizeof(WCHAR)] = UNICODE_NULL;
Dst = (PWSTR)((PCHAR)ProcessParameters->ImagePathName.Buffer +
ProcessParameters->ImagePathName.MaximumLength
);
ProcessParameters->CommandLine.Buffer = Dst;
ProcessParameters->CommandLine.MaximumLength = DOS_MAX_PATH_LENGTH * sizeof( WCHAR );
RtlAppendUnicodeToString(&ProcessParameters->CommandLine,
NtInitialUserProcessBuffer);
#if defined(REMOTE_BOOT)
//
// Pass additional parameters for remote boot clients.
//
if (IoRemoteBootClient && !ExpInTextModeSetup) {
RtlAppendUnicodeToString(&ProcessParameters->CommandLine, L" NETBOOT");
RtlAppendUnicodeToString(&ProcessParameters->CommandLine, L" NETBOOTHAL ");
AnsiDebugString.Length = strlen(NetBootHalName);
AnsiDebugString.MaximumLength = sizeof(NetBootHalName);
AnsiDebugString.Buffer = NetBootHalName;
Status = RtlAnsiStringToUnicodeString(&TmpUnicodeString, &AnsiDebugString, TRUE);
if (!NT_SUCCESS (Status)) {
KeBugCheckEx(SESSION2_INITIALIZATION_FAILED,Status,1,0,0);
}
RtlAppendUnicodeStringToString(&ProcessParameters->CommandLine, &TmpUnicodeString);
(RtlFreeStringRoutine)(TmpUnicodeString.Buffer);
if (NetBootDisconnected) {
RtlAppendUnicodeToString(&ProcessParameters->CommandLine, L" NETBOOTDISCONNECTED");
}
if (NetBootRequiresFormat) {
RtlAppendUnicodeToString(&ProcessParameters->CommandLine, L" NETBOOTFORMAT");
}
}
#endif // defined(REMOTE_BOOT)
NullString.Buffer = L"";
NullString.Length = sizeof(WCHAR);
NullString.MaximumLength = sizeof(WCHAR);
EnvString.Buffer = ProcessParameters->Environment;
EnvString.Length = 0;
EnvString.MaximumLength = (USHORT)Size;
RtlAppendUnicodeToString( &EnvString, L"Path=" );
RtlAppendUnicodeStringToString( &EnvString, &ProcessParameters->DllPath );
RtlAppendUnicodeStringToString( &EnvString, &NullString );
UnicodeSystemDriveString = NtSystemRoot;
UnicodeSystemDriveString.Length = 2 * sizeof( WCHAR );
RtlAppendUnicodeToString( &EnvString, L"SystemDrive=" );
RtlAppendUnicodeStringToString( &EnvString, &UnicodeSystemDriveString );
RtlAppendUnicodeStringToString( &EnvString, &NullString );
RtlAppendUnicodeToString( &EnvString, L"SystemRoot=" );
RtlAppendUnicodeStringToString( &EnvString, &NtSystemRoot );
RtlAppendUnicodeStringToString( &EnvString, &NullString );
#if 0
KdPrint(( "ProcessParameters at %lx\n", ProcessParameters ));
KdPrint(( " CurDir: %wZ\n", &ProcessParameters->CurrentDirectory.DosPath ));
KdPrint(( " DllPath: %wZ\n", &ProcessParameters->DllPath ));
KdPrint(( " ImageFile: %wZ\n", &ProcessParameters->ImagePathName ));
KdPrint(( " Environ: %lx\n", ProcessParameters->Environment ));
Src = ProcessParameters->Environment;
while (*Src) {
KdPrint(( " %ws\n", Src ));
while (*Src++) {
;
}
}
#endif
//
// Notify boot prefetcher of boot progress.
//
CcPfBeginBootPhase(PfSessionManagerInitPhase);
SessionManager = ProcessParameters->ImagePathName;
Status = RtlCreateUserProcess(
&SessionManager,
OBJ_CASE_INSENSITIVE,
RtlDeNormalizeProcessParams( ProcessParameters ),
NULL,
NULL,
NULL,
FALSE,
NULL,
NULL,
&ProcessInformation);
if (InbvBootDriverInstalled)
{
FinalizeBootLogo();
}
if (!NT_SUCCESS(Status)) {
#if DBG
sprintf(DebugBuffer,
"INIT: Unable to create Session Manager. 0x%lx\n",
Status);
RtlInitAnsiString(&AnsiDebugString, DebugBuffer);
if (NT_SUCCESS(RtlAnsiStringToUnicodeString(&UnicodeDebugString,
&AnsiDebugString,
TRUE)) == FALSE) {
KeBugCheck(SESSION3_INITIALIZATION_FAILED);
}
ZwDisplayString(&UnicodeDebugString);
#endif // DBG
KeBugCheckEx(SESSION3_INITIALIZATION_FAILED,Status,0,0,0);
}
Status = ZwResumeThread(ProcessInformation.Thread,NULL);
if ( !NT_SUCCESS(Status) ) {
#if DBG
sprintf(DebugBuffer,
"INIT: Unable to resume Session Manager. 0x%lx\n",
Status);
RtlInitAnsiString(&AnsiDebugString, DebugBuffer);
if (NT_SUCCESS(RtlAnsiStringToUnicodeString(&UnicodeDebugString,
&AnsiDebugString,
TRUE)) == FALSE) {
KeBugCheck(SESSION4_INITIALIZATION_FAILED);
}
ZwDisplayString(&UnicodeDebugString);
#endif // DBG
KeBugCheckEx(SESSION4_INITIALIZATION_FAILED,Status,0,0,0);
}
InbvUpdateProgressBar(100);
//
// Turn on debug output so that we can see chkdsk run.
//
InbvEnableDisplayString(TRUE);
//
// Wait five seconds for the session manager to get started or
// terminate. If the wait times out, then the session manager
// is assumed to be healthy and the zero page thread is called.
//
OldTime.QuadPart = Int32x32To64(5, -(10 * 1000 * 1000));
Status = ZwWaitForSingleObject(
ProcessInformation.Process,
FALSE,
&OldTime
);
if (Status == STATUS_SUCCESS) {
#if DBG
sprintf(DebugBuffer, "INIT: Session Manager terminated.\n");
RtlInitAnsiString(&AnsiDebugString, DebugBuffer);
Status = RtlAnsiStringToUnicodeString(&UnicodeDebugString,
&AnsiDebugString,
TRUE);
if (!NT_SUCCESS (Status)) {
KeBugCheckEx(SESSION4_INITIALIZATION_FAILED,Status,1,0,0);
}
ZwDisplayString(&UnicodeDebugString);
#endif // DBG
KeBugCheck(SESSION5_INITIALIZATION_FAILED);
}
//
// Don't need these handles anymore.
//
ZwClose( ProcessInformation.Thread );
ZwClose( ProcessInformation.Process );
//
// Free up memory used to pass arguments to session manager.
//
Size = 0;
Address = Environment;
ZwFreeVirtualMemory( NtCurrentProcess(),
(PVOID *)&Address,
&Size,
MEM_RELEASE
);
Size = 0;
Address = ProcessParameters;
ZwFreeVirtualMemory( NtCurrentProcess(),
(PVOID *)&Address,
&Size,
MEM_RELEASE
);
InitializationPhase += 1;
#if defined(_X86_)
KiInitializeInterruptTimers();
#endif
MmZeroPageThread();
}
NTSTATUS
CreateSystemRootLink(
IN PLOADER_PARAMETER_BLOCK LoaderBlock
)
{
HANDLE handle;
UNICODE_STRING nameString;
OBJECT_ATTRIBUTES objectAttributes;
STRING linkString;
UNICODE_STRING linkUnicodeString;
NTSTATUS status;
UCHAR deviceNameBuffer[256];
STRING deviceNameString;
UNICODE_STRING deviceNameUnicodeString;
HANDLE linkHandle;
#if DBG
UCHAR debugBuffer[256];
STRING debugString;
UNICODE_STRING debugUnicodeString;
#endif
//
// Create the root directory object for the \ArcName directory.
//
RtlInitUnicodeString( &nameString, L"\\ArcName" );
InitializeObjectAttributes( &objectAttributes,
&nameString,
OBJ_CASE_INSENSITIVE | OBJ_PERMANENT,
NULL,
SePublicDefaultUnrestrictedSd );
status = NtCreateDirectoryObject( &handle,
DIRECTORY_ALL_ACCESS,
&objectAttributes );
if (!NT_SUCCESS( status )) {
KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED,status,1,0,0);
return status;
} else {
(VOID) NtClose( handle );
}
//
// Create the root directory object for the \Device directory.
//
RtlInitUnicodeString( &nameString, L"\\Device" );
InitializeObjectAttributes( &objectAttributes,
&nameString,
OBJ_CASE_INSENSITIVE | OBJ_PERMANENT,
NULL,
SePublicDefaultUnrestrictedSd );
status = NtCreateDirectoryObject( &handle,
DIRECTORY_ALL_ACCESS,
&objectAttributes );
if (!NT_SUCCESS( status )) {
KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED,status,2,0,0);
return status;
} else {
(VOID) NtClose( handle );
}
//
// Create the symbolic link to the root of the system directory.
//
RtlInitAnsiString( &linkString, INIT_SYSTEMROOT_LINKNAME );
status = RtlAnsiStringToUnicodeString( &linkUnicodeString,
&linkString,
TRUE);
if (!NT_SUCCESS( status )) {
KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED,status,3,0,0);
return status;
}
InitializeObjectAttributes( &objectAttributes,
&linkUnicodeString,
OBJ_CASE_INSENSITIVE | OBJ_PERMANENT,
NULL,
SePublicDefaultUnrestrictedSd );
//
// Use ARC device name and system path from loader.
//
sprintf( deviceNameBuffer,
"\\ArcName\\%s%s",
LoaderBlock->ArcBootDeviceName,
LoaderBlock->NtBootPathName);
deviceNameBuffer[strlen(deviceNameBuffer)-1] = '\0';
RtlInitString( &deviceNameString, deviceNameBuffer );
status = RtlAnsiStringToUnicodeString( &deviceNameUnicodeString,
&deviceNameString,
TRUE );
if (!NT_SUCCESS(status)) {
RtlFreeUnicodeString( &linkUnicodeString );
KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED,status,4,0,0);
return status;
}
status = NtCreateSymbolicLinkObject( &linkHandle,
SYMBOLIC_LINK_ALL_ACCESS,
&objectAttributes,
&deviceNameUnicodeString );
RtlFreeUnicodeString( &linkUnicodeString );
RtlFreeUnicodeString( &deviceNameUnicodeString );
if (!NT_SUCCESS(status)) {
KeBugCheckEx(SYMBOLIC_INITIALIZATION_FAILED,status,5,0,0);
return status;
}
#if DBG
sprintf( debugBuffer, "INIT: %s => %s\n",
INIT_SYSTEMROOT_LINKNAME,
deviceNameBuffer );
RtlInitAnsiString( &debugString, debugBuffer );
status = RtlAnsiStringToUnicodeString( &debugUnicodeString,
&debugString,
TRUE );
if (NT_SUCCESS(status)) {
ZwDisplayString( &debugUnicodeString );
RtlFreeUnicodeString( &debugUnicodeString );
}
#endif // DBG
NtClose( linkHandle );
return STATUS_SUCCESS;
}
#if 0
PVOID
LookupImageBaseByName (
IN PLIST_ENTRY ListHead,
IN PSZ Name
)
/*++
Lookups BaseAddress of ImageName - returned value can be used
to find entry points via LookupEntryPoint
--*/
{
PKLDR_DATA_TABLE_ENTRY Entry;
PLIST_ENTRY Next;
PVOID Base;
ANSI_STRING ansiString;
UNICODE_STRING unicodeString;
NTSTATUS status;
Next = ListHead->Flink;
if (!Next) {
return NULL;
}
RtlInitAnsiString(&ansiString, Name);
status = RtlAnsiStringToUnicodeString( &unicodeString, &ansiString, TRUE );
if (!NT_SUCCESS (status)) {
return NULL;
}
Base = NULL;
while (Next != ListHead) {
Entry = CONTAINING_RECORD(Next, KLDR_DATA_TABLE_ENTRY, InLoadOrderLinks);
Next = Next->Flink;
if (RtlEqualUnicodeString (&unicodeString, &Entry->BaseDllName, TRUE)) {
Base = Entry->DllBase;
break;
}
}
RtlFreeUnicodeString( &unicodeString );
return Base;
}
#endif
NTSTATUS
LookupEntryPoint (
IN PVOID DllBase,
IN PSZ NameOfEntryPoint,
OUT PVOID *AddressOfEntryPoint
)
/*++
Routine Description:
Returns the address of an entry point given the DllBase and PSZ
name of the entry point in question
--*/
{
PIMAGE_EXPORT_DIRECTORY ExportDirectory;
ULONG ExportSize;
USHORT Ordinal;
PULONG Addr;
CHAR NameBuffer[64];
ExportDirectory = (PIMAGE_EXPORT_DIRECTORY)
RtlImageDirectoryEntryToData(
DllBase,
TRUE,
IMAGE_DIRECTORY_ENTRY_EXPORT,
&ExportSize);
#if DBG
if (!ExportDirectory) {
DbgPrint("LookupENtryPoint: Can't locate system Export Directory\n");
}
#endif
if ( strlen(NameOfEntryPoint) > sizeof(NameBuffer)-2 ) {
return STATUS_INVALID_PARAMETER;
}
strcpy(NameBuffer,NameOfEntryPoint);
Ordinal = NameToOrdinal(
NameBuffer,
(ULONG_PTR)DllBase,
ExportDirectory->NumberOfNames,
(PULONG)((ULONG_PTR)DllBase + ExportDirectory->AddressOfNames),
(PUSHORT)((ULONG_PTR)DllBase + ExportDirectory->AddressOfNameOrdinals)
);
//
// If Ordinal is not within the Export Address Table,
// then DLL does not implement function.
//
if ( (ULONG)Ordinal >= ExportDirectory->NumberOfFunctions ) {
return STATUS_PROCEDURE_NOT_FOUND;
}
Addr = (PULONG)((ULONG_PTR)DllBase + ExportDirectory->AddressOfFunctions);
*AddressOfEntryPoint = (PVOID)((ULONG_PTR)DllBase + Addr[Ordinal]);
return STATUS_SUCCESS;
}
static USHORT
NameToOrdinal (
IN PSZ NameOfEntryPoint,
IN ULONG_PTR DllBase,
IN ULONG NumberOfNames,
IN PULONG NameTableBase,
IN PUSHORT NameOrdinalTableBase
)
{
ULONG SplitIndex;
LONG CompareResult;
if ( NumberOfNames == 0 ) {
return (USHORT)-1;
}
SplitIndex = NumberOfNames >> 1;
CompareResult = strcmp(NameOfEntryPoint, (PSZ)(DllBase + NameTableBase[SplitIndex]));
if ( CompareResult == 0 ) {
return NameOrdinalTableBase[SplitIndex];
}
if ( NumberOfNames == 1 ) {
return (USHORT)-1;
}
if ( CompareResult < 0 ) {
NumberOfNames = SplitIndex;
} else {
NameTableBase = &NameTableBase[SplitIndex+1];
NameOrdinalTableBase = &NameOrdinalTableBase[SplitIndex+1];
NumberOfNames = NumberOfNames - SplitIndex - 1;
}
return NameToOrdinal(NameOfEntryPoint,DllBase,NumberOfNames,NameTableBase,NameOrdinalTableBase);
}