2014-03-13 17:11:16 +01:00
|
|
|
#include "stdafx.h"
|
2014-06-02 19:27:24 +02:00
|
|
|
#include "Emu/Memory/Memory.h"
|
|
|
|
|
#include "Emu/System.h"
|
2014-03-13 17:11:16 +01:00
|
|
|
#include "Emu/SysCalls/SysCalls.h"
|
2014-09-30 21:06:04 +02:00
|
|
|
#include "Emu/Memory/atomic_type.h"
|
2014-07-21 00:30:16 +02:00
|
|
|
|
2014-06-25 00:38:34 +02:00
|
|
|
#include "sys_spinlock.h"
|
2014-03-13 17:11:16 +01:00
|
|
|
|
|
|
|
|
SysCallBase sys_spinlock("sys_spinlock");
|
|
|
|
|
|
2014-09-27 20:49:33 +02:00
|
|
|
void sys_spinlock_initialize(vm::ptr<atomic_t<u32>> lock)
|
2014-03-13 17:11:16 +01:00
|
|
|
{
|
2014-09-02 03:05:13 +02:00
|
|
|
sys_spinlock.Log("sys_spinlock_initialize(lock_addr=0x%x)", lock.addr());
|
2014-03-13 17:11:16 +01:00
|
|
|
|
2014-07-21 00:30:16 +02:00
|
|
|
// prx: set 0 and sync
|
2014-09-20 01:16:11 +02:00
|
|
|
lock->exchange(be_t<u32>::make(0));
|
2014-03-13 17:11:16 +01:00
|
|
|
}
|
|
|
|
|
|
2014-09-27 20:49:33 +02:00
|
|
|
void sys_spinlock_lock(vm::ptr<atomic_t<u32>> lock)
|
2014-03-13 17:11:16 +01:00
|
|
|
{
|
2014-09-02 03:05:13 +02:00
|
|
|
sys_spinlock.Log("sys_spinlock_lock(lock_addr=0x%x)", lock.addr());
|
2014-03-13 17:11:16 +01:00
|
|
|
|
2014-07-21 00:30:16 +02:00
|
|
|
// prx: exchange with 0xabadcafe, repeat until exchanged with 0
|
2015-01-13 15:54:36 +01:00
|
|
|
while (lock->exchange(be_t<u32>::make(0xabadcafe)).data())
|
2014-03-13 17:11:16 +01:00
|
|
|
{
|
2015-01-13 15:54:36 +01:00
|
|
|
while (lock->read_relaxed().data())
|
2014-07-21 00:30:16 +02:00
|
|
|
{
|
2014-07-21 16:42:43 +02:00
|
|
|
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())
|
|
|
|
|
{
|
2014-09-02 03:05:13 +02:00
|
|
|
sys_spinlock.Warning("sys_spinlock_lock(0x%x) aborted", lock.addr());
|
2014-07-21 00:30:16 +02:00
|
|
|
break;
|
|
|
|
|
}
|
2014-03-13 17:11:16 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-27 20:49:33 +02:00
|
|
|
s32 sys_spinlock_trylock(vm::ptr<atomic_t<u32>> lock)
|
2014-03-13 17:11:16 +01:00
|
|
|
{
|
2014-09-02 03:05:13 +02:00
|
|
|
sys_spinlock.Log("sys_spinlock_trylock(lock_addr=0x%x)", lock.addr());
|
2014-03-13 17:11:16 +01:00
|
|
|
|
2014-07-21 00:30:16 +02:00
|
|
|
// prx: exchange with 0xabadcafe, translate exchanged value
|
2015-01-13 15:54:36 +01:00
|
|
|
if (lock->exchange(be_t<u32>::make(0xabadcafe)).data())
|
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-09-27 20:49:33 +02:00
|
|
|
void sys_spinlock_unlock(vm::ptr<atomic_t<u32>> lock)
|
2014-03-13 17:11:16 +01:00
|
|
|
{
|
2014-09-02 03:05:13 +02:00
|
|
|
sys_spinlock.Log("sys_spinlock_unlock(lock_addr=0x%x)", lock.addr());
|
2014-03-13 17:11:16 +01:00
|
|
|
|
2014-07-21 00:30:16 +02:00
|
|
|
// prx: sync and set 0
|
2014-09-20 01:16:11 +02:00
|
|
|
lock->exchange(be_t<u32>::make(0));
|
2014-12-23 00:31:11 +01:00
|
|
|
}
|