2014-02-05 11:55:32 +00:00
|
|
|
#pragma once
|
2014-06-02 17:27:24 +00:00
|
|
|
#include "BEType.h"
|
2014-02-05 11:55:32 +00:00
|
|
|
|
|
|
|
extern void SM_Sleep();
|
2014-02-26 11:27:06 +00:00
|
|
|
extern size_t SM_GetCurrentThreadId();
|
2014-02-05 11:55:32 +00:00
|
|
|
extern u32 SM_GetCurrentCPUThreadId();
|
|
|
|
extern be_t<u32> SM_GetCurrentCPUThreadIdBE();
|
|
|
|
|
|
|
|
enum SMutexResult
|
|
|
|
{
|
|
|
|
SMR_OK = 0, // succeeded (lock, trylock, unlock)
|
|
|
|
SMR_FAILED, // failed (trylock, unlock)
|
|
|
|
SMR_DEADLOCK, // mutex reached deadlock (lock, trylock)
|
|
|
|
SMR_SIGNAL = SMR_DEADLOCK, // unlock can be used for signaling specific thread
|
|
|
|
SMR_PERMITTED, // not owner of the mutex (unlock)
|
|
|
|
SMR_ABORT, // emulator has been stopped (lock, trylock, unlock)
|
|
|
|
SMR_DESTROYED, // mutex has been destroyed (lock, trylock, unlock)
|
|
|
|
SMR_TIMEOUT, // timed out (lock)
|
|
|
|
};
|
|
|
|
|
|
|
|
template
|
|
|
|
<
|
|
|
|
typename T,
|
2014-02-26 11:27:06 +00:00
|
|
|
u64 free_value = 0,
|
2014-03-31 10:04:34 +00:00
|
|
|
u64 dead_value = 0xffffffff,
|
2014-02-23 16:52:52 +00:00
|
|
|
void (*wait)() = SM_Sleep
|
2014-02-05 11:55:32 +00:00
|
|
|
>
|
|
|
|
class SMutexBase
|
|
|
|
{
|
2014-03-01 23:50:47 +00:00
|
|
|
static_assert(sizeof(T) == sizeof(std::atomic<T>), "Invalid SMutexBase type");
|
2014-02-05 11:55:32 +00:00
|
|
|
std::atomic<T> owner;
|
|
|
|
|
|
|
|
public:
|
|
|
|
SMutexBase()
|
|
|
|
: owner((T)free_value)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-02-20 02:16:17 +00:00
|
|
|
void initialize()
|
|
|
|
{
|
|
|
|
(T&)owner = free_value;
|
|
|
|
}
|
|
|
|
|
2014-02-05 11:55:32 +00:00
|
|
|
~SMutexBase()
|
|
|
|
{
|
|
|
|
lock((T)dead_value);
|
|
|
|
owner = (T)dead_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
__forceinline T GetOwner() const
|
|
|
|
{
|
|
|
|
return (T&)owner;
|
|
|
|
}
|
|
|
|
|
2014-02-20 02:16:17 +00:00
|
|
|
__forceinline T GetFreeValue() const
|
|
|
|
{
|
|
|
|
return (T)free_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
__forceinline T GetDeadValue() const
|
|
|
|
{
|
|
|
|
return (T)dead_value;
|
|
|
|
}
|
|
|
|
|
2014-02-05 11:55:32 +00:00
|
|
|
SMutexResult trylock(T tid)
|
|
|
|
{
|
2014-02-13 11:13:05 +00:00
|
|
|
if (Emu.IsStopped())
|
|
|
|
{
|
|
|
|
return SMR_ABORT;
|
|
|
|
}
|
2014-02-05 11:55:32 +00:00
|
|
|
T old = (T)free_value;
|
|
|
|
|
|
|
|
if (!owner.compare_exchange_strong(old, tid))
|
|
|
|
{
|
|
|
|
if (old == tid)
|
|
|
|
{
|
|
|
|
return SMR_DEADLOCK;
|
|
|
|
}
|
|
|
|
if (old == (T)dead_value)
|
|
|
|
{
|
|
|
|
return SMR_DESTROYED;
|
|
|
|
}
|
|
|
|
return SMR_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SMR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
SMutexResult unlock(T tid, T to = (T)free_value)
|
|
|
|
{
|
2014-02-13 11:13:05 +00:00
|
|
|
if (Emu.IsStopped())
|
|
|
|
{
|
|
|
|
return SMR_ABORT;
|
|
|
|
}
|
2014-02-05 11:55:32 +00:00
|
|
|
T old = tid;
|
|
|
|
|
|
|
|
if (!owner.compare_exchange_strong(old, to))
|
|
|
|
{
|
|
|
|
if (old == (T)free_value)
|
|
|
|
{
|
|
|
|
return SMR_FAILED;
|
|
|
|
}
|
|
|
|
if (old == (T)dead_value)
|
|
|
|
{
|
|
|
|
return SMR_DESTROYED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SMR_PERMITTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SMR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
SMutexResult lock(T tid, u64 timeout = 0)
|
|
|
|
{
|
|
|
|
u64 counter = 0;
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
switch (SMutexResult res = trylock(tid))
|
|
|
|
{
|
|
|
|
case SMR_FAILED: break;
|
|
|
|
default: return res;
|
|
|
|
}
|
|
|
|
|
2014-06-20 11:00:36 +00:00
|
|
|
if (wait) wait();
|
2014-02-05 11:55:32 +00:00
|
|
|
|
|
|
|
if (timeout && counter++ > timeout)
|
|
|
|
{
|
|
|
|
return SMR_TIMEOUT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-06-20 11:00:36 +00:00
|
|
|
template<typename T, typename Tid, Tid (get_tid)()>
|
2014-02-05 11:55:32 +00:00
|
|
|
class SMutexLockerBase
|
|
|
|
{
|
2014-06-20 11:00:36 +00:00
|
|
|
T& sm;
|
2014-02-06 22:55:48 +00:00
|
|
|
public:
|
2014-06-20 11:00:36 +00:00
|
|
|
const Tid tid;
|
2014-02-05 11:55:32 +00:00
|
|
|
|
2014-06-20 11:00:36 +00:00
|
|
|
__forceinline SMutexLockerBase(T& _sm)
|
2014-02-05 11:55:32 +00:00
|
|
|
: sm(_sm)
|
|
|
|
, tid(get_tid())
|
|
|
|
{
|
2014-02-09 11:11:48 +00:00
|
|
|
if (!tid)
|
|
|
|
{
|
2014-03-04 19:18:17 +00:00
|
|
|
if (!Emu.IsStopped())
|
|
|
|
{
|
|
|
|
ConLog.Error("SMutexLockerBase: thread id == 0");
|
|
|
|
Emu.Pause();
|
|
|
|
}
|
|
|
|
return;
|
2014-02-09 11:11:48 +00:00
|
|
|
}
|
2014-02-05 11:55:32 +00:00
|
|
|
sm.lock(tid);
|
|
|
|
}
|
|
|
|
|
2014-03-31 10:04:34 +00:00
|
|
|
__forceinline ~SMutexLockerBase()
|
2014-02-05 11:55:32 +00:00
|
|
|
{
|
2014-03-04 19:18:17 +00:00
|
|
|
if (tid) sm.unlock(tid);
|
2014-02-05 11:55:32 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-02-26 11:27:06 +00:00
|
|
|
typedef SMutexBase<size_t>
|
2014-02-05 11:55:32 +00:00
|
|
|
SMutexGeneral;
|
|
|
|
typedef SMutexBase<u32>
|
|
|
|
SMutex;
|
|
|
|
typedef SMutexBase<be_t<u32>>
|
|
|
|
SMutexBE;
|
|
|
|
|
2014-06-20 11:00:36 +00:00
|
|
|
typedef SMutexLockerBase<SMutexGeneral, size_t, SM_GetCurrentThreadId>
|
2014-02-05 11:55:32 +00:00
|
|
|
SMutexGeneralLocker;
|
2014-06-20 11:00:36 +00:00
|
|
|
typedef SMutexLockerBase<SMutex, u32, SM_GetCurrentCPUThreadId>
|
2014-02-05 11:55:32 +00:00
|
|
|
SMutexLocker;
|
2014-06-20 11:00:36 +00:00
|
|
|
typedef SMutexLockerBase<SMutexBE, be_t<u32>, SM_GetCurrentCPUThreadIdBE>
|
2014-02-23 16:52:52 +00:00
|
|
|
SMutexBELocker;
|