yuzu/src/core/hle/service/service.h

243 lines
8.9 KiB
C
Raw Normal View History

2020-12-28 19:15:37 +04:00
// Copyright 2018 yuzu emulator team
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
#pragma once
#include <cstddef>
2020-12-30 05:38:14 +04:00
#include <mutex>
2020-12-28 19:15:37 +04:00
#include <string>
#include <boost/container/flat_map.hpp>
#include "common/common_types.h"
2020-12-30 05:38:14 +04:00
#include "common/spin_lock.h"
2020-12-28 19:15:37 +04:00
#include "core/hle/kernel/hle_ipc.h"
////////////////////////////////////////////////////////////////////////////////////////////////////
// Namespace Service
namespace Core {
class System;
}
namespace Kernel {
class HLERequestContext;
2021-05-11 04:22:43 +04:00
class KClientPort;
class KServerSession;
2021-06-05 09:15:09 +04:00
class ServiceThread;
2021-05-11 04:22:43 +04:00
} // namespace Kernel
2020-12-28 19:15:37 +04:00
namespace Service {
namespace FileSystem {
class FileSystemController;
}
namespace NVFlinger {
class NVFlinger;
}
namespace SM {
class ServiceManager;
}
2021-06-10 10:49:52 +04:00
/// Default number of maximum connections to a server session.
2021-06-11 00:22:25 +04:00
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");
2020-12-28 19:15:37 +04: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-11 04:22:43 +04:00
/// Invokes a service request routine using the HIPC protocol.
2020-12-28 19:15:37 +04:00
void InvokeRequest(Kernel::HLERequestContext& ctx);
2021-05-11 04:22:43 +04:00
/// Invokes a service request routine using the HIPC protocol.
void InvokeRequestTipc(Kernel::HLERequestContext& ctx);
/// Creates a port pair and registers it on the kernel's global port registry.
2021-06-05 09:15:09 +04:00
Kernel::KClientPort& CreatePort();
2021-05-11 04:22:43 +04:00
2020-12-30 05:38:14 +04:00
/// Handles a synchronization request for the service.
2021-05-11 04:22:43 +04:00
ResultCode HandleSyncRequest(Kernel::KServerSession& session,
Kernel::HLERequestContext& context) override;
2020-12-28 19:15:37 +04:00
protected:
/// Member-function pointer type of SyncRequest handlers.
template <typename Self>
using HandlerFnP = void (Self::*)(Kernel::HLERequestContext&);
2020-12-30 05:38:14 +04: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-12-28 19:15:37 +04:00
/// System context that the service operates under.
Core::System& system;
2021-07-03 13:00:06 +04:00
/// Identifier string used to connect to the service.
std::string service_name;
2020-12-28 19:15:37 +04: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);
explicit ServiceFrameworkBase(Core::System& system_, const char* service_name_,
u32 max_sessions_, InvokerFn* handler_invoker_);
~ServiceFrameworkBase() override;
void RegisterHandlersBase(const FunctionInfoBase* functions, std::size_t n);
2021-05-11 04:22:43 +04:00
void RegisterHandlersBaseTipc(const FunctionInfoBase* functions, std::size_t n);
2020-12-28 19:15:37 +04:00
void ReportUnimplementedFunction(Kernel::HLERequestContext& ctx, const FunctionInfoBase* info);
/// Maximum number of concurrent sessions that this service can handle.
u32 max_sessions;
/// Flag to store if a port was already create/installed to detect multiple install attempts,
/// which is not supported.
2021-07-03 13:00:06 +04:00
bool service_registered = false;
2020-12-28 19:15:37 +04: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-11 04:22:43 +04:00
boost::container::flat_map<u32, FunctionInfoBase> handlers_tipc;
2020-12-30 05:38:14 +04:00
/// Used to gain exclusive access to the service members, e.g. from CoreTiming thread.
Common::SpinLock lock_service;
2020-12-28 19:15:37 +04: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 13:38:19 +04:00
* @param expected_header_ request header in the command buffer which will trigger dispatch
2020-12-28 19:15:37 +04:00
* to this handler
2021-05-16 13:38:19 +04:00
* @param handler_callback_ member function in this service which will be called to handle
2020-12-28 19:15:37 +04:00
* the request
2021-05-16 13:38:19 +04:00
* @param name_ human-friendly name for the request. Used mostly for logging purposes.
2020-12-28 19:15:37 +04:00
*/
2021-05-16 13:38:19 +04:00
FunctionInfo(u32 expected_header_, HandlerFnP<Self> handler_callback_, const char* name_)
2020-12-28 19:15:37 +04:00
: FunctionInfoBase{
2021-05-16 13:38:19 +04:00
expected_header_,
2020-12-28 19:15:37 +04:00
// Type-erase member function pointer by casting it down to the base class.
2021-05-16 13:38:19 +04:00
static_cast<HandlerFnP<ServiceFrameworkBase>>(handler_callback_), name_} {}
2020-12-28 19:15:37 +04:00
};
/**
* Initializes the handler with no functions installed.
*
* @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.
*/
explicit ServiceFramework(Core::System& system_, const char* service_name_,
2021-06-10 10:49:52 +04:00
u32 max_sessions_ = ServerSessionCountMax)
2020-12-28 19:15:37 +04:00
: ServiceFrameworkBase(system_, service_name_, max_sessions_, Invoker) {}
/// Registers handlers in the service.
template <std::size_t N>
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.
*/
void RegisterHandlers(const FunctionInfo* functions, std::size_t n) {
RegisterHandlersBase(functions, n);
}
2021-05-11 04:22:43 +04: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);
}
2020-12-28 19:15:37 +04: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);
}
};
/**
* 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();
private:
std::unique_ptr<NVFlinger::NVFlinger> nv_flinger;
};
} // namespace Service