#include "stdafx.h" #include "sys_lwmutex.h" #include "Emu/IdManager.h" #include "Emu/Cell/ErrorCodes.h" #include "Emu/Cell/PPUThread.h" LOG_CHANNEL(sys_lwmutex); error_code _sys_lwmutex_create(ppu_thread& ppu, vm::ptr lwmutex_id, u32 protocol, vm::ptr control, s32 has_name, u64 name) { vm::temporary_unlock(ppu); sys_lwmutex.warning(u8"_sys_lwmutex_create(lwmutex_id=*0x%x, protocol=0x%x, control=*0x%x, has_name=0x%x, name=0x%llx (“%s”))", lwmutex_id, protocol, control, has_name, name, lv2_obj::name64(std::bit_cast>(name))); if (protocol != SYS_SYNC_FIFO && protocol != SYS_SYNC_RETRY && protocol != SYS_SYNC_PRIORITY) { sys_lwmutex.error("_sys_lwmutex_create(): unknown protocol (0x%x)", protocol); return CELL_EINVAL; } if (!(has_name < 0)) { name = 0; } if (const u32 id = idm::make(protocol, control, name)) { *lwmutex_id = id; return CELL_OK; } return CELL_EAGAIN; } error_code _sys_lwmutex_destroy(ppu_thread& ppu, u32 lwmutex_id) { vm::temporary_unlock(ppu); sys_lwmutex.warning("_sys_lwmutex_destroy(lwmutex_id=0x%x)", lwmutex_id); auto mutex = idm::get(lwmutex_id); if (!mutex) { return CELL_ESRCH; } while (true) { if (std::scoped_lock lock(mutex->mutex); mutex->sq.empty()) { // Set "destroyed" bit if (mutex->lwcond_waiters.fetch_or(INT32_MIN) & 0x7fff'ffff) { // Deschedule if waiters were found lv2_obj::sleep(ppu); } } else { return CELL_EBUSY; } // Wait for all lwcond waiters to quit if (const s32 old = mutex->lwcond_waiters; old != INT32_MIN) { if (old >= 0) { // Sleep queue is no longer empty // Was set to positive value to announce it continue; } mutex->lwcond_waiters.wait(old); if (ppu.is_stopped()) { return 0; } } else { break; } } if (!idm::remove_verify(lwmutex_id, std::move(mutex))) { // Other thread has destroyed the lwmutex earlier return CELL_ESRCH; } return CELL_OK; } error_code _sys_lwmutex_lock(ppu_thread& ppu, u32 lwmutex_id, u64 timeout) { vm::temporary_unlock(ppu); sys_lwmutex.trace("_sys_lwmutex_lock(lwmutex_id=0x%x, timeout=0x%llx)", lwmutex_id, timeout); ppu.gpr[3] = CELL_OK; const auto mutex = idm::get(lwmutex_id, [&](lv2_lwmutex& mutex) { if (mutex.signaled.try_dec(0)) { return true; } std::lock_guard lock(mutex.mutex); auto [old, _] = mutex.signaled.fetch_op([](s32& value) { if (value) { value = 0; return true; } return false; }); if (old) { if (old == INT32_MIN) { ppu.gpr[3] = CELL_EBUSY; } return true; } if (!mutex.add_waiter(&ppu)) { ppu.gpr[3] = CELL_ESRCH; return true; } mutex.sleep(ppu, timeout); return false; }); if (!mutex) { return CELL_ESRCH; } if (mutex.ret) { return not_an_error(ppu.gpr[3]); } while (!ppu.state.test_and_reset(cpu_flag::signal)) { if (ppu.is_stopped()) { return 0; } if (timeout) { if (lv2_obj::wait_timeout(timeout, &ppu)) { std::lock_guard lock(mutex->mutex); if (!mutex->unqueue(mutex->sq, &ppu)) { timeout = 0; continue; } ppu.gpr[3] = CELL_ETIMEDOUT; break; } } else { thread_ctrl::wait(); } } return not_an_error(ppu.gpr[3]); } error_code _sys_lwmutex_trylock(ppu_thread& ppu, u32 lwmutex_id) { vm::temporary_unlock(ppu); sys_lwmutex.trace("_sys_lwmutex_trylock(lwmutex_id=0x%x)", lwmutex_id); const auto mutex = idm::check(lwmutex_id, [&](lv2_lwmutex& mutex) { auto [_, ok] = mutex.signaled.fetch_op([](s32& value) { if (value & 1) { value = 0; return true; } return false; }); return ok; }); if (!mutex) { return CELL_ESRCH; } if (!mutex.ret) { return not_an_error(CELL_EBUSY); } return CELL_OK; } error_code _sys_lwmutex_unlock(ppu_thread& ppu, u32 lwmutex_id) { vm::temporary_unlock(ppu); sys_lwmutex.trace("_sys_lwmutex_unlock(lwmutex_id=0x%x)", lwmutex_id); const auto mutex = idm::check(lwmutex_id, [&](lv2_lwmutex& mutex) { std::lock_guard lock(mutex.mutex); if (const auto cpu = mutex.schedule(mutex.sq, mutex.protocol)) { mutex.awake(cpu); return; } mutex.signaled |= 1; }); if (!mutex) { return CELL_ESRCH; } return CELL_OK; } error_code _sys_lwmutex_unlock2(ppu_thread& ppu, u32 lwmutex_id) { vm::temporary_unlock(ppu); sys_lwmutex.warning("_sys_lwmutex_unlock2(lwmutex_id=0x%x)", lwmutex_id); const auto mutex = idm::check(lwmutex_id, [&](lv2_lwmutex& mutex) { std::lock_guard lock(mutex.mutex); if (const auto cpu = mutex.schedule(mutex.sq, mutex.protocol)) { static_cast(cpu)->gpr[3] = CELL_EBUSY; mutex.awake(cpu); return; } mutex.signaled |= INT32_MIN; }); if (!mutex) { return CELL_ESRCH; } return CELL_OK; }