| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | // Copyright 2016 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2017-05-29 16:45:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | #include <cstring>
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  | #include <tuple>
 | 
					
						
							|  |  |  | #include <type_traits>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | #include "core/hle/ipc.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | #include "core/hle/kernel/hle_ipc.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-21 21:43:25 -07:00
										 |  |  | #include "core/hle/kernel/k_client_port.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-10 15:42:46 -07:00
										 |  |  | #include "core/hle/kernel/k_process.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/k_resource_limit.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  | #include "core/hle/kernel/k_session.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-10 18:23:39 -04:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace IPC { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-10 18:23:39 -04:00
										 |  |  | constexpr ResultCode ERR_REMOTE_PROCESS_DEAD{ErrorModule::HIPC, 301}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | class RequestHelperBase { | 
					
						
							|  |  |  | protected: | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  |     Kernel::HLERequestContext* context = nullptr; | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     u32* cmdbuf; | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |     u32 index = 0; | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2018-07-23 22:40:24 -04:00
										 |  |  |     explicit RequestHelperBase(u32* command_buffer) : cmdbuf(command_buffer) {} | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |     explicit RequestHelperBase(Kernel::HLERequestContext& ctx) | 
					
						
							|  |  |  |         : context(&ctx), cmdbuf(ctx.CommandBuffer()) {} | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-13 08:10:50 -04:00
										 |  |  |     void Skip(u32 size_in_words, bool set_to_null) { | 
					
						
							|  |  |  |         if (set_to_null) { | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |             memset(cmdbuf + index, 0, size_in_words * sizeof(u32)); | 
					
						
							| 
									
										
										
										
											2020-10-13 08:10:50 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         index += size_in_words; | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2017-10-17 18:03:47 -04:00
										 |  |  |      * Aligns the current position forward to a 16-byte boundary, padding with zeros. | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     void AlignWithPadding() { | 
					
						
							| 
									
										
										
										
											2017-10-17 18:03:47 -04:00
										 |  |  |         if (index & 3) { | 
					
						
							| 
									
										
										
										
											2020-10-13 08:10:50 -04:00
										 |  |  |             Skip(static_cast<u32>(4 - (index & 3)), true); | 
					
						
							| 
									
										
										
										
											2017-10-17 18:03:47 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-13 08:10:50 -04:00
										 |  |  |     u32 GetCurrentOffset() const { | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         return index; | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-18 16:54:34 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-13 08:10:50 -04:00
										 |  |  |     void SetCurrentOffset(u32 offset) { | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         index = offset; | 
					
						
							| 
									
										
										
										
											2018-01-18 16:54:34 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  | class ResponseBuilder : public RequestHelperBase { | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2018-01-23 22:33:30 -05:00
										 |  |  |     /// Flags used for customizing the behavior of ResponseBuilder
 | 
					
						
							|  |  |  |     enum class Flags : u32 { | 
					
						
							|  |  |  |         None = 0, | 
					
						
							|  |  |  |         /// Uses move handles to move objects in the response, even when in a domain. This is
 | 
					
						
							|  |  |  |         /// required when PushMoveObjects is used.
 | 
					
						
							|  |  |  |         AlwaysMoveHandles = 1, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |     explicit ResponseBuilder(Kernel::HLERequestContext& ctx, u32 normal_params_size_, | 
					
						
							|  |  |  |                              u32 num_handles_to_copy_ = 0, u32 num_objects_to_move_ = 0, | 
					
						
							| 
									
										
										
										
											2018-07-23 22:40:24 -04:00
										 |  |  |                              Flags flags = Flags::None) | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |         : RequestHelperBase(ctx), normal_params_size(normal_params_size_), | 
					
						
							|  |  |  |           num_handles_to_copy(num_handles_to_copy_), | 
					
						
							|  |  |  |           num_objects_to_move(num_objects_to_move_), kernel{ctx.kernel} { | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         memset(cmdbuf, 0, sizeof(u32) * IPC::COMMAND_BUFFER_LENGTH); | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 12:11:36 -04:00
										 |  |  |         ctx.ClearIncomingObjects(); | 
					
						
							| 
									
										
										
										
											2018-01-15 15:31:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         IPC::CommandHeader header{}; | 
					
						
							| 
									
										
										
										
											2018-01-06 21:14:14 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         // The entire size of the raw data section in u32 units, including the 16 bytes of mandatory
 | 
					
						
							|  |  |  |         // padding.
 | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         u32 raw_data_size = ctx.IsTipc() | 
					
						
							|  |  |  |                                 ? normal_params_size - 1 | 
					
						
							|  |  |  |                                 : sizeof(IPC::DataPayloadHeader) / 4 + 4 + normal_params_size; | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         u32 num_handles_to_move{}; | 
					
						
							|  |  |  |         u32 num_domain_objects{}; | 
					
						
							| 
									
										
										
										
											2018-01-23 22:33:30 -05:00
										 |  |  |         const bool always_move_handles{ | 
					
						
							|  |  |  |             (static_cast<u32>(flags) & static_cast<u32>(Flags::AlwaysMoveHandles)) != 0}; | 
					
						
							| 
									
										
										
										
											2021-05-08 12:11:36 -04:00
										 |  |  |         if (!ctx.Session()->IsDomain() || always_move_handles) { | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  |             num_handles_to_move = num_objects_to_move; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             num_domain_objects = num_objects_to_move; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 12:11:36 -04:00
										 |  |  |         if (ctx.Session()->IsDomain()) { | 
					
						
							| 
									
										
										
										
											2021-05-10 17:45:55 -07:00
										 |  |  |             raw_data_size += static_cast<u32>(sizeof(DomainMessageHeader) / 4 + num_domain_objects); | 
					
						
							| 
									
										
										
										
											2018-01-15 15:31:10 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-06 21:14:14 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         if (ctx.IsTipc()) { | 
					
						
							|  |  |  |             header.type.Assign(ctx.GetCommandType()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-11 12:27:43 -07:00
										 |  |  |         ctx.data_size = static_cast<u32>(raw_data_size); | 
					
						
							| 
									
										
										
										
											2020-10-13 08:10:50 -04:00
										 |  |  |         header.data_size.Assign(static_cast<u32>(raw_data_size)); | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |         if (num_handles_to_copy != 0 || num_handles_to_move != 0) { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |             header.enable_handle_descriptor.Assign(1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PushRaw(header); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (header.enable_handle_descriptor) { | 
					
						
							|  |  |  |             IPC::HandleDescriptorHeader handle_descriptor_header{}; | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |             handle_descriptor_header.num_handles_to_copy.Assign(num_handles_to_copy_); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |             handle_descriptor_header.num_handles_to_move.Assign(num_handles_to_move); | 
					
						
							|  |  |  |             PushRaw(handle_descriptor_header); | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |             ctx.handles_offset = index; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |             Skip(num_handles_to_copy + num_handles_to_move, true); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         if (!ctx.IsTipc()) { | 
					
						
							|  |  |  |             AlignWithPadding(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (ctx.Session()->IsDomain() && ctx.HasDomainMessageHeader()) { | 
					
						
							|  |  |  |                 IPC::DomainMessageHeader domain_header{}; | 
					
						
							|  |  |  |                 domain_header.num_objects = num_domain_objects; | 
					
						
							|  |  |  |                 PushRaw(domain_header); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |             IPC::DataPayloadHeader data_payload_header{}; | 
					
						
							|  |  |  |             data_payload_header.magic = Common::MakeMagic('S', 'F', 'C', 'O'); | 
					
						
							|  |  |  |             PushRaw(data_payload_header); | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         data_payload_index = index; | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         ctx.data_payload_offset = index; | 
					
						
							|  |  |  |         ctx.domain_offset = index + raw_data_size / 4; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  |     template <class T> | 
					
						
							|  |  |  |     void PushIpcInterface(std::shared_ptr<T> iface) { | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  |         if (context->Session()->IsDomain()) { | 
					
						
							| 
									
										
										
										
											2018-01-15 15:31:10 -05:00
										 |  |  |             context->AddDomainObject(std::move(iface)); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2021-05-15 22:34:55 -07:00
										 |  |  |             kernel.CurrentProcess()->GetResourceLimit()->Reserve( | 
					
						
							|  |  |  |                 Kernel::LimitableResource::Sessions, 1); | 
					
						
							| 
									
										
										
										
											2021-05-10 15:42:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  |             auto* session = Kernel::KSession::Create(kernel); | 
					
						
							| 
									
										
										
										
											2021-04-23 17:00:15 -07:00
										 |  |  |             session->Initialize(nullptr, iface->GetServiceName()); | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |             context->AddMoveObject(&session->GetClientSession()); | 
					
						
							| 
									
										
										
										
											2021-04-23 17:00:15 -07:00
										 |  |  |             iface->ClientConnected(&session->GetServerSession()); | 
					
						
							| 
									
										
										
										
											2018-01-15 15:31:10 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  |     template <class T, class... Args> | 
					
						
							|  |  |  |     void PushIpcInterface(Args&&... args) { | 
					
						
							|  |  |  |         PushIpcInterface<T>(std::make_shared<T>(std::forward<Args>(args)...)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void ValidateHeader() { | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |         const std::size_t num_domain_objects = context->NumDomainObjects(); | 
					
						
							|  |  |  |         const std::size_t num_move_objects = context->NumMoveObjects(); | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  |         ASSERT_MSG(!num_domain_objects || !num_move_objects, | 
					
						
							|  |  |  |                    "cannot move normal handles and domain objects"); | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         ASSERT_MSG((index - data_payload_index) == normal_params_size, | 
					
						
							| 
									
										
										
										
											2018-01-23 18:58:25 -05:00
										 |  |  |                    "normal_params_size value is incorrect"); | 
					
						
							|  |  |  |         ASSERT_MSG((num_domain_objects + num_move_objects) == num_objects_to_move, | 
					
						
							|  |  |  |                    "num_objects_to_move value is incorrect"); | 
					
						
							|  |  |  |         ASSERT_MSG(context->NumCopyObjects() == num_handles_to_copy, | 
					
						
							|  |  |  |                    "num_handles_to_copy value is incorrect"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     // Validate on destruction, as there shouldn't be any case where we don't want it
 | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |     ~ResponseBuilder() { | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |         ValidateHeader(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  |     void PushImpl(s8 value); | 
					
						
							|  |  |  |     void PushImpl(s16 value); | 
					
						
							|  |  |  |     void PushImpl(s32 value); | 
					
						
							|  |  |  |     void PushImpl(s64 value); | 
					
						
							|  |  |  |     void PushImpl(u8 value); | 
					
						
							|  |  |  |     void PushImpl(u16 value); | 
					
						
							|  |  |  |     void PushImpl(u32 value); | 
					
						
							|  |  |  |     void PushImpl(u64 value); | 
					
						
							|  |  |  |     void PushImpl(float value); | 
					
						
							|  |  |  |     void PushImpl(double value); | 
					
						
							|  |  |  |     void PushImpl(bool value); | 
					
						
							|  |  |  |     void PushImpl(ResultCode value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     template <typename T> | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  |     void Push(T value) { | 
					
						
							|  |  |  |         return PushImpl(value); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-05 00:29:07 +01:00
										 |  |  |     template <typename First, typename... Other> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     void Push(const First& first_value, const Other&... other_values); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 14:49:13 -04:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Helper function for pushing strongly-typed enumeration values. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @tparam Enum The enumeration type to be pushed | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param value The value to push. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @note The underlying size of the enumeration type is the size of the | 
					
						
							|  |  |  |      *       data that gets pushed. e.g. "enum class SomeEnum : u16" will | 
					
						
							|  |  |  |      *       push a u16-sized amount of data. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <typename Enum> | 
					
						
							|  |  |  |     void PushEnum(Enum value) { | 
					
						
							|  |  |  |         static_assert(std::is_enum_v<Enum>, "T must be an enum type within a PushEnum call."); | 
					
						
							|  |  |  |         static_assert(!std::is_convertible_v<Enum, int>, | 
					
						
							|  |  |  |                       "enum type in PushEnum must be a strongly typed enum."); | 
					
						
							|  |  |  |         Push(static_cast<std::underlying_type_t<Enum>>(value)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * @brief Copies the content of the given trivially copyable class to the buffer as a normal | 
					
						
							|  |  |  |      * param | 
					
						
							|  |  |  |      * @note: The input class must be correctly packed/padded to fit hardware layout. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     void PushRaw(const T& value); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  |     template <typename... O> | 
					
						
							| 
									
										
										
										
											2021-04-03 21:21:22 -07:00
										 |  |  |     void PushMoveObjects(O*... pointers); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-10 02:23:17 -07:00
										 |  |  |     template <typename... O> | 
					
						
							|  |  |  |     void PushMoveObjects(O&... pointers); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     template <typename... O> | 
					
						
							| 
									
										
										
										
											2021-04-03 21:21:22 -07:00
										 |  |  |     void PushCopyObjects(O*... pointers); | 
					
						
							| 
									
										
										
										
											2018-07-23 22:27:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-10 02:23:17 -07:00
										 |  |  |     template <typename... O> | 
					
						
							|  |  |  |     void PushCopyObjects(O&... pointers); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-23 22:27:17 -04:00
										 |  |  | private: | 
					
						
							|  |  |  |     u32 normal_params_size{}; | 
					
						
							|  |  |  |     u32 num_handles_to_copy{}; | 
					
						
							|  |  |  |     u32 num_objects_to_move{}; ///< Domain objects or move handles, context dependent
 | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |     u32 data_payload_index{}; | 
					
						
							| 
									
										
										
										
											2020-11-08 15:49:45 -05:00
										 |  |  |     Kernel::KernelCore& kernel; | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | /// Push ///
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(s32 value) { | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |     cmdbuf[index++] = value; | 
					
						
							| 
									
										
										
										
											2019-01-29 13:09:29 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(u32 value) { | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     cmdbuf[index++] = value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T> | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  | void ResponseBuilder::PushRaw(const T& value) { | 
					
						
							| 
									
										
										
										
											2020-08-05 14:08:26 -04:00
										 |  |  |     static_assert(std::is_trivially_copyable_v<T>, | 
					
						
							|  |  |  |                   "It's undefined behavior to use memcpy with non-trivially copyable objects"); | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     std::memcpy(cmdbuf + index, &value, sizeof(T)); | 
					
						
							| 
									
										
										
										
											2020-10-20 19:07:39 -07:00
										 |  |  |     index += (sizeof(T) + 3) / 4; // round up to word length
 | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(ResultCode value) { | 
					
						
							| 
									
										
										
										
											2018-01-07 09:56:57 -05:00
										 |  |  |     // Result codes are actually 64-bit in the IPC buffer, but only the high part is discarded.
 | 
					
						
							|  |  |  |     Push(value.raw); | 
					
						
							|  |  |  |     Push<u32>(0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(s8 value) { | 
					
						
							| 
									
										
										
										
											2019-01-29 13:09:29 -05:00
										 |  |  |     PushRaw(value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(s16 value) { | 
					
						
							| 
									
										
										
										
											2019-01-29 13:09:29 -05:00
										 |  |  |     PushRaw(value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(s64 value) { | 
					
						
							|  |  |  |     PushImpl(static_cast<u32>(value)); | 
					
						
							|  |  |  |     PushImpl(static_cast<u32>(value >> 32)); | 
					
						
							| 
									
										
										
										
											2019-01-29 13:09:29 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(u8 value) { | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     PushRaw(value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(u16 value) { | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     PushRaw(value); | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(u64 value) { | 
					
						
							|  |  |  |     PushImpl(static_cast<u32>(value)); | 
					
						
							|  |  |  |     PushImpl(static_cast<u32>(value >> 32)); | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(float value) { | 
					
						
							| 
									
										
										
										
											2019-03-16 14:05:01 -04:00
										 |  |  |     u32 integral; | 
					
						
							|  |  |  |     std::memcpy(&integral, &value, sizeof(u32)); | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  |     PushImpl(integral); | 
					
						
							| 
									
										
										
										
											2019-03-16 14:05:01 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(double value) { | 
					
						
							| 
									
										
										
										
											2019-03-16 14:05:01 -04:00
										 |  |  |     u64 integral; | 
					
						
							|  |  |  |     std::memcpy(&integral, &value, sizeof(u64)); | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  |     PushImpl(integral); | 
					
						
							| 
									
										
										
										
											2019-03-16 14:05:01 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | inline void ResponseBuilder::PushImpl(bool value) { | 
					
						
							|  |  |  |     PushImpl(static_cast<u8>(value)); | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename First, typename... Other> | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  | void ResponseBuilder::Push(const First& first_value, const Other&... other_values) { | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     Push(first_value); | 
					
						
							|  |  |  |     Push(other_values...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  | template <typename... O> | 
					
						
							| 
									
										
										
										
											2021-04-03 21:21:22 -07:00
										 |  |  | inline void ResponseBuilder::PushCopyObjects(O*... pointers) { | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     auto objects = {pointers...}; | 
					
						
							|  |  |  |     for (auto& object : objects) { | 
					
						
							| 
									
										
										
										
											2021-04-10 02:23:17 -07:00
										 |  |  |         context->AddCopyObject(object); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename... O> | 
					
						
							|  |  |  | inline void ResponseBuilder::PushCopyObjects(O&... pointers) { | 
					
						
							|  |  |  |     auto objects = {&pointers...}; | 
					
						
							|  |  |  |     for (auto& object : objects) { | 
					
						
							|  |  |  |         context->AddCopyObject(object); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  | template <typename... O> | 
					
						
							| 
									
										
										
										
											2021-04-03 21:21:22 -07:00
										 |  |  | inline void ResponseBuilder::PushMoveObjects(O*... pointers) { | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     auto objects = {pointers...}; | 
					
						
							|  |  |  |     for (auto& object : objects) { | 
					
						
							| 
									
										
										
										
											2021-04-10 02:23:17 -07:00
										 |  |  |         context->AddMoveObject(object); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename... O> | 
					
						
							|  |  |  | inline void ResponseBuilder::PushMoveObjects(O&... pointers) { | 
					
						
							|  |  |  |     auto objects = {&pointers...}; | 
					
						
							|  |  |  |     for (auto& object : objects) { | 
					
						
							|  |  |  |         context->AddMoveObject(object); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | class RequestParser : public RequestHelperBase { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2018-07-23 22:40:24 -04:00
										 |  |  |     explicit RequestParser(u32* command_buffer) : RequestHelperBase(command_buffer) {} | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 12:11:36 -04:00
										 |  |  |     explicit RequestParser(Kernel::HLERequestContext& ctx) : RequestHelperBase(ctx) { | 
					
						
							|  |  |  |         ASSERT_MSG(ctx.GetDataPayloadOffset(), "context is incomplete"); | 
					
						
							|  |  |  |         Skip(ctx.GetDataPayloadOffset(), false); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         // Skip the u64 command id, it's already stored in the context
 | 
					
						
							|  |  |  |         static constexpr u32 CommandIdSize = 2; | 
					
						
							|  |  |  |         Skip(CommandIdSize, false); | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     template <typename T> | 
					
						
							|  |  |  |     T Pop(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  |     void Pop(T& value); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-05 00:29:07 +01:00
										 |  |  |     template <typename First, typename... Other> | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  |     void Pop(First& first_value, Other&... other_values); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-24 00:47:01 -04:00
										 |  |  |     template <typename T> | 
					
						
							|  |  |  |     T PopEnum() { | 
					
						
							|  |  |  |         static_assert(std::is_enum_v<T>, "T must be an enum type within a PopEnum call."); | 
					
						
							|  |  |  |         static_assert(!std::is_convertible_v<T, int>, | 
					
						
							|  |  |  |                       "enum type in PopEnum must be a strongly typed enum."); | 
					
						
							|  |  |  |         return static_cast<T>(Pop<std::underlying_type_t<T>>()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * @brief Reads the next normal parameters as a struct, by copying it | 
					
						
							|  |  |  |      * @note: The output class must be correctly packed/padded to fit hardware layout. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  |     void PopRaw(T& value); | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @brief Reads the next normal parameters as a struct, by copying it into a new value | 
					
						
							|  |  |  |      * @note: The output class must be correctly packed/padded to fit hardware layout. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     T PopRaw(); | 
					
						
							| 
									
										
										
										
											2018-01-07 09:22:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  |     template <class T> | 
					
						
							|  |  |  |     std::shared_ptr<T> PopIpcInterface() { | 
					
						
							|  |  |  |         ASSERT(context->Session()->IsDomain()); | 
					
						
							| 
									
										
										
										
											2019-03-07 16:44:28 -05:00
										 |  |  |         ASSERT(context->GetDomainMessageHeader().input_object_count > 0); | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  |         return context->GetDomainRequestHandler<T>(Pop<u32>() - 1); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Pop ///
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | inline u32 RequestParser::Pop() { | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     return cmdbuf[index++]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-26 18:04:45 -05:00
										 |  |  | template <> | 
					
						
							|  |  |  | inline s32 RequestParser::Pop() { | 
					
						
							|  |  |  |     return static_cast<s32>(Pop<u32>()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | template <typename T> | 
					
						
							|  |  |  | void RequestParser::PopRaw(T& value) { | 
					
						
							| 
									
										
										
										
											2020-08-05 14:08:26 -04:00
										 |  |  |     static_assert(std::is_trivially_copyable_v<T>, | 
					
						
							|  |  |  |                   "It's undefined behavior to use memcpy with non-trivially copyable objects"); | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     std::memcpy(&value, cmdbuf + index, sizeof(T)); | 
					
						
							| 
									
										
										
										
											2020-10-20 19:07:39 -07:00
										 |  |  |     index += (sizeof(T) + 3) / 4; // round up to word length
 | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T> | 
					
						
							|  |  |  | T RequestParser::PopRaw() { | 
					
						
							|  |  |  |     T value; | 
					
						
							|  |  |  |     PopRaw(value); | 
					
						
							|  |  |  |     return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | template <> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | inline u8 RequestParser::Pop() { | 
					
						
							|  |  |  |     return PopRaw<u8>(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | inline u16 RequestParser::Pop() { | 
					
						
							|  |  |  |     return PopRaw<u16>(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | inline u64 RequestParser::Pop() { | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     const u64 lsw = Pop<u32>(); | 
					
						
							|  |  |  |     const u64 msw = Pop<u32>(); | 
					
						
							|  |  |  |     return msw << 32 | lsw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-26 18:04:45 -05:00
										 |  |  | template <> | 
					
						
							|  |  |  | inline s8 RequestParser::Pop() { | 
					
						
							|  |  |  |     return static_cast<s8>(Pop<u8>()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | inline s16 RequestParser::Pop() { | 
					
						
							|  |  |  |     return static_cast<s16>(Pop<u16>()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 21:32:36 -05:00
										 |  |  | template <> | 
					
						
							|  |  |  | inline s64 RequestParser::Pop() { | 
					
						
							|  |  |  |     return static_cast<s64>(Pop<u64>()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-16 14:05:01 -04:00
										 |  |  | template <> | 
					
						
							|  |  |  | inline float RequestParser::Pop() { | 
					
						
							|  |  |  |     const u32 value = Pop<u32>(); | 
					
						
							|  |  |  |     float real; | 
					
						
							|  |  |  |     std::memcpy(&real, &value, sizeof(real)); | 
					
						
							|  |  |  |     return real; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | inline double RequestParser::Pop() { | 
					
						
							|  |  |  |     const u64 value = Pop<u64>(); | 
					
						
							| 
									
										
										
										
											2019-05-18 22:05:29 -04:00
										 |  |  |     double real; | 
					
						
							| 
									
										
										
										
											2019-03-16 14:05:01 -04:00
										 |  |  |     std::memcpy(&real, &value, sizeof(real)); | 
					
						
							|  |  |  |     return real; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | template <> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | inline bool RequestParser::Pop() { | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |     return Pop<u8>() != 0; | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | inline ResultCode RequestParser::Pop() { | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     return ResultCode{Pop<u32>()}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | template <typename T> | 
					
						
							|  |  |  | void RequestParser::Pop(T& value) { | 
					
						
							|  |  |  |     value = Pop<T>(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-05 00:29:07 +01:00
										 |  |  | template <typename First, typename... Other> | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | void RequestParser::Pop(First& first_value, Other&... other_values) { | 
					
						
							|  |  |  |     first_value = Pop<First>(); | 
					
						
							|  |  |  |     Pop(other_values...); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace IPC
 |