#include "stdafx.h" #include "Utilities/Config.h" #include "Emu/Memory/Memory.h" #include "Emu/System.h" #include "Emu/IdManager.h" #include "Emu/Cell/ErrorCodes.h" #include "Emu/Cell/PPUThread.h" #include "sys_ppu_thread.h" LOG_CHANNEL(sys_ppu_thread); void _sys_ppu_thread_exit(PPUThread& ppu, u64 errorcode) { sys_ppu_thread.trace("_sys_ppu_thread_exit(errorcode=0x%llx)", errorcode); LV2_LOCK; // TODO: Should we really unlock mutexes? //// get all sys_mutex objects //for (auto& mutex : idm::get_all()) //{ // // unlock mutex if locked by this thread // if (mutex->owner.get() == &ppu) // { // mutex->unlock(lv2_lock); // } //} ppu.state += cpu_state::exit; // Delete detached thread if (!ppu.is_joinable) { idm::remove(ppu.id); } // Throw if this syscall was not called directly by the SC instruction (hack) if (ppu.GPR[11] != 41 || ppu.custom_task) { throw cpu_state::exit; } } void sys_ppu_thread_yield() { sys_ppu_thread.trace("sys_ppu_thread_yield()"); std::this_thread::yield(); } s32 sys_ppu_thread_join(PPUThread& ppu, u32 thread_id, vm::ptr vptr) { sys_ppu_thread.warning("sys_ppu_thread_join(thread_id=0x%x, vptr=*0x%x)", thread_id, vptr); LV2_LOCK; const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } if (!thread->is_joinable || thread->is_joining) { return CELL_EINVAL; } if (&ppu == thread.get()) { return CELL_EDEADLK; } // mark joining thread->is_joining = true; // join thread while (!(thread->state & cpu_state::exit)) { CHECK_EMU_STATUS; get_current_thread_cv().wait_for(lv2_lock, std::chrono::milliseconds(1)); } // get exit status from the register if (vptr) *vptr = thread->GPR[3]; // cleanup idm::remove(thread->id); return CELL_OK; } s32 sys_ppu_thread_detach(u32 thread_id) { sys_ppu_thread.warning("sys_ppu_thread_detach(thread_id=0x%x)", thread_id); LV2_LOCK; const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } if (!thread->is_joinable) { return CELL_EINVAL; } if (thread->is_joining) { return CELL_EBUSY; } // "detach" thread->is_joinable = false; return CELL_OK; } void sys_ppu_thread_get_join_state(PPUThread& ppu, vm::ptr isjoinable) { sys_ppu_thread.warning("sys_ppu_thread_get_join_state(isjoinable=*0x%x)", isjoinable); LV2_LOCK; *isjoinable = ppu.is_joinable; } s32 sys_ppu_thread_set_priority(u32 thread_id, s32 prio) { sys_ppu_thread.trace("sys_ppu_thread_set_priority(thread_id=0x%x, prio=%d)", thread_id, prio); LV2_LOCK; const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } if (prio < 0 || prio > 3071) { return CELL_EINVAL; } thread->prio = prio; return CELL_OK; } s32 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); LV2_LOCK; const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } *priop = thread->prio; return CELL_OK; } s32 sys_ppu_thread_get_stack_information(PPUThread& 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; } s32 sys_ppu_thread_stop(u32 thread_id) { sys_ppu_thread.todo("sys_ppu_thread_stop(thread_id=0x%x)", thread_id); LV2_LOCK; const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } //t->Stop(); return CELL_OK; } s32 sys_ppu_thread_restart(u32 thread_id) { sys_ppu_thread.todo("sys_ppu_thread_restart(thread_id=0x%x)", thread_id); LV2_LOCK; const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } //t->Stop(); //t->Run(); return CELL_OK; } u32 ppu_thread_create(u32 entry, u64 arg, s32 prio, u32 stacksize, const std::string& name, std::function task) { const auto ppu = idm::make_ptr(name); ppu->prio = prio; ppu->stack_size = stacksize; ppu->custom_task = std::move(task); ppu->cpu_init(); if (entry) { ppu->PC = vm::read32(entry); ppu->GPR[2] = vm::read32(entry + 4); // rtoc } ppu->GPR[3] = arg; ppu->state -= cpu_state::stop; ppu->lock_notify(); return ppu->id; } s32 _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=*0x%x)", thread_id, param, arg, unk, prio, stacksize, flags, threadname); LV2_LOCK; if (prio < 0 || prio > 3071) { return CELL_EINVAL; } const bool is_joinable = (flags & SYS_PPU_THREAD_CREATE_JOINABLE) != 0; const bool is_interrupt = (flags & SYS_PPU_THREAD_CREATE_INTERRUPT) != 0; if (is_joinable && is_interrupt) { return CELL_EPERM; } const auto ppu = idm::make_ptr(threadname ? threadname.get_ptr() : ""); ppu->prio = prio; ppu->stack_size = std::max(stacksize, 0x4000); ppu->cpu_init(); ppu->PC = vm::read32(param->entry); ppu->GPR[2] = vm::read32(param->entry + 4); // rtoc ppu->GPR[3] = arg; ppu->GPR[4] = unk; // actually unknown ppu->GPR[13] = param->tls; ppu->is_joinable = is_joinable; *thread_id = ppu->id; return CELL_OK; } s32 sys_ppu_thread_start(u32 thread_id) { sys_ppu_thread.warning("sys_ppu_thread_start(thread_id=0x%x)", thread_id); LV2_LOCK; const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } thread->state -= cpu_state::stop; thread->lock_notify(); return CELL_OK; } s32 sys_ppu_thread_rename(u32 thread_id, vm::cptr name) { sys_ppu_thread.todo("sys_ppu_thread_rename(thread_id=0x%x, name=*0x%x)", thread_id, name); LV2_LOCK; const auto thread = idm::get(thread_id); if (!thread) { return CELL_ESRCH; } return CELL_OK; }