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.
277 lines
8.5 KiB
C++
277 lines
8.5 KiB
C++
#include "stdafx.h"
|
|
#include "np_contexts.h"
|
|
|
|
#include "Emu/Cell/PPUCallback.h"
|
|
#include "Emu/IdManager.h"
|
|
#include "Emu/Cell/Modules/cellSysutil.h"
|
|
|
|
LOG_CHANNEL(sceNp2);
|
|
|
|
generic_async_transaction_context::generic_async_transaction_context(const SceNpCommunicationId& communicationId, const SceNpCommunicationPassphrase& passphrase, u64 timeout)
|
|
: communicationId(communicationId), passphrase(passphrase), timeout(timeout)
|
|
, idm_id(idm::last_id())
|
|
{
|
|
}
|
|
|
|
generic_async_transaction_context::~generic_async_transaction_context()
|
|
{
|
|
if (thread.joinable())
|
|
thread.join();
|
|
}
|
|
|
|
std::optional<s32> generic_async_transaction_context::get_transaction_status()
|
|
{
|
|
std::lock_guard lock(mutex);
|
|
return result;
|
|
}
|
|
void generic_async_transaction_context::abort_transaction()
|
|
{
|
|
std::lock_guard lock(mutex);
|
|
|
|
result = SCE_NP_COMMUNITY_ERROR_ABORTED;
|
|
wake_cond.notify_one();
|
|
}
|
|
error_code generic_async_transaction_context::wait_for_completion()
|
|
{
|
|
std::unique_lock lock(mutex);
|
|
|
|
if (result)
|
|
{
|
|
return *result;
|
|
}
|
|
|
|
completion_cond.wait(lock);
|
|
|
|
return *result;
|
|
}
|
|
|
|
bool generic_async_transaction_context::set_result_and_wake(error_code err)
|
|
{
|
|
result = err;
|
|
wake_cond.notify_one();
|
|
return true;
|
|
}
|
|
|
|
tus_ctx::tus_ctx(vm::cptr<SceNpCommunicationId> communicationId, vm::cptr<SceNpCommunicationPassphrase> passphrase)
|
|
{
|
|
ensure(!communicationId->data[9] && strlen(communicationId->data) == 9);
|
|
memcpy(&this->communicationId, communicationId.get_ptr(), sizeof(SceNpCommunicationId));
|
|
memcpy(&this->passphrase, passphrase.get_ptr(), sizeof(SceNpCommunicationPassphrase));
|
|
}
|
|
s32 create_tus_context(vm::cptr<SceNpCommunicationId> communicationId, vm::cptr<SceNpCommunicationPassphrase> passphrase)
|
|
{
|
|
s32 tus_id = idm::make<tus_ctx>(communicationId, passphrase);
|
|
|
|
if (tus_id == id_manager::id_traits<tus_ctx>::invalid)
|
|
{
|
|
return SCE_NP_COMMUNITY_ERROR_TOO_MANY_OBJECTS;
|
|
}
|
|
|
|
return static_cast<s32>(tus_id);
|
|
}
|
|
bool destroy_tus_context(s32 ctx_id)
|
|
{
|
|
return idm::remove<tus_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
|
|
tus_transaction_ctx::tus_transaction_ctx(const shared_ptr<tus_ctx>& tus)
|
|
: generic_async_transaction_context(tus->communicationId, tus->passphrase, tus->timeout)
|
|
{
|
|
}
|
|
|
|
s32 create_tus_transaction_context(const shared_ptr<tus_ctx>& tus)
|
|
{
|
|
s32 tus_id = idm::make<tus_transaction_ctx>(tus);
|
|
|
|
if (tus_id == id_manager::id_traits<tus_transaction_ctx>::invalid)
|
|
{
|
|
return SCE_NP_COMMUNITY_ERROR_TOO_MANY_OBJECTS;
|
|
}
|
|
|
|
return static_cast<s32>(tus_id);
|
|
}
|
|
bool destroy_tus_transaction_context(s32 ctx_id)
|
|
{
|
|
return idm::remove<tus_transaction_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
|
|
score_ctx::score_ctx(vm::cptr<SceNpCommunicationId> communicationId, vm::cptr<SceNpCommunicationPassphrase> passphrase)
|
|
{
|
|
ensure(!communicationId->data[9] && strlen(communicationId->data) == 9);
|
|
memcpy(&this->communicationId, communicationId.get_ptr(), sizeof(SceNpCommunicationId));
|
|
memcpy(&this->passphrase, passphrase.get_ptr(), sizeof(SceNpCommunicationPassphrase));
|
|
}
|
|
s32 create_score_context(vm::cptr<SceNpCommunicationId> communicationId, vm::cptr<SceNpCommunicationPassphrase> passphrase)
|
|
{
|
|
s32 score_id = idm::make<score_ctx>(communicationId, passphrase);
|
|
|
|
if (score_id == id_manager::id_traits<score_ctx>::invalid)
|
|
{
|
|
return SCE_NP_COMMUNITY_ERROR_TOO_MANY_OBJECTS;
|
|
}
|
|
|
|
return static_cast<s32>(score_id);
|
|
}
|
|
bool destroy_score_context(s32 ctx_id)
|
|
{
|
|
return idm::remove<score_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
|
|
score_transaction_ctx::score_transaction_ctx(const shared_ptr<score_ctx>& score)
|
|
: generic_async_transaction_context(score->communicationId, score->passphrase, score->timeout)
|
|
{
|
|
pcId = score->pcId;
|
|
}
|
|
|
|
s32 create_score_transaction_context(const shared_ptr<score_ctx>& score)
|
|
{
|
|
s32 trans_id = idm::make<score_transaction_ctx>(score);
|
|
|
|
if (trans_id == id_manager::id_traits<score_transaction_ctx>::invalid)
|
|
{
|
|
return SCE_NP_COMMUNITY_ERROR_TOO_MANY_OBJECTS;
|
|
}
|
|
|
|
return static_cast<s32>(trans_id);
|
|
}
|
|
bool destroy_score_transaction_context(s32 ctx_id)
|
|
{
|
|
return idm::remove<score_transaction_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
|
|
match2_ctx::match2_ctx(vm::cptr<SceNpCommunicationId> communicationId, vm::cptr<SceNpCommunicationPassphrase> passphrase, s32 option)
|
|
{
|
|
ensure(!communicationId->data[9] && strlen(communicationId->data) == 9);
|
|
memcpy(&this->communicationId, communicationId.get_ptr(), sizeof(SceNpCommunicationId));
|
|
memcpy(&this->passphrase, passphrase.get_ptr(), sizeof(SceNpCommunicationPassphrase));
|
|
|
|
include_onlinename = option & SCE_NP_MATCHING2_CONTEXT_OPTION_USE_ONLINENAME;
|
|
include_avatarurl = option & SCE_NP_MATCHING2_CONTEXT_OPTION_USE_AVATARURL;
|
|
}
|
|
u16 create_match2_context(vm::cptr<SceNpCommunicationId> communicationId, vm::cptr<SceNpCommunicationPassphrase> passphrase, s32 option)
|
|
{
|
|
sceNp2.notice("Creating match2 context with communicationId: <%s>", static_cast<const char*>(communicationId->data));
|
|
return static_cast<u16>(idm::make<match2_ctx>(communicationId, passphrase, option));
|
|
}
|
|
bool destroy_match2_context(u16 ctx_id)
|
|
{
|
|
return idm::remove<match2_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
bool check_match2_context(u16 ctx_id)
|
|
{
|
|
return (idm::check_unlocked<match2_ctx>(ctx_id) != nullptr);
|
|
}
|
|
shared_ptr<match2_ctx> get_match2_context(u16 ctx_id)
|
|
{
|
|
return idm::get_unlocked<match2_ctx>(ctx_id);
|
|
}
|
|
|
|
lookup_title_ctx::lookup_title_ctx(vm::cptr<SceNpCommunicationId> communicationId)
|
|
{
|
|
ensure(!communicationId->data[9] && strlen(communicationId->data) == 9);
|
|
memcpy(&this->communicationId, communicationId.get_ptr(), sizeof(SceNpCommunicationId));
|
|
}
|
|
s32 create_lookup_title_context(vm::cptr<SceNpCommunicationId> communicationId)
|
|
{
|
|
return static_cast<s32>(idm::make<lookup_title_ctx>(communicationId));
|
|
}
|
|
bool destroy_lookup_title_context(s32 ctx_id)
|
|
{
|
|
return idm::remove<lookup_title_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
|
|
lookup_transaction_ctx::lookup_transaction_ctx(s32 lt_ctx)
|
|
{
|
|
this->lt_ctx = lt_ctx;
|
|
}
|
|
s32 create_lookup_transaction_context(s32 lt_ctx)
|
|
{
|
|
return static_cast<s32>(idm::make<lookup_transaction_ctx>(lt_ctx));
|
|
}
|
|
bool destroy_lookup_transaction_context(s32 ctx_id)
|
|
{
|
|
return idm::remove<lookup_transaction_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
|
|
commerce2_ctx::commerce2_ctx(u32 version, vm::cptr<SceNpId> npid, vm::ptr<SceNpCommerce2Handler> handler, vm::ptr<void> arg)
|
|
{
|
|
this->version = version;
|
|
memcpy(&this->npid, npid.get_ptr(), sizeof(SceNpId));
|
|
this->context_callback = handler;
|
|
this->context_callback_param = arg;
|
|
}
|
|
s32 create_commerce2_context(u32 version, vm::cptr<SceNpId> npid, vm::ptr<SceNpCommerce2Handler> handler, vm::ptr<void> arg)
|
|
{
|
|
return static_cast<s32>(idm::make<commerce2_ctx>(version, npid, handler, arg));
|
|
}
|
|
bool destroy_commerce2_context(u32 ctx_id)
|
|
{
|
|
return idm::remove<commerce2_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
shared_ptr<commerce2_ctx> get_commerce2_context(u16 ctx_id)
|
|
{
|
|
return idm::get_unlocked<commerce2_ctx>(ctx_id);
|
|
}
|
|
|
|
signaling_ctx::signaling_ctx(vm::ptr<SceNpId> npid, vm::ptr<SceNpSignalingHandler> handler, vm::ptr<void> arg)
|
|
{
|
|
memcpy(&this->npid, npid.get_ptr(), sizeof(SceNpId));
|
|
this->handler = handler;
|
|
this->arg = arg;
|
|
}
|
|
s32 create_signaling_context(vm::ptr<SceNpId> npid, vm::ptr<SceNpSignalingHandler> handler, vm::ptr<void> arg)
|
|
{
|
|
return static_cast<s32>(idm::make<signaling_ctx>(npid, handler, arg));
|
|
}
|
|
bool destroy_signaling_context(u32 ctx_id)
|
|
{
|
|
return idm::remove<signaling_ctx>(static_cast<u32>(ctx_id));
|
|
}
|
|
shared_ptr<signaling_ctx> get_signaling_context(u32 ctx_id)
|
|
{
|
|
return idm::get_unlocked<signaling_ctx>(ctx_id);
|
|
}
|
|
|
|
matching_ctx::matching_ctx(vm::ptr<SceNpId> npId, vm::ptr<SceNpMatchingHandler> handler, vm::ptr<void> arg)
|
|
{
|
|
memcpy(&this->npid, npId.get_ptr(), sizeof(SceNpId));
|
|
this->handler = handler;
|
|
this->arg = arg;
|
|
}
|
|
void matching_ctx::queue_callback(u32 req_id, s32 event, s32 error_code)
|
|
{
|
|
if (handler)
|
|
{
|
|
sysutil_register_cb([=, handler = this->handler, ctx_id = this->ctx_id, arg = this->arg](ppu_thread& cb_ppu) -> s32
|
|
{
|
|
handler(cb_ppu, ctx_id, req_id, event, error_code, arg);
|
|
return 0;
|
|
});
|
|
}
|
|
}
|
|
void matching_ctx::queue_gui_callback(s32 event, s32 error_code)
|
|
{
|
|
if (gui_handler)
|
|
{
|
|
sysutil_register_cb([=, gui_handler = this->gui_handler, ctx_id = this->ctx_id, gui_arg = this->gui_arg](ppu_thread& cb_ppu) -> s32
|
|
{
|
|
gui_handler(cb_ppu, ctx_id, event, error_code, gui_arg);
|
|
return 0;
|
|
});
|
|
}
|
|
}
|
|
s32 create_matching_context(vm::ptr<SceNpId> npId, vm::ptr<SceNpMatchingHandler> handler, vm::ptr<void> arg)
|
|
{
|
|
const u32 ctx_id = idm::make<matching_ctx>(npId, handler, arg);
|
|
auto ctx = get_matching_context(ctx_id);
|
|
ctx->ctx_id = ctx_id;
|
|
return static_cast<s32>(ctx_id);
|
|
}
|
|
shared_ptr<matching_ctx> get_matching_context(u32 ctx_id)
|
|
{
|
|
return idm::get_unlocked<matching_ctx>(ctx_id);
|
|
}
|
|
bool destroy_matching_context(u32 ctx_id)
|
|
{
|
|
return idm::remove<matching_ctx>(static_cast<u32>(ctx_id));
|
|
}
|