121 lines
2.4 KiB
C++
121 lines
2.4 KiB
C++
/**
|
|
* PROJECT: ExectOS
|
|
* COPYRIGHT: See COPYING.md in the top level directory
|
|
* FILE: xtoskrnl/ke/spinlock.cc
|
|
* DESCRIPTION: Spinlocks support
|
|
* DEVELOPERS: Rafal Kupiec <belliash@codingworkshop.eu.org>
|
|
*/
|
|
|
|
#include <xtos.hh>
|
|
|
|
|
|
/* Kernel Library */
|
|
namespace KE
|
|
{
|
|
|
|
/**
|
|
* Acquires a specified queued spinlock.
|
|
*
|
|
* @param LockLevel
|
|
* Supplies the queued spinlock level.
|
|
*
|
|
* @return This routine does not return any value.
|
|
*
|
|
* @since XT 1.0
|
|
*/
|
|
XTFASTCALL
|
|
VOID
|
|
SpinLock::AcquireQueuedSpinLock(IN KSPIN_LOCK_QUEUE_LEVEL LockLevel)
|
|
{
|
|
/* Acquire the queued spinlock */
|
|
AcquireSpinLock(KeGetCurrentProcessorControlBlock()->LockQueue[LockLevel].Lock);
|
|
}
|
|
|
|
/**
|
|
* Acquires a kernel spin lock.
|
|
*
|
|
* @param SpinLock
|
|
* Supplies a pointer to the kernel spin lock.
|
|
*
|
|
* @return This routine does not return any value.
|
|
*
|
|
* @since XT 1.0
|
|
*/
|
|
XTFASTCALL
|
|
VOID
|
|
SpinLock::AcquireSpinLock(IN OUT PKSPIN_LOCK SpinLock)
|
|
{
|
|
/* Try to acquire the lock */
|
|
while(RtlAtomicBitTestAndSet((PLONG)SpinLock, 0))
|
|
{
|
|
/* Wait until locked is cleared */
|
|
while(*(VOLATILE PKSPIN_LOCK)SpinLock & 1)
|
|
{
|
|
/* Yield processor and keep waiting */
|
|
ArYieldProcessor();
|
|
}
|
|
}
|
|
|
|
/* Add an explicit memory barrier */
|
|
ArReadWriteBarrier();
|
|
}
|
|
|
|
/**
|
|
* Initializes a kernel spinlock object.
|
|
*
|
|
* @param SpinLock
|
|
* Supplies a pointer to a kernel spin lock.
|
|
*
|
|
* @return This routine does not return any value.
|
|
*
|
|
* @since NT 3.5
|
|
*/
|
|
XTAPI
|
|
VOID
|
|
SpinLock::InitializeSpinLock(IN PKSPIN_LOCK SpinLock)
|
|
{
|
|
/* Zero initialize spinlock */
|
|
*SpinLock = 0;
|
|
}
|
|
|
|
/**
|
|
* Releases a queued spinlock.
|
|
*
|
|
* @param LockLevel
|
|
* Supplies the queued spinlock level.
|
|
*
|
|
* @return This routine does not return any value.
|
|
*
|
|
* @since XT 1.0
|
|
*/
|
|
XTFASTCALL
|
|
VOID
|
|
SpinLock::ReleaseQueuedSpinLock(IN KSPIN_LOCK_QUEUE_LEVEL LockLevel)
|
|
{
|
|
/* Clear the lock */
|
|
ReleaseSpinLock(KeGetCurrentProcessorControlBlock()->LockQueue[LockLevel].Lock);
|
|
}
|
|
|
|
/**
|
|
* Releases a kernel spin lock.
|
|
*
|
|
* @param SpinLock
|
|
* Supplies a pointer to the kernel spin lock.
|
|
*
|
|
* @return This routine does not return any value.
|
|
*
|
|
* @since XT 1.0
|
|
*/
|
|
XTFASTCALL
|
|
VOID
|
|
SpinLock::ReleaseSpinLock(IN OUT PKSPIN_LOCK SpinLock)
|
|
{
|
|
/* Clear the lock */
|
|
RtlAtomicAnd32((PLONG)SpinLock, 0);
|
|
|
|
/* Add an explicit memory barrier */
|
|
ArReadWriteBarrier();
|
|
}
|
|
|
|
} /* namespace */
|