2014-04-12 02:44:21 +04:00
|
|
|
// Copyright 2014 Citra Emulator Project
|
2014-12-17 08:38:14 +03:00
|
|
|
// Licensed under GPLv2 or any later version
|
2014-04-12 02:44:21 +04:00
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2015-06-21 17:44:11 +03:00
|
|
|
#include <cstddef>
|
2014-04-12 02:44:21 +04:00
|
|
|
#include <string>
|
2015-01-30 21:07:04 +03:00
|
|
|
#include <unordered_map>
|
2015-01-30 21:56:49 +03:00
|
|
|
#include <boost/container/flat_map.hpp>
|
2015-05-06 10:06:12 +03:00
|
|
|
#include "common/common_types.h"
|
2016-12-08 19:06:19 +03:00
|
|
|
#include "core/hle/ipc.h"
|
2016-06-15 02:03:30 +03:00
|
|
|
#include "core/hle/kernel/client_port.h"
|
2016-12-01 06:50:13 +03:00
|
|
|
#include "core/hle/kernel/thread.h"
|
2015-06-21 17:44:11 +03:00
|
|
|
#include "core/hle/result.h"
|
2016-12-01 06:50:13 +03:00
|
|
|
#include "core/memory.h"
|
|
|
|
|
2014-04-12 02:44:21 +04:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Namespace Service
|
|
|
|
|
|
|
|
namespace Service {
|
|
|
|
|
2014-12-14 08:30:11 +03:00
|
|
|
static const int kMaxPortSize = 8; ///< Maximum size of a port name (8 characters)
|
2016-12-05 21:44:41 +03:00
|
|
|
static const u32 DefaultMaxSessions = 10; ///< Arbitrary default number of maximum connections to an HLE service
|
2014-05-08 05:04:55 +04:00
|
|
|
|
2016-12-05 19:02:08 +03: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-12-01 06:50:13 +03: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 20:05:00 +03:00
|
|
|
* @returns ResultCode the result code of the translate operation.
|
2016-12-01 06:50:13 +03:00
|
|
|
*/
|
2016-12-05 20:05:00 +03:00
|
|
|
ResultCode HandleSyncRequest(Kernel::SharedPtr<Kernel::ServerSession> server_session);
|
|
|
|
|
|
|
|
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-01 06:50:13 +03:00
|
|
|
};
|
|
|
|
|
2016-12-05 19:02:08 +03:00
|
|
|
/**
|
|
|
|
* Framework for implementing HLE service handlers which dispatch incoming SyncRequests based on a table mapping header ids to handler functions.
|
|
|
|
*/
|
2016-12-01 06:50:13 +03:00
|
|
|
class Interface : public SessionRequestHandler {
|
2014-04-12 02:44:21 +04:00
|
|
|
public:
|
2016-12-05 21:44:41 +03: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 19:06:19 +03:00
|
|
|
Interface(u32 max_sessions = DefaultMaxSessions);
|
2016-12-05 21:44:41 +03:00
|
|
|
|
2016-12-08 19:06:19 +03:00
|
|
|
virtual ~Interface();
|
2016-12-05 21:44:41 +03:00
|
|
|
|
2016-06-18 01:09:43 +03:00
|
|
|
std::string GetName() const {
|
2016-09-18 03:38:01 +03:00
|
|
|
return GetPortName();
|
|
|
|
}
|
2014-04-12 02:44:21 +04:00
|
|
|
|
2016-11-20 08:50:48 +03:00
|
|
|
virtual void SetVersion(u32 raw_version) {
|
|
|
|
version.raw = raw_version;
|
|
|
|
}
|
|
|
|
|
2016-06-15 02:03:30 +03:00
|
|
|
/**
|
2016-12-05 21:44:41 +03:00
|
|
|
* Gets the maximum allowed number of sessions that can be connected to this service at the same time.
|
2016-06-15 02:03:30 +03:00
|
|
|
* @returns The maximum number of connections allowed.
|
|
|
|
*/
|
2016-12-05 21:44:41 +03:00
|
|
|
u32 GetMaxSessions() const { return max_sessions; }
|
2016-06-15 02:03:30 +03:00
|
|
|
|
2014-04-25 06:16:54 +04:00
|
|
|
typedef void (*Function)(Interface*);
|
|
|
|
|
|
|
|
struct FunctionInfo {
|
2016-09-18 03:38:01 +03:00
|
|
|
u32 id;
|
|
|
|
Function func;
|
2015-01-30 22:15:02 +03:00
|
|
|
const char* name;
|
2014-04-25 06:16:54 +04:00
|
|
|
};
|
|
|
|
|
2014-04-12 02:44:21 +04:00
|
|
|
/**
|
|
|
|
* Gets the string name used by CTROS for a service
|
|
|
|
* @return Port name of service
|
|
|
|
*/
|
2014-08-18 07:03:22 +04:00
|
|
|
virtual std::string GetPortName() const {
|
2014-04-12 02:44:21 +04:00
|
|
|
return "[UNKNOWN SERVICE PORT]";
|
|
|
|
}
|
|
|
|
|
2014-04-16 06:40:19 +04:00
|
|
|
protected:
|
2016-12-05 20:05:00 +03:00
|
|
|
void HandleSyncRequestImpl(Kernel::SharedPtr<Kernel::ServerSession> server_session) override;
|
|
|
|
|
2014-04-16 06:40:19 +04:00
|
|
|
/**
|
|
|
|
* Registers the functions in the service
|
|
|
|
*/
|
2015-01-30 21:56:49 +03:00
|
|
|
template <size_t N>
|
2015-04-14 21:22:09 +03:00
|
|
|
inline void Register(const FunctionInfo (&functions)[N]) {
|
|
|
|
Register(functions, N);
|
2014-04-16 06:40:19 +04:00
|
|
|
}
|
|
|
|
|
2015-04-14 21: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-12 02:44:21 +04:00
|
|
|
private:
|
2016-12-05 21:44:41 +03:00
|
|
|
u32 max_sessions; ///< Maximum number of concurrent sessions that this service can handle.
|
2015-01-30 21:56:49 +03:00
|
|
|
boost::container::flat_map<u32, FunctionInfo> m_functions;
|
2014-04-12 02:44:21 +04:00
|
|
|
};
|
|
|
|
|
2014-04-13 05:55:36 +04:00
|
|
|
/// Initialize ServiceManager
|
|
|
|
void Init();
|
|
|
|
|
|
|
|
/// Shutdown ServiceManager
|
|
|
|
void Shutdown();
|
|
|
|
|
2015-01-30 21:07:04 +03:00
|
|
|
/// Map of named ports managed by the kernel, which can be retrieved using the ConnectToPort SVC.
|
2016-12-05 19:02:08 +03:00
|
|
|
extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports;
|
2015-01-30 21:07:04 +03:00
|
|
|
/// Map of services registered with the "srv:" service, retrieved using GetServiceHandle.
|
2016-12-05 19:02:08 +03:00
|
|
|
extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_srv_services;
|
2014-04-13 05:55:36 +04:00
|
|
|
|
2015-02-27 05:13:08 +03:00
|
|
|
/// Adds a service to the services table
|
|
|
|
void AddService(Interface* interface_);
|
|
|
|
|
2014-04-12 02:44:21 +04:00
|
|
|
} // namespace
|