NT4/private/ntos/nthals/halalcor/alpha/alinitnt.c
2020-09-30 17:12:29 +02:00

796 lines
15 KiB
C
Raw Blame History

This file contains invisible Unicode characters

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

/*++
Copyright (c) 1993 Digital Equipment Corporation
Module Name:
alinitnt.c
Abstract:
This module implements the platform-specific initialization for
an Alcor system.
Author:
Joe Notarangelo 19-Jul-1994
Environment:
Kernel mode only.
Revision History:
--*/
#include "halp.h"
#include "pcrtc.h"
#include "alcor.h"
#include "iousage.h"
#include "stdio.h"
#include "fwcallbk.h"
#include <ntverp.h> // to get the product build number.
//
// Define extern global buffer for the Uncorrectable Error Frame.
// declared in halalpha\inithal.c
//
extern PERROR_FRAME PUncorrectableError;
//
// Define the Product Naming data.
//
PCHAR HalpProductName;
PCHAR HalpFamilyName = "AlphaStation";
ULONG HalpProcessorNumber = 5;
#define MAX_INIT_MSG (80)
//
// Define the bus type, this value allows us to distinguish between
// EISA and ISA systems. We're only interested in distinguishing
// between just those two buses.
//
ULONG HalpBusType = MACHINE_TYPE_EISA;
//
// Define global data used to communicate new clock rates to the clock
// interrupt service routine.
//
ULONG HalpCurrentTimeIncrement;
ULONG HalpNextRateSelect;
ULONG HalpNextTimeIncrement;
ULONG HalpNewTimeIncrement;
//
// Define external references.
//
extern ULONG HalDisablePCIParityChecking;
//
// Function prototypes.
//
BOOLEAN
HalpInitializeAlcorInterrupts (
VOID
);
VOID
HalpParseLoaderBlock(
PLOADER_PARAMETER_BLOCK LoaderBlock
);
VOID
HalpAlcorErrorInterrupt(
VOID
);
VOID
HalpRegisterPlatformResources(
PUCHAR HalName
);
//
// Irql mask and tables
//
// irql 0 - passive
// irql 1 - sfw apc level
// irql 2 - sfw dispatch level
// irql 3 - device low
// irql 4 - device high
// irql 5 - interval clock
// irql 6 - not used
// irql 7 - error, mchk, nmi, performance counters
//
//
//
// The hardware interrupt pins are used as follows for Alcor
//
// IRQ0 = CIA_INT
// IRQ1 = SYS_INT (PCI and ESC interrupts)
// IRQ2 = Interval Clock
// IRQ3 = Reserved
//
BOOLEAN
HalpInitializeInterrupts (
VOID
)
/*++
Routine Description:
This function initializes interrupts for an Alcor system.
Arguments:
None.
Return Value:
A value of TRUE is returned if the initialization is successfully
completed. Otherwise a value of FALSE is returned.
--*/
{
extern ULONG Halp21164CorrectedErrorInterrupt();
extern ULONG HalpCiaErrorInterrupt();
extern ULONG HalpDeviceInterrupt();
extern ULONG HalpHaltInterrupt();
ULONG Vector;
//
// Initialize HAL processor parameters based on estimated CPU speed.
// This must be done before HalpStallExecution is called. Compute integral
// megahertz first to avoid rounding errors due to imprecise cycle clock
// period values.
//
HalpInitializeProcessorParameters();
//
// Start the periodic interrupt from the RTC
//
HalpProgramIntervalTimer(MAXIMUM_RATE_SELECT);
//
// Initialize Alcor interrupts.
//
HalpInitializeAlcorInterrupts();
//
// Initialize the EV5 (21164) interrupts.
//
HalpInitialize21164Interrupts();
PCR->InterruptRoutine[EV5_IRQ0_VECTOR] =
(PKINTERRUPT_ROUTINE)HalpCiaErrorInterrupt;
PCR->InterruptRoutine[EV5_IRQ1_VECTOR] =
(PKINTERRUPT_ROUTINE)HalpDeviceInterrupt;
PCR->InterruptRoutine[EV5_IRQ2_VECTOR] =
(PKINTERRUPT_ROUTINE)HalpClockInterrupt;
PCR->InterruptRoutine[EV5_HALT_VECTOR] =
(PKINTERRUPT_ROUTINE)HalpHaltInterrupt;
PCR->InterruptRoutine[EV5_MCHK_VECTOR] =
(PKINTERRUPT_ROUTINE)HalpAlcorErrorInterrupt;
PCR->InterruptRoutine[EV5_CRD_VECTOR] =
(PKINTERRUPT_ROUTINE)Halp21164CorrectedErrorInterrupt;
HalpStart21164Interrupts();
return TRUE;
}
VOID
HalpSetTimeIncrement(
VOID
)
/*++
Routine Description:
This routine is responsible for setting the time increment for an EV5
based machine via a call into the kernel.
Arguments:
None.
Return Value:
None.
--*/
{
//
// Set the time increment value.
//
HalpCurrentTimeIncrement = MAXIMUM_INCREMENT;
HalpNextTimeIncrement = MAXIMUM_INCREMENT;
HalpNextRateSelect = 0;
KeSetTimeIncrement( MAXIMUM_INCREMENT, MINIMUM_INCREMENT );
}
VOID
HalpInitializeClockInterrupts(
VOID
)
/*++
Routine Description:
This function is a NOOP for Alcor.
Arguments:
None.
Return Value:
None.
--*/
{
return;
}
VOID
HalpEstablishErrorHandler(
VOID
)
/*++
Routine Description:
This routine performs the initialization necessary for the HAL to
begin servicing machine checks.
Arguments:
None.
Return Value:
None.
--*/
{
BOOLEAN PciParityChecking;
BOOLEAN ReportCorrectables;
//
// Connect the machine check handler via the PCR.
//
PCR->MachineCheckError = HalMachineCheck;
HalpInitializeCiaMachineChecks( ReportCorrectables = FALSE,
PciParityChecking = FALSE );
return;
}
VOID
HalpInitializeMachineDependent(
IN ULONG Phase,
IN PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
This function performs any Platform-specific initialization based on
the current phase on initialization.
Arguments:
Phase - Supplies an indicator for phase of initialization, phase 0 or
phase 1.
LoaderBlock - supplies a pointer to the loader block.
Return Value:
None.
--*/
{
ULONG BusIrql;
ULONG BusNumber;
ULONG Model;
UCHAR MsgBuffer[MAX_INIT_MSG];
BOOLEAN ReportCorrectables;
BOOLEAN PciParityChecking;
if( Phase == 0 ){
//
// Phase 0 Initialization.
//
//
// Parse the Loader Parameter block looking for PCI entry to determine
// if PCI parity should be disabled
//
HalpParseLoaderBlock( LoaderBlock );
//
// Establish the error handler, to reflect the PCI parity checking.
//
if( HalDisablePCIParityChecking != 0 ){
PciParityChecking = FALSE;
} else {
PciParityChecking = TRUE;
}
HalpInitializeCiaMachineChecks( ReportCorrectables = TRUE,
PciParityChecking );
} else {
//
// Phase 1 Initialization.
//
//
// Initialize the existing bus handlers.
//
HalpRegisterInternalBusHandlers();
//
// Initialize PCI Bus.
//
HalpInitializePCIBus(LoaderBlock);
//
// Initialize profiler.
//
HalpInitializeProfiler();
//
// Determine model type.
//
Model= READ_GRU_REGISTER(&((PGRU_INTERRUPT_CSRS)GRU_CSRS_QVA)->IntReq);
Model= (Model >> 24) & 0xf;
if (Model >= 0x8) {
//
// Maverick.
//
HalpProductName = "500";
//
// Print a message with version number. Yes, it's the
// final official name now.
//
sprintf(MsgBuffer,
"Digital Equipment Corporation %s %s/%d",
HalpFamilyName,
HalpProductName,
HalpClockMegaHertz );
} else {
//
// Alcor.
//
HalpProductName = "600";
//
// Print a message with version number.
//
sprintf(MsgBuffer,
"Digital Equipment Corporation %s %s %d/%d",
HalpFamilyName,
HalpProductName,
HalpProcessorNumber,
HalpClockMegaHertz );
}
HalDisplayString( MsgBuffer );
HalDisplayString("\n");
//
// Build the string to register the HAL.
//
strcat(MsgBuffer, " PCI/EISA HAL");
HalpRegisterPlatformResources( MsgBuffer );
}
return;
}
VOID
HalpRegisterPlatformResources(
PUCHAR HalName
)
/*++
Routine Description:
Register I/O resources used by the HAL.
Arguments:
HalName - Supplies a pointer to the name for the HAL.
Return Value:
None.
--*/
{
RESOURCE_USAGE Resource;
//
// Register the buses.
//
HalpRegisterBusUsage(Internal);
HalpRegisterBusUsage(Eisa);
HalpRegisterBusUsage(Isa);
HalpRegisterBusUsage(PCIBus);
//
// Register the name of the HAL.
//
HalpRegisterHalName( HalName );
//
// Register the interrupt vector used for the cascaded interrupt
// on the 8254s.
//
Resource.BusType = Isa;
Resource.BusNumber = 0;
Resource.ResourceType = CmResourceTypeInterrupt;
Resource.u.InterruptMode = Latched;
Resource.u.BusInterruptVector = 2;
Resource.u.SystemInterruptVector = 2;
Resource.u.SystemIrql = 2;
HalpRegisterResourceUsage(&Resource);
//
// Register machine specific io/memory addresses.
//
Resource.BusType = Isa;
Resource.BusNumber = 0;
Resource.ResourceType = CmResourceTypePort;
Resource.u.Start = I2C_INTERFACE_DATA_PORT;
Resource.u.Length = I2C_INTERFACE_LENGTH;
HalpRegisterResourceUsage(&Resource);
Resource.u.Start = SUPERIO_INDEX_PORT;
Resource.u.Length = SUPERIO_PORT_LENGTH;
HalpRegisterResourceUsage(&Resource);
//
// Register the DMA channel used for the cascade.
//
Resource.BusType = Isa;
Resource.BusNumber = 0;
Resource.ResourceType = CmResourceTypeDma;
Resource.u.DmaChannel = 0x4;
Resource.u.DmaPort = 0x0;
HalpRegisterResourceUsage(&Resource);
}
ULONG
HalSetTimeIncrement (
IN ULONG DesiredIncrement
)
/*++
Routine Description:
This function is called to set the clock interrupt rate to the frequency
required by the specified time increment value.
Arguments:
DesiredIncrement - Supplies desired number of 100ns units between clock
interrupts.
Return Value:
The actual time increment in 100ns units.
--*/
{
ULONG NewTimeIncrement;
ULONG NextRateSelect;
KIRQL OldIrql;
//
// Raise IRQL to the highest level, set the new clock interrupt
// parameters, lower IRQl, and return the new time increment value.
//
KeRaiseIrql(HIGH_LEVEL, &OldIrql);
if (DesiredIncrement < MINIMUM_INCREMENT) {
DesiredIncrement = MINIMUM_INCREMENT;
}
if (DesiredIncrement > MAXIMUM_INCREMENT) {
DesiredIncrement = MAXIMUM_INCREMENT;
}
//
// Find the allowed increment that is less than or equal to
// the desired increment.
//
if (DesiredIncrement >= RTC_PERIOD_IN_CLUNKS4) {
NewTimeIncrement = RTC_PERIOD_IN_CLUNKS4;
NextRateSelect = RTC_RATE_SELECT4;
} else if (DesiredIncrement >= RTC_PERIOD_IN_CLUNKS3) {
NewTimeIncrement = RTC_PERIOD_IN_CLUNKS3;
NextRateSelect = RTC_RATE_SELECT3;
} else if (DesiredIncrement >= RTC_PERIOD_IN_CLUNKS2) {
NewTimeIncrement = RTC_PERIOD_IN_CLUNKS2;
NextRateSelect = RTC_RATE_SELECT2;
} else {
NewTimeIncrement = RTC_PERIOD_IN_CLUNKS1;
NextRateSelect = RTC_RATE_SELECT1;
}
HalpNextRateSelect = NextRateSelect;
HalpNewTimeIncrement = NewTimeIncrement;
KeLowerIrql(OldIrql);
return NewTimeIncrement;
}
VOID
HalpGetMachineDependentErrorFrameSizes(
PULONG RawProcessorSize,
PULONG RawSystemInfoSize
)
/*++
Routine Description:
This function is called from HalpAllocateUncorrectableErrorFrame.
This function retuns the size of the system specific error frame
sizes.
Arguments:
RawProcessorSize - Processor-specific uncorrectable frame size.
RawSystemInfoSize - system-specific uncorrectable frame size.
Return Value:
None.
--*/
{
*RawProcessorSize = sizeof(PROCESSOR_EV5_UNCORRECTABLE);
*RawSystemInfoSize = sizeof(CIA_UNCORRECTABLE_FRAME);
return;
}
VOID
HalpGetSystemInfo(SYSTEM_INFORMATION *SystemInfo)
/*++
Routine Description:
Fills in the system information structure.
NOTE: Must later investigate the Fw call to get the firmware revision
ID. Must also figure out a way to get the OS version (preferebly the
build number).
Arguments:
SystemInfo - Pointer to the SYSTEM_INFORMATION structure.
Return Value:
None
--*/
{
char systemtype[] = "Alcor";
EXTENDED_SYSTEM_INFORMATION FwExtSysInfo;
VenReturnExtendedSystemInformation(&FwExtSysInfo);
RtlCopyMemory(SystemInfo->FirmwareRevisionId,
FwExtSysInfo.FirmwareVersion,
16);
RtlCopyMemory(SystemInfo->SystemType,systemtype, 8);
SystemInfo->ClockSpeed =
((1000 * 1000) + (PCR->CycleClockPeriod >> 1)) / PCR->CycleClockPeriod;
SystemInfo->SystemRevision = PCR->SystemRevision;
RtlCopyMemory(SystemInfo->SystemSerialNumber,
PCR->SystemSerialNumber,
16);
SystemInfo->SystemVariant = PCR->SystemVariant;
SystemInfo->PalMajorVersion = PCR->PalMajorVersion;
SystemInfo->PalMinorVersion = PCR->PalMinorVersion;
SystemInfo->OsRevisionId = VER_PRODUCTBUILD;
//
// For now fill in dummy values.
//
SystemInfo->ModuleVariant = 1UL;
SystemInfo->ModuleRevision = 1UL;
SystemInfo->ModuleSerialNumber = 0;
return;
}
VOID
HalpInitializeUncorrectableErrorFrame (
VOID
)
/*++
Routine Description:
This function Allocates an Uncorrectable Error frame for this
system and initializes the frame with certain constant/global
values.
This is routine called during machine dependent system
Initialization.
Arguments:
none
Return Value:
none
--*/
{
//
// If the Uncorrectable error buffer is not set then simply return
//
if(PUncorrectableError == NULL)
return;
PUncorrectableError->Signature = ERROR_FRAME_SIGNATURE;
PUncorrectableError->FrameType = UncorrectableFrame;
//
// ERROR_FRAME_VERSION is define in errframe.h and will
// change as and when there is a change in the errframe.h.
// This Version number helps the service, that reads this
// information from the dumpfile, to check if it knows about
// this frmae version type to decode. If it doesn't know, it
// will dump the entire frame to the EventLog with a message
// "Error Frame Version Mismatch".
//
PUncorrectableError->VersionNumber = ERROR_FRAME_VERSION;
//
// The sequence number will always be 1 for Uncorrectable errors.
//
PUncorrectableError->SequenceNumber = 1;
//
// The PerformanceCounterValue field is not used for Uncorrectable
// errors.
//
PUncorrectableError->PerformanceCounterValue = 0;
//
// We will fill in the UncorrectableFrame.SystemInfo here.
//
HalpGetSystemInfo(&PUncorrectableError->UncorrectableFrame.System);
PUncorrectableError->UncorrectableFrame.Flags.SystemInformationValid = 1;
return;
}
//
//jnfix
//
// This routine is bogus and does not apply to Alcor and the call should be
// ripped out of fwreturn (or at least changed to something that is more
// abstract).
//
VOID
HalpResetHAERegisters(
VOID
)
{
return;
}
//
//jnfix - this variable is needed because the clock interrupt handler
// - in intsup.s was made to be familiar with ev4prof.c, unfortunate
// - since we don't use ev4prof.c, so for now this is a hack, later
// - we will either fix intsup.s or create a new intsup.s that does
// - not have this hack
//
ULONG HalpNumberOfTicksReload;