| 
									
										
										
										
											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, | 
					
						
							|  |  |  |                    u32 num_handles_to_copy = 0, u32 num_handles_to_move = 0) | 
					
						
							|  |  |  |         : RequestHelperBase(context) { | 
					
						
							|  |  |  |         memset(cmdbuf, 0, 64); | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         context.ClearIncomingObjects(); | 
					
						
							| 
									
										
										
										
											2017-06-08 21:30:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         IPC::CommandHeader header{}; | 
					
						
							|  |  |  |         header.data_size.Assign(normal_params_size * sizeof(u32)); | 
					
						
							|  |  |  |         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()) { | 
					
						
							|  |  |  |             PushRaw(IPC::DomainMessageHeader{}); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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() { | 
					
						
							|  |  |  |         auto& request_handlers = context->Domain()->request_handlers; | 
					
						
							|  |  |  |         request_handlers.push_back(std::move(std::make_shared<T>()->shared_from_this())); | 
					
						
							|  |  |  |         Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |         AlignWithPadding(); | 
					
						
							|  |  |  |         Push<u32>(static_cast<u32>(request_handlers.size())); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  |     // TODO : ensure that translate params are added after all regular params
 | 
					
						
							|  |  |  |     template <typename... H> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     void PushCopyHandles(H... handles); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     template <typename... H> | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     void PushMoveHandles(H... handles); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  |     template <typename... O> | 
					
						
							|  |  |  |     void PushObjects(Kernel::SharedPtr<O>... pointers); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     void PushCurrentPIDHandle(); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-27 00:26:09 +01:00
										 |  |  |     void PushStaticBuffer(VAddr buffer_vaddr, size_t size, u8 buffer_id); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-27 00:26:09 +01:00
										 |  |  |     void PushMappedBuffer(VAddr buffer_vaddr, size_t size, MappedBufferPermissions perms); | 
					
						
							| 
									
										
										
										
											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...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename... H> | 
					
						
							|  |  |  | inline void RequestBuilder::PushCopyHandles(H... handles) { | 
					
						
							|  |  |  |     Push(CopyHandleDesc(sizeof...(H))); | 
					
						
							|  |  |  |     Push(static_cast<Kernel::Handle>(handles)...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename... H> | 
					
						
							|  |  |  | inline void RequestBuilder::PushMoveHandles(H... handles) { | 
					
						
							|  |  |  |     Push(MoveHandleDesc(sizeof...(H))); | 
					
						
							|  |  |  |     Push(static_cast<Kernel::Handle>(handles)...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  | template <typename... O> | 
					
						
							|  |  |  | inline void RequestBuilder::PushObjects(Kernel::SharedPtr<O>... pointers) { | 
					
						
							|  |  |  |     PushMoveHandles(context->AddOutgoingHandle(std::move(pointers))...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  | inline void RequestBuilder::PushCurrentPIDHandle() { | 
					
						
							|  |  |  |     Push(CallingPidDesc()); | 
					
						
							|  |  |  |     Push(u32(0)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-27 00:26:09 +01:00
										 |  |  | inline void RequestBuilder::PushStaticBuffer(VAddr buffer_vaddr, size_t size, u8 buffer_id) { | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |     Push(StaticBufferDesc(size, buffer_id)); | 
					
						
							|  |  |  |     Push(buffer_vaddr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-27 00:26:09 +01:00
										 |  |  | inline void RequestBuilder::PushMappedBuffer(VAddr buffer_vaddr, size_t size, | 
					
						
							| 
									
										
										
										
											2016-12-30 15:54:40 +01:00
										 |  |  |                                              MappedBufferPermissions perms) { | 
					
						
							|  |  |  |     Push(MappedBufferDesc(size, perms)); | 
					
						
							|  |  |  |     Push(buffer_vaddr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  |     /// Equivalent to calling `PopHandles<1>()[0]`.
 | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  |     Kernel::Handle PopHandle(); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Pops a descriptor containing `N` handles. The handles are returned as an array. The | 
					
						
							|  |  |  |      * descriptor must contain exactly `N` handles, it is not permitted to, for example, call | 
					
						
							|  |  |  |      * PopHandles<1>() twice to read a multi-handle descriptor with 2 handles, or to make a single | 
					
						
							|  |  |  |      * PopHandles<2>() call to read 2 single-handle descriptors. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <unsigned int N> | 
					
						
							|  |  |  |     std::array<Kernel::Handle, N> PopHandles(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Convenience wrapper around PopHandles() which assigns the handles to the passed references.
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  |     template <typename... H> | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  |     void PopHandles(H&... handles) { | 
					
						
							|  |  |  |         std::tie(handles...) = PopHandles<sizeof...(H)>(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Equivalent to calling `PopGenericObjects<1>()[0]`.
 | 
					
						
							|  |  |  |     Kernel::SharedPtr<Kernel::Object> PopGenericObject(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Equivalent to calling `std::get<0>(PopObjects<T>())`.
 | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     Kernel::SharedPtr<T> PopObject(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Pop a descriptor containing `N` handles and resolves them to Kernel::Object pointers. If a | 
					
						
							|  |  |  |      * handle is invalid, null is returned for that object instead. The same caveats from | 
					
						
							|  |  |  |      * PopHandles() apply regarding `N` matching the number of handles in the descriptor. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <unsigned int N> | 
					
						
							|  |  |  |     std::array<Kernel::SharedPtr<Kernel::Object>, N> PopGenericObjects(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Resolves handles to Kernel::Objects as in PopGenericsObjects(), but then also casts them to | 
					
						
							|  |  |  |      * the passed `T` types, while verifying that the cast is valid. If the type of an object does | 
					
						
							|  |  |  |      * not match, null is returned instead. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <typename... T> | 
					
						
							|  |  |  |     std::tuple<Kernel::SharedPtr<T>...> PopObjects(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Convenience wrapper around PopObjects() which assigns the handles to the passed references.
 | 
					
						
							|  |  |  |     template <typename... T> | 
					
						
							|  |  |  |     void PopObjects(Kernel::SharedPtr<T>&... pointers) { | 
					
						
							|  |  |  |         std::tie(pointers...) = PopObjects<T...>(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @brief Pops the static buffer vaddr | 
					
						
							|  |  |  |      * @return                  The virtual address of the buffer | 
					
						
							|  |  |  |      * @param[out] data_size    If non-null, the pointed value will be set to the size of the data | 
					
						
							|  |  |  |      * @param[out] useStaticBuffersToGetVaddr Indicates if we should read the vaddr from the static | 
					
						
							|  |  |  |      * buffers (which is the correct thing to do, but no service presently implement it) instead of | 
					
						
							|  |  |  |      * using the same value as the process who sent the request | 
					
						
							|  |  |  |      * given by the source process | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Static buffers must be set up before any IPC request using those is sent. | 
					
						
							|  |  |  |      * It is the duty of the process (usually services) to allocate and set up the receiving static | 
					
						
							|  |  |  |      * buffer information | 
					
						
							|  |  |  |      * Please note that the setup uses virtual addresses. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  |     VAddr PopStaticBuffer(size_t* data_size = nullptr, bool useStaticBuffersToGetVaddr = false); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @brief Pops the mapped buffer vaddr | 
					
						
							|  |  |  |      * @return                  The virtual address of the buffer | 
					
						
							|  |  |  |      * @param[out] data_size    If non-null, the pointed value will be set to the size of the data | 
					
						
							|  |  |  |      * given by the source process | 
					
						
							|  |  |  |      * @param[out] buffer_perms If non-null, the pointed value will be set to the permissions of the | 
					
						
							|  |  |  |      * buffer | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     VAddr PopMappedBuffer(size_t* data_size = nullptr, | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  |                           MappedBufferPermissions* buffer_perms = nullptr); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | inline Kernel::Handle RequestParser::PopHandle() { | 
					
						
							|  |  |  |     const u32 handle_descriptor = Pop<u32>(); | 
					
						
							|  |  |  |     DEBUG_ASSERT_MSG(IsHandleDescriptor(handle_descriptor), | 
					
						
							|  |  |  |                      "Tried to pop handle(s) but the descriptor is not a handle descriptor"); | 
					
						
							|  |  |  |     DEBUG_ASSERT_MSG(HandleNumberFromDesc(handle_descriptor) == 1, | 
					
						
							|  |  |  |                      "Descriptor indicates that there isn't exactly one handle"); | 
					
						
							|  |  |  |     return Pop<Kernel::Handle>(); | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 00:51:18 -07:00
										 |  |  | template <unsigned int N> | 
					
						
							|  |  |  | std::array<Kernel::Handle, N> RequestParser::PopHandles() { | 
					
						
							|  |  |  |     u32 handle_descriptor = Pop<u32>(); | 
					
						
							|  |  |  |     ASSERT_MSG(IsHandleDescriptor(handle_descriptor), | 
					
						
							|  |  |  |                "Tried to pop handle(s) but the descriptor is not a handle descriptor"); | 
					
						
							|  |  |  |     ASSERT_MSG(N == HandleNumberFromDesc(handle_descriptor), | 
					
						
							|  |  |  |                "Number of handles doesn't match the descriptor"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::array<Kernel::Handle, N> handles{}; | 
					
						
							|  |  |  |     for (Kernel::Handle& handle : handles) { | 
					
						
							|  |  |  |         handle = Pop<Kernel::Handle>(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return handles; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline Kernel::SharedPtr<Kernel::Object> RequestParser::PopGenericObject() { | 
					
						
							|  |  |  |     Kernel::Handle handle = PopHandle(); | 
					
						
							|  |  |  |     return context->GetIncomingHandle(handle); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T> | 
					
						
							|  |  |  | Kernel::SharedPtr<T> RequestParser::PopObject() { | 
					
						
							|  |  |  |     return Kernel::DynamicObjectCast<T>(PopGenericObject()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <unsigned int N> | 
					
						
							|  |  |  | inline std::array<Kernel::SharedPtr<Kernel::Object>, N> RequestParser::PopGenericObjects() { | 
					
						
							|  |  |  |     std::array<Kernel::Handle, N> handles = PopHandles<N>(); | 
					
						
							|  |  |  |     std::array<Kernel::SharedPtr<Kernel::Object>, N> pointers; | 
					
						
							|  |  |  |     for (int i = 0; i < N; ++i) { | 
					
						
							|  |  |  |         pointers[i] = context->GetIncomingHandle(handles[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return pointers; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace detail { | 
					
						
							|  |  |  | template <typename... T, size_t... I> | 
					
						
							|  |  |  | std::tuple<Kernel::SharedPtr<T>...> PopObjectsHelper( | 
					
						
							|  |  |  |     std::array<Kernel::SharedPtr<Kernel::Object>, sizeof...(T)>&& pointers, | 
					
						
							|  |  |  |     std::index_sequence<I...>) { | 
					
						
							|  |  |  |     return std::make_tuple(Kernel::DynamicObjectCast<T>(std::move(pointers[I]))...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | } // namespace detail
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename... T> | 
					
						
							|  |  |  | inline std::tuple<Kernel::SharedPtr<T>...> RequestParser::PopObjects() { | 
					
						
							|  |  |  |     return detail::PopObjectsHelper<T...>(PopGenericObjects<sizeof...(T)>(), | 
					
						
							|  |  |  |                                           std::index_sequence_for<T...>{}); | 
					
						
							| 
									
										
										
										
											2016-12-26 14:42:06 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline VAddr RequestParser::PopStaticBuffer(size_t* data_size, bool useStaticBuffersToGetVaddr) { | 
					
						
							|  |  |  |     const u32 sbuffer_descriptor = Pop<u32>(); | 
					
						
							|  |  |  |     StaticBufferDescInfo bufferInfo{sbuffer_descriptor}; | 
					
						
							|  |  |  |     if (data_size != nullptr) | 
					
						
							|  |  |  |         *data_size = bufferInfo.size; | 
					
						
							|  |  |  |     if (!useStaticBuffersToGetVaddr) | 
					
						
							|  |  |  |         return Pop<VAddr>(); | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         ASSERT_MSG(0, "remove the assert if multiprocess/IPC translation are implemented."); | 
					
						
							|  |  |  |         // The buffer has already been copied to the static buffer by the kernel during
 | 
					
						
							|  |  |  |         // translation
 | 
					
						
							|  |  |  |         Pop<VAddr>(); // Pop the calling process buffer address
 | 
					
						
							|  |  |  |                       // and get the vaddr from the static buffers
 | 
					
						
							|  |  |  |         return cmdbuf[(0x100 >> 2) + bufferInfo.buffer_id * 2 + 1]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline VAddr RequestParser::PopMappedBuffer(size_t* data_size, | 
					
						
							|  |  |  |                                             MappedBufferPermissions* buffer_perms) { | 
					
						
							|  |  |  |     const u32 sbuffer_descriptor = Pop<u32>(); | 
					
						
							|  |  |  |     MappedBufferDescInfo bufferInfo{sbuffer_descriptor}; | 
					
						
							|  |  |  |     if (data_size != nullptr) | 
					
						
							|  |  |  |         *data_size = bufferInfo.size; | 
					
						
							|  |  |  |     if (buffer_perms != nullptr) | 
					
						
							|  |  |  |         *buffer_perms = bufferInfo.perms; | 
					
						
							|  |  |  |     return Pop<VAddr>(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 13:39:45 +01:00
										 |  |  | } // namespace IPC
 |