| 
									
										
											  
											
												chore: make yuzu REUSE compliant
[REUSE] is a specification that aims at making file copyright
information consistent, so that it can be both human and machine
readable. It basically requires that all files have a header containing
copyright and licensing information. When this isn't possible, like
when dealing with binary assets, generated files or embedded third-party
dependencies, it is permitted to insert copyright information in the
`.reuse/dep5` file.
Oh, and it also requires that all the licenses used in the project are
present in the `LICENSES` folder, that's why the diff is so huge.
This can be done automatically with `reuse download --all`.
The `reuse` tool also contains a handy subcommand that analyzes the
project and tells whether or not the project is (still) compliant,
`reuse lint`.
Following REUSE has a few advantages over the current approach:
- Copyright information is easy to access for users / downstream
- Files like `dist/license.md` do not need to exist anymore, as
  `.reuse/dep5` is used instead
- `reuse lint` makes it easy to ensure that copyright information of
  files like binary assets / images is always accurate and up to date
To add copyright information of files that didn't have it I looked up
who committed what and when, for each file. As yuzu contributors do not
have to sign a CLA or similar I couldn't assume that copyright ownership
was of the "yuzu Emulator Project", so I used the name and/or email of
the commit author instead.
[REUSE]: https://reuse.software
Follow-up to 01cf05bc75b1e47beb08937439f3ed9339e7b254
											
										 
											2022-05-15 02:06:02 +02:00
										 |  |  | // SPDX-FileCopyrightText: 2016 Citra Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2017-05-29 16:45:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | #include <cstring>
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  | #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-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 { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | constexpr Result ERR_REMOTE_PROCESS_DEAD{ErrorModule::HIPC, 301}; | 
					
						
							| 
									
										
										
										
											2019-03-10 18:23:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-15 23:49:03 -07:00
										 |  |  |         u32 raw_data_size = ctx.write_size = | 
					
						
							|  |  |  |             ctx.IsTipc() ? normal_params_size - 1 : 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-15 23:49:03 -07:00
										 |  |  |             raw_data_size += | 
					
						
							|  |  |  |                 static_cast<u32>(sizeof(DomainMessageHeader) / sizeof(u32) + num_domain_objects); | 
					
						
							|  |  |  |             ctx.write_size += 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-15 23:49:03 -07:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             raw_data_size += static_cast<u32>(sizeof(IPC::DataPayloadHeader) / sizeof(u32) + 4 + | 
					
						
							|  |  |  |                                               normal_params_size); | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |         header.data_size.Assign(raw_data_size); | 
					
						
							|  |  |  |         if (num_handles_to_copy || num_handles_to_move) { | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |         ctx.write_size += index; | 
					
						
							|  |  |  |         ctx.domain_offset = static_cast<u32>(index + raw_data_size / sizeof(u32)); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2022-10-11 18:16:56 -04:00
										 |  |  |             session->Initialize(nullptr, iface->GetServiceName(), | 
					
						
							|  |  |  |                                 std::make_shared<Kernel::SessionRequestManager>(kernel)); | 
					
						
							| 
									
										
										
										
											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)...)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  |     void PushImpl(Result value); | 
					
						
							| 
									
										
										
										
											2020-08-31 10:34:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | inline void ResponseBuilder::PushImpl(Result 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) { | 
					
						
							| 
									
										
										
										
											2021-06-24 18:39:43 -07:00
										 |  |  |         // TIPC does not have data payload offset
 | 
					
						
							|  |  |  |         if (!ctx.IsTipc()) { | 
					
						
							|  |  |  |             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> | 
					
						
							| 
									
										
										
										
											2022-03-10 23:45:54 -08:00
										 |  |  |     std::weak_ptr<T> PopIpcInterface() { | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  |         ASSERT(context->Session()->IsDomain()); | 
					
						
							| 
									
										
										
										
											2019-03-07 16:44:28 -05:00
										 |  |  |         ASSERT(context->GetDomainMessageHeader().input_object_count > 0); | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |         return context->GetDomainHandler<T>(Pop<u32>() - 1); | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											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>()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 11:29:44 -05:00
										 |  |  | // Ignore the -Wclass-memaccess warning on memcpy for non-trivially default constructible objects.
 | 
					
						
							|  |  |  | #if defined(__GNUC__)
 | 
					
						
							|  |  |  | #pragma GCC diagnostic push
 | 
					
						
							|  |  |  | #pragma GCC diagnostic ignored "-Wclass-memaccess"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-02-05 11:29:44 -05:00
										 |  |  | #if defined(__GNUC__)
 | 
					
						
							|  |  |  | #pragma GCC diagnostic pop
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											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 <> | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | inline Result RequestParser::Pop() { | 
					
						
							|  |  |  |     return Result{Pop<u32>()}; | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
 |