#include "stdafx.h" #include "sys_interrupt.h" #include "Emu/IdManager.h" #include "Emu/Cell/ErrorCodes.h" #include "Emu/Cell/PPUThread.h" #include "Emu/Cell/PPUOpcodes.h" LOG_CHANNEL(sys_interrupt); lv2_int_tag::lv2_int_tag() noexcept : id(idm::last_id()) { exists.release(1); } lv2_int_serv::lv2_int_serv(const std::shared_ptr>& thread, u64 arg1, u64 arg2) noexcept : id(idm::last_id()) , thread(thread) , arg1(arg1) , arg2(arg2) { exists.release(1); } void lv2_int_serv::exec() const { thread->cmd_list ({ { ppu_cmd::reset_stack, 0 }, { ppu_cmd::set_args, 2 }, arg1, arg2, { ppu_cmd::opd_call, 0 }, thread->entry_func, { ppu_cmd::sleep, 0 } }); thread->cmd_notify++; thread->cmd_notify.notify_one(); } bool ppu_thread_exit(ppu_thread& ppu); void lv2_int_serv::join() const { thread->cmd_list ({ { ppu_cmd::ptr_call, 0 }, std::bit_cast(&ppu_thread_exit) }); thread->cmd_notify++; thread->cmd_notify.notify_one(); (*thread)(); idm::remove_verify>(thread->id, static_cast>>(thread)); } error_code sys_interrupt_tag_destroy(ppu_thread& ppu, u32 intrtag) { ppu.state += cpu_flag::wait; sys_interrupt.warning("sys_interrupt_tag_destroy(intrtag=0x%x)", intrtag); const auto tag = idm::withdraw(intrtag, [](lv2_int_tag& tag) -> CellError { if (tag.handler && tag.handler->exists) { return CELL_EBUSY; } tag.exists.release(0); return {}; }); if (!tag) { return CELL_ESRCH; } if (tag.ret) { return tag.ret; } return CELL_OK; } error_code _sys_interrupt_thread_establish(ppu_thread& ppu, vm::ptr ih, u32 intrtag, u32 intrthread, u64 arg1, u64 arg2) { ppu.state += cpu_flag::wait; sys_interrupt.warning("_sys_interrupt_thread_establish(ih=*0x%x, intrtag=0x%x, intrthread=0x%x, arg1=0x%llx, arg2=0x%llx)", ih, intrtag, intrthread, arg1, arg2); CellError error = CELL_EAGAIN; const u32 id = idm::import([&]() { std::shared_ptr result; // Get interrupt tag const auto tag = idm::check_unlocked(intrtag); if (!tag) { error = CELL_ESRCH; return result; } // Get interrupt thread const auto it = idm::get_unlocked>(intrthread); if (!it) { error = CELL_ESRCH; return result; } // If interrupt thread is running, it's already established on another interrupt tag if (cpu_flag::stop - it->state) { error = CELL_EAGAIN; return result; } // It's unclear if multiple handlers can be established on single interrupt tag if (tag->handler && tag->handler->exists) { error = CELL_ESTAT; return result; } result = std::make_shared(it, arg1, arg2); tag->handler = result; it->state -= cpu_flag::stop; it->state.notify_one(cpu_flag::stop); return result; }); if (id) { *ih = id; return CELL_OK; } return error; } error_code _sys_interrupt_thread_disestablish(ppu_thread& ppu, u32 ih, vm::ptr r13) { ppu.state += cpu_flag::wait; sys_interrupt.warning("_sys_interrupt_thread_disestablish(ih=0x%x, r13=*0x%x)", ih, r13); const auto handler = idm::withdraw(ih, [](lv2_obj& obj) { obj.exists.release(0); }); if (!handler) { if (const auto thread = idm::withdraw>(ih)) { *r13 = thread->gpr[13]; return CELL_OK; } return CELL_ESRCH; } lv2_obj::sleep(ppu); // Wait for sys_interrupt_thread_eoi() and destroy interrupt thread handler->join(); // Save TLS base *r13 = handler->thread->gpr[13]; return CELL_OK; } void sys_interrupt_thread_eoi(ppu_thread& ppu) { ppu.state += cpu_flag::wait; sys_interrupt.trace("sys_interrupt_thread_eoi()"); ppu.state += cpu_flag::ret; }