| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											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:07:04 -02:00
										 |  |  | #include <unordered_map>
 | 
					
						
							| 
									
										
										
										
											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"
 | 
					
						
							| 
									
										
										
										
											2016-12-08 11:06:19 -05:00
										 |  |  | #include "core/hle/ipc.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | #include "core/hle/kernel/client_port.h"
 | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-21 15:44:11 +01:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Namespace Service
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Service { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-14 03:30:11 -02:00
										 |  |  | static const int kMaxPortSize = 8; ///< Maximum size of a port name (8 characters)
 | 
					
						
							| 
									
										
										
										
											2016-12-05 13:44:41 -05:00
										 |  |  | static const u32 DefaultMaxSessions = 10; ///< Arbitrary default number of maximum connections to an HLE service
 | 
					
						
							| 
									
										
										
										
											2014-05-07 21:04:55 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-05 11:02:08 -05: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. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | class SessionRequestHandler { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Dispatches and 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. | 
					
						
							| 
									
										
										
										
											2016-12-05 12:05:00 -05:00
										 |  |  |      * @returns ResultCode the result code of the translate operation. | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-05 12:05:00 -05:00
										 |  |  |     ResultCode HandleSyncRequest(Kernel::SharedPtr<Kernel::ServerSession> server_session); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:01:10 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Signals that a client has just connected to this HLE handler and keeps the | 
					
						
							|  |  |  |      * associated ServerSession alive for the duration of the connection. | 
					
						
							|  |  |  |      * @param server_session Owning pointer to the ServerSession associated with the connection. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void ClientConnected(Kernel::SharedPtr<Kernel::ServerSession> server_session); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Signals that a client has just disconnected from this HLE handler and releases the | 
					
						
							|  |  |  |      * associated ServerSession. | 
					
						
							|  |  |  |      * @param server_session ServerSession associated with the connection. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void ClientDisconnected(Kernel::SharedPtr<Kernel::ServerSession> server_session); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-05 12:05:00 -05:00
										 |  |  | protected: | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Handles a sync request from the emulated application and writes the response to the command buffer. | 
					
						
							|  |  |  |      * TODO(Subv): Use a wrapper structure to hold all the information relevant to | 
					
						
							|  |  |  |      * this request (ServerSession, Originator thread, Translated command buffer, etc). | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     virtual void HandleSyncRequestImpl(Kernel::SharedPtr<Kernel::ServerSession> server_session) = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Performs command buffer translation for this request. | 
					
						
							|  |  |  |      * The command buffer from the ServerSession thread's TLS is copied into a | 
					
						
							|  |  |  |      * buffer and all descriptors in the buffer are processed. | 
					
						
							|  |  |  |      * TODO(Subv): Implement this function, currently we do not support multiple processes running at once, | 
					
						
							|  |  |  |      * but once that is implemented we'll need to properly translate all descriptors in the command buffer. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ResultCode TranslateRequest(Kernel::SharedPtr<Kernel::ServerSession> server_session); | 
					
						
							| 
									
										
										
										
											2016-12-08 15:01:10 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// List of sessions that are connected to this handler.
 | 
					
						
							|  |  |  |     /// A ServerSession whose server endpoint is an HLE implementation is kept alive by this list for the duration of the connection.
 | 
					
						
							|  |  |  |     std::vector<Kernel::SharedPtr<Kernel::ServerSession>> connected_sessions; | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-05 11:02:08 -05:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Framework for implementing HLE service handlers which dispatch incoming SyncRequests based on a table mapping header ids to handler functions. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | class Interface : public SessionRequestHandler { | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2016-12-05 13:44:41 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Creates an HLE interface with the specified max sessions. | 
					
						
							|  |  |  |      * @param max_sessions Maximum number of sessions that can be | 
					
						
							|  |  |  |      * connected to this service at the same time. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-08 11:06:19 -05:00
										 |  |  |     Interface(u32 max_sessions = DefaultMaxSessions); | 
					
						
							| 
									
										
										
										
											2016-12-05 13:44:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 11:06:19 -05:00
										 |  |  |     virtual ~Interface(); | 
					
						
							| 
									
										
										
										
											2016-12-05 13:44:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-17 17:09:43 -05:00
										 |  |  |     std::string GetName() const { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         return GetPortName(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-20 08:50:48 +03:00
										 |  |  |     virtual void SetVersion(u32 raw_version) { | 
					
						
							|  |  |  |         version.raw = raw_version; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2016-12-05 13:44:41 -05:00
										 |  |  |      * Gets the maximum allowed number of sessions that can be connected to this service at the same time. | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  |      * @returns The maximum number of connections allowed. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-05 13:44:41 -05:00
										 |  |  |     u32 GetMaxSessions() const { return max_sessions; } | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-24 22:16:54 -04:00
										 |  |  |     typedef void (*Function)(Interface*); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct FunctionInfo { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         u32 id; | 
					
						
							|  |  |  |         Function func; | 
					
						
							| 
									
										
										
										
											2015-01-30 17:15:02 -02:00
										 |  |  |         const char* name; | 
					
						
							| 
									
										
										
										
											2014-04-24 22:16:54 -04:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Gets the string name used by CTROS for a service | 
					
						
							|  |  |  |      * @return Port name of service | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-08-17 23:03:22 -04:00
										 |  |  |     virtual std::string GetPortName() const { | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  |         return "[UNKNOWN SERVICE PORT]"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-15 22:40:19 -04:00
										 |  |  | protected: | 
					
						
							| 
									
										
										
										
											2016-12-05 12:05:00 -05:00
										 |  |  |     void HandleSyncRequestImpl(Kernel::SharedPtr<Kernel::ServerSession> server_session) override; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-15 22:40:19 -04:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Registers the functions in the service | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-01-30 16:56:49 -02:00
										 |  |  |     template <size_t N> | 
					
						
							| 
									
										
										
										
											2015-04-14 15:22:09 -03:00
										 |  |  |     inline void Register(const FunctionInfo (&functions)[N]) { | 
					
						
							|  |  |  |         Register(functions, N); | 
					
						
							| 
									
										
										
										
											2014-04-15 22:40:19 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-14 15:22:09 -03:00
										 |  |  |     void Register(const FunctionInfo* functions, size_t n); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-20 08:50:48 +03:00
										 |  |  |     union { | 
					
						
							|  |  |  |         u32 raw; | 
					
						
							|  |  |  |         BitField<0, 8, u32> major; | 
					
						
							|  |  |  |         BitField<8, 8, u32> minor; | 
					
						
							|  |  |  |         BitField<16, 8, u32> build; | 
					
						
							|  |  |  |         BitField<24, 8, u32> revision; | 
					
						
							|  |  |  |     } version = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2016-12-05 13:44:41 -05:00
										 |  |  |     u32 max_sessions; ///< Maximum number of concurrent sessions that this service can handle.
 | 
					
						
							| 
									
										
										
										
											2015-01-30 16:56:49 -02:00
										 |  |  |     boost::container::flat_map<u32, FunctionInfo> m_functions; | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | /// Initialize ServiceManager
 | 
					
						
							|  |  |  | void Init(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Shutdown ServiceManager
 | 
					
						
							|  |  |  | void Shutdown(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-30 16:07:04 -02:00
										 |  |  | /// Map of named ports managed by the kernel, which can be retrieved using the ConnectToPort SVC.
 | 
					
						
							| 
									
										
										
										
											2016-12-05 11:02:08 -05:00
										 |  |  | extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports; | 
					
						
							| 
									
										
										
										
											2015-01-30 16:07:04 -02:00
										 |  |  | /// Map of services registered with the "srv:" service, retrieved using GetServiceHandle.
 | 
					
						
							| 
									
										
										
										
											2016-12-05 11:02:08 -05:00
										 |  |  | extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_srv_services; | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | /// Adds a service to the services table
 | 
					
						
							|  |  |  | void AddService(Interface* interface_); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 18:44:21 -04:00
										 |  |  | } // namespace
 |