mirror of
https://github.com/reactos/reactos.git
synced 2025-01-07 06:45:24 +00:00
189 lines
5.5 KiB
C
189 lines
5.5 KiB
C
/*
|
|
* PROJECT: ReactOS Kernel
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
* FILE: ntoskrnl/ke/amd64/interrupt.c
|
|
* PURPOSE: Manages the Kernel's IRQ support for external drivers,
|
|
* for the purpopses of connecting, disconnecting and setting
|
|
* up ISRs for drivers. The backend behind the Io* Interrupt
|
|
* routines.
|
|
* PROGRAMMERS: Timo Kreuzer (timo.kreuzer@reactos.org)
|
|
* Alex Ionescu (alex.ionescu@reactos.org)
|
|
*/
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
#include <ntoskrnl.h>
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
extern UCHAR KiInterruptDispatchTemplate[16];
|
|
extern UCHAR KiUnexpectedRange[];
|
|
extern UCHAR KiUnexpectedRangeEnd[];
|
|
void KiInterruptDispatch(void);
|
|
|
|
|
|
/* FUNCTIONS ****************************************************************/
|
|
|
|
VOID
|
|
NTAPI
|
|
KeInitializeInterrupt(
|
|
IN PKINTERRUPT Interrupt,
|
|
IN PKSERVICE_ROUTINE ServiceRoutine,
|
|
IN PVOID ServiceContext,
|
|
IN PKSPIN_LOCK SpinLock,
|
|
IN ULONG Vector,
|
|
IN KIRQL Irql,
|
|
IN KIRQL SynchronizeIrql,
|
|
IN KINTERRUPT_MODE InterruptMode,
|
|
IN BOOLEAN ShareVector,
|
|
IN CHAR ProcessorNumber,
|
|
IN BOOLEAN FloatingSave)
|
|
{
|
|
|
|
/* Initialize the header */
|
|
Interrupt->Type = InterruptObject;
|
|
Interrupt->Size = sizeof(KINTERRUPT);
|
|
|
|
/* If no Spinlock is given, use the internal */
|
|
if (!SpinLock) SpinLock = &Interrupt->SpinLock;
|
|
KeInitializeSpinLock(&Interrupt->SpinLock);
|
|
|
|
/* Set the given parameters */
|
|
Interrupt->ServiceRoutine = ServiceRoutine;
|
|
Interrupt->ServiceContext = ServiceContext;
|
|
Interrupt->ActualLock = SpinLock;
|
|
Interrupt->Vector = Vector;
|
|
Interrupt->Irql = Irql;
|
|
Interrupt->SynchronizeIrql = SynchronizeIrql;
|
|
Interrupt->Mode = InterruptMode;
|
|
Interrupt->ShareVector = ShareVector;
|
|
Interrupt->Number = ProcessorNumber;
|
|
Interrupt->FloatingSave = FloatingSave;
|
|
|
|
/* Set initial values */
|
|
Interrupt->TickCount = 0;
|
|
Interrupt->Connected = FALSE;
|
|
Interrupt->ServiceCount = 0;
|
|
Interrupt->DispatchCount = 0;
|
|
Interrupt->TrapFrame = NULL;
|
|
Interrupt->Reserved = 0;
|
|
|
|
/* Copy the dispatch code (its location independent, no need to patch it) */
|
|
RtlCopyMemory(Interrupt->DispatchCode,
|
|
KiInterruptDispatchTemplate,
|
|
sizeof(Interrupt->DispatchCode));
|
|
|
|
Interrupt->DispatchAddress = 0;
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
KeConnectInterrupt(IN PKINTERRUPT Interrupt)
|
|
{
|
|
PVOID CurrentHandler;
|
|
PKINTERRUPT ConnectedInterrupt;
|
|
|
|
ASSERT(Interrupt->Vector >= PRIMARY_VECTOR_BASE);
|
|
ASSERT(Interrupt->Vector <= MAXIMUM_IDTVECTOR);
|
|
ASSERT(Interrupt->Number < KeNumberProcessors);
|
|
ASSERT(Interrupt->Irql <= HIGH_LEVEL);
|
|
ASSERT(Interrupt->SynchronizeIrql >= Interrupt->Irql);
|
|
ASSERT(Interrupt->Irql == (Interrupt->Vector >> 4));
|
|
|
|
/* Check if its already connected */
|
|
if (Interrupt->Connected) return TRUE;
|
|
|
|
/* Query the current handler */
|
|
CurrentHandler = KeQueryInterruptHandler(Interrupt->Vector);
|
|
|
|
/* Check if the vector is unused */
|
|
if ((CurrentHandler >= (PVOID)KiUnexpectedRange) &&
|
|
(CurrentHandler <= (PVOID)KiUnexpectedRangeEnd))
|
|
{
|
|
/* Initialize the list for chained interrupts */
|
|
InitializeListHead(&Interrupt->InterruptListEntry);
|
|
|
|
/* Set normal dispatch address */
|
|
Interrupt->DispatchAddress = KiInterruptDispatch;
|
|
|
|
/* Set the new handler */
|
|
KeRegisterInterruptHandler(Interrupt->Vector,
|
|
Interrupt->DispatchCode);
|
|
|
|
/* Enable the interrupt */
|
|
if (!HalEnableSystemInterrupt(Interrupt->Vector,
|
|
Interrupt->Irql,
|
|
Interrupt->Mode))
|
|
{
|
|
/* Didn't work, restore old handler */
|
|
DPRINT1("HalEnableSystemInterrupt failed\n");
|
|
KeRegisterInterruptHandler(Interrupt->Vector, CurrentHandler);
|
|
return FALSE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Get the connected interrupt */
|
|
ConnectedInterrupt = CONTAINING_RECORD(CurrentHandler, KINTERRUPT, DispatchCode);
|
|
|
|
/* Check if sharing is ok */
|
|
if ((Interrupt->ShareVector == 0) ||
|
|
(ConnectedInterrupt->ShareVector == 0) ||
|
|
(Interrupt->Mode != ConnectedInterrupt->Mode))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* Insert the new interrupt into the connected interrupt's list */
|
|
InsertTailList(&ConnectedInterrupt->InterruptListEntry,
|
|
&Interrupt->InterruptListEntry);
|
|
}
|
|
|
|
/* Mark as connected */
|
|
Interrupt->Connected = TRUE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
KeDisconnectInterrupt(IN PKINTERRUPT Interrupt)
|
|
{
|
|
/* If the interrupt wasn't connected, there's nothing to do */
|
|
if (!Interrupt->Connected)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
UNIMPLEMENTED;
|
|
__debugbreak();
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
KeSynchronizeExecution(IN OUT PKINTERRUPT Interrupt,
|
|
IN PKSYNCHRONIZE_ROUTINE SynchronizeRoutine,
|
|
IN PVOID SynchronizeContext OPTIONAL)
|
|
{
|
|
BOOLEAN Success;
|
|
KIRQL OldIrql;
|
|
|
|
/* Raise IRQL */
|
|
OldIrql = KfRaiseIrql(Interrupt->SynchronizeIrql);
|
|
|
|
/* Acquire interrupt spinlock */
|
|
KeAcquireSpinLockAtDpcLevel(Interrupt->ActualLock);
|
|
|
|
/* Call the routine */
|
|
Success = SynchronizeRoutine(SynchronizeContext);
|
|
|
|
/* Release lock */
|
|
KeReleaseSpinLockFromDpcLevel(Interrupt->ActualLock);
|
|
|
|
/* Lower IRQL */
|
|
KeLowerIrql(OldIrql);
|
|
|
|
/* Return status */
|
|
return Success;
|
|
}
|