mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
6e97b4314f
And make it thread-safe. Co-authored-by: Thomas Faber <thomas.faber@reactos.org>
348 lines
9.5 KiB
C++
348 lines
9.5 KiB
C++
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS Kernel Streaming
|
|
* FILE: drivers/wdm/audio/backpln/portcls/interrupt.cpp
|
|
* PURPOSE: portcls interrupt object
|
|
* PROGRAMMER: Johannes Anderwald
|
|
*/
|
|
|
|
#include "private.hpp"
|
|
|
|
#ifndef YDEBUG
|
|
#define NDEBUG
|
|
#endif
|
|
|
|
#include <debug.h>
|
|
|
|
typedef struct
|
|
{
|
|
LIST_ENTRY ListEntry;
|
|
PINTERRUPTSYNCROUTINE SyncRoutine;
|
|
PVOID DynamicContext;
|
|
}SYNC_ENTRY, *PSYNC_ENTRY;
|
|
|
|
class CInterruptSync : public CUnknownImpl<IInterruptSync>
|
|
{
|
|
public:
|
|
STDMETHODIMP QueryInterface( REFIID InterfaceId, PVOID* Interface);
|
|
|
|
IMP_IInterruptSync;
|
|
CInterruptSync(IUnknown *OuterUnknown){}
|
|
virtual ~CInterruptSync(){}
|
|
|
|
public:
|
|
|
|
KSPIN_LOCK m_Lock;
|
|
LIST_ENTRY m_ServiceRoutines;
|
|
PKINTERRUPT m_Interrupt;
|
|
INTERRUPTSYNCMODE m_Mode;
|
|
PRESOURCELIST m_ResourceList;
|
|
ULONG m_ResourceIndex;
|
|
|
|
PINTERRUPTSYNCROUTINE m_SyncRoutine;
|
|
PVOID m_DynamicContext;
|
|
NTSTATUS m_Status;
|
|
|
|
friend BOOLEAN NTAPI CInterruptSynchronizedRoutine(IN PVOID ServiceContext);
|
|
friend BOOLEAN NTAPI IInterruptServiceRoutine(IN PKINTERRUPT Interrupt, IN PVOID ServiceContext);
|
|
};
|
|
|
|
|
|
//---------------------------------------------------------------
|
|
// IUnknown methods
|
|
//
|
|
|
|
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
CInterruptSync::QueryInterface(
|
|
IN REFIID refiid,
|
|
OUT PVOID* Output)
|
|
{
|
|
UNICODE_STRING GuidString;
|
|
|
|
DPRINT("CInterruptSync::QueryInterface: this %p\n", this);
|
|
|
|
if (IsEqualGUIDAligned(refiid, IID_IInterruptSync) ||
|
|
IsEqualGUIDAligned(refiid, IID_IUnknown))
|
|
{
|
|
*Output = PVOID(PUNKNOWN(this));
|
|
PUNKNOWN(*Output)->AddRef();
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
if (RtlStringFromGUID(refiid, &GuidString) == STATUS_SUCCESS)
|
|
{
|
|
DPRINT1("CInterruptSync::QueryInterface: no interface!!! iface %S\n", GuidString.Buffer);
|
|
RtlFreeUnicodeString(&GuidString);
|
|
}
|
|
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
//---------------------------------------------------------------
|
|
// CInterruptSync methods
|
|
//
|
|
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
CInterruptSynchronizedRoutine(
|
|
IN PVOID ServiceContext)
|
|
{
|
|
CInterruptSync * This = (CInterruptSync*)ServiceContext;
|
|
This->m_Status = This->m_SyncRoutine(This, This->m_DynamicContext);
|
|
|
|
DPRINT("CInterruptSynchronizedRoutine this %p SyncRoutine %p Context %p Status %x\n", This, This->m_SyncRoutine, This->m_DynamicContext, This->m_Status);
|
|
return TRUE;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
CInterruptSync::CallSynchronizedRoutine(
|
|
IN PINTERRUPTSYNCROUTINE Routine,
|
|
IN PVOID DynamicContext)
|
|
{
|
|
KIRQL OldIrql;
|
|
|
|
DPRINT("CInterruptSync::CallSynchronizedRoutine this %p Routine %p DynamicContext %p Irql %x Interrupt %p\n", this, Routine, DynamicContext, KeGetCurrentIrql(), m_Interrupt);
|
|
|
|
if (!m_Interrupt)
|
|
{
|
|
DPRINT("CInterruptSync_CallSynchronizedRoutine %p no interrupt connected\n", this);
|
|
if (KeGetCurrentIrql() > DISPATCH_LEVEL)
|
|
return STATUS_UNSUCCESSFUL;
|
|
|
|
KeAcquireSpinLock(&m_Lock, &OldIrql);
|
|
m_SyncRoutine = Routine;
|
|
m_DynamicContext = DynamicContext;
|
|
CInterruptSynchronizedRoutine((PVOID)this);
|
|
KeReleaseSpinLock(&m_Lock, OldIrql);
|
|
|
|
return m_Status;
|
|
}
|
|
|
|
m_SyncRoutine = Routine;
|
|
m_DynamicContext = DynamicContext;
|
|
|
|
if (KeSynchronizeExecution(m_Interrupt, CInterruptSynchronizedRoutine, (PVOID)this))
|
|
return m_Status;
|
|
else
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
PKINTERRUPT
|
|
NTAPI
|
|
CInterruptSync::GetKInterrupt()
|
|
{
|
|
DPRINT("CInterruptSynchronizedRoutine\n");
|
|
|
|
return m_Interrupt;
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
IInterruptServiceRoutine(
|
|
IN PKINTERRUPT Interrupt,
|
|
IN PVOID ServiceContext)
|
|
{
|
|
PLIST_ENTRY CurEntry;
|
|
PSYNC_ENTRY Entry;
|
|
NTSTATUS Status;
|
|
BOOL Success, Ret;
|
|
|
|
CInterruptSync * This = (CInterruptSync*)ServiceContext;
|
|
|
|
DPRINT("IInterruptServiceRoutine Mode %u\n", This->m_Mode);
|
|
|
|
Ret = FALSE;
|
|
if (This->m_Mode == InterruptSyncModeNormal)
|
|
{
|
|
CurEntry = This->m_ServiceRoutines.Flink;
|
|
while (CurEntry != &This->m_ServiceRoutines)
|
|
{
|
|
Entry = CONTAINING_RECORD(CurEntry, SYNC_ENTRY, ListEntry);
|
|
Status = Entry->SyncRoutine((CInterruptSync*)This, Entry->DynamicContext);
|
|
if (Status == STATUS_SUCCESS)
|
|
{
|
|
/* Mark as handled and break on the first success */
|
|
Ret = TRUE;
|
|
break;
|
|
}
|
|
CurEntry = CurEntry->Flink;
|
|
}
|
|
return Ret;
|
|
}
|
|
else if (This->m_Mode == InterruptSyncModeAll)
|
|
{
|
|
CurEntry = This->m_ServiceRoutines.Flink;
|
|
while (CurEntry != &This->m_ServiceRoutines)
|
|
{
|
|
Entry = CONTAINING_RECORD(CurEntry, SYNC_ENTRY, ListEntry);
|
|
Status = Entry->SyncRoutine((CInterruptSync*)This, Entry->DynamicContext);
|
|
if (Status == STATUS_SUCCESS)
|
|
{
|
|
/* Mark as handled but don't break */
|
|
Ret = TRUE;
|
|
}
|
|
CurEntry = CurEntry->Flink;
|
|
}
|
|
return Ret;
|
|
}
|
|
else if (This->m_Mode == InterruptSyncModeRepeat)
|
|
{
|
|
do
|
|
{
|
|
Success = FALSE;
|
|
CurEntry = This->m_ServiceRoutines.Flink;
|
|
while (CurEntry != &This->m_ServiceRoutines)
|
|
{
|
|
Entry = CONTAINING_RECORD(CurEntry, SYNC_ENTRY, ListEntry);
|
|
Status = Entry->SyncRoutine((CInterruptSync*)This, Entry->DynamicContext);
|
|
if (Status == STATUS_SUCCESS)
|
|
{
|
|
/* Mark as handled if it works at least once */
|
|
Success = TRUE;
|
|
Ret = TRUE;
|
|
}
|
|
CurEntry = CurEntry->Flink;
|
|
}
|
|
} while(Success);
|
|
return Ret;
|
|
}
|
|
else
|
|
{
|
|
DPRINT("Unknown mode %u\n", This->m_Mode);
|
|
return Ret;
|
|
}
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
CInterruptSync::Connect()
|
|
{
|
|
NTSTATUS Status;
|
|
PCM_PARTIAL_RESOURCE_DESCRIPTOR Descriptor;
|
|
|
|
DPRINT("CInterruptSync::Connect\n");
|
|
PC_ASSERT_IRQL_EQUAL(PASSIVE_LEVEL);
|
|
|
|
Descriptor = m_ResourceList->FindTranslatedEntry(CmResourceTypeInterrupt, m_ResourceIndex);
|
|
if (!Descriptor)
|
|
return STATUS_UNSUCCESSFUL;
|
|
|
|
if (IsListEmpty(&m_ServiceRoutines))
|
|
return STATUS_UNSUCCESSFUL;
|
|
|
|
DPRINT1("Vector %u Level %u Flags %x Affinity %x\n", Descriptor->u.Interrupt.Vector, Descriptor->u.Interrupt.Level, Descriptor->Flags, Descriptor->u.Interrupt.Affinity);
|
|
|
|
Status = IoConnectInterrupt(&m_Interrupt,
|
|
IInterruptServiceRoutine,
|
|
(PVOID)this,
|
|
NULL, //&m_Lock,
|
|
Descriptor->u.Interrupt.Vector,
|
|
(KIRQL)Descriptor->u.Interrupt.Level,
|
|
(KIRQL)Descriptor->u.Interrupt.Level,
|
|
(KINTERRUPT_MODE)(Descriptor->Flags & CM_RESOURCE_INTERRUPT_LATCHED),
|
|
(Descriptor->ShareDisposition != CmResourceShareDeviceExclusive),
|
|
Descriptor->u.Interrupt.Affinity,
|
|
FALSE);
|
|
|
|
DPRINT1("CInterruptSync::Connect result %x\n", Status);
|
|
return Status;
|
|
}
|
|
|
|
|
|
VOID
|
|
NTAPI
|
|
CInterruptSync::Disconnect()
|
|
{
|
|
DPRINT("CInterruptSync::Disconnect\n");
|
|
PC_ASSERT_IRQL_EQUAL(PASSIVE_LEVEL);
|
|
|
|
if (!m_Interrupt)
|
|
{
|
|
DPRINT("CInterruptSync_Disconnect %p no interrupt connected\n", this);
|
|
return;
|
|
}
|
|
|
|
IoDisconnectInterrupt(m_Interrupt);
|
|
m_Interrupt = NULL;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
CInterruptSync::RegisterServiceRoutine(
|
|
IN PINTERRUPTSYNCROUTINE Routine,
|
|
IN PVOID DynamicContext,
|
|
IN BOOLEAN First)
|
|
{
|
|
PSYNC_ENTRY NewEntry;
|
|
|
|
DPRINT("CInterruptSync::RegisterServiceRoutine\n");
|
|
PC_ASSERT_IRQL_EQUAL(PASSIVE_LEVEL);
|
|
|
|
NewEntry = (PSYNC_ENTRY)AllocateItem(NonPagedPool, sizeof(SYNC_ENTRY), TAG_PORTCLASS);
|
|
if (!NewEntry)
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
NewEntry->SyncRoutine = Routine;
|
|
NewEntry->DynamicContext = DynamicContext;
|
|
|
|
if (First)
|
|
InsertHeadList(&m_ServiceRoutines, &NewEntry->ListEntry);
|
|
else
|
|
InsertTailList(&m_ServiceRoutines, &NewEntry->ListEntry);
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
PcNewInterruptSync(
|
|
OUT PINTERRUPTSYNC* OutInterruptSync,
|
|
IN PUNKNOWN OuterUnknown OPTIONAL,
|
|
IN PRESOURCELIST ResourceList,
|
|
IN ULONG ResourceIndex,
|
|
IN INTERRUPTSYNCMODE Mode)
|
|
{
|
|
CInterruptSync * This;
|
|
NTSTATUS Status;
|
|
|
|
DPRINT("PcNewInterruptSync entered OutInterruptSync %p OuterUnknown %p ResourceList %p ResourceIndex %u Mode %d\n",
|
|
OutInterruptSync, OuterUnknown, ResourceList, ResourceIndex, Mode);
|
|
|
|
if (!OutInterruptSync || !ResourceList || Mode < InterruptSyncModeNormal || Mode > InterruptSyncModeRepeat)
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
if (ResourceIndex > ResourceList->NumberOfEntriesOfType(CmResourceTypeInterrupt))
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
This = new(NonPagedPool, TAG_PORTCLASS)CInterruptSync(OuterUnknown);
|
|
if (!This)
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
Status = This->QueryInterface(IID_IInterruptSync, (PVOID*)OutInterruptSync);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
delete This;
|
|
return Status;
|
|
}
|
|
|
|
ResourceList->AddRef();
|
|
|
|
//
|
|
// initialize object
|
|
//
|
|
This->m_Mode = Mode;
|
|
This->m_ResourceIndex = ResourceIndex;
|
|
This->m_ResourceList = ResourceList;
|
|
InitializeListHead(&This->m_ServiceRoutines);
|
|
KeInitializeSpinLock(&This->m_Lock);
|
|
|
|
return Status;
|
|
}
|