mirror of
https://github.com/alliedmodders/hl2sdk.git
synced 2025-01-03 16:13:22 +08:00
PVKII branch support (#111)
* Update tier0, vstdlib with PVKII's libs * PVKII threadtools changes backported to C++14 * Rebuild mathlib, tier1 * Update Windows tier0/vstdlib import libs * Rebuild tier1/mathlib for Windows
This commit is contained in:
parent
78dab58646
commit
6769441f9f
BIN
lib/public/linux32/libtier0.so
Normal file → Executable file
BIN
lib/public/linux32/libtier0.so
Normal file → Executable file
Binary file not shown.
Binary file not shown.
BIN
lib/public/linux32/libvstdlib.so
Normal file → Executable file
BIN
lib/public/linux32/libvstdlib.so
Normal file → Executable file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -213,7 +213,7 @@ short *CBoneCache::CachedToStudio()
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Construct a singleton
|
// Construct a singleton
|
||||||
static CDataManager<CBoneCache, bonecacheparams_t, CBoneCache *, CThreadFastMutex> g_StudioBoneCache( 128 * 1024L );
|
static CDataManager<CBoneCache, bonecacheparams_t, CBoneCache *, CThreadFastMutexRecursive> g_StudioBoneCache( 128 * 1024L );
|
||||||
|
|
||||||
CBoneCache *Studio_GetBoneCache( memhandle_t cacheHandle )
|
CBoneCache *Studio_GetBoneCache( memhandle_t cacheHandle )
|
||||||
{
|
{
|
||||||
@ -2625,14 +2625,14 @@ public:
|
|||||||
X[i] = P[i];
|
X[i] = P[i];
|
||||||
normalize(X);
|
normalize(X);
|
||||||
|
|
||||||
// Its y axis is perpendicular to P, so Y = unit( E - X(E·X) ).
|
// Its y axis is perpendicular to P, so Y = unit( E - X(E<EFBFBD>X) ).
|
||||||
|
|
||||||
float dDOTx = dot(D,X);
|
float dDOTx = dot(D,X);
|
||||||
for (i = 0 ; i < 3 ; i++)
|
for (i = 0 ; i < 3 ; i++)
|
||||||
Y[i] = D[i] - dDOTx * X[i];
|
Y[i] = D[i] - dDOTx * X[i];
|
||||||
normalize(Y);
|
normalize(Y);
|
||||||
|
|
||||||
// Its z axis is perpendicular to both X and Y, so Z = X×Y.
|
// Its z axis is perpendicular to both X and Y, so Z = X<EFBFBD>Y.
|
||||||
|
|
||||||
cross(X,Y,Z);
|
cross(X,Y,Z);
|
||||||
|
|
||||||
|
@ -1724,7 +1724,7 @@ struct virtualmodel_t
|
|||||||
return &m_group[ m_seq[ sequence ].group ];
|
return &m_group[ m_seq[ sequence ].group ];
|
||||||
} // Note: user must manage mutex for this
|
} // Note: user must manage mutex for this
|
||||||
|
|
||||||
CThreadFastMutex m_Lock;
|
CThreadFastMutexRecursive m_Lock;
|
||||||
|
|
||||||
CUtlVector< virtualsequence_t > m_seq;
|
CUtlVector< virtualsequence_t > m_seq;
|
||||||
CUtlVector< virtualgeneric_t > m_anim;
|
CUtlVector< virtualgeneric_t > m_anim;
|
||||||
|
@ -11,6 +11,11 @@
|
|||||||
|
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
|
|
||||||
|
#include <limits>
|
||||||
|
#include <atomic>
|
||||||
|
#include <mutex>
|
||||||
|
#include <shared_mutex>
|
||||||
|
|
||||||
#include "tier0/platform.h"
|
#include "tier0/platform.h"
|
||||||
#include "tier0/dbg.h"
|
#include "tier0/dbg.h"
|
||||||
#include "tier0/vcrmode.h"
|
#include "tier0/vcrmode.h"
|
||||||
@ -589,64 +594,24 @@ private:
|
|||||||
//
|
//
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
|
|
||||||
class PLATFORM_CLASS CThreadMutex
|
class CThreadMutex
|
||||||
{
|
{
|
||||||
|
std::mutex mutex;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
CThreadMutex();
|
bool TryLock() { return mutex.try_lock(); }
|
||||||
~CThreadMutex();
|
void Lock() { mutex.lock(); }
|
||||||
|
void Unlock() { mutex.unlock(); }
|
||||||
|
};
|
||||||
|
|
||||||
//------------------------------------------------------
|
class CThreadMutexRecursive
|
||||||
// Mutex acquisition/release. Const intentionally defeated.
|
{
|
||||||
//------------------------------------------------------
|
std::recursive_mutex mutex;
|
||||||
void Lock();
|
|
||||||
void Lock() const { (const_cast<CThreadMutex *>(this))->Lock(); }
|
|
||||||
void Unlock();
|
|
||||||
void Unlock() const { (const_cast<CThreadMutex *>(this))->Unlock(); }
|
|
||||||
|
|
||||||
bool TryLock();
|
public:
|
||||||
bool TryLock() const { return (const_cast<CThreadMutex *>(this))->TryLock(); }
|
bool TryLock() { return mutex.try_lock(); }
|
||||||
|
void Lock() { mutex.lock(); }
|
||||||
//------------------------------------------------------
|
void Unlock() { mutex.unlock(); }
|
||||||
// Use this to make deadlocks easier to track by asserting
|
|
||||||
// when it is expected that the current thread owns the mutex
|
|
||||||
//------------------------------------------------------
|
|
||||||
bool AssertOwnedByCurrentThread();
|
|
||||||
|
|
||||||
//------------------------------------------------------
|
|
||||||
// Enable tracing to track deadlock problems
|
|
||||||
//------------------------------------------------------
|
|
||||||
void SetTrace( bool );
|
|
||||||
|
|
||||||
private:
|
|
||||||
// Disallow copying
|
|
||||||
CThreadMutex( const CThreadMutex & );
|
|
||||||
CThreadMutex &operator=( const CThreadMutex & );
|
|
||||||
|
|
||||||
#if defined( _WIN32 )
|
|
||||||
// Efficient solution to breaking the windows.h dependency, invariant is tested.
|
|
||||||
#ifdef _WIN64
|
|
||||||
#define TT_SIZEOF_CRITICALSECTION 40
|
|
||||||
#else
|
|
||||||
#ifndef _X360
|
|
||||||
#define TT_SIZEOF_CRITICALSECTION 24
|
|
||||||
#else
|
|
||||||
#define TT_SIZEOF_CRITICALSECTION 28
|
|
||||||
#endif // !_XBOX
|
|
||||||
#endif // _WIN64
|
|
||||||
byte m_CriticalSection[TT_SIZEOF_CRITICALSECTION];
|
|
||||||
#elif defined(POSIX)
|
|
||||||
pthread_mutex_t m_Mutex;
|
|
||||||
pthread_mutexattr_t m_Attr;
|
|
||||||
#else
|
|
||||||
#error
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef THREAD_MUTEX_TRACING_SUPPORTED
|
|
||||||
// Debugging (always here to allow mixed debug/release builds w/o changing size)
|
|
||||||
uint m_currentOwnerID;
|
|
||||||
uint16 m_lockCount;
|
|
||||||
bool m_bTrace;
|
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
@ -661,118 +626,11 @@ private:
|
|||||||
|
|
||||||
#if !defined(THREAD_PROFILER)
|
#if !defined(THREAD_PROFILER)
|
||||||
|
|
||||||
class CThreadFastMutex
|
// Just make these the same thing, if we really need to, we can recreate these using atomics
|
||||||
{
|
// But honestly the standard mutex objects will likely be superior anyway
|
||||||
public:
|
// Implementations already behave as spinlocks for brief periods before sleeping like our old one
|
||||||
CThreadFastMutex()
|
using CThreadFastMutex = CThreadMutex;
|
||||||
: m_ownerID( 0 ),
|
using CThreadFastMutexRecursive = CThreadMutexRecursive;
|
||||||
m_depth( 0 )
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
FORCEINLINE bool TryLockInline( const uint32 threadId ) volatile
|
|
||||||
{
|
|
||||||
if ( threadId != m_ownerID && !ThreadInterlockedAssignIf( (volatile long *)&m_ownerID, (long)threadId, 0 ) )
|
|
||||||
return false;
|
|
||||||
|
|
||||||
ThreadMemoryBarrier();
|
|
||||||
++m_depth;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryLock( const uint32 threadId ) volatile
|
|
||||||
{
|
|
||||||
return TryLockInline( threadId );
|
|
||||||
}
|
|
||||||
|
|
||||||
PLATFORM_CLASS void Lock( const uint32 threadId, unsigned nSpinSleepTime ) volatile;
|
|
||||||
|
|
||||||
public:
|
|
||||||
bool TryLock() volatile
|
|
||||||
{
|
|
||||||
#ifdef _DEBUG
|
|
||||||
if ( m_depth == INT_MAX )
|
|
||||||
DebuggerBreak();
|
|
||||||
|
|
||||||
if ( m_depth < 0 )
|
|
||||||
DebuggerBreak();
|
|
||||||
#endif
|
|
||||||
return TryLockInline( ThreadGetCurrentId() );
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifndef _DEBUG
|
|
||||||
FORCEINLINE
|
|
||||||
#endif
|
|
||||||
void Lock( unsigned int nSpinSleepTime = 0 ) volatile
|
|
||||||
{
|
|
||||||
const uint32 threadId = ThreadGetCurrentId();
|
|
||||||
|
|
||||||
if ( !TryLockInline( threadId ) )
|
|
||||||
{
|
|
||||||
ThreadPause();
|
|
||||||
Lock( threadId, nSpinSleepTime );
|
|
||||||
}
|
|
||||||
#ifdef _DEBUG
|
|
||||||
if ( m_ownerID != ThreadGetCurrentId() )
|
|
||||||
DebuggerBreak();
|
|
||||||
|
|
||||||
if ( m_depth == INT_MAX )
|
|
||||||
DebuggerBreak();
|
|
||||||
|
|
||||||
if ( m_depth < 0 )
|
|
||||||
DebuggerBreak();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifndef _DEBUG
|
|
||||||
FORCEINLINE
|
|
||||||
#endif
|
|
||||||
void Unlock() volatile
|
|
||||||
{
|
|
||||||
#ifdef _DEBUG
|
|
||||||
if ( m_ownerID != ThreadGetCurrentId() )
|
|
||||||
DebuggerBreak();
|
|
||||||
|
|
||||||
if ( m_depth <= 0 )
|
|
||||||
DebuggerBreak();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
--m_depth;
|
|
||||||
if ( !m_depth )
|
|
||||||
{
|
|
||||||
ThreadMemoryBarrier();
|
|
||||||
ThreadInterlockedExchange( &m_ownerID, 0 );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef WIN32
|
|
||||||
bool TryLock() const volatile { return (const_cast<CThreadFastMutex *>(this))->TryLock(); }
|
|
||||||
void Lock(unsigned nSpinSleepTime = 1 ) const volatile { (const_cast<CThreadFastMutex *>(this))->Lock( nSpinSleepTime ); }
|
|
||||||
void Unlock() const volatile { (const_cast<CThreadFastMutex *>(this))->Unlock(); }
|
|
||||||
#endif
|
|
||||||
// To match regular CThreadMutex:
|
|
||||||
bool AssertOwnedByCurrentThread() { return true; }
|
|
||||||
void SetTrace( bool ) {}
|
|
||||||
|
|
||||||
uint32 GetOwnerId() const { return m_ownerID; }
|
|
||||||
int GetDepth() const { return m_depth; }
|
|
||||||
private:
|
|
||||||
volatile uint32 m_ownerID;
|
|
||||||
int m_depth;
|
|
||||||
};
|
|
||||||
|
|
||||||
class ALIGN128 CAlignedThreadFastMutex : public CThreadFastMutex
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
CAlignedThreadFastMutex()
|
|
||||||
{
|
|
||||||
Assert( (size_t)this % 128 == 0 && sizeof(*this) == 128 );
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
uint8 pad[128-sizeof(CThreadFastMutex)];
|
|
||||||
} ALIGN128_POST;
|
|
||||||
|
|
||||||
#else
|
#else
|
||||||
typedef CThreadMutex CThreadFastMutex;
|
typedef CThreadMutex CThreadFastMutex;
|
||||||
@ -866,7 +724,7 @@ private:
|
|||||||
MUTEX_TYPE &m_lock;
|
MUTEX_TYPE &m_lock;
|
||||||
|
|
||||||
// Disallow copying
|
// Disallow copying
|
||||||
CAutoLockT<MUTEX_TYPE>( const CAutoLockT<MUTEX_TYPE> & );
|
CAutoLockT( const CAutoLockT<MUTEX_TYPE> & );
|
||||||
CAutoLockT<MUTEX_TYPE> &operator=( const CAutoLockT<MUTEX_TYPE> & );
|
CAutoLockT<MUTEX_TYPE> &operator=( const CAutoLockT<MUTEX_TYPE> & );
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -878,8 +736,6 @@ template <int size> struct CAutoLockTypeDeducer {};
|
|||||||
template <> struct CAutoLockTypeDeducer<sizeof(CThreadMutex)> { typedef CThreadMutex Type_t; };
|
template <> struct CAutoLockTypeDeducer<sizeof(CThreadMutex)> { typedef CThreadMutex Type_t; };
|
||||||
template <> struct CAutoLockTypeDeducer<sizeof(CThreadNullMutex)> { typedef CThreadNullMutex Type_t; };
|
template <> struct CAutoLockTypeDeducer<sizeof(CThreadNullMutex)> { typedef CThreadNullMutex Type_t; };
|
||||||
#if !defined(THREAD_PROFILER)
|
#if !defined(THREAD_PROFILER)
|
||||||
template <> struct CAutoLockTypeDeducer<sizeof(CThreadFastMutex)> { typedef CThreadFastMutex Type_t; };
|
|
||||||
template <> struct CAutoLockTypeDeducer<sizeof(CAlignedThreadFastMutex)> { typedef CAlignedThreadFastMutex Type_t; };
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define AUTO_LOCK_( type, mutex ) \
|
#define AUTO_LOCK_( type, mutex ) \
|
||||||
@ -893,7 +749,7 @@ template<typename T> T strip_cv_quals_for_mutex(volatile T&);
|
|||||||
template<typename T> T strip_cv_quals_for_mutex(const volatile T&);
|
template<typename T> T strip_cv_quals_for_mutex(const volatile T&);
|
||||||
|
|
||||||
#define AUTO_LOCK( mutex ) \
|
#define AUTO_LOCK( mutex ) \
|
||||||
AUTO_LOCK_( typeof(::strip_cv_quals_for_mutex(mutex)), mutex )
|
AUTO_LOCK_( __typeof__(::strip_cv_quals_for_mutex(mutex)), mutex )
|
||||||
|
|
||||||
#else // GNUC
|
#else // GNUC
|
||||||
|
|
||||||
@ -1087,35 +943,15 @@ inline int ThreadWaitForEvents( int nEvents, CThreadEvent * const *pEvents, bool
|
|||||||
//
|
//
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
|
|
||||||
class PLATFORM_CLASS CThreadRWLock
|
class CThreadRWLock
|
||||||
{
|
{
|
||||||
|
std::shared_timed_mutex mutex{};
|
||||||
public:
|
public:
|
||||||
CThreadRWLock();
|
void LockForRead() { mutex.lock_shared(); }
|
||||||
|
void UnlockRead() { mutex.unlock_shared(); }
|
||||||
|
|
||||||
void LockForRead();
|
void LockForWrite() { mutex.lock(); }
|
||||||
void UnlockRead();
|
void UnlockWrite() { mutex.unlock(); }
|
||||||
void LockForWrite();
|
|
||||||
void UnlockWrite();
|
|
||||||
|
|
||||||
void LockForRead() const { const_cast<CThreadRWLock *>(this)->LockForRead(); }
|
|
||||||
void UnlockRead() const { const_cast<CThreadRWLock *>(this)->UnlockRead(); }
|
|
||||||
void LockForWrite() const { const_cast<CThreadRWLock *>(this)->LockForWrite(); }
|
|
||||||
void UnlockWrite() const { const_cast<CThreadRWLock *>(this)->UnlockWrite(); }
|
|
||||||
|
|
||||||
private:
|
|
||||||
void WaitForRead();
|
|
||||||
|
|
||||||
#ifdef WIN32
|
|
||||||
CThreadFastMutex m_mutex;
|
|
||||||
#else
|
|
||||||
CThreadMutex m_mutex;
|
|
||||||
#endif
|
|
||||||
CThreadEvent m_CanWrite;
|
|
||||||
CThreadEvent m_CanRead;
|
|
||||||
|
|
||||||
int m_nWriters;
|
|
||||||
int m_nActiveReaders;
|
|
||||||
int m_nPendingReaders;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
@ -1124,41 +960,107 @@ private:
|
|||||||
//
|
//
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
|
|
||||||
class ALIGN8 PLATFORM_CLASS CThreadSpinRWLock
|
class CThreadSpinRWLock
|
||||||
{
|
{
|
||||||
|
std::atomic<uint32_t> lock{};
|
||||||
|
std::atomic_bool writePending{};
|
||||||
|
|
||||||
|
static constexpr uint32_t WRITE_MODE = (std::numeric_limits<uint32_t>::max)();
|
||||||
|
static constexpr uint32_t READ_MAX = WRITE_MODE - 1;
|
||||||
public:
|
public:
|
||||||
CThreadSpinRWLock() { COMPILE_TIME_ASSERT( sizeof( LockInfo_t ) == sizeof( int64 ) ); Assert( (intp)this % 8 == 0 ); memset( this, 0, sizeof( *this ) ); }
|
|
||||||
|
|
||||||
bool TryLockForWrite();
|
bool TryLockForWrite()
|
||||||
bool TryLockForRead();
|
|
||||||
|
|
||||||
void LockForRead();
|
|
||||||
void UnlockRead();
|
|
||||||
void LockForWrite();
|
|
||||||
void UnlockWrite();
|
|
||||||
|
|
||||||
bool TryLockForWrite() const { return const_cast<CThreadSpinRWLock *>(this)->TryLockForWrite(); }
|
|
||||||
bool TryLockForRead() const { return const_cast<CThreadSpinRWLock *>(this)->TryLockForRead(); }
|
|
||||||
void LockForRead() const { const_cast<CThreadSpinRWLock *>(this)->LockForRead(); }
|
|
||||||
void UnlockRead() const { const_cast<CThreadSpinRWLock *>(this)->UnlockRead(); }
|
|
||||||
void LockForWrite() const { const_cast<CThreadSpinRWLock *>(this)->LockForWrite(); }
|
|
||||||
void UnlockWrite() const { const_cast<CThreadSpinRWLock *>(this)->UnlockWrite(); }
|
|
||||||
|
|
||||||
private:
|
|
||||||
struct LockInfo_t
|
|
||||||
{
|
{
|
||||||
uint32 m_writerId;
|
uint32_t was = lock;
|
||||||
int m_nReaders;
|
|
||||||
|
if ( was == 0 )
|
||||||
|
{
|
||||||
|
return lock.compare_exchange_strong( was, WRITE_MODE );
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TryLockForRead()
|
||||||
|
{
|
||||||
|
uint32_t was = lock;
|
||||||
|
|
||||||
|
if ( was == 0 && writePending.load() )
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if ( was < READ_MAX )
|
||||||
|
return lock.compare_exchange_strong( was, was + 1 );
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void LockForRead()
|
||||||
|
{
|
||||||
|
while ( true )
|
||||||
|
{
|
||||||
|
uint32_t was = lock;
|
||||||
|
|
||||||
|
if ( was == 0 && writePending.load() )
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if ( was < READ_MAX )
|
||||||
|
{
|
||||||
|
if ( lock.compare_exchange_weak( was, was + 1 ) )
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void UnlockRead()
|
||||||
|
{
|
||||||
|
while ( true )
|
||||||
|
{
|
||||||
|
uint32_t was = lock;
|
||||||
|
|
||||||
|
if ( was == 0 )
|
||||||
|
return;
|
||||||
|
|
||||||
|
if ( was <= READ_MAX )
|
||||||
|
{
|
||||||
|
if ( lock.compare_exchange_weak( was, was - 1 ) )
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void LockForWrite()
|
||||||
|
{
|
||||||
|
writePending.exchange(true);
|
||||||
|
|
||||||
|
while ( true )
|
||||||
|
{
|
||||||
|
uint32_t was = lock;
|
||||||
|
|
||||||
|
if ( was == 0 )
|
||||||
|
{
|
||||||
|
if ( lock.compare_exchange_weak( was, WRITE_MODE ) )
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void UnlockWrite()
|
||||||
|
{
|
||||||
|
writePending.store(false);
|
||||||
|
|
||||||
|
while ( true )
|
||||||
|
{
|
||||||
|
uint32_t was = lock;
|
||||||
|
|
||||||
|
if ( was != WRITE_MODE )
|
||||||
|
return;
|
||||||
|
|
||||||
|
if ( lock.compare_exchange_weak( was, 0 ) )
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
bool AssignIf( const LockInfo_t &newValue, const LockInfo_t &comperand );
|
|
||||||
bool TryLockForWrite( const uint32 threadId );
|
|
||||||
void SpinLockForWrite( const uint32 threadId );
|
|
||||||
|
|
||||||
volatile LockInfo_t m_lockInfo;
|
|
||||||
CInterlockedInt m_nWriters;
|
|
||||||
} ALIGN8_POST;
|
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
//
|
//
|
||||||
// A thread wrapper similar to a Java thread.
|
// A thread wrapper similar to a Java thread.
|
||||||
@ -1506,257 +1408,6 @@ public:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// CThreadMutex. Inlining to reduce overhead and to allow client code
|
|
||||||
// to decide debug status (tracing)
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
typedef struct _RTL_CRITICAL_SECTION RTL_CRITICAL_SECTION;
|
|
||||||
typedef RTL_CRITICAL_SECTION CRITICAL_SECTION;
|
|
||||||
|
|
||||||
#ifndef _X360
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
void __declspec(dllimport) __stdcall InitializeCriticalSection(CRITICAL_SECTION *);
|
|
||||||
void __declspec(dllimport) __stdcall EnterCriticalSection(CRITICAL_SECTION *);
|
|
||||||
void __declspec(dllimport) __stdcall LeaveCriticalSection(CRITICAL_SECTION *);
|
|
||||||
void __declspec(dllimport) __stdcall DeleteCriticalSection(CRITICAL_SECTION *);
|
|
||||||
};
|
|
||||||
#endif
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline void CThreadMutex::Lock()
|
|
||||||
{
|
|
||||||
#ifdef THREAD_MUTEX_TRACING_ENABLED
|
|
||||||
uint thisThreadID = ThreadGetCurrentId();
|
|
||||||
if ( m_bTrace && m_currentOwnerID && ( m_currentOwnerID != thisThreadID ) )
|
|
||||||
Msg( "Thread %u about to wait for lock %p owned by %u\n", ThreadGetCurrentId(), (CRITICAL_SECTION *)&m_CriticalSection, m_currentOwnerID );
|
|
||||||
#endif
|
|
||||||
|
|
||||||
VCRHook_EnterCriticalSection((CRITICAL_SECTION *)&m_CriticalSection);
|
|
||||||
|
|
||||||
#ifdef THREAD_MUTEX_TRACING_ENABLED
|
|
||||||
if (m_lockCount == 0)
|
|
||||||
{
|
|
||||||
// we now own it for the first time. Set owner information
|
|
||||||
m_currentOwnerID = thisThreadID;
|
|
||||||
if ( m_bTrace )
|
|
||||||
Msg( "Thread %u now owns lock %p\n", m_currentOwnerID, (CRITICAL_SECTION *)&m_CriticalSection );
|
|
||||||
}
|
|
||||||
m_lockCount++;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline void CThreadMutex::Unlock()
|
|
||||||
{
|
|
||||||
#ifdef THREAD_MUTEX_TRACING_ENABLED
|
|
||||||
AssertMsg( m_lockCount >= 1, "Invalid unlock of thread lock" );
|
|
||||||
m_lockCount--;
|
|
||||||
if (m_lockCount == 0)
|
|
||||||
{
|
|
||||||
if ( m_bTrace )
|
|
||||||
Msg( "Thread %u releasing lock %p\n", m_currentOwnerID, (CRITICAL_SECTION *)&m_CriticalSection );
|
|
||||||
m_currentOwnerID = 0;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
LeaveCriticalSection((CRITICAL_SECTION *)&m_CriticalSection);
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline bool CThreadMutex::AssertOwnedByCurrentThread()
|
|
||||||
{
|
|
||||||
#ifdef THREAD_MUTEX_TRACING_ENABLED
|
|
||||||
if (ThreadGetCurrentId() == m_currentOwnerID)
|
|
||||||
return true;
|
|
||||||
AssertMsg3( 0, "Expected thread %u as owner of lock %p, but %u owns", ThreadGetCurrentId(), (CRITICAL_SECTION *)&m_CriticalSection, m_currentOwnerID );
|
|
||||||
return false;
|
|
||||||
#else
|
|
||||||
return true;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline void CThreadMutex::SetTrace( bool bTrace )
|
|
||||||
{
|
|
||||||
#ifdef THREAD_MUTEX_TRACING_ENABLED
|
|
||||||
m_bTrace = bTrace;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
#elif defined(POSIX)
|
|
||||||
|
|
||||||
inline CThreadMutex::CThreadMutex()
|
|
||||||
{
|
|
||||||
// enable recursive locks as we need them
|
|
||||||
pthread_mutexattr_init( &m_Attr );
|
|
||||||
pthread_mutexattr_settype( &m_Attr, PTHREAD_MUTEX_RECURSIVE );
|
|
||||||
pthread_mutex_init( &m_Mutex, &m_Attr );
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline CThreadMutex::~CThreadMutex()
|
|
||||||
{
|
|
||||||
pthread_mutex_destroy( &m_Mutex );
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline void CThreadMutex::Lock()
|
|
||||||
{
|
|
||||||
pthread_mutex_lock( &m_Mutex );
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline void CThreadMutex::Unlock()
|
|
||||||
{
|
|
||||||
pthread_mutex_unlock( &m_Mutex );
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline bool CThreadMutex::AssertOwnedByCurrentThread()
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------
|
|
||||||
|
|
||||||
inline void CThreadMutex::SetTrace(bool fTrace)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // POSIX
|
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// CThreadRWLock inline functions
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
inline CThreadRWLock::CThreadRWLock()
|
|
||||||
: m_CanRead( true ),
|
|
||||||
m_nWriters( 0 ),
|
|
||||||
m_nActiveReaders( 0 ),
|
|
||||||
m_nPendingReaders( 0 )
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void CThreadRWLock::LockForRead()
|
|
||||||
{
|
|
||||||
m_mutex.Lock();
|
|
||||||
if ( m_nWriters)
|
|
||||||
{
|
|
||||||
WaitForRead();
|
|
||||||
}
|
|
||||||
m_nActiveReaders++;
|
|
||||||
m_mutex.Unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void CThreadRWLock::UnlockRead()
|
|
||||||
{
|
|
||||||
m_mutex.Lock();
|
|
||||||
m_nActiveReaders--;
|
|
||||||
if ( m_nActiveReaders == 0 && m_nWriters != 0 )
|
|
||||||
{
|
|
||||||
m_CanWrite.Set();
|
|
||||||
}
|
|
||||||
m_mutex.Unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// CThreadSpinRWLock inline functions
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
inline bool CThreadSpinRWLock::AssignIf( const LockInfo_t &newValue, const LockInfo_t &comperand )
|
|
||||||
{
|
|
||||||
return ThreadInterlockedAssignIf64( (int64 *)&m_lockInfo, *((int64 *)&newValue), *((int64 *)&comperand) );
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool CThreadSpinRWLock::TryLockForWrite( const uint32 threadId )
|
|
||||||
{
|
|
||||||
// In order to grab a write lock, there can be no readers and no owners of the write lock
|
|
||||||
if ( m_lockInfo.m_nReaders > 0 || ( m_lockInfo.m_writerId && m_lockInfo.m_writerId != threadId ) )
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
static const LockInfo_t oldValue = { 0, 0 };
|
|
||||||
LockInfo_t newValue = { threadId, 0 };
|
|
||||||
const bool bSuccess = AssignIf( newValue, oldValue );
|
|
||||||
#if defined(_X360)
|
|
||||||
if ( bSuccess )
|
|
||||||
{
|
|
||||||
// X360TBD: Serious perf implications. Not Yet. __sync();
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
return bSuccess;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool CThreadSpinRWLock::TryLockForWrite()
|
|
||||||
{
|
|
||||||
m_nWriters++;
|
|
||||||
if ( !TryLockForWrite( ThreadGetCurrentId() ) )
|
|
||||||
{
|
|
||||||
m_nWriters--;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool CThreadSpinRWLock::TryLockForRead()
|
|
||||||
{
|
|
||||||
if ( m_nWriters != 0 )
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
// In order to grab a write lock, the number of readers must not change and no thread can own the write
|
|
||||||
LockInfo_t oldValue;
|
|
||||||
LockInfo_t newValue;
|
|
||||||
|
|
||||||
oldValue.m_nReaders = m_lockInfo.m_nReaders;
|
|
||||||
oldValue.m_writerId = 0;
|
|
||||||
newValue.m_nReaders = oldValue.m_nReaders + 1;
|
|
||||||
newValue.m_writerId = 0;
|
|
||||||
|
|
||||||
const bool bSuccess = AssignIf( newValue, oldValue );
|
|
||||||
#if defined(_X360)
|
|
||||||
if ( bSuccess )
|
|
||||||
{
|
|
||||||
// X360TBD: Serious perf implications. Not Yet. __sync();
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
return bSuccess;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void CThreadSpinRWLock::LockForWrite()
|
|
||||||
{
|
|
||||||
const uint32 threadId = ThreadGetCurrentId();
|
|
||||||
|
|
||||||
m_nWriters++;
|
|
||||||
|
|
||||||
if ( !TryLockForWrite( threadId ) )
|
|
||||||
{
|
|
||||||
ThreadPause();
|
|
||||||
SpinLockForWrite( threadId );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// read data from a memory address
|
// read data from a memory address
|
||||||
template<class T> FORCEINLINE T ReadVolatileMemory( T const *pPtr )
|
template<class T> FORCEINLINE T ReadVolatileMemory( T const *pPtr )
|
||||||
{
|
{
|
||||||
|
Loading…
x
Reference in New Issue
Block a user