You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
256 lines
7.4 KiB
C
256 lines
7.4 KiB
C
/*
|
|
* Copyright (c) Contributors to the Open 3D Engine Project.
|
|
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0 OR MIT
|
|
*
|
|
*/
|
|
|
|
|
|
#pragma once
|
|
|
|
#include <AzCore/PlatformIncl.h>
|
|
#include <AzCore/std/parallel/semaphore.h> // for CreateSemaphore
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// CryLock_WinMutex
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CryLock_WinMutex::CryLock_WinMutex()
|
|
: m_hdl(CreateMutex(NULL, FALSE, NULL)) {}
|
|
CryLock_WinMutex::~CryLock_WinMutex()
|
|
{
|
|
CloseHandle(m_hdl);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryLock_WinMutex::Lock()
|
|
{
|
|
WaitForSingleObject(m_hdl, INFINITE);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryLock_WinMutex::Unlock()
|
|
{
|
|
ReleaseMutex(m_hdl);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
bool CryLock_WinMutex::TryLock()
|
|
{
|
|
return WaitForSingleObject(m_hdl, 0) != WAIT_TIMEOUT;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// CryLock_CritSection
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CryLock_CritSection::CryLock_CritSection()
|
|
{
|
|
InitializeCriticalSection((CRITICAL_SECTION*)&m_cs);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CryLock_CritSection::~CryLock_CritSection()
|
|
{
|
|
DeleteCriticalSection((CRITICAL_SECTION*)&m_cs);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryLock_CritSection::Lock()
|
|
{
|
|
EnterCriticalSection((CRITICAL_SECTION*)&m_cs);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryLock_CritSection::Unlock()
|
|
{
|
|
LeaveCriticalSection((CRITICAL_SECTION*)&m_cs);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
bool CryLock_CritSection::TryLock()
|
|
{
|
|
return TryEnterCriticalSection((CRITICAL_SECTION*)&m_cs) != FALSE;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// most of this is taken from http://www.cs.wustl.edu/~schmidt/win32-cv-1.html
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CryConditionVariable::CryConditionVariable()
|
|
{
|
|
m_waitersCount = 0;
|
|
m_wasBroadcast = 0;
|
|
m_sema = CreateSemaphore(NULL, 0, 0x7fffffff, NULL);
|
|
InitializeCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
m_waitersDone = CreateEvent(NULL, FALSE, FALSE, NULL);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CryConditionVariable::~CryConditionVariable()
|
|
{
|
|
CloseHandle(m_sema);
|
|
DeleteCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
CloseHandle(m_waitersDone);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryConditionVariable::Wait(LockType& lock)
|
|
{
|
|
EnterCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
m_waitersCount++;
|
|
LeaveCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
|
|
SignalObjectAndWait(lock._get_win32_handle(), m_sema, INFINITE, FALSE);
|
|
|
|
EnterCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
m_waitersCount--;
|
|
bool lastWaiter = m_wasBroadcast && m_waitersCount == 0;
|
|
LeaveCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
|
|
if (lastWaiter)
|
|
{
|
|
SignalObjectAndWait(m_waitersDone, lock._get_win32_handle(), INFINITE, FALSE);
|
|
}
|
|
else
|
|
{
|
|
WaitForSingleObject(lock._get_win32_handle(), INFINITE);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
bool CryConditionVariable::TimedWait(LockType& lock, uint32 millis)
|
|
{
|
|
EnterCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
m_waitersCount++;
|
|
LeaveCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
|
|
bool ok = true;
|
|
if (WAIT_TIMEOUT == SignalObjectAndWait(lock._get_win32_handle(), m_sema, millis, FALSE))
|
|
{
|
|
ok = false;
|
|
}
|
|
|
|
EnterCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
m_waitersCount--;
|
|
bool lastWaiter = m_wasBroadcast && m_waitersCount == 0;
|
|
LeaveCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
|
|
if (lastWaiter)
|
|
{
|
|
SignalObjectAndWait(m_waitersDone, lock._get_win32_handle(), INFINITE, FALSE);
|
|
}
|
|
else
|
|
{
|
|
WaitForSingleObject(lock._get_win32_handle(), INFINITE);
|
|
}
|
|
|
|
return ok;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryConditionVariable::NotifySingle()
|
|
{
|
|
EnterCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
bool haveWaiters = m_waitersCount > 0;
|
|
LeaveCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
if (haveWaiters)
|
|
{
|
|
ReleaseSemaphore(m_sema, 1, 0);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryConditionVariable::Notify()
|
|
{
|
|
EnterCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
bool haveWaiters = false;
|
|
if (m_waitersCount > 0)
|
|
{
|
|
m_wasBroadcast = 1;
|
|
haveWaiters = true;
|
|
}
|
|
if (haveWaiters)
|
|
{
|
|
ReleaseSemaphore(m_sema, m_waitersCount, 0);
|
|
LeaveCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
WaitForSingleObject(m_waitersDone, INFINITE);
|
|
m_wasBroadcast = 0;
|
|
}
|
|
else
|
|
{
|
|
LeaveCriticalSection((CRITICAL_SECTION*)&m_waitersCountLock);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CrySemaphore::CrySemaphore(int nMaximumCount, int nInitialCount)
|
|
{
|
|
m_Semaphore = (void*)CreateSemaphore(NULL, nInitialCount, nMaximumCount, NULL);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CrySemaphore::~CrySemaphore()
|
|
{
|
|
CloseHandle((HANDLE)m_Semaphore);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CrySemaphore::Acquire()
|
|
{
|
|
WaitForSingleObject((HANDLE)m_Semaphore, INFINITE);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CrySemaphore::Release()
|
|
{
|
|
ReleaseSemaphore((HANDLE)m_Semaphore, 1, NULL);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CryFastSemaphore::CryFastSemaphore(int nMaximumCount, int nInitialCount)
|
|
: m_Semaphore(nMaximumCount)
|
|
, m_nCounter(nInitialCount)
|
|
{
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
CryFastSemaphore::~CryFastSemaphore()
|
|
{
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryFastSemaphore::Acquire()
|
|
{
|
|
int nCount = ~0;
|
|
do
|
|
{
|
|
nCount = *const_cast<volatile int*>(&m_nCounter);
|
|
} while (CryInterlockedCompareExchange(alias_cast<volatile LONG*>(&m_nCounter), nCount - 1, nCount) != nCount);
|
|
|
|
// if the count would have been 0 or below, go to kernel semaphore
|
|
if ((nCount - 1) < 0)
|
|
{
|
|
m_Semaphore.Acquire();
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void CryFastSemaphore::Release()
|
|
{
|
|
int nCount = ~0;
|
|
do
|
|
{
|
|
nCount = *const_cast<volatile int*>(&m_nCounter);
|
|
} while (CryInterlockedCompareExchange(alias_cast<volatile LONG*>(&m_nCounter), nCount + 1, nCount) != nCount);
|
|
|
|
// wake up kernel semaphore if we have waiter
|
|
if (nCount < 0)
|
|
{
|
|
m_Semaphore.Release();
|
|
}
|
|
}
|