| 
									
										
										
										
											2018-01-13 16:22:39 -05:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 15:44:11 +01:00
										 |  |  | #include <cstddef>
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2015-01-30 16:56:49 -02:00
										 |  |  | #include <boost/container/flat_map.hpp>
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "core/hle/kernel/hle_ipc.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | #include "core/hle/kernel/object.h"
 | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Namespace Service
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | namespace Core { | 
					
						
							|  |  |  | class System; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							|  |  |  | class VfsFilesystem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 01:29:46 -07:00
										 |  |  | namespace Kernel { | 
					
						
							|  |  |  | class ClientPort; | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | class ServerPort; | 
					
						
							| 
									
										
										
										
											2017-06-06 01:29:46 -07:00
										 |  |  | class ServerSession; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | class HLERequestContext; | 
					
						
							| 
									
										
										
										
											2018-01-20 00:48:02 -07:00
										 |  |  | } // namespace Kernel
 | 
					
						
							| 
									
										
										
										
											2017-06-06 01:29:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | namespace Service { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | namespace SM { | 
					
						
							|  |  |  | class ServiceManager; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-14 03:30:11 -02:00
										 |  |  | static const int kMaxPortSize = 8; ///< Maximum size of a port name (8 characters)
 | 
					
						
							| 
									
										
										
										
											2016-12-14 12:33:49 -05:00
										 |  |  | /// Arbitrary default number of maximum connections to an HLE service.
 | 
					
						
							|  |  |  | static const u32 DefaultMaxSessions = 10; | 
					
						
							| 
									
										
										
										
											2014-05-07 21:04:55 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This is an non-templated base of ServiceFramework to reduce code bloat and compilation times, it | 
					
						
							|  |  |  |  * is not meant to be used directly. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @see ServiceFramework | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class ServiceFrameworkBase : public Kernel::SessionRequestHandler { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     /// Returns the string identifier used to connect to the service.
 | 
					
						
							|  |  |  |     std::string GetServiceName() const { | 
					
						
							|  |  |  |         return service_name; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Returns the maximum number of sessions that can be connected to this service at the same | 
					
						
							|  |  |  |      * time. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     u32 GetMaxSessions() const { | 
					
						
							|  |  |  |         return max_sessions; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Creates a port pair and registers this service with the given ServiceManager.
 | 
					
						
							|  |  |  |     void InstallAsService(SM::ServiceManager& service_manager); | 
					
						
							|  |  |  |     /// Creates a port pair and registers it on the kernel's global port registry.
 | 
					
						
							|  |  |  |     void InstallAsNamedPort(); | 
					
						
							| 
									
										
										
										
											2017-10-18 21:34:19 -04:00
										 |  |  |     /// Creates and returns an unregistered port for the service.
 | 
					
						
							|  |  |  |     Kernel::SharedPtr<Kernel::ClientPort> CreatePort(); | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     void InvokeRequest(Kernel::HLERequestContext& ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     ResultCode HandleSyncRequest(Kernel::HLERequestContext& context) override; | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     /// Member-function pointer type of SyncRequest handlers.
 | 
					
						
							|  |  |  |     template <typename Self> | 
					
						
							|  |  |  |     using HandlerFnP = void (Self::*)(Kernel::HLERequestContext&); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     friend class ServiceFramework; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct FunctionInfoBase { | 
					
						
							|  |  |  |         u32 expected_header; | 
					
						
							|  |  |  |         HandlerFnP<ServiceFrameworkBase> handler_callback; | 
					
						
							|  |  |  |         const char* name; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     using InvokerFn = void(ServiceFrameworkBase* object, HandlerFnP<ServiceFrameworkBase> member, | 
					
						
							|  |  |  |                            Kernel::HLERequestContext& ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ServiceFrameworkBase(const char* service_name, u32 max_sessions, InvokerFn* handler_invoker); | 
					
						
							| 
									
										
										
										
											2019-04-04 12:16:56 -04:00
										 |  |  |     ~ServiceFrameworkBase() override; | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     void RegisterHandlersBase(const FunctionInfoBase* functions, std::size_t n); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     void ReportUnimplementedFunction(Kernel::HLERequestContext& ctx, const FunctionInfoBase* info); | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Identifier string used to connect to the service.
 | 
					
						
							|  |  |  |     std::string service_name; | 
					
						
							|  |  |  |     /// Maximum number of concurrent sessions that this service can handle.
 | 
					
						
							|  |  |  |     u32 max_sessions; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-12 20:07:14 +01:00
										 |  |  |     /// Flag to store if a port was already create/installed to detect multiple install attempts,
 | 
					
						
							|  |  |  |     /// which is not supported.
 | 
					
						
							|  |  |  |     bool port_installed = false; | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Function used to safely up-cast pointers to the derived class before invoking a handler.
 | 
					
						
							|  |  |  |     InvokerFn* handler_invoker; | 
					
						
							|  |  |  |     boost::container::flat_map<u32, FunctionInfoBase> handlers; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Framework for implementing HLE services. Dispatches on the header id of incoming SyncRequests | 
					
						
							|  |  |  |  * based on a table mapping header ids to handler functions. Service implementations should inherit | 
					
						
							|  |  |  |  * from ServiceFramework using the CRTP (`class Foo : public ServiceFramework<Foo> { ... };`) and | 
					
						
							|  |  |  |  * populate it with handlers by calling #RegisterHandlers. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * In order to avoid duplicating code in the binary and exposing too many implementation details in | 
					
						
							|  |  |  |  * the header, this class is split into a non-templated base (ServiceFrameworkBase) and a template | 
					
						
							|  |  |  |  * deriving from it (ServiceFramework). The functions in this class will mostly only erase the type | 
					
						
							|  |  |  |  * of the passed in function pointers and then delegate the actual work to the implementation in the | 
					
						
							|  |  |  |  * base class. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template <typename Self> | 
					
						
							|  |  |  | class ServiceFramework : public ServiceFrameworkBase { | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     /// Contains information about a request type which is handled by the service.
 | 
					
						
							|  |  |  |     struct FunctionInfo : FunctionInfoBase { | 
					
						
							|  |  |  |         // TODO(yuriks): This function could be constexpr, but clang is the only compiler that
 | 
					
						
							|  |  |  |         // doesn't emit an ICE or a wrong diagnostic because of the static_cast.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /**
 | 
					
						
							|  |  |  |          * Constructs a FunctionInfo for a function. | 
					
						
							|  |  |  |          * | 
					
						
							|  |  |  |          * @param expected_header request header in the command buffer which will trigger dispatch | 
					
						
							|  |  |  |          *     to this handler | 
					
						
							|  |  |  |          * @param handler_callback member function in this service which will be called to handle | 
					
						
							|  |  |  |          *     the request | 
					
						
							|  |  |  |          * @param name human-friendly name for the request. Used mostly for logging purposes. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         FunctionInfo(u32 expected_header, HandlerFnP<Self> handler_callback, const char* name) | 
					
						
							|  |  |  |             : FunctionInfoBase{ | 
					
						
							|  |  |  |                   expected_header, | 
					
						
							|  |  |  |                   // Type-erase member function pointer by casting it down to the base class.
 | 
					
						
							|  |  |  |                   static_cast<HandlerFnP<ServiceFrameworkBase>>(handler_callback), name} {} | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Initializes the handler with no functions installed. | 
					
						
							|  |  |  |      * @param max_sessions Maximum number of sessions that can be | 
					
						
							|  |  |  |      * connected to this service at the same time. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-07-23 23:10:48 -04:00
										 |  |  |     explicit ServiceFramework(const char* service_name, u32 max_sessions = DefaultMaxSessions) | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |         : ServiceFrameworkBase(service_name, max_sessions, Invoker) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Registers handlers in the service.
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     template <std::size_t N> | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |     void RegisterHandlers(const FunctionInfo (&functions)[N]) { | 
					
						
							|  |  |  |         RegisterHandlers(functions, N); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Registers handlers in the service. Usually prefer using the other RegisterHandlers | 
					
						
							|  |  |  |      * overload in order to avoid needing to specify the array size. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     void RegisterHandlers(const FunctionInfo* functions, std::size_t n) { | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |         RegisterHandlersBase(functions, n); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * This function is used to allow invocation of pointers to handlers stored in the base class | 
					
						
							|  |  |  |      * without needing to expose the type of this derived class. Pointers-to-member may require a | 
					
						
							|  |  |  |      * fixup when being up or downcast, and thus code that does that needs to know the concrete type | 
					
						
							|  |  |  |      * of the derived class in order to invoke one of it's functions through a pointer. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     static void Invoker(ServiceFrameworkBase* object, HandlerFnP<ServiceFrameworkBase> member, | 
					
						
							|  |  |  |                         Kernel::HLERequestContext& ctx) { | 
					
						
							|  |  |  |         // Cast back up to our original types and call the member function
 | 
					
						
							|  |  |  |         (static_cast<Self*>(object)->*static_cast<HandlerFnP<Self>>(member))(ctx); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | /// Initialize ServiceManager
 | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | void Init(std::shared_ptr<SM::ServiceManager>& sm, Core::System& system, | 
					
						
							|  |  |  |           FileSys::VfsFilesystem& vfs); | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | /// Shutdown ServiceManager
 | 
					
						
							|  |  |  | void Shutdown(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 00:48:02 -07:00
										 |  |  | } // namespace Service
 |