From fadab1d5f365c11f0b4c33e74b0d297756f2f3f9 Mon Sep 17 00:00:00 2001 From: bunnei Date: Mon, 30 Apr 2018 23:24:31 -0400 Subject: ipc: Add support for PopIpcInterface() method. - This can be used for domain objects as inputs to service functions. --- src/core/hle/ipc.h | 1 + src/core/hle/ipc_helpers.h | 7 +++++++ src/core/hle/kernel/hle_ipc.h | 12 ++++++++++++ src/core/hle/kernel/server_session.cpp | 3 +++ 4 files changed, 23 insertions(+) (limited to 'src/core') diff --git a/src/core/hle/ipc.h b/src/core/hle/ipc.h index a6602e12c..ef6595550 100644 --- a/src/core/hle/ipc.h +++ b/src/core/hle/ipc.h @@ -167,6 +167,7 @@ struct DomainMessageHeader { struct { union { BitField<0, 8, CommandType> command; + BitField<8, 8, u32_le> input_object_count; BitField<16, 16, u32_le> size; }; u32_le object_id; diff --git a/src/core/hle/ipc_helpers.h b/src/core/hle/ipc_helpers.h index 3f87c4297..24605a273 100644 --- a/src/core/hle/ipc_helpers.h +++ b/src/core/hle/ipc_helpers.h @@ -298,6 +298,13 @@ public: template Kernel::SharedPtr GetCopyObject(size_t index); + + template + std::shared_ptr PopIpcInterface() { + ASSERT(context->Session()->IsDomain()); + ASSERT(context->GetDomainMessageHeader()->input_object_count > 0); + return context->GetDomainRequestHandler(Pop() - 1); + } }; /// Pop /// diff --git a/src/core/hle/kernel/hle_ipc.h b/src/core/hle/kernel/hle_ipc.h index 6d4ed7648..376263eac 100644 --- a/src/core/hle/kernel/hle_ipc.h +++ b/src/core/hle/kernel/hle_ipc.h @@ -202,6 +202,16 @@ public: domain_objects.emplace_back(std::move(object)); } + template + std::shared_ptr GetDomainRequestHandler(size_t index) const { + return std::static_pointer_cast(domain_request_handlers[index]); + } + + void SetDomainRequestHandlers( + const std::vector>& handlers) { + domain_request_handlers = handlers; + } + /// Clears the list of objects so that no lingering objects are written accidentally to the /// response buffer. void ClearIncomingObjects() { @@ -245,6 +255,8 @@ private: unsigned data_payload_offset{}; unsigned buffer_c_offset{}; u32_le command{}; + + std::vector> domain_request_handlers; }; } // namespace Kernel diff --git a/src/core/hle/kernel/server_session.cpp b/src/core/hle/kernel/server_session.cpp index b1f8e771c..ed33c8600 100644 --- a/src/core/hle/kernel/server_session.cpp +++ b/src/core/hle/kernel/server_session.cpp @@ -61,6 +61,9 @@ void ServerSession::Acquire(Thread* thread) { ResultCode ServerSession::HandleDomainSyncRequest(Kernel::HLERequestContext& context) { auto& domain_message_header = context.GetDomainMessageHeader(); if (domain_message_header) { + // Set domain handlers in HLE context, used for domain objects (IPC interfaces) as inputs + context.SetDomainRequestHandlers(domain_request_handlers); + // If there is a DomainMessageHeader, then this is CommandType "Request" const u32 object_id{context.GetDomainMessageHeader()->object_id}; switch (domain_message_header->command) { -- cgit v1.2.3