490 lines
11 KiB
C
490 lines
11 KiB
C
|
/*++
|
|||
|
|
|||
|
Copyright (c) 1990 Microsoft Corporation
|
|||
|
Copyright (c) 1992, 1993, 1994 Digital Equipment Corporation
|
|||
|
|
|||
|
Module Name:
|
|||
|
|
|||
|
pciesc.c
|
|||
|
|
|||
|
Abstract:
|
|||
|
|
|||
|
The module provides the interrupt support for the PCI ESC's
|
|||
|
cascaded 82c59 programmable interrupt controllers.
|
|||
|
|
|||
|
Author:
|
|||
|
|
|||
|
Eric Rehm (DEC) 4-Feburary-1994
|
|||
|
|
|||
|
Revision History:
|
|||
|
|
|||
|
James Livingston 29-Apr-1994
|
|||
|
Adapted from pcisio.c module for Intel 82374EB (ESC).
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
#include "halp.h"
|
|||
|
#include "eisa.h"
|
|||
|
|
|||
|
//
|
|||
|
// Define the context structure for use by interrupt service routines.
|
|||
|
//
|
|||
|
|
|||
|
typedef BOOLEAN (*PSECOND_LEVEL_DISPATCH)(
|
|||
|
PKINTERRUPT InterruptObject
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// Import save area for ESC interrupt mask registers.
|
|||
|
//
|
|||
|
|
|||
|
UCHAR HalpEisaInterrupt1Mask;
|
|||
|
UCHAR HalpEisaInterrupt2Mask;
|
|||
|
UCHAR HalpEisaInterrupt1Level;
|
|||
|
UCHAR HalpEisaInterrupt2Level;
|
|||
|
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
HalpInitializeEisaInterrupts (
|
|||
|
VOID
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine initializes the standard dual 82c59 programmable interrupt
|
|||
|
controller.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
{
|
|||
|
UCHAR DataByte;
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the ESC interrupt controller. There are two cascaded
|
|||
|
// interrupt controllers, each of which must be initialized with 4
|
|||
|
// control words.
|
|||
|
//
|
|||
|
|
|||
|
DataByte = 0;
|
|||
|
((PINITIALIZATION_COMMAND_1) &DataByte)->Icw4Needed = 1;
|
|||
|
((PINITIALIZATION_COMMAND_1) &DataByte)->InitializationFlag = 1;
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort0,
|
|||
|
DataByte
|
|||
|
);
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort0,
|
|||
|
DataByte
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// The second intitialization control word sets the interrupt vector to
|
|||
|
// 0-15.
|
|||
|
//
|
|||
|
|
|||
|
DataByte = 0;
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort1,
|
|||
|
DataByte
|
|||
|
);
|
|||
|
|
|||
|
DataByte = 0x08;
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort1,
|
|||
|
DataByte
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// The third initialization control word sets the controls for slave mode.
|
|||
|
// The master ICW3 uses bit position and the slave ICW3 uses a numeric.
|
|||
|
//
|
|||
|
|
|||
|
DataByte = 1 << SLAVE_IRQL_LEVEL;
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort1,
|
|||
|
DataByte
|
|||
|
);
|
|||
|
|
|||
|
DataByte = SLAVE_IRQL_LEVEL;
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort1,
|
|||
|
DataByte
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// The fourth initialization control word is used to specify normal
|
|||
|
// end-of-interrupt mode and not special-fully-nested mode.
|
|||
|
//
|
|||
|
|
|||
|
DataByte = 0;
|
|||
|
((PINITIALIZATION_COMMAND_4) &DataByte)->I80x86Mode = 1;
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort1,
|
|||
|
DataByte
|
|||
|
);
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort1,
|
|||
|
DataByte
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Disable all of the interrupts except the slave.
|
|||
|
//
|
|||
|
|
|||
|
HalpEisaInterrupt1Mask = (UCHAR)(~(1 << SLAVE_IRQL_LEVEL));
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort1,
|
|||
|
HalpEisaInterrupt1Mask
|
|||
|
);
|
|||
|
|
|||
|
HalpEisaInterrupt2Mask = 0xFF;
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort1,
|
|||
|
HalpEisaInterrupt2Mask
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the edge/level register masks to 0, which is the default
|
|||
|
// edge-sensitive value.
|
|||
|
//
|
|||
|
|
|||
|
HalpEisaInterrupt1Level = 0;
|
|||
|
HalpEisaInterrupt2Level = 0;
|
|||
|
|
|||
|
return (TRUE);
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
HalpDisableEisaInterrupt(
|
|||
|
IN ULONG Vector
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function Disables the EISA interrupt specified by Vector.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Vector - Supplies the vector of the ESIA interrupt that is Disabled.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
//
|
|||
|
// Calculate the EISA interrupt vector.
|
|||
|
//
|
|||
|
|
|||
|
Vector -= EISA_VECTORS;
|
|||
|
|
|||
|
//
|
|||
|
// Determine if this vector is for interrupt controller 1 or 2.
|
|||
|
//
|
|||
|
|
|||
|
if (Vector & 0x08) {
|
|||
|
|
|||
|
//
|
|||
|
// The interrupt is for controller 2.
|
|||
|
//
|
|||
|
|
|||
|
Vector &= 0x7;
|
|||
|
|
|||
|
HalpEisaInterrupt2Mask |= (UCHAR) 1 << Vector;
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort1,
|
|||
|
HalpEisaInterrupt2Mask
|
|||
|
);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// The interrupt is in controller 1.
|
|||
|
//
|
|||
|
|
|||
|
Vector &= 0x7;
|
|||
|
|
|||
|
//
|
|||
|
// never disable IRQL2; it is the slave interrupt
|
|||
|
//
|
|||
|
|
|||
|
if (Vector != SLAVE_IRQL_LEVEL) {
|
|||
|
HalpEisaInterrupt1Mask |= (ULONG) 1 << Vector;
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort1,
|
|||
|
HalpEisaInterrupt1Mask
|
|||
|
);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
HalpEnableEisaInterrupt(
|
|||
|
IN ULONG Vector,
|
|||
|
IN KINTERRUPT_MODE InterruptMode
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function enables the EISA interrupt specified by Vector.
|
|||
|
Arguments:
|
|||
|
|
|||
|
Vector - Supplies the vector of the EISA interrupt that is enabled.
|
|||
|
|
|||
|
InterruptMode - Supplies the mode of the interrupt; LevelSensitive or
|
|||
|
Latched.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
//
|
|||
|
// Calculate the EISA interrupt vector.
|
|||
|
//
|
|||
|
|
|||
|
Vector -= EISA_VECTORS;
|
|||
|
|
|||
|
//
|
|||
|
// Determine if this vector is for interrupt controller 1 or 2.
|
|||
|
//
|
|||
|
|
|||
|
if (Vector & 0x08) {
|
|||
|
|
|||
|
//
|
|||
|
// The interrupt is in controller 2.
|
|||
|
//
|
|||
|
|
|||
|
Vector &= 0x7;
|
|||
|
|
|||
|
HalpEisaInterrupt2Mask &= (UCHAR) ~(1 << Vector);
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort1,
|
|||
|
HalpEisaInterrupt2Mask
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// Set the level/edge control register.
|
|||
|
//
|
|||
|
|
|||
|
if (InterruptMode == LevelSensitive) {
|
|||
|
|
|||
|
HalpEisaInterrupt2Level |= (UCHAR) (1 << Vector);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
HalpEisaInterrupt2Level &= (UCHAR) ~(1 << Vector);
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2EdgeLevel,
|
|||
|
HalpEisaInterrupt2Level
|
|||
|
);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// The interrupt is in controller 1.
|
|||
|
//
|
|||
|
|
|||
|
Vector &= 0x7;
|
|||
|
|
|||
|
HalpEisaInterrupt1Mask &= (UCHAR) ~(1 << Vector);
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort1,
|
|||
|
HalpEisaInterrupt1Mask
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// Set the level/edge control register.
|
|||
|
//
|
|||
|
|
|||
|
if (InterruptMode == LevelSensitive) {
|
|||
|
|
|||
|
HalpEisaInterrupt1Level |= (UCHAR) (1 << Vector);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
HalpEisaInterrupt1Level &= (UCHAR) ~(1 << Vector);
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1EdgeLevel,
|
|||
|
HalpEisaInterrupt1Level
|
|||
|
);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
HalpEisaDispatch(
|
|||
|
IN PKINTERRUPT Interrupt,
|
|||
|
IN PVOID ServiceContext,
|
|||
|
IN PKTRAP_FRAME TrapFrame
|
|||
|
)
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine is entered as the result of an interrupt being generated
|
|||
|
via the vector that is connected to an interrupt object that describes
|
|||
|
the EISA device interrupts. Its function is to call the second-level
|
|||
|
interrupt dispatch routine and acknowledge the interrupt at the ESC
|
|||
|
controller.
|
|||
|
|
|||
|
This service routine could be connected as follows:
|
|||
|
|
|||
|
KeInitializeInterrupt(&Interrupt, HalpDispatch,
|
|||
|
EISA_VIRTUAL_BASE,
|
|||
|
(PKSPIN_LOCK)NULL, ISA_LEVEL, ISA_LEVEL, ISA_LEVEL,
|
|||
|
LevelSensitive, TRUE, 0, FALSE);
|
|||
|
KeConnectInterrupt(&Interrupt);
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Interrupt - Supplies a pointer to the interrupt object.
|
|||
|
|
|||
|
ServiceContext - Supplies a pointer to the EISA interrupt acknowledge
|
|||
|
register.
|
|||
|
|
|||
|
TrapFrame - Supplies a pointer to the trap frame for this interrupt.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
Returns the value returned from the second level routine.
|
|||
|
|
|||
|
--*/
|
|||
|
{
|
|||
|
UCHAR EISAVector;
|
|||
|
PKPRCB Prcb;
|
|||
|
BOOLEAN returnValue;
|
|||
|
USHORT PCRInOffset;
|
|||
|
UCHAR Int1Isr;
|
|||
|
UCHAR Int2Isr;
|
|||
|
PULONG DispatchCode;
|
|||
|
PKINTERRUPT InterruptObject;
|
|||
|
|
|||
|
//
|
|||
|
// Acknowledge the Interrupt controller and receive the returned
|
|||
|
// interrupt vector.
|
|||
|
//
|
|||
|
|
|||
|
EISAVector = HalpAcknowledgeEisaInterrupt(ServiceContext);
|
|||
|
|
|||
|
|
|||
|
if ((EISAVector & 0x07) == 0x07) {
|
|||
|
|
|||
|
//
|
|||
|
// Check for a passive release by looking at the inservice register.
|
|||
|
// If there is a real IRQL7 interrupt, just go along normally. If there
|
|||
|
// is not, then it is a passive release. So just dismiss it.
|
|||
|
//
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort0,
|
|||
|
0x0B
|
|||
|
);
|
|||
|
|
|||
|
Int1Isr = READ_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort0);
|
|||
|
|
|||
|
//
|
|||
|
// do second controller
|
|||
|
//
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort0,
|
|||
|
0x0B
|
|||
|
);
|
|||
|
|
|||
|
Int2Isr = READ_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort0);
|
|||
|
|
|||
|
|
|||
|
if (!(Int2Isr & 0x80) && !(Int1Isr & 0x80)) {
|
|||
|
|
|||
|
//
|
|||
|
// Clear the master controller to clear situation
|
|||
|
//
|
|||
|
|
|||
|
if (!(Int2Isr & 0x80)) {
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort0,
|
|||
|
NONSPECIFIC_END_OF_INTERRUPT
|
|||
|
);
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
return FALSE; // ecrfix - now returns a value
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Dispatch to the secondary interrupt service routine.
|
|||
|
//
|
|||
|
|
|||
|
PCRInOffset = EISAVector + EISA_VECTORS;
|
|||
|
DispatchCode = (PULONG)PCR->InterruptRoutine[PCRInOffset];
|
|||
|
InterruptObject = CONTAINING_RECORD(DispatchCode,
|
|||
|
KINTERRUPT,
|
|||
|
DispatchCode);
|
|||
|
|
|||
|
returnValue = ((PSECOND_LEVEL_DISPATCH)
|
|||
|
InterruptObject->DispatchAddress)(InterruptObject);
|
|||
|
|
|||
|
//
|
|||
|
// Dismiss the interrupt in the ESC interrupt controllers.
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// If this is a cascaded interrupt then the interrupt must be dismissed in
|
|||
|
// both controlles.
|
|||
|
//
|
|||
|
|
|||
|
if (EISAVector & 0x08) {
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt2ControlPort0,
|
|||
|
NONSPECIFIC_END_OF_INTERRUPT
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
WRITE_PORT_UCHAR(
|
|||
|
&((PEISA_CONTROL) HalpEisaControlBase)->Interrupt1ControlPort0,
|
|||
|
NONSPECIFIC_END_OF_INTERRUPT
|
|||
|
);
|
|||
|
|
|||
|
return(returnValue);
|
|||
|
}
|