mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
- Implement KiAcquireDispatcherObject, KiReleaseDispatcherObject, used on SMP for dispatcher objects which use the volatile long member in the dispatcher header (such as gates).
- Implement KiTryThreadLock which tries to acquire the lock but exits if it can't, isntead of looping. - Re-factor KeWaitForGate to make it SMP-safe by acquiring various locks when required. - Fixed up KeSignalGateBoostPriority so it can also be SMP-safe, and also fix what exactly happens when the gate is signaled (directly make the other thread ready instead of doing a KiUnwaitThread on it). - Split KxAcquireSpinLock/KxReleaseSpinLock into SMP vs UP like the other routines. svn path=/trunk/; revision=24103
This commit is contained in:
parent
d53352c677
commit
20425875ce
3 changed files with 275 additions and 97 deletions
|
@ -43,6 +43,7 @@ Author:
|
||||||
// Object Type Mask for Kernel Dispatcher Objects
|
// Object Type Mask for Kernel Dispatcher Objects
|
||||||
//
|
//
|
||||||
#define KOBJECT_TYPE_MASK 0x7F
|
#define KOBJECT_TYPE_MASK 0x7F
|
||||||
|
#define KOBJECT_LOCK_BIT 0x80
|
||||||
|
|
||||||
//
|
//
|
||||||
// Dispatcher Priority increments
|
// Dispatcher Priority increments
|
||||||
|
|
|
@ -384,45 +384,16 @@ KxUnwaitThreadForEvent(IN PKEVENT Event,
|
||||||
} while (WaitEntry != WaitList);
|
} while (WaitEntry != WaitList);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef _CONFIG_SMP
|
||||||
//
|
//
|
||||||
// Spinlock Acquisition at IRQL >= DISPATCH_LEVEL
|
// Spinlock Acquire at IRQL >= DISPATCH_LEVEL
|
||||||
//
|
//
|
||||||
FORCEINLINE
|
FORCEINLINE
|
||||||
VOID
|
VOID
|
||||||
KxAcquireSpinLock(IN PKSPIN_LOCK SpinLock)
|
KxAcquireSpinLock(IN PKSPIN_LOCK SpinLock)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_SMP
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
/* Try to acquire it */
|
|
||||||
if (InterlockedBitTestAndSet((PLONG)SpinLock, 0))
|
|
||||||
{
|
|
||||||
/* Value changed... wait until it's locked */
|
|
||||||
while (*(volatile KSPIN_LOCK *)SpinLock == 1)
|
|
||||||
{
|
|
||||||
#ifdef DBG
|
|
||||||
/* On debug builds, we use a much slower but useful routine */
|
|
||||||
Kii386SpinOnSpinLock(SpinLock, 5);
|
|
||||||
#else
|
|
||||||
/* Otherwise, just yield and keep looping */
|
|
||||||
YieldProcessor();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
#ifdef DBG
|
|
||||||
/* On debug builds, we OR in the KTHREAD */
|
|
||||||
*SpinLock = KeGetCurrentThread() | 1;
|
|
||||||
#endif
|
|
||||||
/* All is well, break out */
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
/* On UP builds, spinlocks don't exist at IRQL >= DISPATCH */
|
/* On UP builds, spinlocks don't exist at IRQL >= DISPATCH */
|
||||||
UNREFERENCED_PARAMETER(SpinLock);
|
UNREFERENCED_PARAMETER(SpinLock);
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
|
@ -432,27 +403,30 @@ FORCEINLINE
|
||||||
VOID
|
VOID
|
||||||
KxReleaseSpinLock(IN PKSPIN_LOCK SpinLock)
|
KxReleaseSpinLock(IN PKSPIN_LOCK SpinLock)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_SMP
|
|
||||||
#ifdef DBG
|
|
||||||
/* Make sure that the threads match */
|
|
||||||
if ((KeGetCurrentThread() | 1) != *SpinLock)
|
|
||||||
{
|
|
||||||
/* They don't, bugcheck */
|
|
||||||
KeBugCheckEx(SPIN_LOCK_NOT_OWNED, SpinLock, 0, 0, 0);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
/* Clear the lock */
|
|
||||||
InterlockedAnd(SpinLock, 0);
|
|
||||||
#else
|
|
||||||
/* On UP builds, spinlocks don't exist at IRQL >= DISPATCH */
|
/* On UP builds, spinlocks don't exist at IRQL >= DISPATCH */
|
||||||
UNREFERENCED_PARAMETER(SpinLock);
|
UNREFERENCED_PARAMETER(SpinLock);
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// Thread Scheduling Routines
|
// This routine protects against multiple CPU acquires, it's meaningless on UP.
|
||||||
//
|
//
|
||||||
#ifndef _CONFIG_SMP
|
VOID
|
||||||
|
FORCEINLINE
|
||||||
|
KiAcquireDispatcherObject(IN DISPATCHER_HEADER* Object)
|
||||||
|
{
|
||||||
|
UNREFERENCED_PARAMETER(Object);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// This routine protects against multiple CPU acquires, it's meaningless on UP.
|
||||||
|
//
|
||||||
|
VOID
|
||||||
|
FORCEINLINE
|
||||||
|
KiReleaseDispatcherObject(IN DISPATCHER_HEADER* Object)
|
||||||
|
{
|
||||||
|
UNREFERENCED_PARAMETER(Object);
|
||||||
|
}
|
||||||
|
|
||||||
KIRQL
|
KIRQL
|
||||||
FORCEINLINE
|
FORCEINLINE
|
||||||
KiAcquireDispatcherLock(VOID)
|
KiAcquireDispatcherLock(VOID)
|
||||||
|
@ -560,6 +534,17 @@ KiReleaseThreadLock(IN PKTHREAD Thread)
|
||||||
UNREFERENCED_PARAMETER(Thread);
|
UNREFERENCED_PARAMETER(Thread);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// This routine protects against multiple CPU acquires, it's meaningless on UP.
|
||||||
|
//
|
||||||
|
FORCEINLINE
|
||||||
|
BOOLEAN
|
||||||
|
KiTryThreadLock(IN PKTHREAD Thread)
|
||||||
|
{
|
||||||
|
UNREFERENCED_PARAMETER(Thread);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
FORCEINLINE
|
FORCEINLINE
|
||||||
VOID
|
VOID
|
||||||
KiCheckDeferredReadyList(IN PKPRCB Prcb)
|
KiCheckDeferredReadyList(IN PKPRCB Prcb)
|
||||||
|
@ -597,6 +582,98 @@ KiRequestApcInterrupt(IN BOOLEAN NeedApc,
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
|
//
|
||||||
|
// Spinlock Acquisition at IRQL >= DISPATCH_LEVEL
|
||||||
|
//
|
||||||
|
FORCEINLINE
|
||||||
|
VOID
|
||||||
|
KxAcquireSpinLock(IN PKSPIN_LOCK SpinLock)
|
||||||
|
{
|
||||||
|
for (;;)
|
||||||
|
{
|
||||||
|
/* Try to acquire it */
|
||||||
|
if (InterlockedBitTestAndSet((PLONG)SpinLock, 0))
|
||||||
|
{
|
||||||
|
/* Value changed... wait until it's locked */
|
||||||
|
while (*(volatile KSPIN_LOCK *)SpinLock == 1)
|
||||||
|
{
|
||||||
|
#ifdef DBG
|
||||||
|
/* On debug builds, we use a much slower but useful routine */
|
||||||
|
Kii386SpinOnSpinLock(SpinLock, 5);
|
||||||
|
#else
|
||||||
|
/* Otherwise, just yield and keep looping */
|
||||||
|
YieldProcessor();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
#ifdef DBG
|
||||||
|
/* On debug builds, we OR in the KTHREAD */
|
||||||
|
*SpinLock = KeGetCurrentThread() | 1;
|
||||||
|
#endif
|
||||||
|
/* All is well, break out */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Spinlock Release at IRQL >= DISPATCH_LEVEL
|
||||||
|
//
|
||||||
|
FORCEINLINE
|
||||||
|
VOID
|
||||||
|
KxReleaseSpinLock(IN PKSPIN_LOCK SpinLock)
|
||||||
|
{
|
||||||
|
#ifdef DBG
|
||||||
|
/* Make sure that the threads match */
|
||||||
|
if ((KeGetCurrentThread() | 1) != *SpinLock)
|
||||||
|
{
|
||||||
|
/* They don't, bugcheck */
|
||||||
|
KeBugCheckEx(SPIN_LOCK_NOT_OWNED, SpinLock, 0, 0, 0);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
/* Clear the lock */
|
||||||
|
InterlockedAnd(SpinLock, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
KIRQL
|
||||||
|
FORCEINLINE
|
||||||
|
KiAcquireDispatcherObject(IN DISPATCHER_HEADER* Object)
|
||||||
|
{
|
||||||
|
LONG OldValue, NewValue;
|
||||||
|
|
||||||
|
/* Make sure we're at a safe level to touch the lock */
|
||||||
|
ASSERT(KeGetCurrentIrql() >= DISPATCH_LEVEL);
|
||||||
|
|
||||||
|
/* Start acquire loop */
|
||||||
|
do
|
||||||
|
{
|
||||||
|
/* Loop until the other CPU releases it */
|
||||||
|
while ((UCHAR)Object->Lock & KOBJECT_LOCK_BIT)
|
||||||
|
{
|
||||||
|
/* Let the CPU know that this is a loop */
|
||||||
|
YieldProcessor();
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Try acquiring the lock now */
|
||||||
|
NewValue = InterlockedCompareExchange(&Object->Lock,
|
||||||
|
OldValue | KOBJECT_LOCK_BIT,
|
||||||
|
OldValue);
|
||||||
|
} while (NewValue != OldValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
KIRQL
|
||||||
|
FORCEINLINE
|
||||||
|
KiReleaseDispatcherObject(IN DISPATCHER_HEADER* Object)
|
||||||
|
{
|
||||||
|
/* Make sure we're at a safe level to touch the lock */
|
||||||
|
ASSERT(KeGetCurrentIrql() >= DISPATCH_LEVEL);
|
||||||
|
|
||||||
|
/* Release it */
|
||||||
|
InterlockedAnd(&Object->Lock, ~KOBJECT_LOCK_BIT);
|
||||||
|
}
|
||||||
|
|
||||||
KIRQL
|
KIRQL
|
||||||
FORCEINLINE
|
FORCEINLINE
|
||||||
KiAcquireDispatcherLock(VOID)
|
KiAcquireDispatcherLock(VOID)
|
||||||
|
@ -753,6 +830,23 @@ KiReleaseThreadLock(IN PKTHREAD Thread)
|
||||||
InterlockedAnd(&Thread->ThreadLock, 0);
|
InterlockedAnd(&Thread->ThreadLock, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
FORCEINLINE
|
||||||
|
BOOLEAN
|
||||||
|
KiTryThreadLock(IN PKTHREAD Thread)
|
||||||
|
{
|
||||||
|
LONG Value;
|
||||||
|
|
||||||
|
/* If the lock isn't acquired, return false */
|
||||||
|
if (!Thread->ThreadLock) return FALSE;
|
||||||
|
|
||||||
|
/* Otherwise, try to acquire it and check the result */
|
||||||
|
Value = 1;
|
||||||
|
Value = InterlockedExchange(&Thread->ThreadLock, &Value);
|
||||||
|
|
||||||
|
/* Return the lock state */
|
||||||
|
return (Value == TRUE);
|
||||||
|
}
|
||||||
|
|
||||||
FORCEINLINE
|
FORCEINLINE
|
||||||
VOID
|
VOID
|
||||||
KiCheckDeferredReadyList(IN PKPRCB Prcb)
|
KiCheckDeferredReadyList(IN PKPRCB Prcb)
|
||||||
|
|
|
@ -33,21 +33,23 @@ KeWaitForGate(IN PKGATE Gate,
|
||||||
IN KPROCESSOR_MODE WaitMode)
|
IN KPROCESSOR_MODE WaitMode)
|
||||||
{
|
{
|
||||||
KLOCK_QUEUE_HANDLE ApcLock;
|
KLOCK_QUEUE_HANDLE ApcLock;
|
||||||
PKTHREAD CurrentThread = KeGetCurrentThread();
|
PKTHREAD Thread = KeGetCurrentThread();
|
||||||
PKWAIT_BLOCK GateWaitBlock;
|
PKWAIT_BLOCK GateWaitBlock;
|
||||||
NTSTATUS Status;
|
NTSTATUS Status;
|
||||||
|
PKQUEUE Queue;
|
||||||
ASSERT_GATE(Gate);
|
ASSERT_GATE(Gate);
|
||||||
ASSERT_IRQL_LESS_OR_EQUAL(DISPATCH_LEVEL);
|
ASSERT_IRQL_LESS_OR_EQUAL(DISPATCH_LEVEL);
|
||||||
|
ASSERT(FALSE);
|
||||||
|
|
||||||
/* Start wait loop */
|
/* Start wait loop */
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
/* Acquire the APC lock */
|
/* Acquire the APC lock */
|
||||||
KiAcquireApcLock(CurrentThread, &ApcLock);
|
KiAcquireApcLock(Thread, &ApcLock);
|
||||||
|
|
||||||
/* Check if a kernel APC is pending and we're below APC_LEVEL */
|
/* Check if a kernel APC is pending and we're below APC_LEVEL */
|
||||||
if ((CurrentThread->ApcState.KernelApcPending) &&
|
if ((Thread->ApcState.KernelApcPending) &&
|
||||||
!(CurrentThread->SpecialApcDisable) &&
|
!(Thread->SpecialApcDisable) &&
|
||||||
(ApcLock.OldIrql < APC_LEVEL))
|
(ApcLock.OldIrql < APC_LEVEL))
|
||||||
{
|
{
|
||||||
/* Release the lock, this will fire the APC */
|
/* Release the lock, this will fire the APC */
|
||||||
|
@ -55,41 +57,71 @@ KeWaitForGate(IN PKGATE Gate,
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
/* Check if we have a queue and lock the dispatcher if so */
|
||||||
|
Queue = Thread->Queue;
|
||||||
|
if (Queue) KiAcquireDispatcherLockAtDpcLevel();
|
||||||
|
|
||||||
|
/* Lock the thread */
|
||||||
|
KiAcquireThreadLock(Thread);
|
||||||
|
|
||||||
|
/* Lock the gate */
|
||||||
|
KiAcquireDispatcherObject(&Gate->Header);
|
||||||
|
|
||||||
/* Check if it's already signaled */
|
/* Check if it's already signaled */
|
||||||
if (Gate->Header.SignalState)
|
if (Gate->Header.SignalState)
|
||||||
{
|
{
|
||||||
/* Unsignal it */
|
/* Unsignal it */
|
||||||
Gate->Header.SignalState = 0;
|
Gate->Header.SignalState = 0;
|
||||||
|
|
||||||
|
/* Release the gate and thread locks */
|
||||||
|
KiReleaseDispatcherObject(&Gate->Header);
|
||||||
|
KiReleaseThreadLock(Thread);
|
||||||
|
|
||||||
/* Release the APC lock and return */
|
/* Release the APC lock and return */
|
||||||
KiReleaseApcLock(&ApcLock);
|
KiReleaseApcLock(&ApcLock);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Setup a Wait Block */
|
/* Setup a Wait Block */
|
||||||
GateWaitBlock = &CurrentThread->WaitBlock[0];
|
GateWaitBlock = &Thread->WaitBlock[0];
|
||||||
GateWaitBlock->Object = (PVOID)Gate;
|
GateWaitBlock->Object = (PVOID)Gate;
|
||||||
GateWaitBlock->Thread = CurrentThread;
|
GateWaitBlock->Thread = Thread;
|
||||||
|
|
||||||
/* Set the Thread Wait Data */
|
/* Set the Thread Wait Data */
|
||||||
CurrentThread->WaitMode = WaitMode;
|
Thread->WaitMode = WaitMode;
|
||||||
CurrentThread->WaitReason = WaitReason;
|
Thread->WaitReason = WaitReason;
|
||||||
CurrentThread->WaitIrql = ApcLock.OldIrql;
|
Thread->WaitIrql = ApcLock.OldIrql;
|
||||||
CurrentThread->State = GateWait;
|
Thread->State = GateWait;
|
||||||
CurrentThread->GateObject = Gate;
|
Thread->GateObject = Gate;
|
||||||
|
|
||||||
/* Insert into the Wait List */
|
/* Insert into the Wait List */
|
||||||
InsertTailList(&Gate->Header.WaitListHead,
|
InsertTailList(&Gate->Header.WaitListHead,
|
||||||
&GateWaitBlock->WaitListEntry);
|
&GateWaitBlock->WaitListEntry);
|
||||||
|
|
||||||
/* Handle Kernel Queues */
|
/* Release the gate lock */
|
||||||
if (CurrentThread->Queue) KiWakeQueue(CurrentThread->Queue);
|
KiReleaseDispatcherObject(&Gate->Header);
|
||||||
|
|
||||||
|
/* Set swap busy */
|
||||||
|
KiSetThreadSwapBusy(Thread);
|
||||||
|
|
||||||
|
/* Release the thread lock */
|
||||||
|
KiReleaseThreadLock(Thread);
|
||||||
|
|
||||||
|
/* Check if we had a queue */
|
||||||
|
if (Queue)
|
||||||
|
{
|
||||||
|
/* Wake it up */
|
||||||
|
KiWakeQueue(Queue);
|
||||||
|
|
||||||
|
/* Release the dispatcher lock */
|
||||||
|
KiReleaseDispatcherLockFromDpcLevel();
|
||||||
|
}
|
||||||
|
|
||||||
/* Release the APC lock but stay at DPC level */
|
/* Release the APC lock but stay at DPC level */
|
||||||
KiReleaseApcLockFromDpcLevel(&ApcLock);
|
KiReleaseApcLockFromDpcLevel(&ApcLock);
|
||||||
|
|
||||||
/* Find a new thread to run */
|
/* Find a new thread to run */
|
||||||
Status = KiSwapThread(CurrentThread, KeGetCurrentPrcb());
|
Status = KiSwapThread(Thread, KeGetCurrentPrcb());
|
||||||
|
|
||||||
/* Check if we were executing an APC */
|
/* Check if we were executing an APC */
|
||||||
if (Status != STATUS_KERNEL_APC) return;
|
if (Status != STATUS_KERNEL_APC) return;
|
||||||
|
@ -104,13 +136,19 @@ KeSignalGateBoostPriority(IN PKGATE Gate)
|
||||||
PKTHREAD WaitThread;
|
PKTHREAD WaitThread;
|
||||||
PKWAIT_BLOCK WaitBlock;
|
PKWAIT_BLOCK WaitBlock;
|
||||||
KIRQL OldIrql;
|
KIRQL OldIrql;
|
||||||
NTSTATUS WaitStatus = STATUS_SUCCESS;
|
|
||||||
ASSERT_GATE(Gate);
|
ASSERT_GATE(Gate);
|
||||||
ASSERT_IRQL_LESS_OR_EQUAL(DISPATCH_LEVEL);
|
ASSERT_IRQL_LESS_OR_EQUAL(DISPATCH_LEVEL);
|
||||||
|
ASSERT(FALSE);
|
||||||
|
|
||||||
|
/* Start entry loop */
|
||||||
|
for (;;)
|
||||||
|
{
|
||||||
/* Raise to synch level */
|
/* Raise to synch level */
|
||||||
OldIrql = KeRaiseIrqlToSynchLevel();
|
OldIrql = KeRaiseIrqlToSynchLevel();
|
||||||
|
|
||||||
|
/* Lock the gate */
|
||||||
|
KiAcquireDispatcherObject(&Gate->Header);
|
||||||
|
|
||||||
/* Make sure we're not already signaled or that the list is empty */
|
/* Make sure we're not already signaled or that the list is empty */
|
||||||
if (Gate->Header.SignalState)
|
if (Gate->Header.SignalState)
|
||||||
{
|
{
|
||||||
|
@ -132,21 +170,66 @@ KeSignalGateBoostPriority(IN PKGATE Gate)
|
||||||
KWAIT_BLOCK,
|
KWAIT_BLOCK,
|
||||||
WaitListEntry);
|
WaitListEntry);
|
||||||
|
|
||||||
/* Remove it */
|
|
||||||
RemoveEntryList(&WaitBlock->WaitListEntry);
|
|
||||||
|
|
||||||
/* Get the Associated thread */
|
/* Get the Associated thread */
|
||||||
WaitThread = WaitBlock->Thread;
|
WaitThread = WaitBlock->Thread;
|
||||||
|
|
||||||
/* Increment the Queue's active threads */
|
/* Check to see if the waiting thread is locked */
|
||||||
if (WaitThread->Queue) WaitThread->Queue->CurrentCount++;
|
if (!KiTryThreadLock(WaitThread))
|
||||||
|
{
|
||||||
|
/* Unlock the gate */
|
||||||
|
KiReleaseDispatcherObject(&Gate->Header);
|
||||||
|
|
||||||
/* FIXME: This isn't really correct!!! */
|
/* Lower IRQL and loop again */
|
||||||
KiUnwaitThread(WaitThread, WaitStatus, EVENT_INCREMENT);
|
KeLowerIrql(OldIrql);
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Remove it */
|
||||||
|
RemoveEntryList(&WaitBlock->WaitListEntry);
|
||||||
|
|
||||||
|
/* Clear wait status */
|
||||||
|
WaitThread->WaitStatus = 0;
|
||||||
|
|
||||||
|
/* Set state and CPU */
|
||||||
|
WaitThread->State = DeferredReady;
|
||||||
|
WaitThread->DeferredProcessor = KeGetCurrentPrcb()->Number;
|
||||||
|
|
||||||
|
/* Release the gate lock */
|
||||||
|
KiReleaseDispatcherObject(&Gate->Header);
|
||||||
|
|
||||||
|
/* Release the thread lock */
|
||||||
|
KiReleaseThreadLock(WaitThread);
|
||||||
|
|
||||||
|
/* FIXME: Boosting */
|
||||||
|
|
||||||
|
/* Check if we have a queue */
|
||||||
|
if (WaitThread->Queue)
|
||||||
|
{
|
||||||
|
/* Acquire the dispatcher lock */
|
||||||
|
KiAcquireDispatcherLockAtDpcLevel();
|
||||||
|
|
||||||
|
/* Check if we still have one */
|
||||||
|
if (WaitThread->Queue)
|
||||||
|
{
|
||||||
|
/* Increment active threads */
|
||||||
|
WaitThread->Queue->CurrentCount++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Release lock */
|
||||||
|
KiReleaseDispatcherLockFromDpcLevel();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Make the thread ready */
|
||||||
|
KiReadyThread(WaitThread);
|
||||||
|
|
||||||
/* Exit the dispatcher */
|
/* Exit the dispatcher */
|
||||||
KiExitDispatcher(OldIrql);
|
KiExitDispatcher(OldIrql);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If we got here, then there's no rescheduling. */
|
||||||
|
KiReleaseDispatcherObject(&Gate->Header);
|
||||||
|
KeLowerIrql(OldIrql);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* EOF */
|
/* EOF */
|
||||||
|
|
Loading…
Reference in a new issue