#include "stdafx.h" #include "Emu/Memory/vm.h" #include "Emu/System.h" #include "Emu/IdManager.h" #include "Emu/Cell/ErrorCodes.h" #include "Emu/Cell/PPUThread.h" #include "sys_ppu_thread.h" #include "sys_event.h" #include "sys_mmapper.h" logs::channel sys_ppu_thread("sys_ppu_thread"); void _sys_ppu_thread_exit(ppu_thread& ppu, u64 errorcode) { vm::temporary_unlock(ppu); sys_ppu_thread.trace("_sys_ppu_thread_exit(errorcode=0x%llx)", errorcode); ppu.state += cpu_flag::exit; // Get joiner ID const u32 jid = ppu.joiner.fetch_op([](u32& value) { if (value == 0) { // Joinable, not joined value = -3; } else if (value != -1) { // Joinable, joined value = -2; } // Detached otherwise }); if (jid == -1) { // Delete detached thread and unqueue idm::remove(ppu.id); } else if (jid != 0) { writer_lock lock(id_manager::g_mutex); // Schedule joiner and unqueue lv2_obj::awake(*idm::check_unlocked(jid), -2); } // Unqueue lv2_obj::sleep(ppu); // Remove suspend state (TODO) ppu.state -= cpu_flag::suspend; } void sys_ppu_thread_yield(ppu_thread& ppu) { sys_ppu_thread.trace("sys_ppu_thread_yield()"); lv2_obj::awake(ppu, -4); } error_code sys_ppu_thread_join(ppu_thread& ppu, u32 thread_id, vm::ptr vptr) { vm::temporary_unlock(ppu); sys_ppu_thread.trace("sys_ppu_thread_join(thread_id=0x%x, vptr=*0x%x)", thread_id, vptr); const auto thread = idm::get(thread_id, [&](ppu_thread& thread) -> CellError { CellError result = thread.joiner.atomic_op([&](u32& value) -> CellError { if (value == -3) { value = -2; return CELL_EBUSY; } if (value == -2) { return CELL_ESRCH; } if (value) { return CELL_EINVAL; } // TODO: check precedence? if (&ppu == &thread) { return CELL_EDEADLK; } value = ppu.id; return {}; }); if (!result) { lv2_obj::sleep(ppu); } return result; }); if (!thread) { return CELL_ESRCH; } if (thread.ret && thread.ret != CELL_EBUSY) { return thread.ret; } // Wait for cleanup thread->join(); // Get the exit status from the register if (vptr) { ppu.test_state(); *vptr = thread->gpr[3]; } // Cleanup idm::remove(thread->id); return CELL_OK; } error_code sys_ppu_thread_detach(u32 thread_id) { sys_ppu_thread.trace("sys_ppu_thread_detach(thread_id=0x%x)", thread_id); const auto thread = idm::check(thread_id, [&](ppu_thread& thread) -> CellError { return thread.joiner.atomic_op([&](u32& value) -> CellError { if (value == -3) { value = -2; return CELL_EAGAIN; } if (value == -2) { return CELL_ESRCH; } if (value == -1) { return CELL_EINVAL; } if (value) { return CELL_EBUSY; } value = -1; return {}; }); }); if (!thread) { return CELL_ESRCH; } if (thread.ret && thread.ret != CELL_EAGAIN) { return thread.ret; } if (thread.ret == CELL_EAGAIN) { idm::remove(thread_id); } return CELL_OK; } void sys_ppu_thread_get_join_state(ppu_thread& ppu, vm::ptr isjoinable) { sys_ppu_thread.trace("sys_ppu_thread_get_join_state(isjoinable=*0x%x)", isjoinable); *isjoinable = ppu.joiner != -1; } error_code sys_ppu_thread_set_priority(ppu_thread& ppu, u32 thread_id, s32 prio) { sys_ppu_thread.trace("sys_ppu_thread_set_priority(thread_id=0x%x, prio=%d)", thread_id, prio); if (prio < 0 || prio > 3071) { return CELL_EINVAL; } const auto thread = idm::check(thread_id, [&](ppu_thread& thread) { if (thread.prio != prio && thread.prio.exchange(prio) != prio) { lv2_obj::awake(thread, prio); } }); if (!thread) { return CELL_ESRCH; } return CELL_OK; } error_code sys_ppu_thread_get_priority(u32 thread_id, vm::ptr priop) { sys_ppu_thread.trace("sys_ppu_thread_get_priority(thread_id=0x%x, priop=*0x%x)", thread_id, priop); const auto thread = idm::check(thread_id, [&](ppu_thread& thread) { *priop = thread.prio; }); if (!thread) { return CELL_ESRCH; } return CELL_OK; } error_code sys_ppu_thread_get_stack_information(ppu_thread& ppu, vm::ptr sp) { sys_ppu_thread.trace("sys_ppu_thread_get_stack_information(sp=*0x%x)", sp); sp->pst_addr = ppu.stack_addr; sp->pst_size = ppu.stack_size; return CELL_OK; } error_code sys_ppu_thread_stop(u32 thread_id) { sys_ppu_thread.todo("sys_ppu_thread_stop(thread_id=0x%x)", thread_id); const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } return CELL_OK; } error_code sys_ppu_thread_restart(u32 thread_id) { sys_ppu_thread.todo("sys_ppu_thread_restart(thread_id=0x%x)", thread_id); const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } return CELL_OK; } error_code _sys_ppu_thread_create(vm::ptr thread_id, vm::ptr param, u64 arg, u64 unk, s32 prio, u32 stacksize, u64 flags, vm::cptr threadname) { sys_ppu_thread.warning("_sys_ppu_thread_create(thread_id=*0x%x, param=*0x%x, arg=0x%llx, unk=0x%llx, prio=%d, stacksize=0x%x, flags=0x%llx, threadname=%s)", thread_id, param, arg, unk, prio, stacksize, flags, threadname); if (prio < 0 || prio > 3071) { return CELL_EINVAL; } if ((flags & 3) == 3) // Check two flags: joinable + interrupt not allowed { return CELL_EPERM; } const u32 tid = idm::import([&]() { auto ppu = std::make_shared(threadname ? threadname.get_ptr() : "", prio, stacksize); if ((flags & SYS_PPU_THREAD_CREATE_JOINABLE) != 0) { ppu->joiner = 0; } ppu->gpr[13] = param->tls.value(); if ((flags & SYS_PPU_THREAD_CREATE_INTERRUPT) == 0) { // Initialize thread entry point ppu->cmd_list ({ { ppu_cmd::set_args, 2 }, arg, unk, // Actually unknown { ppu_cmd::lle_call, param->entry.value() }, }); } else { // Save entry for further use (workaround) ppu->gpr[2] = param->entry.value(); } return ppu; }); if (!tid) { return CELL_EAGAIN; } *thread_id = tid; return CELL_OK; } error_code sys_ppu_thread_start(ppu_thread& ppu, u32 thread_id) { sys_ppu_thread.trace("sys_ppu_thread_start(thread_id=0x%x)", thread_id); const auto thread = idm::get(thread_id, [&](ppu_thread& thread) { lv2_obj::awake(thread, -2); }); if (!thread) { return CELL_ESRCH; } if (!thread->state.test_and_reset(cpu_flag::stop)) { // TODO: what happens there? return CELL_EPERM; } else { thread->notify(); // Dirty hack for sound: confirm the creation of _mxr000 event queue if (thread->m_name == "_cellsurMixerMain") { lv2_obj::sleep(ppu); while (!idm::select([](u32, lv2_event_queue& eq) { //some games do not set event queue name, though key seems constant for them return (eq.name == "_mxr000\0"_u64) || (eq.key == 0x8000cafe02460300); })) { thread_ctrl::wait_for(50000); } ppu.test_state(); } } return CELL_OK; } error_code sys_ppu_thread_rename(u32 thread_id, vm::cptr name) { sys_ppu_thread.todo("sys_ppu_thread_rename(thread_id=0x%x, name=%s)", thread_id, name); const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } return CELL_OK; } error_code sys_ppu_thread_recover_page_fault(u32 thread_id) { sys_ppu_thread.warning("sys_ppu_thread_recover_page_fault(thread_id=0x%x)", thread_id); const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } // We can only wake a thread if it is being suspended for a page fault. auto pf_events = fxm::get_always(); auto pf_event_ind = pf_events->events.begin(); for (auto event_ind = pf_events->events.begin(); event_ind != pf_events->events.end(); ++event_ind) { if (event_ind->thread_id == thread_id) { pf_event_ind = event_ind; break; } } if (pf_event_ind == pf_events->events.end()) { // if not found... return CELL_EINVAL; } pf_events->events.erase(pf_event_ind); lv2_obj::awake(*thread); return CELL_OK; } error_code sys_ppu_thread_get_page_fault_context(u32 thread_id, vm::ptr ctxt) { sys_ppu_thread.todo("sys_ppu_thread_get_page_fault_context(thread_id=0x%x, ctxt=*0x%x)", thread_id, ctxt); const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } // We can only get a context if the thread is being suspended for a page fault. auto pf_events = fxm::get_always(); bool found = false; for (const auto& ev : pf_events->events) { if (ev.thread_id == thread_id) { found = true; break; } } if (!found) { return CELL_EINVAL; } // TODO: Fill ctxt with proper information. return CELL_OK; }