| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											2020-12-28 18:23:42 -08:00
										 |  |  | #include <mutex>
 | 
					
						
							| 
									
										
										
										
											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"
 | 
					
						
							| 
									
										
										
										
											2020-12-28 18:23:42 -08:00
										 |  |  | #include "common/spin_lock.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "core/hle/kernel/hle_ipc.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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 01:29:46 -07:00
										 |  |  | namespace Kernel { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | class HLERequestContext; | 
					
						
							| 
									
										
										
										
											2021-05-10 16:08:06 -07:00
										 |  |  | class KClientPort; | 
					
						
							|  |  |  | class KServerSession; | 
					
						
							| 
									
										
										
										
											2021-06-04 19:26:48 -07:00
										 |  |  | class ServiceThread; | 
					
						
							| 
									
										
										
										
											2021-05-10 16:08:06 -07:00
										 |  |  | } // namespace Kernel
 | 
					
						
							| 
									
										
										
										
											2017-06-06 01:29:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | namespace Service { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 17:53:58 -04:00
										 |  |  | namespace FileSystem { | 
					
						
							|  |  |  | class FileSystemController; | 
					
						
							| 
									
										
										
										
											2020-11-24 14:31:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace NVFlinger { | 
					
						
							|  |  |  | class NVFlinger; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-22 17:53:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | namespace SM { | 
					
						
							|  |  |  | class ServiceManager; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 22:04:36 -07:00
										 |  |  | /// Default number of maximum connections to a server session.
 | 
					
						
							|  |  |  | static constexpr u32 ServerSessionCountMax = 0x40; | 
					
						
							|  |  |  | static_assert(ServerSessionCountMax == 0x40, | 
					
						
							|  |  |  |               "ServerSessionCountMax isn't 0x40 somehow, this assert is a reminder that this will " | 
					
						
							|  |  |  |               "break lots of things"); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2021-05-10 15:57:59 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Invokes a service request routine using the HIPC protocol.
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     void InvokeRequest(Kernel::HLERequestContext& ctx); | 
					
						
							| 
									
										
										
										
											2021-05-10 16:08:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Invokes a service request routine using the HIPC protocol.
 | 
					
						
							|  |  |  |     void InvokeRequestTipc(Kernel::HLERequestContext& ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 15:57:59 -07:00
										 |  |  |     /// Creates a port pair and registers it on the kernel's global port registry.
 | 
					
						
							| 
									
										
										
										
											2021-06-04 19:26:48 -07:00
										 |  |  |     Kernel::KClientPort& CreatePort(); | 
					
						
							| 
									
										
										
										
											2021-05-10 15:57:59 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-28 18:23:42 -08:00
										 |  |  |     /// Handles a synchronization request for the service.
 | 
					
						
							| 
									
										
										
										
											2021-05-10 16:05:37 -07:00
										 |  |  |     ResultCode HandleSyncRequest(Kernel::KServerSession& session, | 
					
						
							|  |  |  |                                  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&); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-28 18:23:42 -08:00
										 |  |  |     /// Used to gain exclusive access to the service members, e.g. from CoreTiming thread.
 | 
					
						
							|  |  |  |     [[nodiscard]] std::scoped_lock<Common::SpinLock> LockService() { | 
					
						
							|  |  |  |         return std::scoped_lock{lock_service}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-26 15:19:08 -05:00
										 |  |  |     /// System context that the service operates under.
 | 
					
						
							|  |  |  |     Core::System& system; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-26 15:19:08 -05:00
										 |  |  |     explicit ServiceFrameworkBase(Core::System& system_, 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); | 
					
						
							| 
									
										
										
										
											2021-05-10 16:08:06 -07:00
										 |  |  |     void RegisterHandlersBaseTipc(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; | 
					
						
							| 
									
										
										
										
											2021-05-10 16:08:06 -07:00
										 |  |  |     boost::container::flat_map<u32, FunctionInfoBase> handlers_tipc; | 
					
						
							| 
									
										
										
										
											2020-12-28 18:23:42 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Used to gain exclusive access to the service members, e.g. from CoreTiming thread.
 | 
					
						
							|  |  |  |     Common::SpinLock lock_service; | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |          * | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |          * @param expected_header_ request header in the command buffer which will trigger dispatch | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |          *     to this handler | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |          * @param handler_callback_ member function in this service which will be called to handle | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |          *     the request | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |          * @param name_ human-friendly name for the request. Used mostly for logging purposes. | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |          */ | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |         FunctionInfo(u32 expected_header_, HandlerFnP<Self> handler_callback_, const char* name_) | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |             : FunctionInfoBase{ | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |                   expected_header_, | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |                   // Type-erase member function pointer by casting it down to the base class.
 | 
					
						
							| 
									
										
										
										
											2021-05-16 01:46:30 -04:00
										 |  |  |                   static_cast<HandlerFnP<ServiceFrameworkBase>>(handler_callback_), name_} {} | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Initializes the handler with no functions installed. | 
					
						
							| 
									
										
										
										
											2020-11-26 15:19:08 -05:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @param system_       The system context to construct this service under. | 
					
						
							|  |  |  |      * @param service_name_ Name of the service. | 
					
						
							|  |  |  |      * @param max_sessions_ Maximum number of sessions that can be | 
					
						
							|  |  |  |      *                      connected to this service at the same time. | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-11-26 15:19:08 -05:00
										 |  |  |     explicit ServiceFramework(Core::System& system_, const char* service_name_, | 
					
						
							| 
									
										
										
										
											2021-06-09 22:04:36 -07:00
										 |  |  |                               u32 max_sessions_ = ServerSessionCountMax) | 
					
						
							| 
									
										
										
										
											2020-11-26 15:19:08 -05:00
										 |  |  |         : ServiceFrameworkBase(system_, service_name_, max_sessions_, Invoker) {} | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// 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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 16:08:06 -07:00
										 |  |  |     /// Registers handlers in the service.
 | 
					
						
							|  |  |  |     template <std::size_t N> | 
					
						
							|  |  |  |     void RegisterHandlersTipc(const FunctionInfo (&functions)[N]) { | 
					
						
							|  |  |  |         RegisterHandlersTipc(functions, N); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Registers handlers in the service. Usually prefer using the other RegisterHandlers | 
					
						
							|  |  |  |      * overload in order to avoid needing to specify the array size. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void RegisterHandlersTipc(const FunctionInfo* functions, std::size_t n) { | 
					
						
							|  |  |  |         RegisterHandlersBaseTipc(functions, n); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-24 14:31:58 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * The purpose of this class is to own any objects that need to be shared across the other service | 
					
						
							|  |  |  |  * implementations. Will be torn down when the global system instance is shutdown. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class Services final { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit Services(std::shared_ptr<SM::ServiceManager>& sm, Core::System& system); | 
					
						
							|  |  |  |     ~Services(); | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-24 14:31:58 -08:00
										 |  |  | private: | 
					
						
							|  |  |  |     std::unique_ptr<NVFlinger::NVFlinger> nv_flinger; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 00:48:02 -07:00
										 |  |  | } // namespace Service
 |