mirror of https://github.com/tongzx/nt5src
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.
652 lines
21 KiB
652 lines
21 KiB
/*++
|
|
|
|
Copyright (c) 1991 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
ntsetup.c
|
|
|
|
Abstract:
|
|
|
|
This module is the tail-end of the OS loader program. It performs all
|
|
MIPS specific allocations and initialize. The OS loader invokes this
|
|
this routine immediately before calling the loaded kernel image.
|
|
|
|
Author:
|
|
|
|
John Vert (jvert) 20-Jun-1991
|
|
|
|
Environment:
|
|
|
|
Kernel mode
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include "bldr.h"
|
|
#include "stdio.h"
|
|
|
|
//
|
|
// Define macro to round structure size to next 16-byte boundary
|
|
//
|
|
|
|
#define ROUND_UP(x) ((sizeof(x) + 15) & (~15))
|
|
|
|
//
|
|
// Configuration Data Header
|
|
// The following structure is copied from fw\mips\oli2msft.h
|
|
// NOTE shielint - Somehow, this structure got incorporated into
|
|
// firmware EISA configuration data. We need to know the size of the
|
|
// header and remove it before writing eisa configuration data to
|
|
// registry.
|
|
//
|
|
|
|
typedef struct _CONFIGURATION_DATA_HEADER {
|
|
USHORT Version;
|
|
USHORT Revision;
|
|
PCHAR Type;
|
|
PCHAR Vendor;
|
|
PCHAR ProductName;
|
|
PCHAR SerialNumber;
|
|
} CONFIGURATION_DATA_HEADER;
|
|
|
|
#define CONFIGURATION_DATA_HEADER_SIZE sizeof(CONFIGURATION_DATA_HEADER)
|
|
|
|
//
|
|
// Internal function references
|
|
//
|
|
|
|
ARC_STATUS
|
|
ReorganizeEisaConfigurationTree(
|
|
IN PCONFIGURATION_COMPONENT_DATA RootEntry
|
|
);
|
|
|
|
ARC_STATUS
|
|
CreateEisaConfigurationData (
|
|
IN PCONFIGURATION_COMPONENT_DATA RootEntry
|
|
);
|
|
|
|
VOID
|
|
BlQueryImplementationAndRevision (
|
|
OUT PULONG ProcessorId,
|
|
OUT PULONG FloatingId
|
|
);
|
|
|
|
ARC_STATUS
|
|
BlSetupForNt(
|
|
IN PLOADER_PARAMETER_BLOCK BlLoaderBlock
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function initializes the MIPS specific kernel data structures
|
|
required by the NT system.
|
|
|
|
Arguments:
|
|
|
|
BlLoaderBlock - Supplies the address of the loader parameter block.
|
|
|
|
Return Value:
|
|
|
|
ESUCCESS is returned if the setup is successfully complete. Otherwise,
|
|
an unsuccessful status is returned.
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
PCONFIGURATION_COMPONENT_DATA ConfigEntry;
|
|
ULONG FloatingId;
|
|
CHAR Identifier[256];
|
|
ULONG KernelPage;
|
|
ULONG LinesPerBlock;
|
|
ULONG LineSize;
|
|
PCHAR NewIdentifier;
|
|
ULONG PrcbPage;
|
|
ULONG ProcessorId;
|
|
ARC_STATUS Status;
|
|
|
|
//
|
|
// If the host configuration is not a multiprocessor machine, then add
|
|
// the processor and floating point coprocessor identification to the
|
|
// processor identification string.
|
|
//
|
|
|
|
if (SYSTEM_BLOCK->RestartBlock == NULL) {
|
|
ConfigEntry = KeFindConfigurationEntry(BlLoaderBlock->ConfigurationRoot,
|
|
ProcessorClass,
|
|
CentralProcessor,
|
|
NULL);
|
|
|
|
if (ConfigEntry != NULL) {
|
|
BlQueryImplementationAndRevision(&ProcessorId, &FloatingId);
|
|
sprintf(&Identifier[0],
|
|
"%s - Pr %d/%d, Fp %d/%d",
|
|
ConfigEntry->ComponentEntry.Identifier,
|
|
(ProcessorId >> 8) & 0xff,
|
|
ProcessorId & 0xff,
|
|
(FloatingId >> 8) & 0xff,
|
|
FloatingId & 0xff);
|
|
|
|
NewIdentifier = (PCHAR)BlAllocateHeap(strlen(&Identifier[0]) + 1);
|
|
if (NewIdentifier != NULL) {
|
|
strcpy(NewIdentifier, &Identifier[0]);
|
|
ConfigEntry->ComponentEntry.IdentifierLength = strlen(NewIdentifier);
|
|
ConfigEntry->ComponentEntry.Identifier = NewIdentifier;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Find System entry and check each of its direct child to
|
|
// look for EisaAdapter.
|
|
//
|
|
|
|
ConfigEntry = KeFindConfigurationEntry(BlLoaderBlock->ConfigurationRoot,
|
|
SystemClass,
|
|
ArcSystem,
|
|
NULL);
|
|
if (ConfigEntry) {
|
|
ConfigEntry = ConfigEntry->Child;
|
|
}
|
|
|
|
while (ConfigEntry) {
|
|
|
|
if ((ConfigEntry->ComponentEntry.Class == AdapterClass) &&
|
|
(ConfigEntry->ComponentEntry.Type == EisaAdapter)) {
|
|
|
|
//
|
|
// Convert EISA format configuration data to our CM_ format.
|
|
//
|
|
|
|
Status = ReorganizeEisaConfigurationTree(ConfigEntry);
|
|
if (Status != ESUCCESS) {
|
|
return(Status);
|
|
}
|
|
}
|
|
ConfigEntry = ConfigEntry->Sibling;
|
|
}
|
|
|
|
//
|
|
// Find the primary data and instruction cache configuration entries, and
|
|
// compute the fill size and cache size for each cache. These entries MUST
|
|
// be present on all ARC compliant systems.
|
|
//
|
|
|
|
ConfigEntry = KeFindConfigurationEntry(BlLoaderBlock->ConfigurationRoot,
|
|
CacheClass,
|
|
PrimaryDcache,
|
|
NULL);
|
|
|
|
if (ConfigEntry != NULL) {
|
|
LinesPerBlock = ConfigEntry->ComponentEntry.Key >> 24;
|
|
LineSize = 1 << ((ConfigEntry->ComponentEntry.Key >> 16) & 0xff);
|
|
BlLoaderBlock->u.Mips.FirstLevelDcacheFillSize = LinesPerBlock * LineSize;
|
|
BlLoaderBlock->u.Mips.FirstLevelDcacheSize =
|
|
1 << ((ConfigEntry->ComponentEntry.Key & 0xffff) + PAGE_SHIFT);
|
|
|
|
} else {
|
|
return EINVAL;
|
|
}
|
|
|
|
ConfigEntry = KeFindConfigurationEntry(BlLoaderBlock->ConfigurationRoot,
|
|
CacheClass,
|
|
PrimaryIcache,
|
|
NULL);
|
|
|
|
if (ConfigEntry != NULL) {
|
|
LinesPerBlock = ConfigEntry->ComponentEntry.Key >> 24;
|
|
LineSize = 1 << ((ConfigEntry->ComponentEntry.Key >> 16) & 0xff);
|
|
BlLoaderBlock->u.Mips.FirstLevelIcacheFillSize = LinesPerBlock * LineSize;
|
|
BlLoaderBlock->u.Mips.FirstLevelIcacheSize =
|
|
1 << ((ConfigEntry->ComponentEntry.Key & 0xffff) + PAGE_SHIFT);
|
|
|
|
} else {
|
|
return EINVAL;
|
|
}
|
|
|
|
//
|
|
// Find the secondary data and instruction cache configuration entries,
|
|
// and if present, compute the fill size and cache size for each cache.
|
|
// These entries are optional, and may or may not, be present.
|
|
//
|
|
|
|
ConfigEntry = KeFindConfigurationEntry(BlLoaderBlock->ConfigurationRoot,
|
|
CacheClass,
|
|
SecondaryCache,
|
|
NULL);
|
|
|
|
if (ConfigEntry != NULL) {
|
|
LinesPerBlock = ConfigEntry->ComponentEntry.Key >> 24;
|
|
LineSize = 1 << ((ConfigEntry->ComponentEntry.Key >> 16) & 0xff);
|
|
BlLoaderBlock->u.Mips.SecondLevelDcacheFillSize = LinesPerBlock * LineSize;
|
|
BlLoaderBlock->u.Mips.SecondLevelDcacheSize =
|
|
1 << ((ConfigEntry->ComponentEntry.Key & 0xffff) + PAGE_SHIFT);
|
|
|
|
BlLoaderBlock->u.Mips.SecondLevelIcacheSize = 0;
|
|
BlLoaderBlock->u.Mips.SecondLevelIcacheFillSize = 0;
|
|
|
|
} else {
|
|
ConfigEntry = KeFindConfigurationEntry(BlLoaderBlock->ConfigurationRoot,
|
|
CacheClass,
|
|
SecondaryDcache,
|
|
NULL);
|
|
|
|
if (ConfigEntry != NULL) {
|
|
LinesPerBlock = ConfigEntry->ComponentEntry.Key >> 24;
|
|
LineSize = 1 << ((ConfigEntry->ComponentEntry.Key >> 16) & 0xff);
|
|
BlLoaderBlock->u.Mips.SecondLevelDcacheFillSize = LinesPerBlock * LineSize;
|
|
BlLoaderBlock->u.Mips.SecondLevelDcacheSize =
|
|
1 << ((ConfigEntry->ComponentEntry.Key & 0xffff) + PAGE_SHIFT);
|
|
|
|
ConfigEntry = KeFindConfigurationEntry(BlLoaderBlock->ConfigurationRoot,
|
|
CacheClass,
|
|
SecondaryIcache,
|
|
NULL);
|
|
|
|
if (ConfigEntry != NULL) {
|
|
LinesPerBlock = ConfigEntry->ComponentEntry.Key >> 24;
|
|
LineSize = 1 << ((ConfigEntry->ComponentEntry.Key >> 16) & 0xff);
|
|
BlLoaderBlock->u.Mips.SecondLevelIcacheFillSize = LinesPerBlock * LineSize;
|
|
BlLoaderBlock->u.Mips.SecondLevelIcacheSize =
|
|
1 << ((ConfigEntry->ComponentEntry.Key & 0xffff) + PAGE_SHIFT);
|
|
|
|
} else {
|
|
BlLoaderBlock->u.Mips.SecondLevelIcacheSize = 0;
|
|
BlLoaderBlock->u.Mips.SecondLevelIcacheFillSize = 0;
|
|
}
|
|
|
|
} else {
|
|
BlLoaderBlock->u.Mips.SecondLevelDcacheSize = 0;
|
|
BlLoaderBlock->u.Mips.SecondLevelDcacheFillSize = 0;
|
|
BlLoaderBlock->u.Mips.SecondLevelIcacheSize = 0;
|
|
BlLoaderBlock->u.Mips.SecondLevelIcacheFillSize = 0;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Allocate DPC stack pages for the boot processor.
|
|
//
|
|
|
|
Status = BlAllocateDescriptor(LoaderStartupDpcStack,
|
|
0,
|
|
KERNEL_STACK_SIZE >> PAGE_SHIFT,
|
|
&KernelPage);
|
|
|
|
if (Status != ESUCCESS) {
|
|
return(Status);
|
|
}
|
|
|
|
BlLoaderBlock->u.Mips.InterruptStack =
|
|
(KSEG0_BASE | (KernelPage << PAGE_SHIFT)) + KERNEL_STACK_SIZE;
|
|
|
|
//
|
|
// Allocate kernel stack pages for the boot processor idle thread.
|
|
//
|
|
|
|
Status = BlAllocateDescriptor(LoaderStartupKernelStack,
|
|
0,
|
|
KERNEL_STACK_SIZE >> PAGE_SHIFT,
|
|
&KernelPage);
|
|
|
|
if (Status != ESUCCESS) {
|
|
return(Status);
|
|
}
|
|
|
|
BlLoaderBlock->KernelStack =
|
|
(KSEG0_BASE | (KernelPage << PAGE_SHIFT)) + KERNEL_STACK_SIZE;
|
|
|
|
//
|
|
// Allocate panic stack pages for the boot processor.
|
|
//
|
|
|
|
Status = BlAllocateDescriptor(LoaderStartupPanicStack,
|
|
0,
|
|
KERNEL_STACK_SIZE >> PAGE_SHIFT,
|
|
&KernelPage);
|
|
|
|
if (Status != ESUCCESS) {
|
|
return(Status);
|
|
}
|
|
|
|
BlLoaderBlock->u.Mips.PanicStack =
|
|
(KSEG0_BASE | (KernelPage << PAGE_SHIFT)) + KERNEL_STACK_SIZE;
|
|
|
|
//
|
|
// Allocate and zero two pages for the PCR.
|
|
//
|
|
|
|
Status = BlAllocateDescriptor(LoaderStartupPcrPage,
|
|
0,
|
|
2,
|
|
&BlLoaderBlock->u.Mips.PcrPage);
|
|
|
|
if (Status != ESUCCESS) {
|
|
return(Status);
|
|
}
|
|
|
|
BlLoaderBlock->u.Mips.PcrPage2 = BlLoaderBlock->u.Mips.PcrPage + 1;
|
|
RtlZeroMemory((PVOID)(KSEG0_BASE | (BlLoaderBlock->u.Mips.PcrPage << PAGE_SHIFT)),
|
|
PAGE_SIZE * 2);
|
|
|
|
//
|
|
// Allocate and zero two pages for the PDR and one page of memory for
|
|
// the initial processor block, idle process, and idle thread structures.
|
|
//
|
|
|
|
Status = BlAllocateDescriptor(LoaderStartupPdrPage,
|
|
0,
|
|
3,
|
|
&BlLoaderBlock->u.Mips.PdrPage);
|
|
|
|
if (Status != ESUCCESS) {
|
|
return(Status);
|
|
}
|
|
|
|
RtlZeroMemory((PVOID)(KSEG0_BASE | (BlLoaderBlock->u.Mips.PdrPage << PAGE_SHIFT)),
|
|
PAGE_SIZE * 3);
|
|
|
|
//
|
|
// The storage for processor control block, the idle thread object, and
|
|
// the idle thread process object are allocated from the third page of the
|
|
// PDR allocation. The addresses of these data structures are computed
|
|
// and stored in the loader parameter block and the memory is zeroed.
|
|
//
|
|
|
|
PrcbPage = BlLoaderBlock->u.Mips.PdrPage + 2;
|
|
if (PAGE_SIZE >= (ROUND_UP(KPRCB) + ROUND_UP(EPROCESS) + ROUND_UP(ETHREAD))) {
|
|
BlLoaderBlock->Prcb = KSEG0_BASE | (PrcbPage << PAGE_SHIFT);
|
|
BlLoaderBlock->Process = BlLoaderBlock->Prcb + ROUND_UP(KPRCB);
|
|
BlLoaderBlock->Thread = BlLoaderBlock->Process + ROUND_UP(EPROCESS);
|
|
|
|
} else {
|
|
return(ENOMEM);
|
|
}
|
|
|
|
//
|
|
// Flush all caches.
|
|
//
|
|
|
|
if (SYSTEM_BLOCK->FirmwareVectorLength > (sizeof(PVOID) * FlushAllCachesRoutine)) {
|
|
ArcFlushAllCaches();
|
|
}
|
|
|
|
return(ESUCCESS);
|
|
}
|
|
|
|
ARC_STATUS
|
|
ReorganizeEisaConfigurationTree(
|
|
IN PCONFIGURATION_COMPONENT_DATA RootEntry
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine sorts the eisa adapter configuration tree based on
|
|
the slot the component resided in. It also creates a new configuration
|
|
data for EisaAdapter component to contain ALL the eisa slot and function
|
|
information. Finally the Eisa tree will be wiped out.
|
|
|
|
Arguments:
|
|
|
|
RootEntry - Supplies a pointer to a EisaAdapter component. This is
|
|
the root of Eisa adapter tree.
|
|
|
|
|
|
Returns:
|
|
|
|
ESUCCESS is returned if the reorganization is successfully complete.
|
|
Otherwise, an unsuccessful status is returned.
|
|
|
|
--*/
|
|
{
|
|
|
|
PCONFIGURATION_COMPONENT_DATA CurrentEntry, PreviousEntry;
|
|
PCONFIGURATION_COMPONENT_DATA EntryFound, EntryFoundPrevious;
|
|
PCONFIGURATION_COMPONENT_DATA AttachedEntry, DetachedList;
|
|
ARC_STATUS Status;
|
|
|
|
//
|
|
// We sort the direct children of EISA adapter tree based on the slot
|
|
// they reside in. Only the direct children of EISA root need to be
|
|
// sorted.
|
|
// Note the "Key" field of CONFIGURATION_COMPONENT contains
|
|
// EISA slot number.
|
|
//
|
|
|
|
//
|
|
// First, detach all the children from EISA root.
|
|
//
|
|
|
|
AttachedEntry = NULL; // Child list of Eisa root
|
|
DetachedList = RootEntry->Child; // Detached child list
|
|
PreviousEntry = NULL;
|
|
|
|
while (DetachedList) {
|
|
|
|
//
|
|
// Find the component with the smallest slot number from detached
|
|
// list.
|
|
//
|
|
|
|
EntryFound = DetachedList;
|
|
EntryFoundPrevious = NULL;
|
|
CurrentEntry = DetachedList->Sibling;
|
|
PreviousEntry = DetachedList;
|
|
while (CurrentEntry) {
|
|
if (CurrentEntry->ComponentEntry.Key <
|
|
EntryFound->ComponentEntry.Key) {
|
|
EntryFound = CurrentEntry;
|
|
EntryFoundPrevious = PreviousEntry;
|
|
}
|
|
PreviousEntry = CurrentEntry;
|
|
CurrentEntry = CurrentEntry->Sibling;
|
|
}
|
|
|
|
//
|
|
// Remove the component from the detached child list.
|
|
// If the component is not the head of the detached list, we remove it
|
|
// by setting its previous entry's sibling to the component's sibling.
|
|
// Otherwise, we simply update Detach list head to point to the
|
|
// component's sibling.
|
|
//
|
|
|
|
if (EntryFoundPrevious) {
|
|
EntryFoundPrevious->Sibling = EntryFound->Sibling;
|
|
} else {
|
|
DetachedList = EntryFound->Sibling;
|
|
}
|
|
|
|
//
|
|
// Attach the component to the child list of Eisa root.
|
|
//
|
|
|
|
if (AttachedEntry) {
|
|
AttachedEntry->Sibling = EntryFound;
|
|
} else {
|
|
RootEntry->Child = EntryFound;
|
|
}
|
|
AttachedEntry = EntryFound;
|
|
AttachedEntry->Sibling = NULL;
|
|
}
|
|
|
|
//
|
|
// Finally, we traverse the Eisa tree to collect all the Eisa slot
|
|
// and function information and put it to the configuration data of
|
|
// Eisa root entry.
|
|
//
|
|
|
|
Status = CreateEisaConfigurationData(RootEntry);
|
|
|
|
//
|
|
// Wipe out all the children of EISA tree.
|
|
// NOTE shielint - For each child component, we should convert its
|
|
// configuration data from EISA format to our CM_ format.
|
|
//
|
|
|
|
RootEntry->Child = NULL;
|
|
return(Status);
|
|
|
|
}
|
|
|
|
ARC_STATUS
|
|
CreateEisaConfigurationData (
|
|
IN PCONFIGURATION_COMPONENT_DATA RootEntry
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine traverses Eisa configuration tree to collect all the
|
|
slot and function information and attaches it to the configuration data
|
|
of Eisa RootEntry.
|
|
|
|
Note that this routine assumes that the EISA tree has been sorted based
|
|
on the slot number.
|
|
|
|
Arguments:
|
|
|
|
RootEntry - Supplies a pointer to the Eisa configuration
|
|
component entry.
|
|
|
|
Returns:
|
|
|
|
ESUCCESS is returned if the new EisaAdapter configuration data is
|
|
successfully created. Otherwise, an unsuccessful status is returned.
|
|
|
|
--*/
|
|
{
|
|
ULONG DataSize, NextSlot = 0, i;
|
|
PCM_PARTIAL_RESOURCE_LIST Descriptor;
|
|
PCONFIGURATION_COMPONENT Component;
|
|
PCONFIGURATION_COMPONENT_DATA CurrentEntry;
|
|
PUCHAR DataPointer;
|
|
CM_EISA_SLOT_INFORMATION EmptySlot =
|
|
{EISA_EMPTY_SLOT, 0, 0, 0, 0, 0, 0, 0};
|
|
|
|
//
|
|
// Remove the configuration data of Eisa Adapter
|
|
//
|
|
|
|
RootEntry->ConfigurationData = NULL;
|
|
RootEntry->ComponentEntry.ConfigurationDataLength = 0;
|
|
|
|
//
|
|
// If the EISA stree contains valid slot information, i.e.
|
|
// root has children attaching to it.
|
|
//
|
|
|
|
if (RootEntry->Child) {
|
|
|
|
//
|
|
// First find out how much memory is needed to store EISA config
|
|
// data.
|
|
//
|
|
|
|
DataSize = sizeof(CM_PARTIAL_RESOURCE_LIST);
|
|
CurrentEntry = RootEntry->Child;
|
|
|
|
while (CurrentEntry) {
|
|
Component = &CurrentEntry->ComponentEntry;
|
|
if (CurrentEntry->ConfigurationData) {
|
|
if (Component->Key > NextSlot) {
|
|
|
|
//
|
|
// If there is any empty slot between current slot
|
|
// and previous checked slot, we need to count the
|
|
// space for the empty slots.
|
|
//
|
|
|
|
DataSize += (Component->Key - NextSlot) *
|
|
sizeof(CM_EISA_SLOT_INFORMATION);
|
|
}
|
|
DataSize += Component->ConfigurationDataLength + 1 -
|
|
CONFIGURATION_DATA_HEADER_SIZE;
|
|
NextSlot = Component->Key + 1;
|
|
}
|
|
CurrentEntry = CurrentEntry->Sibling;
|
|
}
|
|
|
|
//
|
|
// Allocate memory from heap to hold the EISA configuration data.
|
|
//
|
|
|
|
DataPointer = BlAllocateHeap(DataSize);
|
|
|
|
if (DataPointer == NULL) {
|
|
return ENOMEM;
|
|
} else {
|
|
RootEntry->ConfigurationData = DataPointer;
|
|
RootEntry->ComponentEntry.ConfigurationDataLength = DataSize;
|
|
}
|
|
|
|
//
|
|
// Create a CM_PARTIAL_RESOURCE_LIST for the new configuration data.
|
|
//
|
|
|
|
Descriptor = (PCM_PARTIAL_RESOURCE_LIST)DataPointer;
|
|
Descriptor->Version = 0;
|
|
Descriptor->Revision = 0;
|
|
Descriptor->Count = 1;
|
|
Descriptor->PartialDescriptors[0].Type = CmResourceTypeDeviceSpecific;
|
|
Descriptor->PartialDescriptors[0].ShareDisposition = 0;
|
|
Descriptor->PartialDescriptors[0].Flags = 0;
|
|
Descriptor->PartialDescriptors[0].u.DeviceSpecificData.Reserved1 = 0;
|
|
Descriptor->PartialDescriptors[0].u.DeviceSpecificData.Reserved2 = 0;
|
|
Descriptor->PartialDescriptors[0].u.DeviceSpecificData.DataSize =
|
|
DataSize - sizeof(CM_PARTIAL_RESOURCE_LIST);
|
|
|
|
//
|
|
// Visit each child of the RootEntry and copy its ConfigurationData
|
|
// to the new configuration data area.
|
|
// N.B. The configuration data includes a slot information and zero
|
|
// or more function information. The slot information provided
|
|
// by ARC eisa data does not have "ReturnedCode" as defined in
|
|
// our CM_EISA_SLOT_INFORMATION. This code will convert the
|
|
// standard EISA slot information to our CM format.
|
|
//
|
|
|
|
CurrentEntry = RootEntry->Child;
|
|
DataPointer += sizeof(CM_PARTIAL_RESOURCE_LIST);
|
|
NextSlot = 0;
|
|
|
|
while (CurrentEntry) {
|
|
Component = &CurrentEntry->ComponentEntry;
|
|
if (CurrentEntry->ConfigurationData) {
|
|
|
|
//
|
|
// Check if there is any empty slot. If yes, create empty
|
|
// slot information. Also make sure the config data area is
|
|
// big enough.
|
|
//
|
|
|
|
if (Component->Key > NextSlot) {
|
|
for (i = NextSlot; i < CurrentEntry->ComponentEntry.Key; i++ ) {
|
|
*(PCM_EISA_SLOT_INFORMATION)DataPointer = EmptySlot;
|
|
DataPointer += sizeof(CM_EISA_SLOT_INFORMATION);
|
|
}
|
|
}
|
|
|
|
*DataPointer++ = 0; // See comment above
|
|
RtlMoveMemory( // Skip config data header
|
|
DataPointer,
|
|
(PUCHAR)CurrentEntry->ConfigurationData +
|
|
CONFIGURATION_DATA_HEADER_SIZE,
|
|
Component->ConfigurationDataLength -
|
|
CONFIGURATION_DATA_HEADER_SIZE
|
|
);
|
|
DataPointer += Component->ConfigurationDataLength -
|
|
CONFIGURATION_DATA_HEADER_SIZE;
|
|
NextSlot = Component->Key + 1;
|
|
}
|
|
CurrentEntry = CurrentEntry->Sibling;
|
|
}
|
|
}
|
|
return(ESUCCESS);
|
|
}
|
|
|