rpcsx/rpcs3/Emu/SysCalls/lv2/sys_spinlock.cpp

63 lines
1.5 KiB
C++
Raw Normal View History

2014-03-13 17:11:16 +01:00
#include "stdafx.h"
#include "Utilities/Log.h"
#include "Emu/Memory/Memory.h"
#include "Emu/System.h"
#include "Emu/Cell/PPUThread.h"
2014-03-13 17:11:16 +01:00
#include "Emu/SysCalls/SysCalls.h"
2014-07-21 00:30:16 +02:00
#include "sys_spinlock.h"
2014-03-13 17:11:16 +01:00
SysCallBase sys_spinlock("sys_spinlock");
2014-07-21 00:30:16 +02:00
void sys_spinlock_initialize(mem_ptr_t<std::atomic<be_t<u32>>> lock)
2014-03-13 17:11:16 +01:00
{
sys_spinlock.Log("sys_spinlock_initialize(lock_addr=0x%x)", lock.GetAddr());
2014-07-21 00:30:16 +02:00
// prx: set 0 and sync
*lock = be_t<u32>::MakeFromBE(0);
2014-03-13 17:11:16 +01:00
}
2014-07-21 00:30:16 +02:00
void sys_spinlock_lock(mem_ptr_t<std::atomic<be_t<u32>>> lock)
2014-03-13 17:11:16 +01:00
{
sys_spinlock.Log("sys_spinlock_lock(lock_addr=0x%x)", lock.GetAddr());
2014-07-21 00:30:16 +02:00
// prx: exchange with 0xabadcafe, repeat until exchanged with 0
while (lock->exchange(be_t<u32>::MakeFromBE(se32(0xabadcafe))).ToBE())
2014-03-13 17:11:16 +01:00
{
2014-07-21 00:30:16 +02:00
while (lock->load(std::memory_order_relaxed).ToBE())
{
std::this_thread::sleep_for(std::chrono::milliseconds(1)); // hack
2014-07-21 00:30:16 +02:00
if (Emu.IsStopped())
{
break;
}
}
if (Emu.IsStopped())
{
LOG_WARNING(HLE, "sys_spinlock_lock(0x%x) aborted", lock.GetAddr());
break;
}
2014-03-13 17:11:16 +01:00
}
}
2014-07-21 00:30:16 +02:00
s32 sys_spinlock_trylock(mem_ptr_t<std::atomic<be_t<u32>>> lock)
2014-03-13 17:11:16 +01:00
{
sys_spinlock.Log("sys_spinlock_trylock(lock_addr=0x%x)", lock.GetAddr());
2014-07-21 00:30:16 +02:00
// prx: exchange with 0xabadcafe, translate exchanged value
if (lock->exchange(be_t<u32>::MakeFromBE(se32(0xabadcafe))).ToBE())
2014-03-13 17:11:16 +01:00
{
2014-07-21 00:30:16 +02:00
return CELL_EBUSY;
2014-03-13 17:11:16 +01:00
}
return CELL_OK;
}
2014-07-21 00:30:16 +02:00
void sys_spinlock_unlock(mem_ptr_t<std::atomic<be_t<u32>>> lock)
2014-03-13 17:11:16 +01:00
{
sys_spinlock.Log("sys_spinlock_unlock(lock_addr=0x%x)", lock.GetAddr());
2014-07-21 00:30:16 +02:00
// prx: sync and set 0
*lock = be_t<u32>::MakeFromBE(0);
2014-03-13 17:11:16 +01:00
}