mirror of
				https://github.com/PabloMK7/citra.git
				synced 2025-11-03 23:28:48 +00:00 
			
		
		
		
	Kernel/IPC: Use Ports and Sessions as the fundamental building block of Inter Process Communication.
All handles obtained via srv::GetServiceHandle or svcConnectToPort are references to ClientSessions. Service modules will wait on the counterpart of those ClientSessions (Called ServerSessions) using svcReplyAndReceive or svcWaitSynchronization[1|N], and will be awoken when a SyncRequest is performed. HLE Interfaces are now ClientPorts which override the HandleSyncRequest virtual member function to perform command handling immediately.
This commit is contained in:
		
							parent
							
								
									68c00ee771
								
							
						
					
					
						commit
						073653e858
					
				
					 16 changed files with 315 additions and 89 deletions
				
			
		| 
						 | 
					@ -36,6 +36,7 @@ set(SRCS
 | 
				
			||||||
            hle/applets/swkbd.cpp
 | 
					            hle/applets/swkbd.cpp
 | 
				
			||||||
            hle/kernel/address_arbiter.cpp
 | 
					            hle/kernel/address_arbiter.cpp
 | 
				
			||||||
            hle/kernel/client_port.cpp
 | 
					            hle/kernel/client_port.cpp
 | 
				
			||||||
 | 
					            hle/kernel/client_session.cpp
 | 
				
			||||||
            hle/kernel/event.cpp
 | 
					            hle/kernel/event.cpp
 | 
				
			||||||
            hle/kernel/kernel.cpp
 | 
					            hle/kernel/kernel.cpp
 | 
				
			||||||
            hle/kernel/memory.cpp
 | 
					            hle/kernel/memory.cpp
 | 
				
			||||||
| 
						 | 
					@ -44,7 +45,7 @@ set(SRCS
 | 
				
			||||||
            hle/kernel/resource_limit.cpp
 | 
					            hle/kernel/resource_limit.cpp
 | 
				
			||||||
            hle/kernel/semaphore.cpp
 | 
					            hle/kernel/semaphore.cpp
 | 
				
			||||||
            hle/kernel/server_port.cpp
 | 
					            hle/kernel/server_port.cpp
 | 
				
			||||||
            hle/kernel/session.cpp
 | 
					            hle/kernel/server_session.cpp
 | 
				
			||||||
            hle/kernel/shared_memory.cpp
 | 
					            hle/kernel/shared_memory.cpp
 | 
				
			||||||
            hle/kernel/thread.cpp
 | 
					            hle/kernel/thread.cpp
 | 
				
			||||||
            hle/kernel/timer.cpp
 | 
					            hle/kernel/timer.cpp
 | 
				
			||||||
| 
						 | 
					@ -184,6 +185,7 @@ set(HEADERS
 | 
				
			||||||
            hle/applets/swkbd.h
 | 
					            hle/applets/swkbd.h
 | 
				
			||||||
            hle/kernel/address_arbiter.h
 | 
					            hle/kernel/address_arbiter.h
 | 
				
			||||||
            hle/kernel/client_port.h
 | 
					            hle/kernel/client_port.h
 | 
				
			||||||
 | 
					            hle/kernel/client_session.h
 | 
				
			||||||
            hle/kernel/event.h
 | 
					            hle/kernel/event.h
 | 
				
			||||||
            hle/kernel/kernel.h
 | 
					            hle/kernel/kernel.h
 | 
				
			||||||
            hle/kernel/memory.h
 | 
					            hle/kernel/memory.h
 | 
				
			||||||
| 
						 | 
					@ -192,7 +194,7 @@ set(HEADERS
 | 
				
			||||||
            hle/kernel/resource_limit.h
 | 
					            hle/kernel/resource_limit.h
 | 
				
			||||||
            hle/kernel/semaphore.h
 | 
					            hle/kernel/semaphore.h
 | 
				
			||||||
            hle/kernel/server_port.h
 | 
					            hle/kernel/server_port.h
 | 
				
			||||||
            hle/kernel/session.h
 | 
					            hle/kernel/server_session.h
 | 
				
			||||||
            hle/kernel/shared_memory.h
 | 
					            hle/kernel/shared_memory.h
 | 
				
			||||||
            hle/kernel/thread.h
 | 
					            hle/kernel/thread.h
 | 
				
			||||||
            hle/kernel/timer.h
 | 
					            hle/kernel/timer.h
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -6,10 +6,17 @@
 | 
				
			||||||
#include "core/hle/kernel/client_port.h"
 | 
					#include "core/hle/kernel/client_port.h"
 | 
				
			||||||
#include "core/hle/kernel/kernel.h"
 | 
					#include "core/hle/kernel/kernel.h"
 | 
				
			||||||
#include "core/hle/kernel/server_port.h"
 | 
					#include "core/hle/kernel/server_port.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/server_session.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace Kernel {
 | 
					namespace Kernel {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ClientPort::ClientPort() {}
 | 
					ClientPort::ClientPort() {}
 | 
				
			||||||
ClientPort::~ClientPort() {}
 | 
					ClientPort::~ClientPort() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void ClientPort::AddWaitingSession(SharedPtr<ServerSession> server_session) {
 | 
				
			||||||
 | 
					    server_port->pending_sessions.push_back(server_session);
 | 
				
			||||||
 | 
					    // Wake the threads waiting on the ServerPort
 | 
				
			||||||
 | 
					    server_port->WakeupAllWaitingThreads();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
} // namespace
 | 
					} // namespace
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -11,16 +11,27 @@
 | 
				
			||||||
namespace Kernel {
 | 
					namespace Kernel {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class ServerPort;
 | 
					class ServerPort;
 | 
				
			||||||
 | 
					class ServerSession;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class ClientPort : public Object {
 | 
					class ClientPort : public Object {
 | 
				
			||||||
public:
 | 
					public:
 | 
				
			||||||
    friend class ServerPort;
 | 
					    friend class ServerPort;
 | 
				
			||||||
    std::string GetTypeName() const override {
 | 
					
 | 
				
			||||||
        return "ClientPort";
 | 
					    /**
 | 
				
			||||||
    }
 | 
					     * Adds the specified server session to the queue of pending sessions of the associated ServerPort
 | 
				
			||||||
    std::string GetName() const override {
 | 
					     * @param server_session Server session to add to the queue
 | 
				
			||||||
        return name;
 | 
					     */
 | 
				
			||||||
    }
 | 
					    virtual void AddWaitingSession(SharedPtr<ServerSession> server_session);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Handle a sync request from the emulated application.
 | 
				
			||||||
 | 
					     * Only HLE services should override this function.
 | 
				
			||||||
 | 
					     * @returns ResultCode from the operation.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    virtual ResultCode HandleSyncRequest() { return RESULT_SUCCESS; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    std::string GetTypeName() const override { return "ClientPort"; }
 | 
				
			||||||
 | 
					    std::string GetName() const override { return name; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    static const HandleType HANDLE_TYPE = HandleType::ClientPort;
 | 
					    static const HandleType HANDLE_TYPE = HandleType::ClientPort;
 | 
				
			||||||
    HandleType GetHandleType() const override {
 | 
					    HandleType GetHandleType() const override {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										42
									
								
								src/core/hle/kernel/client_session.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								src/core/hle/kernel/client_session.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,42 @@
 | 
				
			||||||
 | 
					// Copyright 2016 Citra Emulator Project
 | 
				
			||||||
 | 
					// Licensed under GPLv2 or any later version
 | 
				
			||||||
 | 
					// Refer to the license.txt file included.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "common/assert.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "core/hle/kernel/client_port.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/client_session.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/server_session.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/kernel.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace Kernel {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ClientSession::ClientSession() {}
 | 
				
			||||||
 | 
					ClientSession::~ClientSession() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ResultVal<SharedPtr<ClientSession>> ClientSession::Create(SharedPtr<ServerSession> server_session, SharedPtr<ClientPort> client_port, std::string name) {
 | 
				
			||||||
 | 
					    SharedPtr<ClientSession> client_session(new ClientSession);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    client_session->name = std::move(name);
 | 
				
			||||||
 | 
					    client_session->server_session = server_session;
 | 
				
			||||||
 | 
					    client_session->client_port = client_port;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return MakeResult<SharedPtr<ClientSession>>(std::move(client_session));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ResultCode ClientSession::HandleSyncRequest() {
 | 
				
			||||||
 | 
					    // Signal the server session that new data is available
 | 
				
			||||||
 | 
					    ResultCode result = server_session->HandleSyncRequest();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (result.IsError())
 | 
				
			||||||
 | 
					        return result;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Tell the client port to handle the request in case it's an HLE service.
 | 
				
			||||||
 | 
					    // The client port can be nullptr for port-less sessions (Like for example File and Directory sessions).
 | 
				
			||||||
 | 
					    if (client_port != nullptr)
 | 
				
			||||||
 | 
					        result = client_port->HandleSyncRequest();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return result;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					} // namespace
 | 
				
			||||||
							
								
								
									
										50
									
								
								src/core/hle/kernel/client_session.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/core/hle/kernel/client_session.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,50 @@
 | 
				
			||||||
 | 
					// Copyright 2016 Citra Emulator Project
 | 
				
			||||||
 | 
					// Licensed under GPLv2 or any later version
 | 
				
			||||||
 | 
					// Refer to the license.txt file included.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#pragma once
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <string>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "common/common_types.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "core/hle/kernel/kernel.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace Kernel {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class ClientPort;
 | 
				
			||||||
 | 
					class ServerSession;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class ClientSession final : public Object {
 | 
				
			||||||
 | 
					public:
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Creates a client session.
 | 
				
			||||||
 | 
					     * @param server_session The server session associated with this client session
 | 
				
			||||||
 | 
					     * @param client_port The client port which this session is connected to
 | 
				
			||||||
 | 
					     * @param name Optional name of client session
 | 
				
			||||||
 | 
					     * @return The created client session
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    static ResultVal<SharedPtr<ClientSession>> Create(SharedPtr<ServerSession> server_session, SharedPtr<ClientPort> client_port, std::string name = "Unknown");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    std::string GetTypeName() const override { return "ClientSession"; }
 | 
				
			||||||
 | 
					    std::string GetName() const override { return name; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    static const HandleType HANDLE_TYPE = HandleType::ClientSession;
 | 
				
			||||||
 | 
					    HandleType GetHandleType() const override { return HANDLE_TYPE; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Handle a SyncRequest from the emulated application.
 | 
				
			||||||
 | 
					     * @return ResultCode of the operation.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    ResultCode HandleSyncRequest();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    std::string name;                           ///< Name of client port (optional)
 | 
				
			||||||
 | 
					    SharedPtr<ServerSession> server_session;    ///< The server session associated with this client session.
 | 
				
			||||||
 | 
					    SharedPtr<ClientPort> client_port;          ///< The client port which this session is connected to.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private:
 | 
				
			||||||
 | 
					    ClientSession();
 | 
				
			||||||
 | 
					    ~ClientSession() override;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					} // namespace
 | 
				
			||||||
| 
						 | 
					@ -31,22 +31,24 @@ enum KernelHandle : Handle {
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
enum class HandleType : u32 {
 | 
					enum class HandleType : u32 {
 | 
				
			||||||
    Unknown = 0,
 | 
					    Unknown         = 0,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    Session = 2,
 | 
					
 | 
				
			||||||
    Event = 3,
 | 
					    Event           = 3,
 | 
				
			||||||
    Mutex = 4,
 | 
					    Mutex           = 4,
 | 
				
			||||||
    SharedMemory = 5,
 | 
					    SharedMemory    = 5,
 | 
				
			||||||
    Redirection = 6,
 | 
					    Redirection     = 6,
 | 
				
			||||||
    Thread = 7,
 | 
					    Thread          = 7,
 | 
				
			||||||
    Process = 8,
 | 
					    Process         = 8,
 | 
				
			||||||
    AddressArbiter = 9,
 | 
					    AddressArbiter  = 9,
 | 
				
			||||||
    Semaphore = 10,
 | 
					    Semaphore       = 10,
 | 
				
			||||||
    Timer = 11,
 | 
					    Timer           = 11,
 | 
				
			||||||
    ResourceLimit = 12,
 | 
					    ResourceLimit   = 12,
 | 
				
			||||||
    CodeSet = 13,
 | 
					    CodeSet         = 13,
 | 
				
			||||||
    ClientPort = 14,
 | 
					    ClientPort      = 14,
 | 
				
			||||||
    ServerPort = 15,
 | 
					    ServerPort      = 15,
 | 
				
			||||||
 | 
					    ClientSession   = 16,
 | 
				
			||||||
 | 
					    ServerSession   = 17,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
enum {
 | 
					enum {
 | 
				
			||||||
| 
						 | 
					@ -82,7 +84,7 @@ public:
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    bool IsWaitable() const {
 | 
					    bool IsWaitable() const {
 | 
				
			||||||
        switch (GetHandleType()) {
 | 
					        switch (GetHandleType()) {
 | 
				
			||||||
        case HandleType::Session:
 | 
					        case HandleType::ServerSession:
 | 
				
			||||||
        case HandleType::ServerPort:
 | 
					        case HandleType::ServerPort:
 | 
				
			||||||
        case HandleType::Event:
 | 
					        case HandleType::Event:
 | 
				
			||||||
        case HandleType::Mutex:
 | 
					        case HandleType::Mutex:
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										58
									
								
								src/core/hle/kernel/server_session.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/core/hle/kernel/server_session.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,58 @@
 | 
				
			||||||
 | 
					// Copyright 2016 Citra Emulator Project
 | 
				
			||||||
 | 
					// Licensed under GPLv2 or any later version
 | 
				
			||||||
 | 
					// Refer to the license.txt file included.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <tuple>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "core/hle/kernel/client_port.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/client_session.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/server_session.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/thread.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace Kernel {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ServerSession::ServerSession() {}
 | 
				
			||||||
 | 
					ServerSession::~ServerSession() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ResultVal<SharedPtr<ServerSession>> ServerSession::Create(std::string name) {
 | 
				
			||||||
 | 
					    SharedPtr<ServerSession> server_session(new ServerSession);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    server_session->name = std::move(name);
 | 
				
			||||||
 | 
					    server_session->signaled = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return MakeResult<SharedPtr<ServerSession>>(std::move(server_session));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool ServerSession::ShouldWait() {
 | 
				
			||||||
 | 
					    return !signaled;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void ServerSession::Acquire() {
 | 
				
			||||||
 | 
					    ASSERT_MSG(!ShouldWait(), "object unavailable!");
 | 
				
			||||||
 | 
					    signaled = false;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ResultCode ServerSession::HandleSyncRequest() {
 | 
				
			||||||
 | 
					    // The ServerSession received a sync request, this means that there's new data available
 | 
				
			||||||
 | 
					    // from one of its ClientSessions, so wake up any threads that may be waiting on a svcReplyAndReceive or similar.
 | 
				
			||||||
 | 
					    signaled = true;
 | 
				
			||||||
 | 
					    WakeupAllWaitingThreads();
 | 
				
			||||||
 | 
					    return RESULT_SUCCESS;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					SharedPtr<ClientSession> ServerSession::CreateClientSession() {
 | 
				
			||||||
 | 
					    // In Citra, some types of ServerSessions (File and Directory sessions) are not created as a pair of Server-Client sessions,
 | 
				
			||||||
 | 
					    // but are instead created as a single ServerSession, which then hands over a ClientSession on demand (When opening the File or Directory).
 | 
				
			||||||
 | 
					    // The real kernel (Or more specifically, the real FS service) does create the pair of Sessions at the same time (via svcCreateSession), and simply
 | 
				
			||||||
 | 
					    // stores the ClientSession until it is needed.
 | 
				
			||||||
 | 
					    return ClientSession::Create(SharedPtr<ServerSession>(this), nullptr, name + "Client").MoveFrom();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					std::tuple<SharedPtr<ServerSession>, SharedPtr<ClientSession>> ServerSession::CreateSessionPair(SharedPtr<ClientPort> client_port, std::string name) {
 | 
				
			||||||
 | 
					    auto server_session = ServerSession::Create(name + "Server").MoveFrom();
 | 
				
			||||||
 | 
					    auto client_session = ClientSession::Create(server_session, client_port, name + "Client").MoveFrom();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return std::make_tuple(server_session, client_session);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -162,57 +162,64 @@ inline u32* GetCommandBuffer(const int offset = 0) {
 | 
				
			||||||
                                    offset);
 | 
					                                    offset);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class ClientSession;
 | 
				
			||||||
 | 
					class ClientPort;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * Kernel object representing the client endpoint of an IPC session. Sessions are the basic CTR-OS
 | 
					 * Kernel object representing the server endpoint of an IPC session. Sessions are the basic CTR-OS
 | 
				
			||||||
 * primitive for communication between different processes, and are used to implement service calls
 | 
					 * primitive for communication between different processes, and are used to implement service calls
 | 
				
			||||||
 * to the various system services.
 | 
					 * to the various system services.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * To make a service call, the client must write the command header and parameters to the buffer
 | 
					 * To make a service call, the client must write the command header and parameters to the buffer
 | 
				
			||||||
 * located at offset 0x80 of the TLS (Thread-Local Storage) area, then execute a SendSyncRequest
 | 
					 * located at offset 0x80 of the TLS (Thread-Local Storage) area, then execute a SendSyncRequest
 | 
				
			||||||
 * SVC call with its Session handle. The kernel will read the command header, using it to marshall
 | 
					 * SVC call with its ClientSession handle. The kernel will read the command header, using it to marshall
 | 
				
			||||||
 * the parameters to the process at the server endpoint of the session. After the server replies to
 | 
					 * the parameters to the process at the server endpoint of the session. After the server replies to
 | 
				
			||||||
 * the request, the response is marshalled back to the caller's TLS buffer and control is
 | 
					 * the request, the response is marshalled back to the caller's TLS buffer and control is
 | 
				
			||||||
 * transferred back to it.
 | 
					 * transferred back to it.
 | 
				
			||||||
 *
 | 
					 | 
				
			||||||
 * In Citra, only the client endpoint is currently implemented and only HLE calls, where the IPC
 | 
					 | 
				
			||||||
 * request is answered by C++ code in the emulator, are supported. When SendSyncRequest is called
 | 
					 | 
				
			||||||
 * with the session handle, this class's SyncRequest method is called, which should read the TLS
 | 
					 | 
				
			||||||
 * buffer and emulate the call accordingly. Since the code can directly read the emulated memory,
 | 
					 | 
				
			||||||
 * no parameter marshalling is done.
 | 
					 | 
				
			||||||
 *
 | 
					 | 
				
			||||||
 * In the long term, this should be turned into the full-fledged IPC mechanism implemented by
 | 
					 | 
				
			||||||
 * CTR-OS so that IPC calls can be optionally handled by the real implementations of processes, as
 | 
					 | 
				
			||||||
 * opposed to HLE simulations.
 | 
					 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
class Session : public WaitObject {
 | 
					class ServerSession : public WaitObject {
 | 
				
			||||||
public:
 | 
					public:
 | 
				
			||||||
    Session();
 | 
					    ServerSession();
 | 
				
			||||||
    ~Session() override;
 | 
					    ~ServerSession() override;
 | 
				
			||||||
 | 
					 | 
				
			||||||
    std::string GetTypeName() const override {
 | 
					 | 
				
			||||||
        return "Session";
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    static const HandleType HANDLE_TYPE = HandleType::Session;
 | 
					 | 
				
			||||||
    HandleType GetHandleType() const override {
 | 
					 | 
				
			||||||
        return HANDLE_TYPE;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * Handles a synchronous call to this session using HLE emulation. Emulated <-> emulated calls
 | 
					     * Creates a server session.
 | 
				
			||||||
     * aren't supported yet.
 | 
					     * @param name Optional name of the server session
 | 
				
			||||||
 | 
					     * @return The created server session
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    virtual ResultVal<bool> SyncRequest() = 0;
 | 
					    static ResultVal<SharedPtr<ServerSession>> Create(std::string name = "Unknown");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO(bunnei): These functions exist to satisfy a hardware test with a Session object
 | 
					    std::string GetTypeName() const override { return "ServerSession"; }
 | 
				
			||||||
    // passed into WaitSynchronization. Figure out the meaning of them.
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    bool ShouldWait() override {
 | 
					    static const HandleType HANDLE_TYPE = HandleType::ServerSession;
 | 
				
			||||||
        return true;
 | 
					    HandleType GetHandleType() const override { return HANDLE_TYPE; }
 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    void Acquire() override {
 | 
					    /**
 | 
				
			||||||
        ASSERT_MSG(!ShouldWait(), "object unavailable!");
 | 
					     * Creates a pair of ServerSession and an associated ClientSession.
 | 
				
			||||||
    }
 | 
					     * @param client_port ClientPort to which the sessions are connected
 | 
				
			||||||
 | 
					     * @param name Optional name of the ports
 | 
				
			||||||
 | 
					     * @return The created session tuple
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    static std::tuple<SharedPtr<ServerSession>, SharedPtr<ClientSession>> CreateSessionPair(SharedPtr<ClientPort> client_port, std::string name = "Unknown");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Creates a portless ClientSession and associates it with this ServerSession.
 | 
				
			||||||
 | 
					     * @returns ClientSession The newly created ClientSession.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    SharedPtr<ClientSession> CreateClientSession();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Handle a sync request from the emulated application.
 | 
				
			||||||
 | 
					     * Only HLE services should override this function.
 | 
				
			||||||
 | 
					     * @returns ResultCode from the operation.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    virtual ResultCode HandleSyncRequest();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    bool ShouldWait() override;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void Acquire() override;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    std::string name; ///< The name of this session (optional)
 | 
				
			||||||
 | 
					    bool signaled;    ///< Whether there's new data available to this ServerSession
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -92,7 +92,7 @@ File::File(std::unique_ptr<FileSys::FileBackend>&& backend, const FileSys::Path&
 | 
				
			||||||
 | 
					
 | 
				
			||||||
File::~File() {}
 | 
					File::~File() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ResultVal<bool> File::SyncRequest() {
 | 
					ResultCode File::HandleSyncRequest() {
 | 
				
			||||||
    u32* cmd_buff = Kernel::GetCommandBuffer();
 | 
					    u32* cmd_buff = Kernel::GetCommandBuffer();
 | 
				
			||||||
    FileCommand cmd = static_cast<FileCommand>(cmd_buff[0]);
 | 
					    FileCommand cmd = static_cast<FileCommand>(cmd_buff[0]);
 | 
				
			||||||
    switch (cmd) {
 | 
					    switch (cmd) {
 | 
				
			||||||
| 
						 | 
					@ -193,10 +193,10 @@ ResultVal<bool> File::SyncRequest() {
 | 
				
			||||||
        LOG_ERROR(Service_FS, "Unknown command=0x%08X!", cmd);
 | 
					        LOG_ERROR(Service_FS, "Unknown command=0x%08X!", cmd);
 | 
				
			||||||
        ResultCode error = UnimplementedFunction(ErrorModule::FS);
 | 
					        ResultCode error = UnimplementedFunction(ErrorModule::FS);
 | 
				
			||||||
        cmd_buff[1] = error.raw; // TODO(Link Mauve): use the correct error code for that.
 | 
					        cmd_buff[1] = error.raw; // TODO(Link Mauve): use the correct error code for that.
 | 
				
			||||||
        return error;
 | 
					        return ServerSession::HandleSyncRequest();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    cmd_buff[1] = RESULT_SUCCESS.raw; // No error
 | 
					    cmd_buff[1] = RESULT_SUCCESS.raw; // No error
 | 
				
			||||||
    return MakeResult<bool>(false);
 | 
					    return ServerSession::HandleSyncRequest();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
Directory::Directory(std::unique_ptr<FileSys::DirectoryBackend>&& backend,
 | 
					Directory::Directory(std::unique_ptr<FileSys::DirectoryBackend>&& backend,
 | 
				
			||||||
| 
						 | 
					@ -205,7 +205,7 @@ Directory::Directory(std::unique_ptr<FileSys::DirectoryBackend>&& backend,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
Directory::~Directory() {}
 | 
					Directory::~Directory() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ResultVal<bool> Directory::SyncRequest() {
 | 
					ResultCode Directory::HandleSyncRequest() {
 | 
				
			||||||
    u32* cmd_buff = Kernel::GetCommandBuffer();
 | 
					    u32* cmd_buff = Kernel::GetCommandBuffer();
 | 
				
			||||||
    DirectoryCommand cmd = static_cast<DirectoryCommand>(cmd_buff[0]);
 | 
					    DirectoryCommand cmd = static_cast<DirectoryCommand>(cmd_buff[0]);
 | 
				
			||||||
    switch (cmd) {
 | 
					    switch (cmd) {
 | 
				
			||||||
| 
						 | 
					@ -236,10 +236,10 @@ ResultVal<bool> Directory::SyncRequest() {
 | 
				
			||||||
        LOG_ERROR(Service_FS, "Unknown command=0x%08X!", cmd);
 | 
					        LOG_ERROR(Service_FS, "Unknown command=0x%08X!", cmd);
 | 
				
			||||||
        ResultCode error = UnimplementedFunction(ErrorModule::FS);
 | 
					        ResultCode error = UnimplementedFunction(ErrorModule::FS);
 | 
				
			||||||
        cmd_buff[1] = error.raw; // TODO(Link Mauve): use the correct error code for that.
 | 
					        cmd_buff[1] = error.raw; // TODO(Link Mauve): use the correct error code for that.
 | 
				
			||||||
        return MakeResult<bool>(false);
 | 
					        return ServerSession::HandleSyncRequest();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    cmd_buff[1] = RESULT_SUCCESS.raw; // No error
 | 
					    cmd_buff[1] = RESULT_SUCCESS.raw; // No error
 | 
				
			||||||
    return MakeResult<bool>(false);
 | 
					    return ServerSession::HandleSyncRequest();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -8,7 +8,7 @@
 | 
				
			||||||
#include <string>
 | 
					#include <string>
 | 
				
			||||||
#include "common/common_types.h"
 | 
					#include "common/common_types.h"
 | 
				
			||||||
#include "core/file_sys/archive_backend.h"
 | 
					#include "core/file_sys/archive_backend.h"
 | 
				
			||||||
#include "core/hle/kernel/session.h"
 | 
					#include "core/hle/kernel/server_session.h"
 | 
				
			||||||
#include "core/hle/result.h"
 | 
					#include "core/hle/result.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace FileSys {
 | 
					namespace FileSys {
 | 
				
			||||||
| 
						 | 
					@ -41,7 +41,7 @@ enum class MediaType : u32 { NAND = 0, SDMC = 1 };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
typedef u64 ArchiveHandle;
 | 
					typedef u64 ArchiveHandle;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class File : public Kernel::Session {
 | 
					class File : public Kernel::ServerSession {
 | 
				
			||||||
public:
 | 
					public:
 | 
				
			||||||
    File(std::unique_ptr<FileSys::FileBackend>&& backend, const FileSys::Path& path);
 | 
					    File(std::unique_ptr<FileSys::FileBackend>&& backend, const FileSys::Path& path);
 | 
				
			||||||
    ~File();
 | 
					    ~File();
 | 
				
			||||||
| 
						 | 
					@ -49,14 +49,15 @@ public:
 | 
				
			||||||
    std::string GetName() const override {
 | 
					    std::string GetName() const override {
 | 
				
			||||||
        return "Path: " + path.DebugStr();
 | 
					        return "Path: " + path.DebugStr();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    ResultVal<bool> SyncRequest() override;
 | 
					
 | 
				
			||||||
 | 
					    ResultCode HandleSyncRequest() override;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    FileSys::Path path; ///< Path of the file
 | 
					    FileSys::Path path; ///< Path of the file
 | 
				
			||||||
    u32 priority;       ///< Priority of the file. TODO(Subv): Find out what this means
 | 
					    u32 priority;       ///< Priority of the file. TODO(Subv): Find out what this means
 | 
				
			||||||
    std::unique_ptr<FileSys::FileBackend> backend; ///< File backend interface
 | 
					    std::unique_ptr<FileSys::FileBackend> backend; ///< File backend interface
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class Directory : public Kernel::Session {
 | 
					class Directory : public Kernel::ServerSession {
 | 
				
			||||||
public:
 | 
					public:
 | 
				
			||||||
    Directory(std::unique_ptr<FileSys::DirectoryBackend>&& backend, const FileSys::Path& path);
 | 
					    Directory(std::unique_ptr<FileSys::DirectoryBackend>&& backend, const FileSys::Path& path);
 | 
				
			||||||
    ~Directory();
 | 
					    ~Directory();
 | 
				
			||||||
| 
						 | 
					@ -64,7 +65,8 @@ public:
 | 
				
			||||||
    std::string GetName() const override {
 | 
					    std::string GetName() const override {
 | 
				
			||||||
        return "Directory: " + path.DebugStr();
 | 
					        return "Directory: " + path.DebugStr();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    ResultVal<bool> SyncRequest() override;
 | 
					
 | 
				
			||||||
 | 
					    ResultCode HandleSyncRequest() override;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    FileSys::Path path;                                 ///< Path of the directory
 | 
					    FileSys::Path path;                                 ///< Path of the directory
 | 
				
			||||||
    std::unique_ptr<FileSys::DirectoryBackend> backend; ///< File backend interface
 | 
					    std::unique_ptr<FileSys::DirectoryBackend> backend; ///< File backend interface
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -8,6 +8,7 @@
 | 
				
			||||||
#include "common/logging/log.h"
 | 
					#include "common/logging/log.h"
 | 
				
			||||||
#include "common/scope_exit.h"
 | 
					#include "common/scope_exit.h"
 | 
				
			||||||
#include "common/string_util.h"
 | 
					#include "common/string_util.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/client_session.h"
 | 
				
			||||||
#include "core/hle/result.h"
 | 
					#include "core/hle/result.h"
 | 
				
			||||||
#include "core/hle/service/fs/archive.h"
 | 
					#include "core/hle/service/fs/archive.h"
 | 
				
			||||||
#include "core/hle/service/fs/fs_user.h"
 | 
					#include "core/hle/service/fs/fs_user.h"
 | 
				
			||||||
| 
						 | 
					@ -17,7 +18,7 @@
 | 
				
			||||||
// Namespace FS_User
 | 
					// Namespace FS_User
 | 
				
			||||||
 | 
					
 | 
				
			||||||
using Kernel::SharedPtr;
 | 
					using Kernel::SharedPtr;
 | 
				
			||||||
using Kernel::Session;
 | 
					using Kernel::ServerSession;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace Service {
 | 
					namespace Service {
 | 
				
			||||||
namespace FS {
 | 
					namespace FS {
 | 
				
			||||||
| 
						 | 
					@ -70,7 +71,7 @@ static void OpenFile(Service::Interface* self) {
 | 
				
			||||||
    ResultVal<SharedPtr<File>> file_res = OpenFileFromArchive(archive_handle, file_path, mode);
 | 
					    ResultVal<SharedPtr<File>> file_res = OpenFileFromArchive(archive_handle, file_path, mode);
 | 
				
			||||||
    cmd_buff[1] = file_res.Code().raw;
 | 
					    cmd_buff[1] = file_res.Code().raw;
 | 
				
			||||||
    if (file_res.Succeeded()) {
 | 
					    if (file_res.Succeeded()) {
 | 
				
			||||||
        cmd_buff[3] = Kernel::g_handle_table.Create(*file_res).MoveFrom();
 | 
					        cmd_buff[3] = Kernel::g_handle_table.Create((*file_res)->CreateClientSession()).MoveFrom();
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
        cmd_buff[3] = 0;
 | 
					        cmd_buff[3] = 0;
 | 
				
			||||||
        LOG_ERROR(Service_FS, "failed to get a handle for file %s", file_path.DebugStr().c_str());
 | 
					        LOG_ERROR(Service_FS, "failed to get a handle for file %s", file_path.DebugStr().c_str());
 | 
				
			||||||
| 
						 | 
					@ -130,7 +131,7 @@ static void OpenFileDirectly(Service::Interface* self) {
 | 
				
			||||||
    ResultVal<SharedPtr<File>> file_res = OpenFileFromArchive(*archive_handle, file_path, mode);
 | 
					    ResultVal<SharedPtr<File>> file_res = OpenFileFromArchive(*archive_handle, file_path, mode);
 | 
				
			||||||
    cmd_buff[1] = file_res.Code().raw;
 | 
					    cmd_buff[1] = file_res.Code().raw;
 | 
				
			||||||
    if (file_res.Succeeded()) {
 | 
					    if (file_res.Succeeded()) {
 | 
				
			||||||
        cmd_buff[3] = Kernel::g_handle_table.Create(*file_res).MoveFrom();
 | 
					        cmd_buff[3] = Kernel::g_handle_table.Create((*file_res)->CreateClientSession()).MoveFrom();
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
        cmd_buff[3] = 0;
 | 
					        cmd_buff[3] = 0;
 | 
				
			||||||
        LOG_ERROR(Service_FS, "failed to get a handle for file %s mode=%u attributes=%u",
 | 
					        LOG_ERROR(Service_FS, "failed to get a handle for file %s mode=%u attributes=%u",
 | 
				
			||||||
| 
						 | 
					@ -391,7 +392,7 @@ static void OpenDirectory(Service::Interface* self) {
 | 
				
			||||||
    ResultVal<SharedPtr<Directory>> dir_res = OpenDirectoryFromArchive(archive_handle, dir_path);
 | 
					    ResultVal<SharedPtr<Directory>> dir_res = OpenDirectoryFromArchive(archive_handle, dir_path);
 | 
				
			||||||
    cmd_buff[1] = dir_res.Code().raw;
 | 
					    cmd_buff[1] = dir_res.Code().raw;
 | 
				
			||||||
    if (dir_res.Succeeded()) {
 | 
					    if (dir_res.Succeeded()) {
 | 
				
			||||||
        cmd_buff[3] = Kernel::g_handle_table.Create(*dir_res).MoveFrom();
 | 
					        cmd_buff[3] = Kernel::g_handle_table.Create((*dir_res)->CreateClientSession()).MoveFrom();
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
        LOG_ERROR(Service_FS, "failed to get a handle for directory type=%d size=%d data=%s",
 | 
					        LOG_ERROR(Service_FS, "failed to get a handle for directory type=%d size=%d data=%s",
 | 
				
			||||||
                  dirname_type, dirname_size, dir_path.DebugStr().c_str());
 | 
					                  dirname_type, dirname_size, dir_path.DebugStr().c_str());
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -41,8 +41,8 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace Service {
 | 
					namespace Service {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
std::unordered_map<std::string, Kernel::SharedPtr<Interface>> g_kernel_named_ports;
 | 
					std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports;
 | 
				
			||||||
std::unordered_map<std::string, Kernel::SharedPtr<Interface>> g_srv_services;
 | 
					std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_srv_services;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * Creates a function string for logging, complete with the name (or header code, depending
 | 
					 * Creates a function string for logging, complete with the name (or header code, depending
 | 
				
			||||||
| 
						 | 
					@ -61,7 +61,7 @@ static std::string MakeFunctionString(const char* name, const char* port_name,
 | 
				
			||||||
    return function_string;
 | 
					    return function_string;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ResultVal<bool> Interface::SyncRequest() {
 | 
					ResultCode Interface::HandleSyncRequest() {
 | 
				
			||||||
    u32* cmd_buff = Kernel::GetCommandBuffer();
 | 
					    u32* cmd_buff = Kernel::GetCommandBuffer();
 | 
				
			||||||
    auto itr = m_functions.find(cmd_buff[0]);
 | 
					    auto itr = m_functions.find(cmd_buff[0]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -75,14 +75,14 @@ ResultVal<bool> Interface::SyncRequest() {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // TODO(bunnei): Hack - ignore error
 | 
					        // TODO(bunnei): Hack - ignore error
 | 
				
			||||||
        cmd_buff[1] = 0;
 | 
					        cmd_buff[1] = 0;
 | 
				
			||||||
        return MakeResult<bool>(false);
 | 
					        return RESULT_SUCCESS;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    LOG_TRACE(Service, "%s",
 | 
					    LOG_TRACE(Service, "%s",
 | 
				
			||||||
              MakeFunctionString(itr->second.name, GetPortName().c_str(), cmd_buff).c_str());
 | 
					              MakeFunctionString(itr->second.name, GetPortName().c_str(), cmd_buff).c_str());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    itr->second.func(this);
 | 
					    itr->second.func(this);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return MakeResult<bool>(false); // TODO: Implement return from actual function
 | 
					    return RESULT_SUCCESS; // TODO: Implement return from actual function, it should fail if the parameter translation fails
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void Interface::Register(const FunctionInfo* functions, size_t n) {
 | 
					void Interface::Register(const FunctionInfo* functions, size_t n) {
 | 
				
			||||||
| 
						 | 
					@ -97,10 +97,16 @@ void Interface::Register(const FunctionInfo* functions, size_t n) {
 | 
				
			||||||
// Module interface
 | 
					// Module interface
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void AddNamedPort(Interface* interface_) {
 | 
					static void AddNamedPort(Interface* interface_) {
 | 
				
			||||||
 | 
					    interface_->name = interface_->GetPortName();
 | 
				
			||||||
 | 
					    interface_->active_sessions = 0;
 | 
				
			||||||
 | 
					    interface_->max_sessions = interface_->GetMaxSessions();
 | 
				
			||||||
    g_kernel_named_ports.emplace(interface_->GetPortName(), interface_);
 | 
					    g_kernel_named_ports.emplace(interface_->GetPortName(), interface_);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void AddService(Interface* interface_) {
 | 
					void AddService(Interface* interface_) {
 | 
				
			||||||
 | 
					    interface_->name = interface_->GetPortName();
 | 
				
			||||||
 | 
					    interface_->active_sessions = 0;
 | 
				
			||||||
 | 
					    interface_->max_sessions = interface_->GetMaxSessions();
 | 
				
			||||||
    g_srv_services.emplace(interface_->GetPortName(), interface_);
 | 
					    g_srv_services.emplace(interface_->GetPortName(), interface_);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -9,7 +9,8 @@
 | 
				
			||||||
#include <unordered_map>
 | 
					#include <unordered_map>
 | 
				
			||||||
#include <boost/container/flat_map.hpp>
 | 
					#include <boost/container/flat_map.hpp>
 | 
				
			||||||
#include "common/common_types.h"
 | 
					#include "common/common_types.h"
 | 
				
			||||||
#include "core/hle/kernel/session.h"
 | 
					#include "core/hle/kernel/client_port.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/server_session.h"
 | 
				
			||||||
#include "core/hle/result.h"
 | 
					#include "core/hle/result.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
				
			||||||
| 
						 | 
					@ -18,9 +19,10 @@
 | 
				
			||||||
namespace Service {
 | 
					namespace Service {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static const int kMaxPortSize = 8; ///< Maximum size of a port name (8 characters)
 | 
					static const int kMaxPortSize = 8; ///< Maximum size of a port name (8 characters)
 | 
				
			||||||
 | 
					static const u32 DefaultMaxSessions = 10; ///< Arbitrary default number of maximum connections to an HLE port
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/// Interface to a CTROS service
 | 
					/// Interface to a CTROS service
 | 
				
			||||||
class Interface : public Kernel::Session {
 | 
					class Interface : public Kernel::ClientPort {
 | 
				
			||||||
    // TODO(yuriks): An "Interface" being a Kernel::Object is mostly non-sense. Interface should be
 | 
					    // TODO(yuriks): An "Interface" being a Kernel::Object is mostly non-sense. Interface should be
 | 
				
			||||||
    // just something that encapsulates a session and acts as a helper to implement service
 | 
					    // just something that encapsulates a session and acts as a helper to implement service
 | 
				
			||||||
    // processes.
 | 
					    // processes.
 | 
				
			||||||
| 
						 | 
					@ -33,6 +35,15 @@ public:
 | 
				
			||||||
        version.raw = raw_version;
 | 
					        version.raw = raw_version;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Gets the maximum allowed number of sessions that can be connected to this port at the same time.
 | 
				
			||||||
 | 
					     * It should be overwritten by each service implementation for more fine-grained control.
 | 
				
			||||||
 | 
					     * @returns The maximum number of connections allowed.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    virtual u32 GetMaxSessions() { return DefaultMaxSessions; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void AddWaitingSession(Kernel::SharedPtr<Kernel::ServerSession> server_session) override { }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    typedef void (*Function)(Interface*);
 | 
					    typedef void (*Function)(Interface*);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    struct FunctionInfo {
 | 
					    struct FunctionInfo {
 | 
				
			||||||
| 
						 | 
					@ -49,7 +60,7 @@ public:
 | 
				
			||||||
        return "[UNKNOWN SERVICE PORT]";
 | 
					        return "[UNKNOWN SERVICE PORT]";
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ResultVal<bool> SyncRequest() override;
 | 
					    ResultCode HandleSyncRequest() override;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
protected:
 | 
					protected:
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
| 
						 | 
					@ -81,9 +92,9 @@ void Init();
 | 
				
			||||||
void Shutdown();
 | 
					void Shutdown();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/// Map of named ports managed by the kernel, which can be retrieved using the ConnectToPort SVC.
 | 
					/// Map of named ports managed by the kernel, which can be retrieved using the ConnectToPort SVC.
 | 
				
			||||||
extern std::unordered_map<std::string, Kernel::SharedPtr<Interface>> g_kernel_named_ports;
 | 
					extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports;
 | 
				
			||||||
/// Map of services registered with the "srv:" service, retrieved using GetServiceHandle.
 | 
					/// Map of services registered with the "srv:" service, retrieved using GetServiceHandle.
 | 
				
			||||||
extern std::unordered_map<std::string, Kernel::SharedPtr<Interface>> g_srv_services;
 | 
					extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_srv_services;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/// Adds a service to the services table
 | 
					/// Adds a service to the services table
 | 
				
			||||||
void AddService(Interface* interface_);
 | 
					void AddService(Interface* interface_);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -11,7 +11,7 @@
 | 
				
			||||||
#include "common/common_types.h"
 | 
					#include "common/common_types.h"
 | 
				
			||||||
#include "common/logging/log.h"
 | 
					#include "common/logging/log.h"
 | 
				
			||||||
#include "common/scope_exit.h"
 | 
					#include "common/scope_exit.h"
 | 
				
			||||||
#include "core/hle/kernel/session.h"
 | 
					#include "core/hle/kernel/server_session.h"
 | 
				
			||||||
#include "core/hle/result.h"
 | 
					#include "core/hle/result.h"
 | 
				
			||||||
#include "core/hle/service/soc_u.h"
 | 
					#include "core/hle/service/soc_u.h"
 | 
				
			||||||
#include "core/memory.h"
 | 
					#include "core/memory.h"
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,8 +2,12 @@
 | 
				
			||||||
// Licensed under GPLv2 or any later version
 | 
					// Licensed under GPLv2 or any later version
 | 
				
			||||||
// Refer to the license.txt file included.
 | 
					// Refer to the license.txt file included.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <tuple>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "common/common_types.h"
 | 
					#include "common/common_types.h"
 | 
				
			||||||
#include "common/logging/log.h"
 | 
					#include "common/logging/log.h"
 | 
				
			||||||
 | 
					#include "core/hle/service/srv.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/client_session.h"
 | 
				
			||||||
#include "core/hle/kernel/event.h"
 | 
					#include "core/hle/kernel/event.h"
 | 
				
			||||||
#include "core/hle/service/srv.h"
 | 
					#include "core/hle/service/srv.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -81,7 +85,18 @@ static void GetServiceHandle(Service::Interface* self) {
 | 
				
			||||||
    auto it = Service::g_srv_services.find(port_name);
 | 
					    auto it = Service::g_srv_services.find(port_name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (it != Service::g_srv_services.end()) {
 | 
					    if (it != Service::g_srv_services.end()) {
 | 
				
			||||||
        cmd_buff[3] = Kernel::g_handle_table.Create(it->second).MoveFrom();
 | 
					        auto client_port = it->second;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Create a new session pair
 | 
				
			||||||
 | 
					        auto sessions = Kernel::ServerSession::CreateSessionPair(client_port, port_name);
 | 
				
			||||||
 | 
					        auto client_session = std::get<Kernel::SharedPtr<Kernel::ClientSession>>(sessions);
 | 
				
			||||||
 | 
					        auto server_session = std::get<Kernel::SharedPtr<Kernel::ServerSession>>(sessions);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Add the server session to the port's queue
 | 
				
			||||||
 | 
					        client_port->AddWaitingSession(server_session);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Return the client session
 | 
				
			||||||
 | 
					        cmd_buff[3] = Kernel::g_handle_table.Create(client_session).MoveFrom();
 | 
				
			||||||
        LOG_TRACE(Service_SRV, "called port=%s, handle=0x%08X", port_name.c_str(), cmd_buff[3]);
 | 
					        LOG_TRACE(Service_SRV, "called port=%s, handle=0x%08X", port_name.c_str(), cmd_buff[3]);
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
        LOG_ERROR(Service_SRV, "(UNIMPLEMENTED) called port=%s", port_name.c_str());
 | 
					        LOG_ERROR(Service_SRV, "(UNIMPLEMENTED) called port=%s", port_name.c_str());
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -13,6 +13,7 @@
 | 
				
			||||||
#include "core/hle/function_wrappers.h"
 | 
					#include "core/hle/function_wrappers.h"
 | 
				
			||||||
#include "core/hle/kernel/address_arbiter.h"
 | 
					#include "core/hle/kernel/address_arbiter.h"
 | 
				
			||||||
#include "core/hle/kernel/client_port.h"
 | 
					#include "core/hle/kernel/client_port.h"
 | 
				
			||||||
 | 
					#include "core/hle/kernel/client_session.h"
 | 
				
			||||||
#include "core/hle/kernel/event.h"
 | 
					#include "core/hle/kernel/event.h"
 | 
				
			||||||
#include "core/hle/kernel/memory.h"
 | 
					#include "core/hle/kernel/memory.h"
 | 
				
			||||||
#include "core/hle/kernel/mutex.h"
 | 
					#include "core/hle/kernel/mutex.h"
 | 
				
			||||||
| 
						 | 
					@ -222,20 +223,31 @@ static ResultCode ConnectToPort(Handle* out_handle, const char* port_name) {
 | 
				
			||||||
        return ERR_NOT_FOUND;
 | 
					        return ERR_NOT_FOUND;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    CASCADE_RESULT(*out_handle, Kernel::g_handle_table.Create(it->second));
 | 
					    auto client_port = it->second;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Create a new session pair
 | 
				
			||||||
 | 
					    auto sessions = Kernel::ServerSession::CreateSessionPair(client_port, port_name);
 | 
				
			||||||
 | 
					    auto client_session = std::get<Kernel::SharedPtr<Kernel::ClientSession>>(sessions);
 | 
				
			||||||
 | 
					    auto server_session = std::get<Kernel::SharedPtr<Kernel::ServerSession>>(sessions);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Add the server session to the port's queue
 | 
				
			||||||
 | 
					    client_port->AddWaitingSession(server_session);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Return the client session
 | 
				
			||||||
 | 
					    CASCADE_RESULT(*out_handle, Kernel::g_handle_table.Create(client_session));
 | 
				
			||||||
    return RESULT_SUCCESS;
 | 
					    return RESULT_SUCCESS;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/// Synchronize to an OS service
 | 
					/// Synchronize to an OS service
 | 
				
			||||||
static ResultCode SendSyncRequest(Handle handle) {
 | 
					static ResultCode SendSyncRequest(Handle handle) {
 | 
				
			||||||
    SharedPtr<Kernel::Session> session = Kernel::g_handle_table.Get<Kernel::Session>(handle);
 | 
					    SharedPtr<Kernel::ClientSession> session = Kernel::g_handle_table.Get<Kernel::ClientSession>(handle);
 | 
				
			||||||
    if (session == nullptr) {
 | 
					    if (session == nullptr) {
 | 
				
			||||||
        return ERR_INVALID_HANDLE;
 | 
					        return ERR_INVALID_HANDLE;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    LOG_TRACE(Kernel_SVC, "called handle=0x%08X(%s)", handle, session->GetName().c_str());
 | 
					    LOG_TRACE(Kernel_SVC, "called handle=0x%08X(%s)", handle, session->GetName().c_str());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return session->SyncRequest().Code();
 | 
					    return session->HandleSyncRequest();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/// Close a handle
 | 
					/// Close a handle
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue