mirror of
https://github.com/RPCS3/rpcs3.git
synced 2025-07-02 13:01:27 +12:00
Replaces `std::shared_pointer` with `stx::atomic_ptr` and `stx::shared_ptr`. Notes to programmers: * This pr kills the use of `dynamic_cast`, `std::dynamic_pointer_cast` and `std::weak_ptr` on IDM objects, possible replacement is to save the object ID on the base object, then use idm::check/get_unlocked to the destination type via the saved ID which may be null. Null pointer check is how you can tell type mismatch (as dynamic cast) or object destruction (as weak_ptr locking). * Double-inheritance on IDM objects should be used with care, `stx::shared_ptr` does not support constant-evaluated pointer offsetting to parent/child type. * `idm::check/get_unlocked` can now be used anywhere. Misc fixes: * Fixes some segfaults with RPCN with interaction with IDM. * Fix deadlocks in access violation handler due locking recursion. * Fixes race condition in process exit-spawn on memory containers read. * Fix bug that theoretically can prevent RPCS3 from booting - fix `id_manager::typeinfo` comparison to compare members instead of `memcmp` which can fail spuriously on padding bytes. * Ensure all IDM inherited types of base, either has `id_base` or `id_type` defined locally, this allows to make getters such as `idm::get_unlocked<lv2_socket, lv2_socket_raw>()` which were broken before. (requires save-states invalidation) * Removes broken operator[] overload of `stx::shared_ptr` and `stx::single_ptr` for non-array types.
252 lines
5.4 KiB
C++
252 lines
5.4 KiB
C++
#include "stdafx.h"
|
|
#include "Emu/System.h"
|
|
#include "Emu/IdManager.h"
|
|
#include "Emu/Cell/PPUModule.h"
|
|
|
|
#include "Emu/Cell/lv2/sys_event.h"
|
|
#include "Emu/Cell/lv2/sys_ppu_thread.h"
|
|
#include "Emu/Cell/Modules/sysPrxForUser.h"
|
|
|
|
LOG_CHANNEL(sys_io);
|
|
|
|
extern void cellPad_init();
|
|
extern void cellKb_init();
|
|
extern void cellMouse_init();
|
|
|
|
struct libio_sys_config
|
|
{
|
|
shared_mutex mtx;
|
|
s32 init_ctr = 0;
|
|
u32 ppu_id = 0;
|
|
u32 queue_id = 0;
|
|
|
|
~libio_sys_config() noexcept
|
|
{
|
|
}
|
|
|
|
void save_or_load(utils::serial& ar)
|
|
{
|
|
ar(init_ctr, ppu_id, queue_id);
|
|
}
|
|
};
|
|
|
|
extern void sys_io_serialize(utils::serial& ar)
|
|
{
|
|
// Do not assign a serialization tag for now, call it from cellPad serialization
|
|
ensure(g_fxo->try_get<libio_sys_config>())->save_or_load(ar);
|
|
}
|
|
|
|
extern bool cellPad_NotifyStateChange(usz index, u64 state, bool lock = true, bool is_blocking = true);
|
|
|
|
void config_event_entry(ppu_thread& ppu)
|
|
{
|
|
ppu.state += cpu_flag::wait;
|
|
|
|
auto& cfg = *ensure(g_fxo->try_get<libio_sys_config>());
|
|
|
|
if (!ppu.loaded_from_savestate)
|
|
{
|
|
// Ensure awake
|
|
ppu.check_state();
|
|
}
|
|
|
|
const u32 queue_id = cfg.queue_id;
|
|
auto queue = idm::get_unlocked<lv2_obj, lv2_event_queue>(queue_id);
|
|
|
|
while (queue && sys_event_queue_receive(ppu, queue_id, vm::null, 0) == CELL_OK)
|
|
{
|
|
if (ppu.is_stopped())
|
|
{
|
|
ppu.state += cpu_flag::again;
|
|
return;
|
|
}
|
|
|
|
// Some delay
|
|
thread_ctrl::wait_for(10000);
|
|
|
|
// Wakeup
|
|
ppu.check_state();
|
|
ppu.state += cpu_flag::wait;
|
|
|
|
const u64 arg1 = ppu.gpr[5];
|
|
const u64 arg2 = ppu.gpr[6];
|
|
const u64 arg3 = ppu.gpr[7];
|
|
|
|
// TODO: Reverse-engineer proper event system
|
|
|
|
if (arg1 == 1)
|
|
{
|
|
while (!cellPad_NotifyStateChange(arg2, arg3, false))
|
|
{
|
|
if (!queue->exists)
|
|
{
|
|
// Exit condition
|
|
queue = null_ptr;
|
|
break;
|
|
}
|
|
|
|
thread_ctrl::wait_for(100);
|
|
}
|
|
}
|
|
}
|
|
|
|
sys_io.notice("config_event_entry(): Exited with the following error code: %s", CellError{static_cast<u32>(ppu.gpr[3])});
|
|
|
|
ppu_execute<&sys_ppu_thread_exit>(ppu, 0);
|
|
}
|
|
|
|
std::unique_lock<shared_mutex> lock_lv2_mutex_alike(shared_mutex& mtx, ppu_thread* ppu)
|
|
{
|
|
std::unique_lock<shared_mutex> lock(mtx, std::defer_lock);
|
|
|
|
while (!lock.try_lock())
|
|
{
|
|
if (ppu)
|
|
{
|
|
// Could not be acquired, put PPU to sleep
|
|
lv2_obj::sleep(*ppu);
|
|
}
|
|
|
|
// Wait for unlock without owning the lock
|
|
mtx.lock_unlock();
|
|
|
|
if (ppu)
|
|
{
|
|
// Awake, still not owning
|
|
ppu->check_state();
|
|
}
|
|
}
|
|
|
|
return lock;
|
|
}
|
|
|
|
extern void send_sys_io_connect_event(usz index, u32 state)
|
|
{
|
|
if (Emu.IsStarting() || Emu.IsReady())
|
|
{
|
|
cellPad_NotifyStateChange(index, state);
|
|
return;
|
|
}
|
|
|
|
auto& cfg = g_fxo->get<libio_sys_config>();
|
|
|
|
auto lock = lock_lv2_mutex_alike(cfg.mtx, cpu_thread::get_current<ppu_thread>());
|
|
|
|
if (cfg.init_ctr)
|
|
{
|
|
if (auto port = idm::get_unlocked<lv2_obj, lv2_event_queue>(cfg.queue_id))
|
|
{
|
|
port->send(0, 1, index, state);
|
|
}
|
|
}
|
|
}
|
|
|
|
error_code sys_config_start(ppu_thread& ppu)
|
|
{
|
|
sys_io.warning("sys_config_start()");
|
|
|
|
auto& cfg = g_fxo->get<libio_sys_config>();
|
|
|
|
auto lock = lock_lv2_mutex_alike(cfg.mtx, &ppu);
|
|
|
|
if (cfg.init_ctr++ == 0)
|
|
{
|
|
// Run thread
|
|
vm::var<u64> _tid;
|
|
vm::var<u32> queue_id;
|
|
vm::var<char[]> _name = vm::make_str("_cfg_evt_hndlr");
|
|
|
|
vm::var<sys_event_queue_attribute_t> attr;
|
|
attr->protocol = SYS_SYNC_PRIORITY;
|
|
attr->type = SYS_PPU_QUEUE;
|
|
attr->name_u64 = 0;
|
|
|
|
ensure(CELL_OK == sys_event_queue_create(ppu, queue_id, attr, 0, 0x20));
|
|
ppu.check_state();
|
|
cfg.queue_id = *queue_id;
|
|
|
|
ensure(CELL_OK == ppu_execute<&sys_ppu_thread_create>(ppu, +_tid, g_fxo->get<ppu_function_manager>().func_addr(FIND_FUNC(config_event_entry)), 0, 512, 0x2000, SYS_PPU_THREAD_CREATE_JOINABLE, +_name));
|
|
ppu.check_state();
|
|
|
|
cfg.ppu_id = static_cast<u32>(*_tid);
|
|
}
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
error_code sys_config_stop(ppu_thread& ppu)
|
|
{
|
|
sys_io.warning("sys_config_stop()");
|
|
|
|
auto& cfg = g_fxo->get<libio_sys_config>();
|
|
|
|
auto lock = lock_lv2_mutex_alike(cfg.mtx, &ppu);
|
|
|
|
if (cfg.init_ctr && cfg.init_ctr-- == 1)
|
|
{
|
|
ensure(CELL_OK == sys_event_queue_destroy(ppu, cfg.queue_id, SYS_EVENT_QUEUE_DESTROY_FORCE));
|
|
ppu.check_state();
|
|
ensure(CELL_OK == sys_ppu_thread_join(ppu, cfg.ppu_id, +vm::var<u64>{}));
|
|
}
|
|
else
|
|
{
|
|
// TODO: Unknown error
|
|
}
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
error_code sys_config_add_service_listener()
|
|
{
|
|
sys_io.todo("sys_config_add_service_listener()");
|
|
return CELL_OK;
|
|
}
|
|
|
|
error_code sys_config_remove_service_listener()
|
|
{
|
|
sys_io.todo("sys_config_remove_service_listener()");
|
|
return CELL_OK;
|
|
}
|
|
|
|
error_code sys_config_register_io_error_handler()
|
|
{
|
|
sys_io.todo("sys_config_register_io_error_handler()");
|
|
return CELL_OK;
|
|
}
|
|
|
|
error_code sys_config_register_service()
|
|
{
|
|
sys_io.todo("sys_config_register_service()");
|
|
return CELL_OK;
|
|
}
|
|
|
|
error_code sys_config_unregister_io_error_handler()
|
|
{
|
|
sys_io.todo("sys_config_unregister_io_error_handler()");
|
|
return CELL_OK;
|
|
}
|
|
|
|
error_code sys_config_unregister_service()
|
|
{
|
|
sys_io.todo("sys_config_unregister_service()");
|
|
return CELL_OK;
|
|
}
|
|
|
|
|
|
DECLARE(ppu_module_manager::sys_io)("sys_io", []()
|
|
{
|
|
cellPad_init();
|
|
cellKb_init();
|
|
cellMouse_init();
|
|
|
|
REG_FUNC(sys_io, sys_config_start);
|
|
REG_FUNC(sys_io, sys_config_stop);
|
|
REG_FUNC(sys_io, sys_config_add_service_listener);
|
|
REG_FUNC(sys_io, sys_config_remove_service_listener);
|
|
REG_FUNC(sys_io, sys_config_register_io_error_handler);
|
|
REG_FUNC(sys_io, sys_config_register_service);
|
|
REG_FUNC(sys_io, sys_config_unregister_io_error_handler);
|
|
REG_FUNC(sys_io, sys_config_unregister_service);
|
|
|
|
REG_HIDDEN_FUNC(config_event_entry);
|
|
});
|