From 9ba9780a96a67191cd7f12f34b343c77668b1831 Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 30 Jan 2024 02:22:45 -0500 Subject: cmif_serialization: fix out layout calculation --- src/core/hle/service/cmif_serialization.h | 50 +++++++++++++++++++++++-------- 1 file changed, 38 insertions(+), 12 deletions(-) diff --git a/src/core/hle/service/cmif_serialization.h b/src/core/hle/service/cmif_serialization.h index 9ee26400d..5eabf51fe 100644 --- a/src/core/hle/service/cmif_serialization.h +++ b/src/core/hle/service/cmif_serialization.h @@ -122,14 +122,14 @@ struct RequestLayout { u32 domain_interface_count; }; -template -constexpr u32 GetArgumentRawDataSize() { +template +constexpr u32 GetInRawDataSize() { if constexpr (ArgIndex >= std::tuple_size_v) { return static_cast(DataOffset); } else { using ArgType = std::tuple_element_t; - if constexpr (ArgumentTraits::Type == Type1 || ArgumentTraits::Type == Type2) { + if constexpr (ArgumentTraits::Type == ArgumentType::InData || ArgumentTraits::Type == ArgumentType::InProcessId) { constexpr size_t ArgAlign = alignof(ArgType); constexpr size_t ArgSize = sizeof(ArgType); @@ -138,9 +138,33 @@ constexpr u32 GetArgumentRawDataSize() { constexpr size_t ArgOffset = Common::AlignUp(DataOffset, ArgAlign); constexpr size_t ArgEnd = ArgOffset + ArgSize; - return GetArgumentRawDataSize(); + return GetInRawDataSize(); + } else { + return GetInRawDataSize(); + } + } +} + +template +constexpr u32 GetOutRawDataSize() { + if constexpr (ArgIndex >= std::tuple_size_v) { + return static_cast(DataOffset); + } else { + using ArgType = std::tuple_element_t; + + if constexpr (ArgumentTraits::Type == ArgumentType::OutData) { + using RawArgType = typename ArgType::Type; + constexpr size_t ArgAlign = alignof(RawArgType); + constexpr size_t ArgSize = sizeof(RawArgType); + + static_assert(PrevAlign <= ArgAlign, "Output argument is not ordered by alignment"); + + constexpr size_t ArgOffset = Common::AlignUp(DataOffset, ArgAlign); + constexpr size_t ArgEnd = ArgOffset + ArgSize; + + return GetOutRawDataSize(); } else { - return GetArgumentRawDataSize(); + return GetOutRawDataSize(); } } } @@ -165,7 +189,7 @@ constexpr RequestLayout GetNonDomainReplyInLayout() { return RequestLayout{ .copy_handle_count = GetArgumentTypeCount(), .move_handle_count = 0, - .cmif_raw_data_size = GetArgumentRawDataSize(), + .cmif_raw_data_size = GetInRawDataSize(), .domain_interface_count = 0, }; } @@ -175,7 +199,7 @@ constexpr RequestLayout GetDomainReplyInLayout() { return RequestLayout{ .copy_handle_count = GetArgumentTypeCount(), .move_handle_count = 0, - .cmif_raw_data_size = GetArgumentRawDataSize(), + .cmif_raw_data_size = GetInRawDataSize(), .domain_interface_count = GetArgumentTypeCount(), }; } @@ -185,7 +209,7 @@ constexpr RequestLayout GetNonDomainReplyOutLayout() { return RequestLayout{ .copy_handle_count = GetArgumentTypeCount(), .move_handle_count = GetArgumentTypeCount() + GetArgumentTypeCount(), - .cmif_raw_data_size = GetArgumentRawDataSize(), + .cmif_raw_data_size = GetOutRawDataSize(), .domain_interface_count = 0, }; } @@ -195,7 +219,7 @@ constexpr RequestLayout GetDomainReplyOutLayout() { return RequestLayout{ .copy_handle_count = GetArgumentTypeCount(), .move_handle_count = GetArgumentTypeCount(), - .cmif_raw_data_size = GetArgumentRawDataSize(), + .cmif_raw_data_size = GetOutRawDataSize(), .domain_interface_count = GetArgumentTypeCount(), }; } @@ -337,13 +361,15 @@ void WriteOutArgument(bool is_domain, CallArguments& args, u8* raw_data, HLERequ using ArgType = std::tuple_element_t; if constexpr (ArgumentTraits::Type == ArgumentType::OutData) { - constexpr size_t ArgAlign = alignof(ArgType); - constexpr size_t ArgSize = sizeof(ArgType); + using RawArgType = decltype(std::get(args).raw); + constexpr size_t ArgAlign = alignof(RawArgType); + constexpr size_t ArgSize = sizeof(RawArgType); static_assert(PrevAlign <= ArgAlign, "Output argument is not ordered by alignment"); static_assert(!RawDataFinished, "All output interface arguments must appear after raw data"); static_assert(!std::is_pointer_v, "Output raw data must not be a pointer"); - static_assert(std::is_trivially_copyable_v(args).raw)>, "Output raw data must be trivially copyable"); + static_assert(!std::is_pointer_v, "Output raw data must not be a pointer"); + static_assert(std::is_trivially_copyable_v, "Output raw data must be trivially copyable"); constexpr size_t ArgOffset = Common::AlignUp(DataOffset, ArgAlign); constexpr size_t ArgEnd = ArgOffset + ArgSize; -- cgit v1.2.3