536 lines
11 KiB
C
536 lines
11 KiB
C
/*++
|
||
|
||
Copyright (c) 1989 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
ixbusdat.c
|
||
|
||
Abstract:
|
||
|
||
This module contains the IoXxx routines for the NT I/O system that
|
||
are hardware dependent. Were these routines not hardware dependent,
|
||
they would reside in the iosubs.c module.
|
||
|
||
Author:
|
||
|
||
Environment:
|
||
|
||
Kernel mode
|
||
|
||
Revision History:
|
||
|
||
|
||
--*/
|
||
|
||
#include "bootx86.h"
|
||
#include "arc.h"
|
||
#include "ixfwhal.h"
|
||
#include "eisa.h"
|
||
#include "mca.h"
|
||
#include "ntconfig.h"
|
||
|
||
ULONG
|
||
HalpGetCmosData(
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG Length
|
||
);
|
||
|
||
ULONG
|
||
HalpGetPosData(
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG Offset,
|
||
IN ULONG Length
|
||
);
|
||
|
||
ULONG
|
||
HalpGetEisaData(
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG Offset,
|
||
IN ULONG Length
|
||
);
|
||
|
||
ULONG
|
||
HalpGetPCIData(
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG Offset,
|
||
IN ULONG Length
|
||
);
|
||
|
||
ULONG
|
||
HalpSetPCIData(
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG Offset,
|
||
IN ULONG Length
|
||
);
|
||
|
||
NTSTATUS
|
||
HalpAssignPCISlotResources (
|
||
IN ULONG BusNumber,
|
||
IN ULONG Slot,
|
||
IN OUT PCM_RESOURCE_LIST *AllocatedResources
|
||
);
|
||
|
||
/*
|
||
*
|
||
* Router functions. Routes each call to specific handler
|
||
*
|
||
*/
|
||
|
||
|
||
ULONG
|
||
HalGetBusData(
|
||
IN BUS_DATA_TYPE BusDataType,
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG Length
|
||
)
|
||
{
|
||
return HalGetBusDataByOffset (BusDataType,BusNumber,SlotNumber,Buffer,0,Length);
|
||
}
|
||
|
||
ULONG
|
||
HalGetBusDataByOffset (
|
||
IN BUS_DATA_TYPE BusDataType,
|
||
IN ULONG BusNumber,
|
||
IN ULONG Slot,
|
||
IN PVOID Buffer,
|
||
IN ULONG Offset,
|
||
IN ULONG Length
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Dispatcher for GetBusData
|
||
|
||
--*/
|
||
{
|
||
switch (BusDataType) {
|
||
case Cmos:
|
||
if (Offset != 0) {
|
||
return 0;
|
||
}
|
||
|
||
return HalpGetCmosData(BusNumber, Slot, Buffer, Length);
|
||
|
||
case Pos:
|
||
return HalpGetPosData(BusNumber, Slot, Buffer, Offset, Length);
|
||
|
||
case EisaConfiguration:
|
||
return HalpGetEisaData(BusNumber, Slot, Buffer, Offset, Length);
|
||
|
||
case PCIConfiguration:
|
||
return HalpGetPCIData(BusNumber, Slot, Buffer, Offset, Length);
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
ULONG
|
||
HalSetBusData(
|
||
IN BUS_DATA_TYPE BusDataType,
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG Length
|
||
)
|
||
{
|
||
return HalSetBusDataByOffset (BusDataType,BusNumber,SlotNumber,Buffer,0,Length);
|
||
}
|
||
|
||
ULONG
|
||
HalSetBusDataByOffset(
|
||
IN BUS_DATA_TYPE BusDataType,
|
||
IN ULONG BusNumber,
|
||
IN ULONG Slot,
|
||
IN PVOID Buffer,
|
||
IN ULONG Offset,
|
||
IN ULONG Length
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Dispatcher for SetBusData
|
||
|
||
--*/
|
||
{
|
||
switch (BusDataType) {
|
||
case PCIConfiguration:
|
||
return HalpSetPCIData(BusNumber, Slot, Buffer, Offset, Length);
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
|
||
NTSTATUS
|
||
HalAssignSlotResources (
|
||
IN PUNICODE_STRING RegistryPath,
|
||
IN PUNICODE_STRING DriverClassName OPTIONAL,
|
||
IN PDRIVER_OBJECT DriverObject,
|
||
IN PDEVICE_OBJECT DeviceObject OPTIONAL,
|
||
IN INTERFACE_TYPE BusType,
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN OUT PCM_RESOURCE_LIST *AllocatedResources
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Dispatcher for AssignSlotResources
|
||
|
||
--*/
|
||
{
|
||
switch (BusType) {
|
||
case PCIBus:
|
||
return HalpAssignPCISlotResources (
|
||
BusNumber,
|
||
SlotNumber,
|
||
AllocatedResources
|
||
);
|
||
default:
|
||
break;
|
||
}
|
||
return STATUS_NOT_FOUND;
|
||
}
|
||
|
||
|
||
|
||
|
||
|
||
/**
|
||
**
|
||
** Standard PC bus functions
|
||
**
|
||
**/
|
||
|
||
|
||
|
||
BOOLEAN
|
||
HalTranslateBusAddress(
|
||
IN INTERFACE_TYPE InterfaceType,
|
||
IN ULONG BusNumber,
|
||
IN PHYSICAL_ADDRESS BusAddress,
|
||
IN OUT PULONG AddressSpace,
|
||
OUT PPHYSICAL_ADDRESS TranslatedAddress
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This function translates a bus-relative address space and address into
|
||
a system physical address.
|
||
|
||
Arguments:
|
||
|
||
BusNumber - Supplies the bus number. This is ignored on
|
||
standard x86 systems
|
||
|
||
BusAddress - Supplies the bus-relative address
|
||
|
||
AddressSpace - Supplies the address space number.
|
||
Returns the host address space number.
|
||
|
||
AddressSpace == 0 => I/O space
|
||
AddressSpace == 1 => memory space
|
||
|
||
TranslatedAddress - Pointer to a physical_address.
|
||
|
||
Return Value:
|
||
|
||
System physical address corresponding to the supplied bus relative
|
||
address and bus address number.
|
||
|
||
--*/
|
||
|
||
{
|
||
TranslatedAddress->HighPart = 0;
|
||
TranslatedAddress->LowPart = BusAddress.LowPart;
|
||
return(TRUE);
|
||
}
|
||
|
||
ULONG
|
||
HalpGetPosData (
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG DOffset,
|
||
IN ULONG Length
|
||
)
|
||
/*--
|
||
|
||
Arguments:
|
||
|
||
BusDataType - Supplies the type of bus.
|
||
|
||
BusNumber - Indicates which bus.
|
||
|
||
Buffer - Supplies the space to store the data.
|
||
|
||
Length - Supplies a count in bytes of the maximum amount to return.
|
||
|
||
Return Value:
|
||
|
||
Returns the amount of data stored into the buffer.
|
||
|
||
--*/
|
||
{
|
||
PVOID McaRegisterBase = 0;
|
||
ULONG Index = 0;
|
||
PUCHAR DataBuffer = Buffer;
|
||
ULONG DataLength = 0;
|
||
PUCHAR PosBase;
|
||
ULONG Offset;
|
||
|
||
if (DOffset != 0 || MachineType != MACHINE_TYPE_MCA) {
|
||
return 0;
|
||
}
|
||
|
||
PosBase = (PUCHAR) &((PMCA_CONTROL) McaRegisterBase)->Pos;
|
||
|
||
//
|
||
// Place the specified adapter into setup mode.
|
||
//
|
||
|
||
WRITE_PORT_UCHAR((PVOID) &((PMCA_CONTROL) McaRegisterBase)->AdapterSetup,
|
||
(UCHAR) ( MCA_ADAPTER_SETUP_ON | SlotNumber ));
|
||
|
||
while (DataLength < Length && DataLength < 6) {
|
||
DataBuffer[DataLength] = READ_PORT_UCHAR( PosBase + DataLength );
|
||
DataLength++;
|
||
}
|
||
|
||
while (DataLength < Length) {
|
||
|
||
WRITE_PORT_UCHAR((PVOID) &((PPROGRAMMABLE_OPTION_SELECT)
|
||
PosBase)->SubaddressExtensionLsb, (UCHAR) Index);
|
||
|
||
WRITE_PORT_UCHAR((PVOID) &((PPROGRAMMABLE_OPTION_SELECT)
|
||
PosBase)->SubaddressExtensionMsb, (UCHAR) (Index >> 8));
|
||
|
||
DataBuffer[Index + 6] = READ_PORT_UCHAR(
|
||
(PVOID) &((PPROGRAMMABLE_OPTION_SELECT)PosBase)->OptionSelectData2);
|
||
|
||
DataLength++;
|
||
|
||
if (DataLength < Length) {
|
||
Offset = DataLength + ((Length - DataLength) / 2);
|
||
DataBuffer[Offset] = READ_PORT_UCHAR(
|
||
(PVOID) &((PPROGRAMMABLE_OPTION_SELECT)PosBase)->OptionSelectData3);
|
||
DataLength++;
|
||
Index++;
|
||
}
|
||
}
|
||
|
||
//
|
||
// Disable adapter setup.
|
||
//
|
||
|
||
WRITE_PORT_UCHAR((PVOID) &((PMCA_CONTROL) McaRegisterBase)->AdapterSetup,
|
||
(UCHAR) ( MCA_ADAPTER_SETUP_OFF ));
|
||
}
|
||
|
||
ULONG
|
||
HalpGetEisaData (
|
||
IN ULONG BusNumber,
|
||
IN ULONG SlotNumber,
|
||
IN PVOID Buffer,
|
||
IN ULONG Offset,
|
||
IN ULONG Length
|
||
)
|
||
/*--
|
||
|
||
Arguments:
|
||
|
||
BusDataType - Supplies the type of bus.
|
||
|
||
BusNumber - Indicates which bus.
|
||
|
||
Buffer - Supplies the space to store the data.
|
||
|
||
Length - Supplies a count in bytes of the maximum amount to return.
|
||
|
||
Return Value:
|
||
|
||
Returns the amount of data stored into the buffer.
|
||
|
||
--*/
|
||
{
|
||
|
||
ULONG DataLength = 0;
|
||
ULONG i;
|
||
ULONG TotalDataSize;
|
||
ULONG SlotDataSize;
|
||
ULONG PartialCount;
|
||
ULONG Index = 0;
|
||
PUCHAR DataBuffer = Buffer;
|
||
PCONFIGURATION_COMPONENT_DATA ConfigData;
|
||
PCM_EISA_SLOT_INFORMATION SlotInformation;
|
||
PCM_PARTIAL_RESOURCE_LIST Descriptor;
|
||
PCM_PARTIAL_RESOURCE_DESCRIPTOR PartialResource;
|
||
BOOLEAN Found = FALSE;
|
||
|
||
if (MachineType != MACHINE_TYPE_EISA) {
|
||
return 0;
|
||
}
|
||
|
||
ConfigData = KeFindConfigurationEntry(
|
||
FwConfigurationTree,
|
||
AdapterClass,
|
||
EisaAdapter,
|
||
NULL
|
||
);
|
||
|
||
if (ConfigData == NULL) {
|
||
DbgPrint("HalGetBusData: KeFindConfigurationEntry failed\n");
|
||
return(0);
|
||
}
|
||
|
||
Descriptor = ConfigData->ConfigurationData;
|
||
PartialResource = Descriptor->PartialDescriptors;
|
||
PartialCount = Descriptor->Count;
|
||
|
||
for (i = 0; i < PartialCount; i++) {
|
||
|
||
//
|
||
// Do each partial Resource
|
||
//
|
||
|
||
switch (PartialResource->Type) {
|
||
case CmResourceTypeNull:
|
||
case CmResourceTypePort:
|
||
case CmResourceTypeInterrupt:
|
||
case CmResourceTypeMemory:
|
||
case CmResourceTypeDma:
|
||
|
||
//
|
||
// We dont care about these.
|
||
//
|
||
|
||
PartialResource++;
|
||
|
||
break;
|
||
|
||
case CmResourceTypeDeviceSpecific:
|
||
|
||
//
|
||
// Bingo!
|
||
//
|
||
|
||
TotalDataSize = PartialResource->u.DeviceSpecificData.DataSize;
|
||
|
||
SlotInformation = (PCM_EISA_SLOT_INFORMATION)
|
||
((PUCHAR)PartialResource +
|
||
sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR));
|
||
|
||
while (((LONG)TotalDataSize) > 0) {
|
||
|
||
if (SlotInformation->ReturnCode == EISA_EMPTY_SLOT) {
|
||
|
||
SlotDataSize = sizeof(CM_EISA_SLOT_INFORMATION);
|
||
|
||
} else {
|
||
|
||
SlotDataSize = sizeof(CM_EISA_SLOT_INFORMATION) +
|
||
SlotInformation->NumberFunctions *
|
||
sizeof(CM_EISA_FUNCTION_INFORMATION);
|
||
}
|
||
|
||
if (SlotDataSize > TotalDataSize) {
|
||
|
||
//
|
||
// Something is wrong again
|
||
//
|
||
|
||
DbgPrint("HalGetBusData: SlotDataSize > TotalDataSize\n");
|
||
|
||
return(0);
|
||
|
||
}
|
||
|
||
if (SlotNumber != 0) {
|
||
|
||
SlotNumber--;
|
||
|
||
SlotInformation = (PCM_EISA_SLOT_INFORMATION)
|
||
((PUCHAR)SlotInformation + SlotDataSize);
|
||
|
||
TotalDataSize -= SlotDataSize;
|
||
|
||
continue;
|
||
|
||
}
|
||
|
||
//
|
||
// This is our slot
|
||
//
|
||
|
||
Found = TRUE;
|
||
break;
|
||
|
||
}
|
||
|
||
//
|
||
// End loop
|
||
//
|
||
|
||
i = PartialCount;
|
||
|
||
break;
|
||
|
||
default:
|
||
|
||
#if DBG
|
||
DbgPrint("Bad Data in registry!\n");
|
||
#endif
|
||
|
||
return(0);
|
||
|
||
}
|
||
|
||
}
|
||
|
||
if (Found) {
|
||
|
||
//
|
||
// As a hack if the length is zero then the buffer points to a
|
||
// PVOID where the pointer to the data should be stored. This is
|
||
// done in the loader because we quickly run out of heap scaning
|
||
// all of the EISA configuration data.
|
||
//
|
||
|
||
if (Length == 0) {
|
||
|
||
//
|
||
// Return the pointer to the mini-port driver.
|
||
//
|
||
|
||
*((PVOID *)Buffer) = SlotInformation;
|
||
return(SlotDataSize);
|
||
}
|
||
|
||
i = Length + Offset;
|
||
if (i > SlotDataSize) {
|
||
i = SlotDataSize;
|
||
}
|
||
|
||
DataLength = i - Offset;
|
||
RtlMoveMemory(Buffer, ((PUCHAR) SlotInformation + Offset), DataLength);
|
||
}
|
||
|
||
return(DataLength);
|
||
}
|