| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2019-11-24 20:15:51 -05:00
										 |  |  | #include <functional>
 | 
					
						
							| 
									
										
										
										
											2017-06-05 22:39:26 -07:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2019-03-07 16:44:28 -05:00
										 |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2023-02-03 00:08:45 -05:00
										 |  |  | #include <span>
 | 
					
						
							| 
									
										
										
										
											2018-03-18 20:22:46 -04:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2018-07-19 16:10:12 -04:00
										 |  |  | #include <type_traits>
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-08-03 21:28:54 +10:00
										 |  |  | #include "common/concepts.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | #include "common/swap.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  | #include "core/hle/ipc.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-24 02:40:31 -07:00
										 |  |  | #include "core/hle/kernel/svc_common.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | union Result; | 
					
						
							| 
									
										
										
										
											2019-03-05 09:20:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-03 12:41:30 -04:00
										 |  |  | namespace Core::Memory { | 
					
						
							|  |  |  | class Memory; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-08 15:49:45 -05:00
										 |  |  | namespace IPC { | 
					
						
							|  |  |  | class ResponseBuilder; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | namespace Service { | 
					
						
							|  |  |  | class ServiceFrameworkBase; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-30 21:13:53 -07:00
										 |  |  | enum class ServiceThreadType { | 
					
						
							|  |  |  |     Default, | 
					
						
							|  |  |  |     CreateNew, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | namespace Kernel { | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  | class Domain; | 
					
						
							|  |  |  | class HLERequestContext; | 
					
						
							| 
									
										
										
										
											2021-10-07 13:30:20 -04:00
										 |  |  | class KAutoObject; | 
					
						
							| 
									
										
										
										
											2020-05-03 12:41:30 -04:00
										 |  |  | class KernelCore; | 
					
						
							| 
									
										
										
										
											2022-10-12 20:26:04 -04:00
										 |  |  | class KEvent; | 
					
						
							| 
									
										
										
										
											2021-04-24 02:40:31 -07:00
										 |  |  | class KHandleTable; | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | class KServerPort; | 
					
						
							| 
									
										
										
										
											2021-04-23 22:04:28 -07:00
										 |  |  | class KProcess; | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  | class KServerSession; | 
					
						
							| 
									
										
										
										
											2020-12-30 23:01:08 -08:00
										 |  |  | class KThread; | 
					
						
							| 
									
										
										
										
											2021-01-29 22:48:06 -08:00
										 |  |  | class KReadableEvent; | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  | class KSession; | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | class SessionRequestManager; | 
					
						
							| 
									
										
										
										
											2021-06-04 19:26:48 -07:00
										 |  |  | class ServiceThread; | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-31 18:09:41 -05:00
										 |  |  | enum class ThreadWakeupReason; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Interface implemented by HLE Session handlers. | 
					
						
							|  |  |  |  * This can be provided to a ServerSession in order to hook into several relevant events | 
					
						
							|  |  |  |  * (such as a new connection or a SyncRequest) so they can be implemented in the emulator. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-06-05 22:39:26 -07:00
										 |  |  | class SessionRequestHandler : public std::enable_shared_from_this<SessionRequestHandler> { | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2022-03-30 21:13:53 -07:00
										 |  |  |     SessionRequestHandler(KernelCore& kernel_, const char* service_name_, | 
					
						
							|  |  |  |                           ServiceThreadType thread_type); | 
					
						
							| 
									
										
										
										
											2018-12-31 18:09:41 -05:00
										 |  |  |     virtual ~SessionRequestHandler(); | 
					
						
							| 
									
										
										
										
											2017-06-06 14:51:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Handles a sync request from the emulated application. | 
					
						
							|  |  |  |      * @param server_session The ServerSession that was triggered for this sync request, | 
					
						
							|  |  |  |      * it should be used to differentiate which client (As in ClientSession) we're answering to. | 
					
						
							|  |  |  |      * TODO(Subv): Use a wrapper structure to hold all the information relevant to | 
					
						
							|  |  |  |      * this request (ServerSession, Originator thread, Translated command buffer, etc). | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  |      * @returns Result the result code of the translate operation. | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  |     virtual Result HandleSyncRequest(Kernel::KServerSession& session, | 
					
						
							|  |  |  |                                      Kernel::HLERequestContext& context) = 0; | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     void AcceptSession(KServerPort* server_port); | 
					
						
							|  |  |  |     void RegisterSession(KServerSession* server_session, | 
					
						
							|  |  |  |                          std::shared_ptr<SessionRequestManager> manager); | 
					
						
							| 
									
										
										
										
											2021-06-04 19:26:48 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-06 19:43:38 -05:00
										 |  |  |     ServiceThread& GetServiceThread() const { | 
					
						
							| 
									
										
										
										
											2021-06-07 21:10:51 -07:00
										 |  |  |         return service_thread; | 
					
						
							| 
									
										
										
										
											2021-06-04 19:26:48 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     KernelCore& kernel; | 
					
						
							| 
									
										
										
										
											2022-11-06 19:43:38 -05:00
										 |  |  |     ServiceThread& service_thread; | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-10 23:45:54 -08:00
										 |  |  | using SessionRequestHandlerWeakPtr = std::weak_ptr<SessionRequestHandler>; | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  | using SessionRequestHandlerPtr = std::shared_ptr<SessionRequestHandler>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Manages the underlying HLE requests for a session, and whether (or not) the session should be | 
					
						
							|  |  |  |  * treated as a domain. This is managed separately from server sessions, as this state is shared | 
					
						
							|  |  |  |  * when objects are cloned. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class SessionRequestManager final { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2021-06-04 19:26:48 -07:00
										 |  |  |     explicit SessionRequestManager(KernelCore& kernel); | 
					
						
							|  |  |  |     ~SessionRequestManager(); | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool IsDomain() const { | 
					
						
							|  |  |  |         return is_domain; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void ConvertToDomain() { | 
					
						
							|  |  |  |         domain_handlers = {session_handler}; | 
					
						
							|  |  |  |         is_domain = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-15 21:57:40 -04:00
										 |  |  |     void ConvertToDomainOnRequestEnd() { | 
					
						
							|  |  |  |         convert_to_domain = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |     std::size_t DomainHandlerCount() const { | 
					
						
							|  |  |  |         return domain_handlers.size(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool HasSessionHandler() const { | 
					
						
							|  |  |  |         return session_handler != nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SessionRequestHandler& SessionHandler() { | 
					
						
							|  |  |  |         return *session_handler; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const SessionRequestHandler& SessionHandler() const { | 
					
						
							|  |  |  |         return *session_handler; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void CloseDomainHandler(std::size_t index) { | 
					
						
							|  |  |  |         if (index < DomainHandlerCount()) { | 
					
						
							|  |  |  |             domain_handlers[index] = nullptr; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2022-06-07 17:02:29 -04:00
										 |  |  |             ASSERT_MSG(false, "Unexpected handler index {}", index); | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-10 23:45:54 -08:00
										 |  |  |     SessionRequestHandlerWeakPtr DomainHandler(std::size_t index) const { | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |         ASSERT_MSG(index < DomainHandlerCount(), "Unexpected handler index {}", index); | 
					
						
							|  |  |  |         return domain_handlers.at(index); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void AppendDomainHandler(SessionRequestHandlerPtr&& handler) { | 
					
						
							|  |  |  |         domain_handlers.emplace_back(std::move(handler)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void SetSessionHandler(SessionRequestHandlerPtr&& handler) { | 
					
						
							|  |  |  |         session_handler = std::move(handler); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-06 19:43:38 -05:00
										 |  |  |     ServiceThread& GetServiceThread() const { | 
					
						
							| 
									
										
										
										
											2021-06-04 19:26:48 -07:00
										 |  |  |         return session_handler->GetServiceThread(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-07 21:55:37 -07:00
										 |  |  |     bool HasSessionRequestHandler(const HLERequestContext& context) const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-15 21:57:40 -04:00
										 |  |  |     Result HandleDomainSyncRequest(KServerSession* server_session, HLERequestContext& context); | 
					
						
							|  |  |  |     Result CompleteSyncRequest(KServerSession* server_session, HLERequestContext& context); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2022-10-15 21:57:40 -04:00
										 |  |  |     bool convert_to_domain{}; | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |     bool is_domain{}; | 
					
						
							|  |  |  |     SessionRequestHandlerPtr session_handler; | 
					
						
							|  |  |  |     std::vector<SessionRequestHandlerPtr> domain_handlers; | 
					
						
							| 
									
										
										
										
											2021-06-04 19:26:48 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     KernelCore& kernel; | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Class containing information about an in-flight IPC request being handled by an HLE service | 
					
						
							|  |  |  |  * implementation. Services should avoid using old global APIs (e.g. Kernel::GetCommandBuffer()) and | 
					
						
							|  |  |  |  * when possible use the APIs in this class to service the request. | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * HLE handle protocol | 
					
						
							|  |  |  |  * =================== | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * To avoid needing HLE services to keep a separate handle table, or having to directly modify the | 
					
						
							|  |  |  |  * requester's table, a tweaked protocol is used to receive and send handles in requests. The kernel | 
					
						
							|  |  |  |  * will decode the incoming handles into object pointers and insert a id in the buffer where the | 
					
						
							|  |  |  |  * handle would normally be. The service then calls GetIncomingHandle() with that id to get the | 
					
						
							|  |  |  |  * pointer to the object. Similarly, instead of inserting a handle into the command buffer, the | 
					
						
							|  |  |  |  * service calls AddOutgoingHandle() and stores the returned id where the handle would normally go. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The end result is similar to just giving services their own real handle tables, but since these | 
					
						
							|  |  |  |  * ids are local to a specific context, it avoids requiring services to manage handles for objects | 
					
						
							|  |  |  |  * across multiple calls and ensuring that unneeded handles are cleaned up. | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |  */ | 
					
						
							|  |  |  | class HLERequestContext { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-05-03 12:41:30 -04:00
										 |  |  |     explicit HLERequestContext(KernelCore& kernel, Core::Memory::Memory& memory, | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  |                                KServerSession* session, KThread* thread); | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |     ~HLERequestContext(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Returns a pointer to the IPC command buffer for this request.
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] u32* CommandBuffer() { | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  |         return cmd_buf.data(); | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Returns the session through which this request was made. This can be used as a map key to | 
					
						
							|  |  |  |      * access per-client data on services. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] Kernel::KServerSession* Session() { | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |         return server_session; | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-18 16:05:12 -07:00
										 |  |  |     /// Populates this context with data from the requesting process/thread.
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  |     Result PopulateFromIncomingCommandBuffer(const KHandleTable& handle_table, u32_le* src_cmdbuf); | 
					
						
							| 
									
										
										
										
											2018-10-20 14:34:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-18 16:05:12 -07:00
										 |  |  |     /// Writes data from this context back to the requesting process/thread.
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  |     Result WriteToOutgoingCommandBuffer(KThread& requesting_thread); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] u32_le GetHipcCommand() const { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         return command; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] u32_le GetTipcCommand() const { | 
					
						
							| 
									
										
										
										
											2021-05-08 02:50:47 -07:00
										 |  |  |         return static_cast<u32_le>(command_header->type.Value()) - | 
					
						
							|  |  |  |                static_cast<u32_le>(IPC::CommandType::TIPC_CommandRegion); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] u32_le GetCommand() const { | 
					
						
							| 
									
										
										
										
											2021-05-08 02:50:47 -07:00
										 |  |  |         return command_header->IsTipc() ? GetTipcCommand() : GetHipcCommand(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] bool IsTipc() const { | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |         return command_header->IsTipc(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] IPC::CommandType GetCommandType() const { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         return command_header->type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] u64 GetPID() const { | 
					
						
							| 
									
										
										
										
											2021-05-16 00:40:19 -04:00
										 |  |  |         return pid; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] u32 GetDataPayloadOffset() const { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         return data_payload_offset; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] const std::vector<IPC::BufferDescriptorX>& BufferDescriptorX() const { | 
					
						
							| 
									
										
										
										
											2017-10-18 21:38:01 -04:00
										 |  |  |         return buffer_x_desciptors; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] const std::vector<IPC::BufferDescriptorABW>& BufferDescriptorA() const { | 
					
						
							| 
									
										
										
										
											2018-01-06 21:14:14 -05:00
										 |  |  |         return buffer_a_desciptors; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] const std::vector<IPC::BufferDescriptorABW>& BufferDescriptorB() const { | 
					
						
							| 
									
										
										
										
											2018-01-07 21:25:01 -05:00
										 |  |  |         return buffer_b_desciptors; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] const std::vector<IPC::BufferDescriptorC>& BufferDescriptorC() const { | 
					
						
							| 
									
										
										
										
											2018-01-18 16:54:34 -03:00
										 |  |  |         return buffer_c_desciptors; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] const IPC::DomainMessageHeader& GetDomainMessageHeader() const { | 
					
						
							| 
									
										
										
										
											2019-03-07 16:44:28 -05:00
										 |  |  |         return domain_message_header.value(); | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] bool HasDomainMessageHeader() const { | 
					
						
							| 
									
										
										
										
											2019-03-07 16:44:28 -05:00
										 |  |  |         return domain_message_header.has_value(); | 
					
						
							| 
									
										
										
										
											2018-10-29 23:20:17 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-03 00:08:45 -05:00
										 |  |  |     /// Helper function to get a span of a buffer using the appropriate buffer descriptor
 | 
					
						
							|  |  |  |     [[nodiscard]] std::span<const u8> ReadBuffer(std::size_t buffer_index = 0) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Helper function to read a copy of a buffer using the appropriate buffer descriptor
 | 
					
						
							|  |  |  |     [[nodiscard]] std::vector<u8> ReadBufferCopy(std::size_t buffer_index = 0) const; | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Helper function to write a buffer using the appropriate buffer descriptor
 | 
					
						
							| 
									
										
										
										
											2020-04-16 22:02:08 -04:00
										 |  |  |     std::size_t WriteBuffer(const void* buffer, std::size_t size, | 
					
						
							|  |  |  |                             std::size_t buffer_index = 0) const; | 
					
						
							| 
									
										
										
										
											2022-07-16 23:48:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Helper function to write buffer B
 | 
					
						
							|  |  |  |     std::size_t WriteBufferB(const void* buffer, std::size_t size, | 
					
						
							|  |  |  |                              std::size_t buffer_index = 0) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Helper function to write buffer C
 | 
					
						
							|  |  |  |     std::size_t WriteBufferC(const void* buffer, std::size_t size, | 
					
						
							|  |  |  |                              std::size_t buffer_index = 0) const; | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-19 16:10:12 -04:00
										 |  |  |     /* Helper function to write a buffer using the appropriate buffer descriptor
 | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2020-08-03 21:28:54 +10:00
										 |  |  |      * @tparam T an arbitrary container that satisfies the | 
					
						
							|  |  |  |      *         ContiguousContainer concept in the C++ standard library or a trivially copyable type. | 
					
						
							| 
									
										
										
										
											2018-07-19 16:10:12 -04:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2020-08-03 21:28:54 +10:00
										 |  |  |      * @param data         The container/data to write into a buffer. | 
					
						
							| 
									
										
										
										
											2018-07-19 16:10:12 -04:00
										 |  |  |      * @param buffer_index The buffer in particular to write to. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-08-03 21:28:54 +10:00
										 |  |  |     template <typename T, typename = std::enable_if_t<!std::is_pointer_v<T>>> | 
					
						
							|  |  |  |     std::size_t WriteBuffer(const T& data, std::size_t buffer_index = 0) const { | 
					
						
							| 
									
										
										
										
											2022-10-26 00:41:54 -04:00
										 |  |  |         if constexpr (Common::IsContiguousContainer<T>) { | 
					
						
							| 
									
										
										
										
											2020-08-03 21:28:54 +10:00
										 |  |  |             using ContiguousType = typename T::value_type; | 
					
						
							|  |  |  |             static_assert(std::is_trivially_copyable_v<ContiguousType>, | 
					
						
							|  |  |  |                           "Container to WriteBuffer must contain trivially copyable objects"); | 
					
						
							|  |  |  |             return WriteBuffer(std::data(data), std::size(data) * sizeof(ContiguousType), | 
					
						
							|  |  |  |                                buffer_index); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             static_assert(std::is_trivially_copyable_v<T>, "T must be trivially copyable"); | 
					
						
							|  |  |  |             return WriteBuffer(&data, sizeof(T), buffer_index); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-19 16:10:12 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-14 00:14:17 -05:00
										 |  |  |     /// Helper function to get the size of the input buffer
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] std::size_t GetReadBufferSize(std::size_t buffer_index = 0) const; | 
					
						
							| 
									
										
										
										
											2018-02-14 00:14:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  |     /// Helper function to get the size of the output buffer
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] std::size_t GetWriteBufferSize(std::size_t buffer_index = 0) const; | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:14:04 -05:00
										 |  |  |     /// Helper function to derive the number of elements able to be contained in the read buffer
 | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     [[nodiscard]] std::size_t GetReadBufferNumElements(std::size_t buffer_index = 0) const { | 
					
						
							|  |  |  |         return GetReadBufferSize(buffer_index) / sizeof(T); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Helper function to derive the number of elements able to be contained in the write buffer
 | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     [[nodiscard]] std::size_t GetWriteBufferNumElements(std::size_t buffer_index = 0) const { | 
					
						
							|  |  |  |         return GetWriteBufferSize(buffer_index) / sizeof(T); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-28 01:18:06 -05:00
										 |  |  |     /// Helper function to test whether the input buffer at buffer_index can be read
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] bool CanReadBuffer(std::size_t buffer_index = 0) const; | 
					
						
							| 
									
										
										
										
											2021-01-28 01:18:06 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Helper function to test whether the output buffer at buffer_index can be written
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] bool CanWriteBuffer(std::size_t buffer_index = 0) const; | 
					
						
							| 
									
										
										
										
											2021-01-28 01:18:06 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] Handle GetCopyHandle(std::size_t index) const { | 
					
						
							| 
									
										
										
										
											2021-05-18 19:52:52 -07:00
										 |  |  |         return incoming_copy_handles.at(index); | 
					
						
							| 
									
										
										
										
											2021-03-12 10:13:31 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] Handle GetMoveHandle(std::size_t index) const { | 
					
						
							| 
									
										
										
										
											2021-05-18 19:52:52 -07:00
										 |  |  |         return incoming_move_handles.at(index); | 
					
						
							| 
									
										
										
										
											2021-03-12 10:13:31 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-20 22:18:56 -07:00
										 |  |  |     void AddMoveObject(KAutoObject* object) { | 
					
						
							| 
									
										
										
										
											2021-05-18 19:52:52 -07:00
										 |  |  |         outgoing_move_objects.emplace_back(object); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-20 22:18:56 -07:00
										 |  |  |     void AddCopyObject(KAutoObject* object) { | 
					
						
							| 
									
										
										
										
											2021-05-18 19:52:52 -07:00
										 |  |  |         outgoing_copy_objects.emplace_back(object); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |     void AddDomainObject(SessionRequestHandlerPtr object) { | 
					
						
							| 
									
										
										
										
											2021-05-18 19:52:52 -07:00
										 |  |  |         outgoing_domain_objects.emplace_back(std::move(object)); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  |     template <typename T> | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |     std::shared_ptr<T> GetDomainHandler(std::size_t index) const { | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         return std::static_pointer_cast<T>(GetManager()->DomainHandler(index).lock()); | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-10 23:45:54 -08:00
										 |  |  |     void SetSessionRequestManager(std::weak_ptr<SessionRequestManager> manager_) { | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         manager = manager_; | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] std::string Description() const; | 
					
						
							| 
									
										
										
										
											2018-02-15 17:22:11 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] KThread& GetThread() { | 
					
						
							| 
									
										
										
										
											2019-11-25 18:28:48 -05:00
										 |  |  |         return *thread; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:08:49 -05:00
										 |  |  |     [[nodiscard]] std::shared_ptr<SessionRequestManager> GetManager() const { | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         return manager.lock(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-18 16:05:12 -07:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-11-08 15:49:45 -05:00
										 |  |  |     friend class IPC::ResponseBuilder; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 02:40:31 -07:00
										 |  |  |     void ParseCommandBuffer(const KHandleTable& handle_table, u32_le* src_cmdbuf, bool incoming); | 
					
						
							| 
									
										
										
										
											2018-10-20 14:34:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  |     std::array<u32, IPC::COMMAND_BUFFER_LENGTH> cmd_buf; | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  |     Kernel::KServerSession* server_session{}; | 
					
						
							| 
									
										
										
										
											2021-04-02 18:02:10 -07:00
										 |  |  |     KThread* thread; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 19:52:52 -07:00
										 |  |  |     std::vector<Handle> incoming_move_handles; | 
					
						
							|  |  |  |     std::vector<Handle> incoming_copy_handles; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::vector<KAutoObject*> outgoing_move_objects; | 
					
						
							|  |  |  |     std::vector<KAutoObject*> outgoing_copy_objects; | 
					
						
							|  |  |  |     std::vector<SessionRequestHandlerPtr> outgoing_domain_objects; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 16:44:28 -05:00
										 |  |  |     std::optional<IPC::CommandHeader> command_header; | 
					
						
							|  |  |  |     std::optional<IPC::HandleDescriptorHeader> handle_descriptor_header; | 
					
						
							|  |  |  |     std::optional<IPC::DataPayloadHeader> data_payload_header; | 
					
						
							|  |  |  |     std::optional<IPC::DomainMessageHeader> domain_message_header; | 
					
						
							| 
									
										
										
										
											2017-10-18 21:38:01 -04:00
										 |  |  |     std::vector<IPC::BufferDescriptorX> buffer_x_desciptors; | 
					
						
							|  |  |  |     std::vector<IPC::BufferDescriptorABW> buffer_a_desciptors; | 
					
						
							|  |  |  |     std::vector<IPC::BufferDescriptorABW> buffer_b_desciptors; | 
					
						
							|  |  |  |     std::vector<IPC::BufferDescriptorABW> buffer_w_desciptors; | 
					
						
							| 
									
										
										
										
											2018-01-18 16:54:34 -03:00
										 |  |  |     std::vector<IPC::BufferDescriptorC> buffer_c_desciptors; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 00:40:19 -04:00
										 |  |  |     u32_le command{}; | 
					
						
							|  |  |  |     u64 pid{}; | 
					
						
							| 
									
										
										
										
											2021-05-15 23:49:03 -07:00
										 |  |  |     u32 write_size{}; | 
					
						
							| 
									
										
										
										
											2021-05-08 02:21:50 -07:00
										 |  |  |     u32 data_payload_offset{}; | 
					
						
							|  |  |  |     u32 handles_offset{}; | 
					
						
							|  |  |  |     u32 domain_offset{}; | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     std::weak_ptr<SessionRequestManager> manager{}; | 
					
						
							| 
									
										
										
										
											2020-05-03 12:41:30 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     KernelCore& kernel; | 
					
						
							|  |  |  |     Core::Memory::Memory& memory; | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | } // namespace Kernel
 |