Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

3186 lines
99 KiB

/*++
Copyright (c) 1989-1993 Microsoft Corporation
Module Name:
ioinit.c
Abstract:
This module contains the code to initialize the I/O system.
Author:
Darryl E. Havens (darrylh) April 27, 1989
Environment:
Kernel mode, system initialization code
Revision History:
--*/
#include "iomgr.h"
#include <setupblk.h>
#include <inbv.h>
#include <ntddstor.h>
#include <hdlsblk.h>
#include <hdlsterm.h>
//
// Define the default number of IRP that can be in progress and allocated
// from a lookaside list.
//
#define DEFAULT_LOOKASIDE_IRP_LIMIT 512
//
// I/O Error logging support
//
PVOID IopErrorLogObject = NULL;
//
// Define a macro for initializing drivers.
//
#define InitializeDriverObject( Object ) { \
ULONG i; \
RtlZeroMemory( Object, \
sizeof( DRIVER_OBJECT ) + sizeof ( DRIVER_EXTENSION )); \
Object->DriverExtension = (PDRIVER_EXTENSION) (Object + 1); \
Object->DriverExtension->DriverObject = Object; \
for (i = 0; i <= IRP_MJ_MAXIMUM_FUNCTION; i++) \
Object->MajorFunction[i] = IopInvalidDeviceRequest; \
Object->Type = IO_TYPE_DRIVER; \
Object->Size = sizeof( DRIVER_OBJECT ); \
}
ULONG IopInitFailCode; // Debugging aid for IoInitSystem
//
// Define external procedures not in common header files
//
VOID
IopInitializeData(
VOID
);
//
// Define the local procedures
//
BOOLEAN
IopCreateObjectTypes(
VOID
);
BOOLEAN
IopCreateRootDirectories(
VOID
);
NTSTATUS
IopInitializeAttributesAndCreateObject(
IN PUNICODE_STRING ObjectName,
IN OUT POBJECT_ATTRIBUTES ObjectAttributes,
OUT PDRIVER_OBJECT *DriverObject
);
BOOLEAN
IopReassignSystemRoot(
IN PLOADER_PARAMETER_BLOCK LoaderBlock,
OUT PSTRING NtDeviceName
);
VOID
IopSetIoRoutines(
IN VOID
);
VOID
IopStoreSystemPartitionInformation(
IN PUNICODE_STRING NtSystemPartitionDeviceName,
IN OUT PUNICODE_STRING OsLoaderPathName
);
//
// The following allows the I/O system's initialization routines to be
// paged out of memory.
//
#ifdef ALLOC_PRAGMA
#pragma alloc_text(INIT,IoInitSystem)
#pragma alloc_text(INIT,IopCreateArcNames)
#pragma alloc_text(INIT,IopCreateObjectTypes)
#pragma alloc_text(INIT,IopCreateRootDirectories)
#pragma alloc_text(INIT,IopInitializeAttributesAndCreateObject)
#pragma alloc_text(INIT,IopInitializeBuiltinDriver)
#pragma alloc_text(INIT,IopMarkBootPartition)
#pragma alloc_text(INIT,IopReassignSystemRoot)
#pragma alloc_text(INIT,IopSetIoRoutines)
#pragma alloc_text(INIT,IopStoreSystemPartitionInformation)
#pragma alloc_text(INIT,IopInitializeReserveIrp)
#endif
BOOLEAN
IoInitSystem(
PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
This routine initializes the I/O system.
Arguments:
LoaderBlock - Supplies a pointer to the loader parameter block that was
created by the OS Loader.
Return Value:
The function value is a BOOLEAN indicating whether or not the I/O system
was successfully initialized.
--*/
{
PDRIVER_OBJECT driverObject;
PDRIVER_OBJECT *nextDriverObject;
STRING ntDeviceName;
UCHAR deviceNameBuffer[256];
ULONG largePacketSize;
ULONG smallPacketSize;
ULONG mdlPacketSize;
PLIST_ENTRY entry;
LARGE_INTEGER deltaTime;
MM_SYSTEMSIZE systemSize;
USHORT completionZoneSize;
USHORT largeIrpZoneSize;
USHORT smallIrpZoneSize;
USHORT mdlZoneSize;
ULONG oldNtGlobalFlag;
NTSTATUS status;
ANSI_STRING ansiString;
UNICODE_STRING eventName;
UNICODE_STRING startTypeName;
OBJECT_ATTRIBUTES objectAttributes;
HANDLE handle;
PGENERAL_LOOKASIDE lookaside;
ULONG lookasideIrpLimit;
ULONG lookasideSize;
ULONG Index;
PKPRCB prcb;
ULONG len;
PKEY_VALUE_PARTIAL_INFORMATION value;
UCHAR valueBuffer[32];
ASSERT( IopQueryOperationLength[FileMaximumInformation] == 0xff );
ASSERT( IopSetOperationLength[FileMaximumInformation] == 0xff );
ASSERT( IopQueryOperationAccess[FileMaximumInformation] == 0xffffffff );
ASSERT( IopSetOperationAccess[FileMaximumInformation] == 0xffffffff );
ASSERT( IopQueryFsOperationLength[FileFsMaximumInformation] == 0xff );
ASSERT( IopSetFsOperationLength[FileFsMaximumInformation] == 0xff );
ASSERT( IopQueryFsOperationAccess[FileFsMaximumInformation] == 0xffffffff );
ASSERT( IopSetFsOperationAccess[FileFsMaximumInformation] == 0xffffffff );
//
// Initialize the I/O database resource, lock, and the file system and
// network file system queue headers. Also allocate the cancel spin
// lock.
//
ntDeviceName.Buffer = deviceNameBuffer;
ntDeviceName.MaximumLength = sizeof(deviceNameBuffer);
ntDeviceName.Length = 0;
ExInitializeResourceLite( &IopDatabaseResource );
ExInitializeResourceLite( &IopSecurityResource );
ExInitializeResourceLite( &IopCrashDumpLock );
InitializeListHead( &IopDiskFileSystemQueueHead );
InitializeListHead( &IopCdRomFileSystemQueueHead );
InitializeListHead( &IopTapeFileSystemQueueHead );
InitializeListHead( &IopNetworkFileSystemQueueHead );
InitializeListHead( &IopBootDriverReinitializeQueueHead );
InitializeListHead( &IopDriverReinitializeQueueHead );
InitializeListHead( &IopNotifyShutdownQueueHead );
InitializeListHead( &IopNotifyLastChanceShutdownQueueHead );
InitializeListHead( &IopFsNotifyChangeQueueHead );
KeInitializeSpinLock( &IoStatisticsLock );
IopSetIoRoutines();
//
// Initialize the unique device object number counter used by IoCreateDevice
// when automatically generating a device object name.
//
IopUniqueDeviceObjectNumber = 0;
//
// Initialize the large I/O Request Packet (IRP) lookaside list head and the
// mutex which guards the list.
//
if (!IopLargeIrpStackLocations) {
IopLargeIrpStackLocations = DEFAULT_LARGE_IRP_LOCATIONS;
IopIrpStackProfiler.Flags |= IOP_ENABLE_AUTO_SIZING;
}
systemSize = MmQuerySystemSize();
switch ( systemSize ) {
case MmSmallSystem :
completionZoneSize = 6;
smallIrpZoneSize = 6;
largeIrpZoneSize = 8;
mdlZoneSize = 16;
lookasideIrpLimit = DEFAULT_LOOKASIDE_IRP_LIMIT;
break;
case MmMediumSystem :
completionZoneSize = 24;
smallIrpZoneSize = 24;
largeIrpZoneSize = 32;
mdlZoneSize = 90;
lookasideIrpLimit = DEFAULT_LOOKASIDE_IRP_LIMIT * 2;
break;
case MmLargeSystem :
if (MmIsThisAnNtAsSystem()) {
completionZoneSize = 96;
smallIrpZoneSize = 96;
largeIrpZoneSize = 128;
mdlZoneSize = 256;
lookasideIrpLimit = DEFAULT_LOOKASIDE_IRP_LIMIT * 128; // 64k
} else {
completionZoneSize = 32;
smallIrpZoneSize = 32;
largeIrpZoneSize = 64;
mdlZoneSize = 128;
lookasideIrpLimit = DEFAULT_LOOKASIDE_IRP_LIMIT * 3;
}
break;
}
//
// Initialize the system I/O completion lookaside list.
//
ExInitializeSystemLookasideList( &IopCompletionLookasideList,
NonPagedPool,
sizeof(IOP_MINI_COMPLETION_PACKET),
' pcI',
completionZoneSize,
&ExSystemLookasideListHead );
//
// Initialize the system large IRP lookaside list.
//
largePacketSize = (ULONG) (sizeof( IRP ) + (IopLargeIrpStackLocations * sizeof( IO_STACK_LOCATION )));
ExInitializeSystemLookasideList( &IopLargeIrpLookasideList,
NonPagedPool,
largePacketSize,
'lprI',
largeIrpZoneSize,
&ExSystemLookasideListHead );
//
// Initialize the system small IRP lookaside list.
//
smallPacketSize = (ULONG) (sizeof( IRP ) + sizeof( IO_STACK_LOCATION ));
ExInitializeSystemLookasideList( &IopSmallIrpLookasideList,
NonPagedPool,
smallPacketSize,
'sprI',
smallIrpZoneSize,
&ExSystemLookasideListHead );
//
// Initialize the system MDL lookaside list.
//
mdlPacketSize = (ULONG) (sizeof( MDL ) + (IOP_FIXED_SIZE_MDL_PFNS * sizeof( PFN_NUMBER )));
ExInitializeSystemLookasideList( &IopMdlLookasideList,
NonPagedPool,
mdlPacketSize,
' ldM',
mdlZoneSize,
&ExSystemLookasideListHead );
//
// Compute the lookaside IRP float credits per processor.
//
lookasideIrpLimit /= KeNumberProcessors;
//
// Initialize the per processor nonpaged lookaside lists and descriptors.
//
// N.B. All the I/O related lookaside list structures are allocated at
// one time to make sure they are aligned, if possible, and to avoid
// pool overhead.
//
lookasideSize = 4 * KeNumberProcessors * sizeof(GENERAL_LOOKASIDE);
lookaside = ExAllocatePoolWithTag( NonPagedPool, lookasideSize, 'oI');
for (Index = 0; Index < (ULONG)KeNumberProcessors; Index += 1) {
prcb = KiProcessorBlock[Index];
//
// Set the per processor IRP float credits.
//
prcb->LookasideIrpFloat = lookasideIrpLimit;
//
// Initialize the I/O completion per processor lookaside pointers
//
prcb->PPLookasideList[LookasideCompletionList].L = &IopCompletionLookasideList;
if (lookaside != NULL) {
ExInitializeSystemLookasideList( lookaside,
NonPagedPool,
sizeof(IOP_MINI_COMPLETION_PACKET),
'PpcI',
completionZoneSize,
&ExSystemLookasideListHead );
prcb->PPLookasideList[LookasideCompletionList].P = lookaside;
lookaside += 1;
} else {
prcb->PPLookasideList[LookasideCompletionList].P = &IopCompletionLookasideList;
}
//
// Initialize the large IRP per processor lookaside pointers.
//
prcb->PPLookasideList[LookasideLargeIrpList].L = &IopLargeIrpLookasideList;
if (lookaside != NULL) {
ExInitializeSystemLookasideList( lookaside,
NonPagedPool,
largePacketSize,
'LprI',
largeIrpZoneSize,
&ExSystemLookasideListHead );
prcb->PPLookasideList[LookasideLargeIrpList].P = lookaside;
lookaside += 1;
} else {
prcb->PPLookasideList[LookasideLargeIrpList].P = &IopLargeIrpLookasideList;
}
//
// Initialize the small IRP per processor lookaside pointers.
//
prcb->PPLookasideList[LookasideSmallIrpList].L = &IopSmallIrpLookasideList;
if (lookaside != NULL) {
ExInitializeSystemLookasideList( lookaside,
NonPagedPool,
smallPacketSize,
'SprI',
smallIrpZoneSize,
&ExSystemLookasideListHead );
prcb->PPLookasideList[LookasideSmallIrpList].P = lookaside;
lookaside += 1;
} else {
prcb->PPLookasideList[LookasideSmallIrpList].P = &IopSmallIrpLookasideList;
}
//
// Initialize the MDL per processor lookaside list pointers.
//
prcb->PPLookasideList[LookasideMdlList].L = &IopMdlLookasideList;
if (lookaside != NULL) {
ExInitializeSystemLookasideList( lookaside,
NonPagedPool,
mdlPacketSize,
'PldM',
mdlZoneSize,
&ExSystemLookasideListHead );
prcb->PPLookasideList[LookasideMdlList].P = lookaside;
lookaside += 1;
} else {
prcb->PPLookasideList[LookasideMdlList].P = &IopMdlLookasideList;
}
}
//
// Initalize the error log spin locks and log list.
//
KeInitializeSpinLock( &IopErrorLogLock );
InitializeListHead( &IopErrorLogListHead );
if (IopInitializeReserveIrp(&IopReserveIrpAllocator) == FALSE) {
IopInitFailCode = 1;
return FALSE;
}
if (IopIrpAutoSizingEnabled() && !NT_SUCCESS(IopInitializeIrpStackProfiler())) {
IopInitFailCode = 13;
return FALSE;
}
//
// Determine if the Error Log service will ever run this boot.
//
InitializeObjectAttributes (&objectAttributes,
&CmRegistryMachineSystemCurrentControlSetServicesEventLog,
OBJ_CASE_INSENSITIVE,
(HANDLE) NULL,
(PSECURITY_DESCRIPTOR) NULL );
status = ZwOpenKey(&handle,
KEY_READ,
&objectAttributes
);
if (NT_SUCCESS (status)) {
RtlInitUnicodeString (&startTypeName, L"Start");
value = (PKEY_VALUE_PARTIAL_INFORMATION) valueBuffer;
status = NtQueryValueKey (handle,
&startTypeName,
KeyValuePartialInformation,
valueBuffer,
sizeof (valueBuffer),
&len);
if (NT_SUCCESS (status) && (value->Type == REG_DWORD)) {
if (SERVICE_DISABLED == (*(PULONG) (value->Data))) {
//
// We are disabled for this boot.
//
IopErrorLogDisabledThisBoot = TRUE;
} else {
IopErrorLogDisabledThisBoot = FALSE;
}
} else {
//
// Didn't find the value so we are not enabled.
//
IopErrorLogDisabledThisBoot = TRUE;
}
} else {
//
// Didn't find the key so we are not enabled
//
IopErrorLogDisabledThisBoot = TRUE;
}
//
// Initialize the timer database and start the timer DPC routine firing
// so that drivers can use it during initialization.
//
deltaTime.QuadPart = - 10 * 1000 * 1000;
KeInitializeSpinLock( &IopTimerLock );
InitializeListHead( &IopTimerQueueHead );
KeInitializeDpc( &IopTimerDpc, IopTimerDispatch, NULL );
KeInitializeTimerEx( &IopTimer, SynchronizationTimer );
(VOID) KeSetTimerEx( &IopTimer, deltaTime, 1000, &IopTimerDpc );
//
// Initialize the IopHardError structure used for informational pop-ups.
//
ExInitializeWorkItem( &IopHardError.ExWorkItem,
IopHardErrorThread,
NULL );
InitializeListHead( &IopHardError.WorkQueue );
KeInitializeSpinLock( &IopHardError.WorkQueueSpinLock );
KeInitializeSemaphore( &IopHardError.WorkQueueSemaphore,
0,
MAXLONG );
IopHardError.ThreadStarted = FALSE;
IopCurrentHardError = NULL;
//
// Create the link tracking named event.
//
RtlInitUnicodeString( &eventName, L"\\Security\\TRKWKS_EVENT" );
InitializeObjectAttributes( &objectAttributes,
&eventName,
OBJ_PERMANENT,
(HANDLE) NULL,
(PSECURITY_DESCRIPTOR) NULL );
status = NtCreateEvent( &handle,
EVENT_ALL_ACCESS,
&objectAttributes,
NotificationEvent,
FALSE );
if (!NT_SUCCESS( status )) {
#if DBG
DbgPrint( "IOINIT: NtCreateEvent failed\n" );
#endif
HeadlessKernelAddLogEntry(HEADLESS_LOG_EVENT_CREATE_FAILED, NULL);
return FALSE;
}
(VOID) ObReferenceObjectByHandle( handle,
0,
ExEventObjectType,
KernelMode,
(PVOID *) &IopLinkTrackingServiceEvent,
NULL );
KeInitializeEvent( &IopLinkTrackingPacket.Event, NotificationEvent, FALSE );
KeInitializeEvent(&IopLinkTrackingPortObject, SynchronizationEvent, TRUE );
//
// Create all of the objects for the I/O system.
//
if (!IopCreateObjectTypes()) {
#if DBG
DbgPrint( "IOINIT: IopCreateObjectTypes failed\n" );
#endif
HeadlessKernelAddLogEntry(HEADLESS_LOG_OBJECT_TYPE_CREATE_FAILED, NULL);
IopInitFailCode = 2;
return FALSE;
}
//
// Create the root directories for the I/O system.
//
if (!IopCreateRootDirectories()) {
#if DBG
DbgPrint( "IOINIT: IopCreateRootDirectories failed\n" );
#endif
HeadlessKernelAddLogEntry(HEADLESS_LOG_ROOT_DIR_CREATE_FAILED, NULL);
IopInitFailCode = 3;
return FALSE;
}
//
// Initialize PlugPlay services phase 0
//
status = IopInitializePlugPlayServices(LoaderBlock, 0);
if (!NT_SUCCESS(status)) {
HeadlessKernelAddLogEntry(HEADLESS_LOG_PNP_PHASE0_INIT_FAILED, NULL);
IopInitFailCode = 4;
return FALSE;
}
//
// Call Power manager to initialize for drivers
//
PoInitDriverServices(0);
//
// Call HAL to initialize PnP bus driver
//
HalInitPnpDriver();
IopMarkHalDeviceNode();
//
// Call WMI to initialize it and allow it to create its driver object
// Note that no calls to WMI can occur until it is initialized here.
//
WMIInitialize(0, (PVOID)LoaderBlock);
//
// Save this for use during PnP enumeration -- we NULL it out later
// before LoaderBlock is reused.
//
IopLoaderBlock = (PVOID)LoaderBlock;
//
// If this is a remote boot, we need to add a few values to the registry.
//
if (IoRemoteBootClient) {
status = IopAddRemoteBootValuesToRegistry(LoaderBlock);
if (!NT_SUCCESS(status)) {
KeBugCheckEx( NETWORK_BOOT_INITIALIZATION_FAILED,
1,
status,
0,
0 );
}
}
//
// Initialize PlugPlay services phase 1 to execute firmware mapper
//
status = IopInitializePlugPlayServices(LoaderBlock, 1);
if (!NT_SUCCESS(status)) {
HeadlessKernelAddLogEntry(HEADLESS_LOG_PNP_PHASE1_INIT_FAILED, NULL);
IopInitFailCode = 5;
return FALSE;
}
//
// Initialize the drivers loaded by the boot loader (OSLOADER)
//
nextDriverObject = &driverObject;
if (!IopInitializeBootDrivers( LoaderBlock,
nextDriverObject )) {
#if DBG
DbgPrint( "IOINIT: Initializing boot drivers failed\n" );
#endif // DBG
HeadlessKernelAddLogEntry(HEADLESS_LOG_BOOT_DRIVERS_INIT_FAILED, NULL);
IopInitFailCode = 6;
return FALSE;
}
//
// Once we have initialized the boot drivers, we don't need the
// copy of the pointer to the loader block any more.
//
IopLoaderBlock = NULL;
//
// If this is a remote boot, start the network and assign
// C: to \Device\LanmanRedirector.
//
if (IoRemoteBootClient) {
status = IopStartNetworkForRemoteBoot(LoaderBlock);
if (!NT_SUCCESS( status )) {
KeBugCheckEx( NETWORK_BOOT_INITIALIZATION_FAILED,
2,
status,
0,
0 );
}
}
//
// Do last known good boot processing. If this is a last known good boot,
// we will copy over the last known good drivers and files. Otherwise we
// will ensure this boot doesn't taint our last good info (in case we crash
// before the boot is marked good). Note that loading of the correct boot
// drivers was handled by the boot loader, who chose an LKG boot in the
// first place.
//
PpLastGoodDoBootProcessing();
//
// Save the current value of the NT Global Flags and enable kernel debugger
// symbol loading while drivers are being loaded so that systems can be
// debugged regardless of whether they are free or checked builds.
//
oldNtGlobalFlag = NtGlobalFlag;
if (!(NtGlobalFlag & FLG_ENABLE_KDEBUG_SYMBOL_LOAD)) {
NtGlobalFlag |= FLG_ENABLE_KDEBUG_SYMBOL_LOAD;
}
status = PsLocateSystemDll();
if (!NT_SUCCESS( status )) {
HeadlessKernelAddLogEntry(HEADLESS_LOG_LOCATE_SYSTEM_DLL_FAILED, NULL);
IopInitFailCode = 7;
return FALSE;
}
//
// Notify the boot prefetcher of boot progress.
//
CcPfBeginBootPhase(PfSystemDriverInitPhase);
//
// Initialize the device drivers for the system.
//
if (!IopInitializeSystemDrivers()) {
#if DBG
DbgPrint( "IOINIT: Initializing system drivers failed\n" );
#endif // DBG
HeadlessKernelAddLogEntry(HEADLESS_LOG_SYSTEM_DRIVERS_INIT_FAILED, NULL);
IopInitFailCode = 8;
return FALSE;
}
IopCallDriverReinitializationRoutines();
//
// Reassign \SystemRoot to NT device name path.
//
if (!IopReassignSystemRoot( LoaderBlock, &ntDeviceName )) {
HeadlessKernelAddLogEntry(HEADLESS_LOG_ASSIGN_SYSTEM_ROOT_FAILED, NULL);
IopInitFailCode = 9;
return FALSE;
}
//
// Protect the system partition of an ARC system if necessary
//
if (!IopProtectSystemPartition( LoaderBlock )) {
HeadlessKernelAddLogEntry(HEADLESS_LOG_PROTECT_SYSTEM_ROOT_FAILED, NULL);
IopInitFailCode = 10;
return FALSE;
}
//
// Assign DOS drive letters to disks and cdroms and define \SystemRoot.
//
ansiString.MaximumLength = NtSystemRoot.MaximumLength / sizeof( WCHAR );
ansiString.Length = 0;
ansiString.Buffer = (RtlAllocateStringRoutine)( ansiString.MaximumLength );
status = RtlUnicodeStringToAnsiString( &ansiString,
&NtSystemRoot,
FALSE
);
if (!NT_SUCCESS( status )) {
DbgPrint( "IOINIT: UnicodeToAnsi( %wZ ) failed - %x\n", &NtSystemRoot, status );
HeadlessKernelAddLogEntry(HEADLESS_LOG_UNICODE_TO_ANSI_FAILED, NULL);
IopInitFailCode = 11;
return FALSE;
}
IoAssignDriveLetters( LoaderBlock,
&ntDeviceName,
ansiString.Buffer,
&ansiString );
status = RtlAnsiStringToUnicodeString( &NtSystemRoot,
&ansiString,
FALSE
);
if (!NT_SUCCESS( status )) {
DbgPrint( "IOINIT: AnsiToUnicode( %Z ) failed - %x\n", &ansiString, status );
HeadlessKernelAddLogEntry(HEADLESS_LOG_ANSI_TO_UNICODE_FAILED, NULL);
IopInitFailCode = 12;
return FALSE;
}
//
// Also restore the NT Global Flags to their original state.
//
NtGlobalFlag = oldNtGlobalFlag;
//
// Let WMI have a second chance to initialize, now that all drivers
// are started and should be ready to get WMI irps
//
WMIInitialize(1, NULL);
//
// Call Power manager to initialize for post-boot drivers
//
PoInitDriverServices(1);
//
// Indicate that the I/O system successfully initialized itself.
//
return TRUE;
}
VOID
IopSetIoRoutines()
{
if (pIofCallDriver == NULL) {
pIofCallDriver = IopfCallDriver;
}
if (pIofCompleteRequest == NULL) {
pIofCompleteRequest = IopfCompleteRequest;
}
if (pIoAllocateIrp == NULL) {
pIoAllocateIrp = IopAllocateIrpPrivate;
}
if (pIoFreeIrp == NULL) {
pIoFreeIrp = IopFreeIrp;
}
}
VOID
IopCreateArcNames(
IN PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
The loader block contains a table of disk signatures and corresponding
ARC names. Each device that the loader can access will appear in the
table. This routine opens each disk device in the system, reads the
signature and compares it to the table. For each match, it creates a
symbolic link between the nt device name and the ARC name.
The checksum value provided by the loader is the ULONG sum of all
elements in the checksum, inverted, plus 1:
checksum = ~sum + 1;
This way the sum of all of the elements can be calculated here and
added to the checksum in the loader block. If the result is zero, then
there is a match.
Arguments:
LoaderBlock - Supplies a pointer to the loader parameter block that was
created by the OS Loader.
Return Value:
None.
--*/
{
STRING arcBootDeviceString;
UCHAR deviceNameBuffer[128];
STRING deviceNameString;
UNICODE_STRING deviceNameUnicodeString;
PDEVICE_OBJECT deviceObject;
UCHAR arcNameBuffer[128];
STRING arcNameString;
UNICODE_STRING arcNameUnicodeString;
PFILE_OBJECT fileObject;
NTSTATUS status;
IO_STATUS_BLOCK ioStatusBlock;
DISK_GEOMETRY diskGeometry;
PDRIVE_LAYOUT_INFORMATION_EX driveLayout;
PLIST_ENTRY listEntry;
PARC_DISK_SIGNATURE diskBlock;
ULONG diskNumber;
ULONG partitionNumber;
PCHAR arcName;
PULONG buffer;
PIRP irp;
KEVENT event;
LARGE_INTEGER offset;
ULONG checkSum;
ULONG i;
PVOID tmpPtr;
BOOLEAN useLegacyEnumeration = FALSE;
BOOLEAN singleBiosDiskFound;
BOOLEAN bootDiskFound = FALSE;
PARC_DISK_INFORMATION arcInformation = LoaderBlock->ArcDiskInformation;
ULONG totalDriverDisksFound = IoGetConfigurationInformation()->DiskCount;
ULONG totalPnpDisksFound = 0;
STRING arcSystemDeviceString;
STRING osLoaderPathString;
UNICODE_STRING osLoaderPathUnicodeString;
PWSTR diskList = NULL;
wchar_t *pDiskNameList;
STORAGE_DEVICE_NUMBER pnpDiskDeviceNumber;
ULONG diskSignature;
//
// ask PNP to give us a list with all the currently active disks
//
pDiskNameList = diskList;
pnpDiskDeviceNumber.DeviceNumber = 0xFFFFFFFF;
status = IoGetDeviceInterfaces(&DiskClassGuid, NULL, 0, &diskList);
if (!NT_SUCCESS(status)) {
useLegacyEnumeration = TRUE;
if (pDiskNameList) {
*pDiskNameList = L'\0';
}
} else {
//
// count the number of disks returned
//
pDiskNameList = diskList;
while (*pDiskNameList != L'\0') {
totalPnpDisksFound++;
pDiskNameList = pDiskNameList + (wcslen(pDiskNameList) + 1);
}
pDiskNameList = diskList;
//
// if the disk returned by PNP are not all the disks in the system
// it means that some legacy driver has generated a disk device object/link.
// In that case we need to enumerate all pnp disks and then using the legacy
// for-loop also enumerate the non-pnp disks
//
if (totalPnpDisksFound < totalDriverDisksFound) {
useLegacyEnumeration = TRUE;
}
}
//
// If a single bios disk was found if there is only a
// single entry on the disk signature list.
//
singleBiosDiskFound = (arcInformation->DiskSignatures.Flink->Flink ==
&arcInformation->DiskSignatures) ? (TRUE) : (FALSE);
//
// Create hal/loader partition name
//
sprintf( arcNameBuffer, "\\ArcName\\%s", LoaderBlock->ArcHalDeviceName );
RtlInitAnsiString( &arcNameString, arcNameBuffer );
RtlAnsiStringToUnicodeString (&IoArcHalDeviceName, &arcNameString, TRUE);
//
// Create boot partition name
//
sprintf( arcNameBuffer, "\\ArcName\\%s", LoaderBlock->ArcBootDeviceName );
RtlInitAnsiString( &arcNameString, arcNameBuffer );
RtlAnsiStringToUnicodeString (&IoArcBootDeviceName, &arcNameString, TRUE);
i = strlen (LoaderBlock->ArcBootDeviceName) + 1;
IoLoaderArcBootDeviceName = ExAllocatePool (PagedPool, i);
if (IoLoaderArcBootDeviceName) {
memcpy (IoLoaderArcBootDeviceName, LoaderBlock->ArcBootDeviceName, i);
}
if (singleBiosDiskFound && strstr(LoaderBlock->ArcBootDeviceName, "cdrom")) {
singleBiosDiskFound = FALSE;
}
//
// Get ARC boot device name from loader block.
//
RtlInitAnsiString( &arcBootDeviceString,
LoaderBlock->ArcBootDeviceName );
//
// Get ARC system device name from loader block.
//
RtlInitAnsiString( &arcSystemDeviceString,
LoaderBlock->ArcHalDeviceName );
//
// If this is a remote boot, create an ArcName for the redirector path.
//
if (IoRemoteBootClient) {
bootDiskFound = TRUE;
RtlInitAnsiString( &deviceNameString, "\\Device\\LanmanRedirector" );
status = RtlAnsiStringToUnicodeString( &deviceNameUnicodeString,
&deviceNameString,
TRUE );
if (NT_SUCCESS( status )) {
sprintf( arcNameBuffer,
"\\ArcName\\%s",
LoaderBlock->ArcBootDeviceName );
RtlInitAnsiString( &arcNameString, arcNameBuffer );
status = RtlAnsiStringToUnicodeString( &arcNameUnicodeString,
&arcNameString,
TRUE );
if (NT_SUCCESS( status )) {
//
// Create symbolic link between NT device name and ARC name.
//
IoCreateSymbolicLink( &arcNameUnicodeString,
&deviceNameUnicodeString );
RtlFreeUnicodeString( &arcNameUnicodeString );
//
// We've found the system partition--store it away in the registry
// to later be transferred to a application-friendly location.
//
RtlInitAnsiString( &osLoaderPathString, LoaderBlock->NtHalPathName );
status = RtlAnsiStringToUnicodeString( &osLoaderPathUnicodeString,
&osLoaderPathString,
TRUE );
#if DBG
if (!NT_SUCCESS( status )) {
DbgPrint("IopCreateArcNames: couldn't allocate unicode string for OsLoader path - %x\n", status);
}
#endif // DBG
if (NT_SUCCESS( status )) {
IopStoreSystemPartitionInformation( &deviceNameUnicodeString,
&osLoaderPathUnicodeString );
RtlFreeUnicodeString( &osLoaderPathUnicodeString );
}
}
RtlFreeUnicodeString( &deviceNameUnicodeString );
}
}
//
// For each disk in the system do the following:
// 1. open the device
// 2. get its geometry
// 3. read the MBR
// 4. determine ARC name via disk signature and checksum
// 5. construct ARC name.
// In order to deal with the case of disk dissappearing before we get to this point
// (due to a failed start on one of many disks present in the system) we ask PNP for a list
// of all the currenttly active disks in the system. If the number of disks returned is
// less than the IoGetConfigurationInformation()->DiskCount, then we have legacy disks
// that we need to enumerate in the for loop.
// In the legacy case, the ending condition for the loop is NOT the total disk on the
// system but an arbitrary number of the max total legacy disks expected in the system..
// Additional note: Legacy disks get assigned symbolic links AFTER all pnp enumeration is complete
//
totalDriverDisksFound = max(totalPnpDisksFound,totalDriverDisksFound);
if (useLegacyEnumeration && (totalPnpDisksFound == 0)) {
//
// search up to a maximum arbitrary number of legacy disks
//
totalDriverDisksFound +=20;
}
for (diskNumber = 0;
diskNumber < totalDriverDisksFound;
diskNumber++) {
//
// Construct the NT name for a disk and obtain a reference.
//
if (pDiskNameList && (*pDiskNameList != L'\0')) {
//
// retrieve the first symbolic linkname from the PNP disk list
//
RtlInitUnicodeString(&deviceNameUnicodeString, pDiskNameList);
pDiskNameList = pDiskNameList + (wcslen(pDiskNameList) + 1);
status = IoGetDeviceObjectPointer( &deviceNameUnicodeString,
FILE_READ_ATTRIBUTES,
&fileObject,
&deviceObject );
if (NT_SUCCESS(status)) {
//
// since PNP gave s just asym link we have to retrieve the actual
// disk number through an IOCTL call to the disk stack.
// Create IRP for get device number device control.
//
irp = IoBuildDeviceIoControlRequest( IOCTL_STORAGE_GET_DEVICE_NUMBER,
deviceObject,
NULL,
0,
&pnpDiskDeviceNumber,
sizeof(STORAGE_DEVICE_NUMBER),
FALSE,
&event,
&ioStatusBlock );
if (!irp) {
ObDereferenceObject( fileObject );
continue;
}
KeInitializeEvent( &event,
NotificationEvent,
FALSE );
status = IoCallDriver( deviceObject,
irp );
if (status == STATUS_PENDING) {
KeWaitForSingleObject( &event,
Executive,
KernelMode,
FALSE,
NULL );
status = ioStatusBlock.Status;
}
if (!NT_SUCCESS( status )) {
ObDereferenceObject( fileObject );
continue;
}
}
if (useLegacyEnumeration && (*pDiskNameList == L'\0') ) {
//
// end of pnp disks
// if there are any legacy disks following we need to update
// the total disk found number to cover the maximum disk number
// a legacy disk could be at. (in a sparse name space)
//
if (pnpDiskDeviceNumber.DeviceNumber == 0xFFFFFFFF) {
pnpDiskDeviceNumber.DeviceNumber = 0;
}
diskNumber = max(diskNumber,pnpDiskDeviceNumber.DeviceNumber);
totalDriverDisksFound = diskNumber + 20;
}
} else {
sprintf( deviceNameBuffer,
"\\Device\\Harddisk%d\\Partition0",
diskNumber );
RtlInitAnsiString( &deviceNameString, deviceNameBuffer );
status = RtlAnsiStringToUnicodeString( &deviceNameUnicodeString,
&deviceNameString,
TRUE );
if (!NT_SUCCESS( status )) {
continue;
}
status = IoGetDeviceObjectPointer( &deviceNameUnicodeString,
FILE_READ_ATTRIBUTES,
&fileObject,
&deviceObject );
RtlFreeUnicodeString( &deviceNameUnicodeString );
//
// set the pnpDiskNumber value so its not used.
//
pnpDiskDeviceNumber.DeviceNumber = 0xFFFFFFFF;
}
if (!NT_SUCCESS( status )) {
continue;
}
//
// Create IRP for get drive geometry device control.
//
irp = IoBuildDeviceIoControlRequest( IOCTL_DISK_GET_DRIVE_GEOMETRY,
deviceObject,
NULL,
0,
&diskGeometry,
sizeof(DISK_GEOMETRY),
FALSE,
&event,
&ioStatusBlock );
if (!irp) {
ObDereferenceObject( fileObject );
continue;
}
KeInitializeEvent( &event,
NotificationEvent,
FALSE );
status = IoCallDriver( deviceObject,
irp );
if (status == STATUS_PENDING) {
KeWaitForSingleObject( &event,
Executive,
KernelMode,
FALSE,
NULL );
status = ioStatusBlock.Status;
}
if (!NT_SUCCESS( status )) {
ObDereferenceObject( fileObject );
continue;
}
//
// Get partition information for this disk.
//
status = IoReadPartitionTableEx(deviceObject,
&driveLayout );
if (!NT_SUCCESS( status )) {
ObDereferenceObject( fileObject );
continue;
}
//
// Make sure sector size is at least 512 bytes.
//
if (diskGeometry.BytesPerSector < 512) {
diskGeometry.BytesPerSector = 512;
}
//
// Check to see if EZ Drive is out there on this disk. If
// it is then zero out the signature in the drive layout since
// this will never be written by anyone AND change to offset to
// actually read sector 1 rather than 0 cause that's what the
// loader actually did.
//
offset.QuadPart = 0;
HalExamineMBR( deviceObject,
diskGeometry.BytesPerSector,
(ULONG)0x55,
&tmpPtr );
if (tmpPtr) {
offset.QuadPart = diskGeometry.BytesPerSector;
ExFreePool(tmpPtr);
}
//
// Allocate buffer for sector read and construct the read request.
//
buffer = ExAllocatePool( NonPagedPoolCacheAlignedMustS,
diskGeometry.BytesPerSector );
if (buffer) {
irp = IoBuildSynchronousFsdRequest( IRP_MJ_READ,
deviceObject,
buffer,
diskGeometry.BytesPerSector,
&offset,
&event,
&ioStatusBlock );
if (!irp) {
ExFreePool(driveLayout);
ExFreePool(buffer);
ObDereferenceObject( fileObject );
continue;
}
} else {
ExFreePool(driveLayout);
ObDereferenceObject( fileObject );
continue;
}
KeInitializeEvent( &event,
NotificationEvent,
FALSE );
status = IoCallDriver( deviceObject,
irp );
if (status == STATUS_PENDING) {
KeWaitForSingleObject( &event,
Executive,
KernelMode,
FALSE,
NULL );
status = ioStatusBlock.Status;
}
if (!NT_SUCCESS( status )) {
ExFreePool(driveLayout);
ExFreePool(buffer);
ObDereferenceObject( fileObject );
continue;
}
ObDereferenceObject( fileObject );
//
// Calculate MBR sector checksum. Only 512 bytes are used.
//
checkSum = 0;
for (i = 0; i < 128; i++) {
checkSum += buffer[i];
}
//
// For each ARC disk information record in the loader block
// match the disk signature and checksum to determine its ARC
// name and construct the NT ARC names symbolic links.
//
for (listEntry = arcInformation->DiskSignatures.Flink;
listEntry != &arcInformation->DiskSignatures;
listEntry = listEntry->Flink) {
//
// Get next record and compare disk signatures.
//
diskBlock = CONTAINING_RECORD( listEntry,
ARC_DISK_SIGNATURE,
ListEntry );
//
// Compare disk signatures.
//
// Or if there is only a single disk drive from
// both the bios and driver viewpoints then
// assign an arc name to that drive.
//
if ((singleBiosDiskFound &&
(totalDriverDisksFound == 1) &&
(driveLayout->PartitionStyle == PARTITION_STYLE_MBR)) ||
(IopVerifyDiskSignature(driveLayout, diskBlock, &diskSignature) &&
!(diskBlock->CheckSum + checkSum))) {
//
// Create unicode device name for physical disk.
//
if (pnpDiskDeviceNumber.DeviceNumber == 0xFFFFFFFF) {
sprintf( deviceNameBuffer,
"\\Device\\Harddisk%d\\Partition0",
diskNumber );
} else {
sprintf( deviceNameBuffer,
"\\Device\\Harddisk%d\\Partition0",
pnpDiskDeviceNumber.DeviceNumber );
}
RtlInitAnsiString( &deviceNameString, deviceNameBuffer );
status = RtlAnsiStringToUnicodeString( &deviceNameUnicodeString,
&deviceNameString,
TRUE );
if (!NT_SUCCESS( status )) {
continue;
}
//
// Create unicode ARC name for this partition.
//
arcName = diskBlock->ArcName;
sprintf( arcNameBuffer,
"\\ArcName\\%s",
arcName );
RtlInitAnsiString( &arcNameString, arcNameBuffer );
status = RtlAnsiStringToUnicodeString( &arcNameUnicodeString,
&arcNameString,
TRUE );
if (!NT_SUCCESS( status )) {
continue;
}
//
// Create symbolic link between NT device name and ARC name.
//
IoCreateSymbolicLink( &arcNameUnicodeString,
&deviceNameUnicodeString );
RtlFreeUnicodeString( &arcNameUnicodeString );
RtlFreeUnicodeString( &deviceNameUnicodeString );
//
// Create an ARC name for every partition on this disk.
//
for (partitionNumber = 0;
partitionNumber < driveLayout->PartitionCount;
partitionNumber++) {
//
// Create unicode NT device name.
//
if (pnpDiskDeviceNumber.DeviceNumber == 0xFFFFFFFF) {
sprintf( deviceNameBuffer,
"\\Device\\Harddisk%d\\Partition%d",
diskNumber,
partitionNumber+1 );
} else {
sprintf( deviceNameBuffer,
"\\Device\\Harddisk%d\\Partition%d",
pnpDiskDeviceNumber.DeviceNumber,
partitionNumber+1 );
}
RtlInitAnsiString( &deviceNameString, deviceNameBuffer );
status = RtlAnsiStringToUnicodeString( &deviceNameUnicodeString,
&deviceNameString,
TRUE );
if (!NT_SUCCESS( status )) {
continue;
}
//
// Create unicode ARC name for this partition and
// check to see if this is the boot disk.
//
sprintf( arcNameBuffer,
"%spartition(%d)",
arcName,
partitionNumber+1 );
RtlInitAnsiString( &arcNameString, arcNameBuffer );
if (RtlEqualString( &arcNameString,
&arcBootDeviceString,
TRUE )) {
bootDiskFound = TRUE;
}
//
// See if this is the system partition.
//
if (RtlEqualString( &arcNameString,
&arcSystemDeviceString,
TRUE )) {
//
// We've found the system partition--store it away in the registry
// to later be transferred to a application-friendly location.
//
RtlInitAnsiString( &osLoaderPathString, LoaderBlock->NtHalPathName );
status = RtlAnsiStringToUnicodeString( &osLoaderPathUnicodeString,
&osLoaderPathString,
TRUE );
#if DBG
if (!NT_SUCCESS( status )) {
DbgPrint("IopCreateArcNames: couldn't allocate unicode string for OsLoader path - %x\n", status);
}
#endif // DBG
if (NT_SUCCESS( status )) {
IopStoreSystemPartitionInformation( &deviceNameUnicodeString,
&osLoaderPathUnicodeString );
RtlFreeUnicodeString( &osLoaderPathUnicodeString );
}
}
//
// Add the NT ARC namespace prefix to the ARC name constructed.
//
sprintf( arcNameBuffer,
"\\ArcName\\%spartition(%d)",
arcName,
partitionNumber+1 );
RtlInitAnsiString( &arcNameString, arcNameBuffer );
status = RtlAnsiStringToUnicodeString( &arcNameUnicodeString,
&arcNameString,
TRUE );
if (!NT_SUCCESS( status )) {
continue;
}
//
// Create symbolic link between NT device name and ARC name.
//
IoCreateSymbolicLink( &arcNameUnicodeString,
&deviceNameUnicodeString );
RtlFreeUnicodeString( &arcNameUnicodeString );
RtlFreeUnicodeString( &deviceNameUnicodeString );
}
} else {
#if DBG
//
// Check key indicators to see if this condition may be
// caused by a viral infection.
//
if (diskBlock->Signature == diskSignature &&
(diskBlock->CheckSum + checkSum) != 0 &&
diskBlock->ValidPartitionTable) {
DbgPrint("IopCreateArcNames: Virus or duplicate disk signatures\n");
}
#endif
}
}
ExFreePool( driveLayout );
ExFreePool( buffer );
}
if (!bootDiskFound) {
//
// Locate the disk block that represents the boot device.
//
diskBlock = NULL;
for (listEntry = arcInformation->DiskSignatures.Flink;
listEntry != &arcInformation->DiskSignatures;
listEntry = listEntry->Flink) {
diskBlock = CONTAINING_RECORD( listEntry,
ARC_DISK_SIGNATURE,
ListEntry );
if (strcmp( diskBlock->ArcName, LoaderBlock->ArcBootDeviceName ) == 0) {
break;
}
diskBlock = NULL;
}
if (diskBlock) {
//
// This could be a CdRom boot. Search all of the NT CdRoms
// to locate a checksum match on the diskBlock found. If
// there is a match, assign the ARC name to the CdRom.
//
irp = NULL;
buffer = ExAllocatePool( NonPagedPoolCacheAlignedMustS,
2048 );
if (buffer) {
//
// Construct the NT names for CdRoms and search each one
// for a checksum match. If found, create the ARC Name
// symbolic link.
//
for (diskNumber = 0; TRUE; diskNumber++) {
sprintf( deviceNameBuffer,
"\\Device\\CdRom%d",
diskNumber );
RtlInitAnsiString( &deviceNameString, deviceNameBuffer );
status = RtlAnsiStringToUnicodeString( &deviceNameUnicodeString,
&deviceNameString,
TRUE );
if (NT_SUCCESS( status )) {
status = IoGetDeviceObjectPointer( &deviceNameUnicodeString,
FILE_READ_ATTRIBUTES,
&fileObject,
&deviceObject );
if (!NT_SUCCESS( status )) {
//
// All CdRoms have been processed.
//
RtlFreeUnicodeString( &deviceNameUnicodeString );
break;
}
//
// Read the block for the checksum calculation.
//
offset.QuadPart = 0x8000;
irp = IoBuildSynchronousFsdRequest( IRP_MJ_READ,
deviceObject,
buffer,
2048,
&offset,
&event,
&ioStatusBlock );
checkSum = 0;
if (irp) {
KeInitializeEvent( &event,
NotificationEvent,
FALSE );
status = IoCallDriver( deviceObject,
irp );
if (status == STATUS_PENDING) {
KeWaitForSingleObject( &event,
Executive,
KernelMode,
FALSE,
NULL );
status = ioStatusBlock.Status;
}
if (NT_SUCCESS( status )) {
//
// Calculate MBR sector checksum.
// 2048 bytes are used.
//
for (i = 0; i < 2048 / sizeof(ULONG) ; i++) {
checkSum += buffer[i];
}
}
}
ObDereferenceObject( fileObject );
if (!(diskBlock->CheckSum + checkSum)) {
//
// This is the boot CdRom. Create the symlink for
// the ARC name from the loader block.
//
sprintf( arcNameBuffer,
"\\ArcName\\%s",
LoaderBlock->ArcBootDeviceName );
RtlInitAnsiString( &arcNameString, arcNameBuffer );
status = RtlAnsiStringToUnicodeString( &arcNameUnicodeString,
&arcNameString,
TRUE );
if (NT_SUCCESS( status )) {
IoCreateSymbolicLink( &arcNameUnicodeString,
&deviceNameUnicodeString );
RtlFreeUnicodeString( &arcNameUnicodeString );
}
RtlFreeUnicodeString( &deviceNameUnicodeString );
break;
}
RtlFreeUnicodeString( &deviceNameUnicodeString );
}
}
ExFreePool(buffer);
}
}
}
if (diskList) {
ExFreePool(diskList);
}
}
#ifdef ALLOC_DATA_PRAGMA
#pragma const_seg("PAGECONST")
#endif // ALLOC_DATA_PRAGMA
const GENERIC_MAPPING IopFileMapping = {
STANDARD_RIGHTS_READ |
FILE_READ_DATA | FILE_READ_ATTRIBUTES | FILE_READ_EA | SYNCHRONIZE,
STANDARD_RIGHTS_WRITE |
FILE_WRITE_DATA | FILE_WRITE_ATTRIBUTES | FILE_WRITE_EA | FILE_APPEND_DATA | SYNCHRONIZE,
STANDARD_RIGHTS_EXECUTE |
SYNCHRONIZE | FILE_READ_ATTRIBUTES | FILE_EXECUTE,
FILE_ALL_ACCESS
};
#ifdef ALLOC_DATA_PRAGMA
#pragma const_seg("INITCONST")
#endif // ALLOC_DATA_PRAGMA
const GENERIC_MAPPING IopCompletionMapping = {
STANDARD_RIGHTS_READ |
IO_COMPLETION_QUERY_STATE,
STANDARD_RIGHTS_WRITE |
IO_COMPLETION_MODIFY_STATE,
STANDARD_RIGHTS_EXECUTE |
SYNCHRONIZE,
IO_COMPLETION_ALL_ACCESS
};
BOOLEAN
IopCreateObjectTypes(
VOID
)
/*++
Routine Description:
This routine creates the object types used by the I/O system and its
components. The object types created are:
Adapter
Controller
Device
Driver
File
I/O Completion
Arguments:
None.
Return Value:
The function value is a BOOLEAN indicating whether or not the object
types were successfully created.
--*/
{
OBJECT_TYPE_INITIALIZER objectTypeInitializer;
UNICODE_STRING nameString;
//
// Initialize the common fields of the Object Type Initializer record
//
RtlZeroMemory( &objectTypeInitializer, sizeof( objectTypeInitializer ) );
objectTypeInitializer.Length = sizeof( objectTypeInitializer );
objectTypeInitializer.InvalidAttributes = OBJ_OPENLINK;
objectTypeInitializer.GenericMapping = IopFileMapping;
objectTypeInitializer.PoolType = NonPagedPool;
objectTypeInitializer.ValidAccessMask = FILE_ALL_ACCESS;
objectTypeInitializer.UseDefaultObject = TRUE;
//
// Create the object type for adapter objects.
//
RtlInitUnicodeString( &nameString, L"Adapter" );
// objectTypeInitializer.DefaultNonPagedPoolCharge = sizeof( struct _ADAPTER_OBJECT );
if (!NT_SUCCESS( ObCreateObjectType( &nameString,
&objectTypeInitializer,
(PSECURITY_DESCRIPTOR) NULL,
&IoAdapterObjectType ))) {
return FALSE;
}
#ifdef _PNP_POWER_
//
// Create the object type for device helper objects.
//
RtlInitUnicodeString( &nameString, L"DeviceHandler" );
if (!NT_SUCCESS( ObCreateObjectType( &nameString,
&objectTypeInitializer,
(PSECURITY_DESCRIPTOR) NULL,
&IoDeviceHandlerObjectType ))) {
return FALSE;
}
IoDeviceHandlerObjectSize = sizeof(DEVICE_HANDLER_OBJECT);
#endif
//
// Create the object type for controller objects.
//
RtlInitUnicodeString( &nameString, L"Controller" );
objectTypeInitializer.DefaultNonPagedPoolCharge = sizeof( CONTROLLER_OBJECT );
if (!NT_SUCCESS( ObCreateObjectType( &nameString,
&objectTypeInitializer,
(PSECURITY_DESCRIPTOR) NULL,
&IoControllerObjectType ))) {
return FALSE;
}
//
// Create the object type for device objects.
//
RtlInitUnicodeString( &nameString, L"Device" );
objectTypeInitializer.DefaultNonPagedPoolCharge = sizeof( DEVICE_OBJECT );
objectTypeInitializer.ParseProcedure = IopParseDevice;
objectTypeInitializer.CaseInsensitive = TRUE;
objectTypeInitializer.DeleteProcedure = IopDeleteDevice;
objectTypeInitializer.SecurityProcedure = IopGetSetSecurityObject;
objectTypeInitializer.QueryNameProcedure = (OB_QUERYNAME_METHOD)NULL;
if (!NT_SUCCESS( ObCreateObjectType( &nameString,
&objectTypeInitializer,
(PSECURITY_DESCRIPTOR) NULL,
&IoDeviceObjectType ))) {
return FALSE;
}
//
// Create the object type for driver objects.
//
RtlInitUnicodeString( &nameString, L"Driver" );
objectTypeInitializer.DefaultNonPagedPoolCharge = sizeof( DRIVER_OBJECT );
objectTypeInitializer.ParseProcedure = (OB_PARSE_METHOD) NULL;
objectTypeInitializer.DeleteProcedure = IopDeleteDriver;
objectTypeInitializer.SecurityProcedure = (OB_SECURITY_METHOD) NULL;
objectTypeInitializer.QueryNameProcedure = (OB_QUERYNAME_METHOD)NULL;
//
// This allows us to get a list of Driver objects.
//
if (IopVerifierOn) {
objectTypeInitializer.MaintainTypeList = TRUE;
}
if (!NT_SUCCESS( ObCreateObjectType( &nameString,
&objectTypeInitializer,
(PSECURITY_DESCRIPTOR) NULL,
&IoDriverObjectType ))) {
return FALSE;
}
//
// Create the object type for I/O completion objects.
//
RtlInitUnicodeString( &nameString, L"IoCompletion" );
objectTypeInitializer.DefaultNonPagedPoolCharge = sizeof( KQUEUE );
objectTypeInitializer.InvalidAttributes = OBJ_PERMANENT | OBJ_OPENLINK;
objectTypeInitializer.GenericMapping = IopCompletionMapping;
objectTypeInitializer.ValidAccessMask = IO_COMPLETION_ALL_ACCESS;
objectTypeInitializer.DeleteProcedure = IopDeleteIoCompletion;
if (!NT_SUCCESS( ObCreateObjectType( &nameString,
&objectTypeInitializer,
(PSECURITY_DESCRIPTOR) NULL,
&IoCompletionObjectType ))) {
return FALSE;
}
//
// Create the object type for file objects.
//
RtlInitUnicodeString( &nameString, L"File" );
objectTypeInitializer.DefaultPagedPoolCharge = IO_FILE_OBJECT_PAGED_POOL_CHARGE;
objectTypeInitializer.DefaultNonPagedPoolCharge = IO_FILE_OBJECT_NON_PAGED_POOL_CHARGE +
sizeof( FILE_OBJECT );
objectTypeInitializer.InvalidAttributes = OBJ_PERMANENT | OBJ_EXCLUSIVE | OBJ_OPENLINK;
objectTypeInitializer.GenericMapping = IopFileMapping;
objectTypeInitializer.ValidAccessMask = FILE_ALL_ACCESS;
objectTypeInitializer.MaintainHandleCount = TRUE;
objectTypeInitializer.CloseProcedure = IopCloseFile;
objectTypeInitializer.DeleteProcedure = IopDeleteFile;
objectTypeInitializer.ParseProcedure = IopParseFile;
objectTypeInitializer.SecurityProcedure = IopGetSetSecurityObject;
objectTypeInitializer.QueryNameProcedure = IopQueryName;
objectTypeInitializer.UseDefaultObject = FALSE;
PERFINFO_MUNG_FILE_OBJECT_TYPE_INITIALIZER(objectTypeInitializer);
if (!NT_SUCCESS( ObCreateObjectType( &nameString,
&objectTypeInitializer,
(PSECURITY_DESCRIPTOR) NULL,
&IoFileObjectType ))) {
return FALSE;
}
PERFINFO_UNMUNG_FILE_OBJECT_TYPE_INITIALIZER(objectTypeInitializer);
return TRUE;
}
BOOLEAN
IopCreateRootDirectories(
VOID
)
/*++
Routine Description:
This routine is invoked to create the object manager directory objects
to contain the various device and file system driver objects.
Arguments:
None.
Return Value:
The function value is a BOOLEAN indicating whether or not the directory
objects were successfully created.
--*/
{
HANDLE handle;
OBJECT_ATTRIBUTES objectAttributes;
UNICODE_STRING nameString;
NTSTATUS status;
//
// Create the root directory object for the \Driver directory.
//
RtlInitUnicodeString( &nameString, L"\\Driver" );
InitializeObjectAttributes( &objectAttributes,
&nameString,
OBJ_PERMANENT,
(HANDLE) NULL,
(PSECURITY_DESCRIPTOR) NULL );
status = NtCreateDirectoryObject( &handle,
DIRECTORY_ALL_ACCESS,
&objectAttributes );
if (!NT_SUCCESS( status )) {
return FALSE;
} else {
(VOID) NtClose( handle );
}
//
// Create the root directory object for the \FileSystem directory.
//
RtlInitUnicodeString( &nameString, L"\\FileSystem" );
status = NtCreateDirectoryObject( &handle,
DIRECTORY_ALL_ACCESS,
&objectAttributes );
if (!NT_SUCCESS( status )) {
return FALSE;
} else {
(VOID) NtClose( handle );
}
//
// Create the root directory object for the \FileSystem\Filters directory.
//
RtlInitUnicodeString( &nameString, L"\\FileSystem\\Filters" );
status = NtCreateDirectoryObject( &handle,
DIRECTORY_ALL_ACCESS,
&objectAttributes );
if (!NT_SUCCESS( status )) {
return FALSE;
} else {
(VOID) NtClose( handle );
}
return TRUE;
}
NTSTATUS
IopInitializeAttributesAndCreateObject(
IN PUNICODE_STRING ObjectName,
IN OUT POBJECT_ATTRIBUTES ObjectAttributes,
OUT PDRIVER_OBJECT *DriverObject
)
/*++
Routine Description:
This routine is invoked to initialize a set of object attributes and
to create a driver object.
Arguments:
ObjectName - Supplies the name of the driver object.
ObjectAttributes - Supplies a pointer to the object attributes structure
to be initialized.
DriverObject - Supplies a variable to receive a pointer to the resultant
created driver object.
Return Value:
The function value is the final status of the operation.
--*/
{
NTSTATUS status;
//
// Simply initialize the object attributes and create the driver object.
//
InitializeObjectAttributes( ObjectAttributes,
ObjectName,
OBJ_PERMANENT | OBJ_CASE_INSENSITIVE,
(HANDLE) NULL,
(PSECURITY_DESCRIPTOR) NULL );
status = ObCreateObject( KeGetPreviousMode(),
IoDriverObjectType,
ObjectAttributes,
KernelMode,
(PVOID) NULL,
(ULONG) (sizeof( DRIVER_OBJECT ) + sizeof ( DRIVER_EXTENSION )),
0,
0,
(PVOID *)DriverObject );
return status;
}
NTSTATUS
IopInitializeBuiltinDriver(
IN PUNICODE_STRING DriverName,
IN PUNICODE_STRING RegistryPath,
IN PDRIVER_INITIALIZE DriverInitializeRoutine,
IN PKLDR_DATA_TABLE_ENTRY DriverEntry,
IN BOOLEAN IsFilter,
IN PDRIVER_OBJECT *Result
)
/*++
Routine Description:
This routine is invoked to initialize a built-in driver.
Arguments:
DriverName - Specifies the name to be used in creating the driver object.
RegistryPath - Specifies the path to be used by the driver to get to
the registry.
DriverInitializeRoutine - Specifies the initialization entry point of
the built-in driver.
DriverEntry - Specifies the driver data table entry to determine if the
driver is a wdm driver.
Return Value:
The function returns a pointer to a DRIVER_OBJECT if the built-in
driver successfully initialized. Otherwise, a value of NULL is returned.
--*/
{
HANDLE handle;
PDRIVER_OBJECT driverObject;
PDRIVER_OBJECT tmpDriverObject;
OBJECT_ATTRIBUTES objectAttributes;
PWSTR buffer;
NTSTATUS status;
HANDLE serviceHandle;
PWSTR pserviceName;
USHORT serviceNameLength;
PDRIVER_EXTENSION driverExtension;
PIMAGE_NT_HEADERS ntHeaders;
PVOID imageBase;
#if DBG
LARGE_INTEGER stime, etime;
ULONG dtime;
#endif
PLIST_ENTRY entry;
PKLDR_DATA_TABLE_ENTRY DataTableEntry;
*Result = NULL;
//
// Log the file name
//
HeadlessKernelAddLogEntry(HEADLESS_LOG_LOADING_FILENAME, DriverName);
//
// Begin by creating the driver object.
//
status = IopInitializeAttributesAndCreateObject( DriverName,
&objectAttributes,
&driverObject );
if (!NT_SUCCESS( status )) {
HeadlessKernelAddLogEntry(HEADLESS_LOG_LOAD_FAILED, NULL);
return status;
}
//
// Initialize the driver object.
//
InitializeDriverObject( driverObject );
driverObject->DriverInit = DriverInitializeRoutine;
//
// Insert the driver object into the object table.
//
status = ObInsertObject( driverObject,
NULL,
FILE_READ_DATA,
0,
(PVOID *) NULL,
&handle );
if (!NT_SUCCESS( status )) {
HeadlessKernelAddLogEntry(HEADLESS_LOG_LOAD_FAILED, NULL);
return status;
}
//
// Reference the handle and obtain a pointer to the driver object so that
// the handle can be deleted without the object going away.
//
status = ObReferenceObjectByHandle( handle,
0,
IoDriverObjectType,
KernelMode,
(PVOID *) &tmpDriverObject,
(POBJECT_HANDLE_INFORMATION) NULL );
ASSERT(status == STATUS_SUCCESS);
//
// Fill in the DriverSection so the image will be automatically unloaded on
// failures. We should use the entry from the PsModuleList.
//
entry = PsLoadedModuleList.Flink;
while (entry != &PsLoadedModuleList && DriverEntry) {
DataTableEntry = CONTAINING_RECORD(entry,
KLDR_DATA_TABLE_ENTRY,
InLoadOrderLinks);
if (RtlEqualString((PSTRING)&DriverEntry->BaseDllName,
(PSTRING)&DataTableEntry->BaseDllName,
TRUE
)) {
driverObject->DriverSection = DataTableEntry;
break;
}
entry = entry->Flink;
}
//
// The boot process takes a while loading drivers. Indicate that
// progress is being made.
//
InbvIndicateProgress();
//
// Get start and sice for the DriverObject.
//
if (DriverEntry) {
imageBase = DriverEntry->DllBase;
ntHeaders = RtlImageNtHeader(imageBase);
driverObject->DriverStart = imageBase;
driverObject->DriverSize = ntHeaders->OptionalHeader.SizeOfImage;
if (!(ntHeaders->OptionalHeader.DllCharacteristics & IMAGE_DLLCHARACTERISTICS_WDM_DRIVER)) {
driverObject->Flags |= DRVO_LEGACY_DRIVER;
}
} else {
ntHeaders = NULL;
driverObject->Flags |= DRVO_LEGACY_DRIVER;
}
//
// Save the name of the driver so that it can be easily located by functions
// such as error logging.
//
buffer = ExAllocatePool( PagedPool, DriverName->MaximumLength + 2 );
if (buffer) {
driverObject->DriverName.Buffer = buffer;
driverObject->DriverName.MaximumLength = DriverName->MaximumLength;
driverObject->DriverName.Length = DriverName->Length;
RtlCopyMemory( driverObject->DriverName.Buffer,
DriverName->Buffer,
DriverName->MaximumLength );
buffer[DriverName->Length >> 1] = (WCHAR) '\0';
}
//
// Save the name of the service key so that it can be easily located by PnP
// mamager.
//
driverExtension = driverObject->DriverExtension;
if (RegistryPath && RegistryPath->Length != 0) {
pserviceName = RegistryPath->Buffer + RegistryPath->Length / sizeof (WCHAR) - 1;
if (*pserviceName == OBJ_NAME_PATH_SEPARATOR) {
pserviceName--;
}
serviceNameLength = 0;
while (pserviceName != RegistryPath->Buffer) {
if (*pserviceName == OBJ_NAME_PATH_SEPARATOR) {
pserviceName++;
break;
} else {
serviceNameLength += sizeof(WCHAR);
pserviceName--;
}
}
if (pserviceName == RegistryPath->Buffer) {
serviceNameLength += sizeof(WCHAR);
}
buffer = ExAllocatePool( NonPagedPool, serviceNameLength + sizeof(UNICODE_NULL) );
if (buffer) {
driverExtension->ServiceKeyName.Buffer = buffer;
driverExtension->ServiceKeyName.MaximumLength = serviceNameLength + sizeof(UNICODE_NULL);
driverExtension->ServiceKeyName.Length = serviceNameLength;
RtlCopyMemory( driverExtension->ServiceKeyName.Buffer,
pserviceName,
serviceNameLength );
buffer[driverExtension->ServiceKeyName.Length >> 1] = UNICODE_NULL;
} else {
status = STATUS_INSUFFICIENT_RESOURCES;
driverExtension->ServiceKeyName.Buffer = NULL;
driverExtension->ServiceKeyName.Length = 0;
goto exit;
}
//
// Prepare driver initialization
//
status = IopOpenRegistryKeyEx( &serviceHandle,
NULL,
RegistryPath,
KEY_ALL_ACCESS
);
if (NT_SUCCESS(status)) {
status = IopPrepareDriverLoading(&driverExtension->ServiceKeyName,
serviceHandle,
imageBase,
IsFilter);
NtClose(serviceHandle);
if (!NT_SUCCESS(status)) {
goto exit;
}
} else {
goto exit;
}
} else {
driverExtension->ServiceKeyName.Buffer = NULL;
driverExtension->ServiceKeyName.MaximumLength = 0;
driverExtension->ServiceKeyName.Length = 0;
}
//
// Load the Registry information in the appropriate fields of the device
// object.
//
driverObject->HardwareDatabase = &CmRegistryMachineHardwareDescriptionSystemName;
#if DBG
KeQuerySystemTime (&stime);
#endif
//
// Now invoke the driver's initialization routine to initialize itself.
//
status = driverObject->DriverInit( driverObject, RegistryPath );
#if DBG
//
// If DriverInit took longer than 5 seconds or the driver did not load,
// print a message.
//
KeQuerySystemTime (&etime);
dtime = (ULONG) ((etime.QuadPart - stime.QuadPart) / 1000000);
if (dtime > 50 || !NT_SUCCESS( status )) {
if (dtime < 10) {
DbgPrint( "IOINIT: Built-in driver %wZ failed to initialize - %lX\n",
DriverName, status );
} else {
DbgPrint( "IOINIT: Built-in driver %wZ took %d.%ds to ",
DriverName, dtime/10, dtime%10 );
if (NT_SUCCESS( status )) {
DbgPrint ("initialize\n");
} else {
DbgPrint ("fail initialization - %lX\n", status);
}
}
}
#endif
exit:
NtClose( handle );
if (NT_SUCCESS( status )) {
IopReadyDeviceObjects( driverObject );
HeadlessKernelAddLogEntry(HEADLESS_LOG_LOAD_SUCCESSFUL, NULL);
*Result = driverObject;
return status;
} else {
if (status != STATUS_PLUGPLAY_NO_DEVICE) {
//
// if STATUS_PLUGPLAY_NO_DEVICE, the driver was disable by hardware profile.
//
IopDriverLoadingFailed(NULL, &driverObject->DriverExtension->ServiceKeyName);
}
HeadlessKernelAddLogEntry(HEADLESS_LOG_LOAD_FAILED, NULL);
ObMakeTemporaryObject(driverObject);
ObDereferenceObject (driverObject);
return status;
}
}
BOOLEAN
IopMarkBootPartition(
IN PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
This routine is invoked to locate and mark the boot partition device object
as a boot device so that subsequent operations can fail more cleanly and
with a better explanation of why the system failed to boot and run properly.
Arguments:
LoaderBlock - Supplies a pointer to the loader parameter block created
by the OS Loader during the boot process. This structure contains
the various system partition and boot device names and paths.
Return Value:
The function value is TRUE if everything worked, otherwise FALSE.
Notes:
If the boot partition device object cannot be found, then the system will
bugcheck.
--*/
{
OBJECT_ATTRIBUTES objectAttributes;
STRING deviceNameString;
UCHAR deviceNameBuffer[256];
UNICODE_STRING deviceNameUnicodeString;
NTSTATUS status;
HANDLE fileHandle;
IO_STATUS_BLOCK ioStatus;
PFILE_OBJECT fileObject;
CHAR ArcNameFmt[12];
ArcNameFmt[0] = '\\';
ArcNameFmt[1] = 'A';
ArcNameFmt[2] = 'r';
ArcNameFmt[3] = 'c';
ArcNameFmt[4] = 'N';
ArcNameFmt[5] = 'a';
ArcNameFmt[6] = 'm';
ArcNameFmt[7] = 'e';
ArcNameFmt[8] = '\\';
ArcNameFmt[9] = '%';
ArcNameFmt[10] = 's';
ArcNameFmt[11] = '\0';
//
// Open the ARC boot device object. The boot device driver should have
// created the object.
//
sprintf( deviceNameBuffer,
ArcNameFmt,
LoaderBlock->ArcBootDeviceName );
RtlInitAnsiString( &deviceNameString, deviceNameBuffer );
status = RtlAnsiStringToUnicodeString( &deviceNameUnicodeString,
&deviceNameString,
TRUE );
if (!NT_SUCCESS( status )) {
return FALSE;
}
InitializeObjectAttributes( &objectAttributes,
&deviceNameUnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL );
status = ZwOpenFile( &fileHandle,
FILE_READ_ATTRIBUTES,
&objectAttributes,
&ioStatus,
0,
FILE_NON_DIRECTORY_FILE );
if (!NT_SUCCESS( status )) {
KeBugCheckEx( INACCESSIBLE_BOOT_DEVICE,
(ULONG_PTR) &deviceNameUnicodeString,
status,
0,
0 );
}
//
// Convert the file handle into a pointer to the device object itself.
//
status = ObReferenceObjectByHandle( fileHandle,
0,
IoFileObjectType,
KernelMode,
(PVOID *) &fileObject,
NULL );
if (!NT_SUCCESS( status )) {
RtlFreeUnicodeString( &deviceNameUnicodeString );
return FALSE;
}
//
// Mark the device object represented by the file object.
//
fileObject->DeviceObject->Flags |= DO_SYSTEM_BOOT_PARTITION;
//
// Save away the characteristics of boot device object for later
// use in WinPE mode
//
if (InitIsWinPEMode) {
if (fileObject->DeviceObject->Characteristics & FILE_REMOVABLE_MEDIA) {
InitWinPEModeType |= INIT_WINPEMODE_REMOVABLE_MEDIA;
}
if (fileObject->DeviceObject->Characteristics & FILE_READ_ONLY_DEVICE) {
InitWinPEModeType |= INIT_WINPEMODE_READONLY_MEDIA;
}
}
//
// Reference the device object and store the reference.
//
ObReferenceObject(fileObject->DeviceObject);
IopErrorLogObject = fileObject->DeviceObject;
RtlFreeUnicodeString( &deviceNameUnicodeString );
//
// Finally, close the handle and dereference the file object.
//
NtClose( fileHandle );
ObDereferenceObject( fileObject );
return TRUE;
}
BOOLEAN
IopReassignSystemRoot(
IN PLOADER_PARAMETER_BLOCK LoaderBlock,
OUT PSTRING NtDeviceName
)
/*++
Routine Description:
This routine is invoked to reassign \SystemRoot from being an ARC path
name to its NT path name equivalent. This is done by looking up the
ARC device name as a symbolic link and determining which NT device object
is referred to by it. The link is then replaced with the new name.
Arguments:
LoaderBlock - Supplies a pointer to the loader parameter block created
by the OS Loader during the boot process. This structure contains
the various system partition and boot device names and paths.
NtDeviceName - Specifies a pointer to a STRING to receive the NT name of
the device from which the system was booted.
Return Value:
The function value is a BOOLEAN indicating whether or not the ARC name
was resolved to an NT name.
--*/
{
OBJECT_ATTRIBUTES objectAttributes;
NTSTATUS status;
UCHAR deviceNameBuffer[256];
WCHAR arcNameUnicodeBuffer[64];
UCHAR arcNameStringBuffer[256];
STRING deviceNameString;
STRING arcNameString;
STRING linkString;
UNICODE_STRING linkUnicodeString;
UNICODE_STRING deviceNameUnicodeString;
UNICODE_STRING arcNameUnicodeString;
HANDLE linkHandle;
#if DBG
UCHAR debugBuffer[256];
STRING debugString;
UNICODE_STRING debugUnicodeString;
#endif
CHAR ArcNameFmt[12];
ArcNameFmt[0] = '\\';
ArcNameFmt[1] = 'A';
ArcNameFmt[2] = 'r';
ArcNameFmt[3] = 'c';
ArcNameFmt[4] = 'N';
ArcNameFmt[5] = 'a';
ArcNameFmt[6] = 'm';
ArcNameFmt[7] = 'e';
ArcNameFmt[8] = '\\';
ArcNameFmt[9] = '%';
ArcNameFmt[10] = 's';
ArcNameFmt[11] = '\0';
//
// Open the ARC boot device symbolic link object. The boot device
// driver should have created the object.
//
sprintf( deviceNameBuffer,
ArcNameFmt,
LoaderBlock->ArcBootDeviceName );
RtlInitAnsiString( &deviceNameString, deviceNameBuffer );
status = RtlAnsiStringToUnicodeString( &deviceNameUnicodeString,
&deviceNameString,
TRUE );
if (!NT_SUCCESS( status )) {
return FALSE;
}
InitializeObjectAttributes( &objectAttributes,
&deviceNameUnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL );
status = NtOpenSymbolicLinkObject( &linkHandle,
SYMBOLIC_LINK_ALL_ACCESS,
&objectAttributes );
if (!NT_SUCCESS( status )) {
#if DBG
sprintf( debugBuffer, "IOINIT: unable to resolve %s, Status == %X\n",
deviceNameBuffer,
status );
RtlInitAnsiString( &debugString, debugBuffer );
status = RtlAnsiStringToUnicodeString( &debugUnicodeString,
&debugString,
TRUE );
if (NT_SUCCESS( status )) {
ZwDisplayString( &debugUnicodeString );
RtlFreeUnicodeString( &debugUnicodeString );
}
#endif // DBG
RtlFreeUnicodeString( &deviceNameUnicodeString );
return FALSE;
}
//
// Get handle to \SystemRoot symbolic link.
//
arcNameUnicodeString.Buffer = arcNameUnicodeBuffer;
arcNameUnicodeString.Length = 0;
arcNameUnicodeString.MaximumLength = sizeof( arcNameUnicodeBuffer );
status = NtQuerySymbolicLinkObject( linkHandle,
&arcNameUnicodeString,
NULL );
if (!NT_SUCCESS( status )) {
return FALSE;
}
arcNameString.Buffer = arcNameStringBuffer;
arcNameString.Length = 0;
arcNameString.MaximumLength = sizeof( arcNameStringBuffer );
status = RtlUnicodeStringToAnsiString( &arcNameString,
&arcNameUnicodeString,
FALSE );
arcNameStringBuffer[arcNameString.Length] = '\0';
NtClose( linkHandle );
RtlFreeUnicodeString( &deviceNameUnicodeString );
RtlInitAnsiString( &linkString, INIT_SYSTEMROOT_LINKNAME );
status = RtlAnsiStringToUnicodeString( &linkUnicodeString,
&linkString,
TRUE);
if (!NT_SUCCESS( status )) {
return FALSE;
}
InitializeObjectAttributes( &objectAttributes,
&linkUnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL );
status = NtOpenSymbolicLinkObject( &linkHandle,
SYMBOLIC_LINK_ALL_ACCESS,
&objectAttributes );
if (!NT_SUCCESS( status )) {
return FALSE;
}
NtMakeTemporaryObject( linkHandle );
NtClose( linkHandle );
sprintf( deviceNameBuffer,
"%Z%s",
&arcNameString,
LoaderBlock->NtBootPathName );
//
// Get NT device name for \SystemRoot assignment.
//
RtlCopyString( NtDeviceName, &arcNameString );
deviceNameBuffer[strlen(deviceNameBuffer)-1] = '\0';
RtlInitAnsiString(&deviceNameString, deviceNameBuffer);
InitializeObjectAttributes( &objectAttributes,
&linkUnicodeString,
OBJ_CASE_INSENSITIVE | OBJ_PERMANENT,
NULL,
NULL );
status = RtlAnsiStringToUnicodeString( &arcNameUnicodeString,
&deviceNameString,
TRUE);
if (!NT_SUCCESS( status )) {
return FALSE;
}
status = NtCreateSymbolicLinkObject( &linkHandle,
SYMBOLIC_LINK_ALL_ACCESS,
&objectAttributes,
&arcNameUnicodeString );
RtlFreeUnicodeString( &arcNameUnicodeString );
RtlFreeUnicodeString( &linkUnicodeString );
NtClose( linkHandle );
#if DBG
if (NT_SUCCESS( status )) {
sprintf( debugBuffer,
"INIT: Reassigned %s => %s\n",
INIT_SYSTEMROOT_LINKNAME,
deviceNameBuffer );
} else {
sprintf( debugBuffer,
"INIT: unable to create %s => %s, Status == %X\n",
INIT_SYSTEMROOT_LINKNAME,
deviceNameBuffer,
status );
}
RtlInitAnsiString( &debugString, debugBuffer );
status = RtlAnsiStringToUnicodeString( &debugUnicodeString,
&debugString,
TRUE );
if (NT_SUCCESS( status )) {
ZwDisplayString( &debugUnicodeString );
RtlFreeUnicodeString( &debugUnicodeString );
}
#endif // DBG
return TRUE;
}
VOID
IopStoreSystemPartitionInformation(
IN PUNICODE_STRING NtSystemPartitionDeviceName,
IN OUT PUNICODE_STRING OsLoaderPathName
)
/*++
Routine Description:
This routine writes two values to the registry (under HKLM\SYSTEM\Setup)--one
containing the NT device name of the system partition and the other containing
the path to the OS loader. These values will later be migrated into a
Win95-compatible registry location (NT path converted to DOS path), so that
installation programs (including our own setup) have a rock-solid way of knowing
what the system partition is, on both ARC and x86.
ERRORS ENCOUNTERED IN THIS ROUTINE ARE NOT CONSIDERED FATAL.
Arguments:
NtSystemPartitionDeviceName - supplies the NT device name of the system partition.
This is the \Device\Harddisk<n>\Partition<m> name, which used to be the actual
device name, but now is a symbolic link to a name of the form \Device\Volume<x>.
We open up this symbolic link, and retrieve the name that it points to. The
target name is the one we store away in the registry.
OsLoaderPathName - supplies the path (on the partition specified in the 1st parameter)
where the OS loader is located. Upon return, this path will have had its trailing
backslash removed (if present, and path isn't root).
Return Value:
None.
--*/
{
NTSTATUS status;
HANDLE linkHandle;
OBJECT_ATTRIBUTES objectAttributes;
HANDLE systemHandle, setupHandle;
UNICODE_STRING nameString, volumeNameString;
WCHAR voumeNameStringBuffer[256];
//
// Declare a unicode buffer big enough to contain the longest string we'll be using.
// (ANSI string in 'sizeof()' below on purpose--we want the number of chars here.)
//
WCHAR nameBuffer[sizeof("SystemPartition")];
//
// Both UNICODE_STRING buffers should be NULL-terminated.
//
ASSERT( NtSystemPartitionDeviceName->MaximumLength >= NtSystemPartitionDeviceName->Length + sizeof(WCHAR) );
ASSERT( NtSystemPartitionDeviceName->Buffer[NtSystemPartitionDeviceName->Length / sizeof(WCHAR)] == L'\0' );
ASSERT( OsLoaderPathName->MaximumLength >= OsLoaderPathName->Length + sizeof(WCHAR) );
ASSERT( OsLoaderPathName->Buffer[OsLoaderPathName->Length / sizeof(WCHAR)] == L'\0' );
//
// Open the NtSystemPartitionDeviceName symbolic link, and find out the volume device
// it points to.
//
InitializeObjectAttributes(&objectAttributes,
NtSystemPartitionDeviceName,
OBJ_CASE_INSENSITIVE,
NULL,
NULL
);
status = NtOpenSymbolicLinkObject(&linkHandle,
SYMBOLIC_LINK_QUERY,
&objectAttributes
);
if (!NT_SUCCESS(status)) {
#if DBG
DbgPrint("IopStoreSystemPartitionInformation: couldn't open symbolic link %wZ - %x\n",
NtSystemPartitionDeviceName,
status
);
#endif // DBG
return;
}
volumeNameString.Buffer = voumeNameStringBuffer;
volumeNameString.Length = 0;
//
// Leave room at the end of the buffer for a terminating null, in case we need to add one.
//
volumeNameString.MaximumLength = sizeof(voumeNameStringBuffer) - sizeof(WCHAR);
status = NtQuerySymbolicLinkObject(linkHandle,
&volumeNameString,
NULL
);
//
// We don't need the handle to the symbolic link any longer.
//
NtClose(linkHandle);
if (!NT_SUCCESS(status)) {
#if DBG
DbgPrint("IopStoreSystemPartitionInformation: couldn't query symbolic link %wZ - %x\n",
NtSystemPartitionDeviceName,
status
);
#endif // DBG
return;
}
//
// Make sure the volume name string is null-terminated.
//
volumeNameString.Buffer[volumeNameString.Length / sizeof(WCHAR)] = L'\0';
//
// Open HKLM\SYSTEM key.
//
status = IopOpenRegistryKeyEx( &systemHandle,
NULL,
&CmRegistryMachineSystemName,
KEY_ALL_ACCESS
);
if (!NT_SUCCESS(status)) {
#if DBG
DbgPrint("IopStoreSystemPartitionInformation: couldn't open \\REGISTRY\\MACHINE\\SYSTEM - %x\n", status);
#endif // DBG
return;
}
//
// Now open/create the setup subkey.
//
ASSERT( sizeof(L"Setup") <= sizeof(nameBuffer) );
nameBuffer[0] = L'S';
nameBuffer[1] = L'e';
nameBuffer[2] = L't';
nameBuffer[3] = L'u';
nameBuffer[4] = L'p';
nameBuffer[5] = L'\0';
nameString.MaximumLength = sizeof(L"Setup");
nameString.Length = sizeof(L"Setup") - sizeof(WCHAR);
nameString.Buffer = nameBuffer;
status = IopCreateRegistryKeyEx( &setupHandle,
systemHandle,
&nameString,
KEY_ALL_ACCESS,
REG_OPTION_NON_VOLATILE,
NULL
);
NtClose(systemHandle); // Don't need the handle to the HKLM\System key anymore.
if (!NT_SUCCESS(status)) {
#if DBG
DbgPrint("IopStoreSystemPartitionInformation: couldn't open Setup subkey - %x\n", status);
#endif // DBG
return;
}
ASSERT( sizeof(L"SystemPartition") <= sizeof(nameBuffer) );
nameBuffer[0] = L'S';
nameBuffer[1] = L'y';
nameBuffer[2] = L's';
nameBuffer[3] = L't';
nameBuffer[4] = L'e';
nameBuffer[5] = L'm';
nameBuffer[6] = L'P';
nameBuffer[7] = L'a';
nameBuffer[8] = L'r';
nameBuffer[9] = L't';
nameBuffer[10] = L'i';
nameBuffer[11] = L't';
nameBuffer[12] = L'i';
nameBuffer[13] = L'o';
nameBuffer[14] = L'n';
nameBuffer[15] = L'\0';
nameString.MaximumLength = sizeof(L"SystemPartition");
nameString.Length = sizeof(L"SystemPartition") - sizeof(WCHAR);
status = NtSetValueKey(setupHandle,
&nameString,
TITLE_INDEX_VALUE,
REG_SZ,
volumeNameString.Buffer,
volumeNameString.Length + sizeof(WCHAR)
);
#if DBG
if (!NT_SUCCESS(status)) {
DbgPrint("IopStoreSystemPartitionInformation: couldn't write SystemPartition value - %x\n", status);
}
#endif // DBG
ASSERT( sizeof(L"OsLoaderPath") <= sizeof(nameBuffer) );
nameBuffer[0] = L'O';
nameBuffer[1] = L's';
nameBuffer[2] = L'L';
nameBuffer[3] = L'o';
nameBuffer[4] = L'a';
nameBuffer[5] = L'd';
nameBuffer[6] = L'e';
nameBuffer[7] = L'r';
nameBuffer[8] = L'P';
nameBuffer[9] = L'a';
nameBuffer[10] = L't';
nameBuffer[11] = L'h';
nameBuffer[12] = L'\0';
nameString.MaximumLength = sizeof(L"OsLoaderPath");
nameString.Length = sizeof(L"OsLoaderPath") - sizeof(WCHAR);
//
// Strip off the trailing backslash from the path (unless, of course, the path is a
// single backslash).
//
if ((OsLoaderPathName->Length > sizeof(WCHAR)) &&
(*(PWCHAR)((PCHAR)OsLoaderPathName->Buffer + OsLoaderPathName->Length - sizeof(WCHAR)) == L'\\')) {
OsLoaderPathName->Length -= sizeof(WCHAR);
*(PWCHAR)((PCHAR)OsLoaderPathName->Buffer + OsLoaderPathName->Length) = L'\0';
}
status = NtSetValueKey(setupHandle,
&nameString,
TITLE_INDEX_VALUE,
REG_SZ,
OsLoaderPathName->Buffer,
OsLoaderPathName->Length + sizeof(WCHAR)
);
#if DBG
if (!NT_SUCCESS(status)) {
DbgPrint("IopStoreSystemPartitionInformation: couldn't write OsLoaderPath value - %x\n", status);
}
#endif // DBG
NtClose(setupHandle);
}
NTSTATUS
IopLogErrorEvent(
IN ULONG SequenceNumber,
IN ULONG UniqueErrorValue,
IN NTSTATUS FinalStatus,
IN NTSTATUS SpecificIOStatus,
IN ULONG LengthOfInsert1,
IN PWCHAR Insert1,
IN ULONG LengthOfInsert2,
IN PWCHAR Insert2
)
/*++
Routine Description:
This routine allocates an error log entry, copies the supplied data
to it, and requests that it be written to the error log file.
Arguments:
SequenceNumber - A value that is unique to an IRP over the life of the irp in
this driver. - 0 generally means an error not associated with an IRP
UniqueErrorValue - A unique long word that identifies the particular
call to this function.
FinalStatus - The final status given to the irp that was associated
with this error. If this log entry is being made during one of
the retries this value will be STATUS_SUCCESS.
SpecificIOStatus - The IO status for a particular error.
LengthOfInsert1 - The length in bytes (including the terminating NULL)
of the first insertion string.
Insert1 - The first insertion string.
LengthOfInsert2 - The length in bytes (including the terminating NULL)
of the second insertion string. NOTE, there must
be a first insertion string for their to be
a second insertion string.
Insert2 - The second insertion string.
Return Value:
STATUS_SUCCESS - Success
STATUS_INVALID_HANDLE - Uninitialized error log device object
STATUS_NO_DATA_DETECTED - NULL Error log entry
--*/
{
PIO_ERROR_LOG_PACKET errorLogEntry;
PUCHAR ptrToFirstInsert;
PUCHAR ptrToSecondInsert;
if (!IopErrorLogObject) {
return(STATUS_INVALID_HANDLE);
}
errorLogEntry = IoAllocateErrorLogEntry(
IopErrorLogObject,
(UCHAR)( sizeof(IO_ERROR_LOG_PACKET) +
LengthOfInsert1 +
LengthOfInsert2) );
if ( errorLogEntry != NULL ) {
errorLogEntry->ErrorCode = SpecificIOStatus;
errorLogEntry->SequenceNumber = SequenceNumber;
errorLogEntry->MajorFunctionCode = 0;
errorLogEntry->RetryCount = 0;
errorLogEntry->UniqueErrorValue = UniqueErrorValue;
errorLogEntry->FinalStatus = FinalStatus;
errorLogEntry->DumpDataSize = 0;
ptrToFirstInsert = (PUCHAR)&errorLogEntry->DumpData[0];
ptrToSecondInsert = ptrToFirstInsert + LengthOfInsert1;
if (LengthOfInsert1) {
errorLogEntry->NumberOfStrings = 1;
errorLogEntry->StringOffset = (USHORT)(ptrToFirstInsert -
(PUCHAR)errorLogEntry);
RtlCopyMemory(
ptrToFirstInsert,
Insert1,
LengthOfInsert1
);
if (LengthOfInsert2) {
errorLogEntry->NumberOfStrings = 2;
RtlCopyMemory(
ptrToSecondInsert,
Insert2,
LengthOfInsert2
);
} //LenghtOfInsert2
} // LenghtOfInsert1
IoWriteErrorLogEntry(errorLogEntry);
return(STATUS_SUCCESS);
} // errorLogEntry != NULL
return(STATUS_NO_DATA_DETECTED);
} //IopLogErrorEvent
BOOLEAN
IopInitializeReserveIrp(
PIOP_RESERVE_IRP_ALLOCATOR Allocator
)
/*++
Routine Description:
This routine initializes the reserve IRP allocator for paging reads.
Arguments:
Allocator - Pointer to the reserve IRP allocator structure.
created by the OS Loader.
Return Value:
The function value is a BOOLEAN indicating whether or not the reserver allocator
was successfully initialized.
--*/
{
Allocator->ReserveIrpStackSize = MAX_RESERVE_IRP_STACK_SIZE;
Allocator->ReserveIrp = IoAllocateIrp(MAX_RESERVE_IRP_STACK_SIZE, FALSE);
if (Allocator->ReserveIrp == NULL) {
return FALSE;
}
Allocator->IrpAllocated = FALSE;
KeInitializeEvent(&Allocator->Event, SynchronizationEvent, FALSE);
return TRUE;
}