| 
									
										
										
										
											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>
 | 
					
						
							|  |  |  | #include <tuple>
 | 
					
						
							|  |  |  | #include <type_traits>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | #include "core/hle/ipc.h"
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  | #include "core/hle/kernel/domain.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-29 16:45:42 -07:00
										 |  |  | #include "core/hle/kernel/handle_table.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | #include "core/hle/kernel/hle_ipc.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace IPC { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     ptrdiff_t index = 0; | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     RequestHelperBase(u32* command_buffer) : cmdbuf(command_buffer) {} | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     RequestHelperBase(Kernel::HLERequestContext& context) | 
					
						
							|  |  |  |         : context(&context), cmdbuf(context.CommandBuffer()) {} | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void ValidateHeader() { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         // DEBUG_ASSERT_MSG(index == TotalSize(), "Operations do not match the header (cmd 0x%x)",
 | 
					
						
							|  |  |  |         //                 header.raw);
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     void Skip(unsigned size_in_words, bool set_to_null) { | 
					
						
							|  |  |  |         if (set_to_null) | 
					
						
							|  |  |  |             memset(cmdbuf + index, 0, size_in_words * sizeof(u32)); | 
					
						
							|  |  |  |         index += size_in_words; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  |             Skip(4 - (index & 3), true); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned GetCurrentOffset() const { | 
					
						
							|  |  |  |         return static_cast<unsigned>(index); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class RequestBuilder : public RequestHelperBase { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     RequestBuilder(u32* command_buffer) : RequestHelperBase(command_buffer) {} | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     RequestBuilder(Kernel::HLERequestContext& context, unsigned normal_params_size, | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |                    u32 num_handles_to_copy = 0, u32 num_handles_to_move = 0, | 
					
						
							|  |  |  |                    u32 num_domain_objects = 0) | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         : RequestHelperBase(context) { | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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.
 | 
					
						
							| 
									
										
										
										
											2018-01-06 21:14:14 -05:00
										 |  |  |         u32 raw_data_size = sizeof(IPC::DataPayloadHeader) / 4 + 4 + normal_params_size; | 
					
						
							|  |  |  |         if (context.IsDomain()) | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |             raw_data_size += sizeof(DomainMessageHeader) / 4 + num_domain_objects; | 
					
						
							| 
									
										
										
										
											2018-01-06 21:14:14 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         header.data_size.Assign(raw_data_size); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         if (num_handles_to_copy || num_handles_to_move) { | 
					
						
							|  |  |  |             header.enable_handle_descriptor.Assign(1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PushRaw(header); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (header.enable_handle_descriptor) { | 
					
						
							|  |  |  |             IPC::HandleDescriptorHeader handle_descriptor_header{}; | 
					
						
							|  |  |  |             handle_descriptor_header.num_handles_to_copy.Assign(num_handles_to_copy); | 
					
						
							|  |  |  |             handle_descriptor_header.num_handles_to_move.Assign(num_handles_to_move); | 
					
						
							|  |  |  |             PushRaw(handle_descriptor_header); | 
					
						
							|  |  |  |             Skip(num_handles_to_copy + num_handles_to_move, true); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         AlignWithPadding(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |         if (context.IsDomain()) { | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |             IPC::DomainMessageHeader domain_header{}; | 
					
						
							| 
									
										
										
										
											2018-01-06 21:14:14 -05:00
										 |  |  |             domain_header.num_objects = num_domain_objects; | 
					
						
							|  |  |  |             PushRaw(domain_header); | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         IPC::DataPayloadHeader data_payload_header{}; | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |         data_payload_header.magic = Common::MakeMagic('S', 'F', 'C', 'O'); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         PushRaw(data_payload_header); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     template <class T> | 
					
						
							|  |  |  |     void PushIpcInterface() { | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         context->AddDomainObject(std::make_shared<T>()); | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     // Validate on destruction, as there shouldn't be any case where we don't want it
 | 
					
						
							|  |  |  |     ~RequestBuilder() { | 
					
						
							|  |  |  |         ValidateHeader(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     void Push(T value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @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> | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     void PushMoveObjects(Kernel::SharedPtr<O>... pointers); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     template <typename... O> | 
					
						
							|  |  |  |     void PushCopyObjects(Kernel::SharedPtr<O>... pointers); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | /// Push ///
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | inline void RequestBuilder::Push(u32 value) { | 
					
						
							|  |  |  |     cmdbuf[index++] = value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T> | 
					
						
							|  |  |  | void RequestBuilder::PushRaw(const T& value) { | 
					
						
							|  |  |  |     std::memcpy(cmdbuf + index, &value, sizeof(T)); | 
					
						
							|  |  |  |     index += (sizeof(T) + 3) / 4; // round up to word length
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | inline void RequestBuilder::Push(u8 value) { | 
					
						
							|  |  |  |     PushRaw(value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | inline void RequestBuilder::Push(u16 value) { | 
					
						
							|  |  |  |     PushRaw(value); | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | inline void RequestBuilder::Push(u64 value) { | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  |     Push(static_cast<u32>(value)); | 
					
						
							|  |  |  |     Push(static_cast<u32>(value >> 32)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | inline void RequestBuilder::Push(bool value) { | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |     Push(static_cast<u8>(value)); | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | inline void RequestBuilder::Push(ResultCode value) { | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  |     Push(value.raw); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | template <typename First, typename... Other> | 
					
						
							|  |  |  | void RequestBuilder::Push(const First& first_value, const Other&... other_values) { | 
					
						
							|  |  |  |     Push(first_value); | 
					
						
							|  |  |  |     Push(other_values...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  | template <typename... O> | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  | inline void RequestBuilder::PushCopyObjects(Kernel::SharedPtr<O>... pointers) { | 
					
						
							|  |  |  |     auto objects = {pointers...}; | 
					
						
							|  |  |  |     for (auto& object : objects) { | 
					
						
							|  |  |  |         context->AddCopyObject(std::move(object)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  | template <typename... O> | 
					
						
							|  |  |  | inline void RequestBuilder::PushMoveObjects(Kernel::SharedPtr<O>... pointers) { | 
					
						
							|  |  |  |     auto objects = {pointers...}; | 
					
						
							|  |  |  |     for (auto& object : objects) { | 
					
						
							|  |  |  |         context->AddMoveObject(std::move(object)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | class RequestParser : public RequestHelperBase { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     RequestParser(u32* command_buffer) : RequestHelperBase(command_buffer) {} | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     RequestParser(Kernel::HLERequestContext& context) : RequestHelperBase(context) { | 
					
						
							|  |  |  |         ASSERT_MSG(context.GetDataPayloadOffset(), "context is incomplete"); | 
					
						
							|  |  |  |         Skip(context.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
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     RequestBuilder MakeBuilder(u32 normal_params_size, u32 num_handles_to_copy, | 
					
						
							|  |  |  |                                u32 num_handles_to_move, bool validate_header = true) { | 
					
						
							|  |  |  |         if (validate_header) { | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |             ValidateHeader(); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return {*context, normal_params_size, num_handles_to_copy, num_handles_to_move}; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @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(); | 
					
						
							| 
									
										
										
										
											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++]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | template <typename T> | 
					
						
							|  |  |  | void RequestParser::PopRaw(T& value) { | 
					
						
							|  |  |  |     std::memcpy(&value, cmdbuf + index, sizeof(T)); | 
					
						
							|  |  |  |     index += (sizeof(T) + 3) / 4; // round up to word length
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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
 |