2016-06-14 19:03:30 -04:00
|
|
|
// 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_session.h"
|
|
|
|
#include "core/hle/kernel/server_session.h"
|
|
|
|
#include "core/hle/kernel/thread.h"
|
|
|
|
|
|
|
|
namespace Kernel {
|
|
|
|
|
2016-12-03 00:58:02 -03:00
|
|
|
ServerSession::ServerSession() = default;
|
2016-12-08 17:01:10 -03:00
|
|
|
ServerSession::~ServerSession() {
|
2016-12-14 14:33:49 -03:00
|
|
|
// This destructor will be called automatically when the last ServerSession handle is closed by
|
|
|
|
// the emulated application.
|
2016-12-08 17:01:10 -03:00
|
|
|
// TODO(Subv): Reduce the ClientPort's connection count,
|
|
|
|
// if the session is still open, set the connection status to 3 (Closed by server),
|
|
|
|
}
|
2016-06-14 19:03:30 -04:00
|
|
|
|
2016-12-14 14:33:49 -03:00
|
|
|
ResultVal<SharedPtr<ServerSession>> ServerSession::Create(
|
|
|
|
std::string name, std::shared_ptr<Service::SessionRequestHandler> hle_handler) {
|
2016-06-14 19:03:30 -04:00
|
|
|
SharedPtr<ServerSession> server_session(new ServerSession);
|
|
|
|
|
|
|
|
server_session->name = std::move(name);
|
|
|
|
server_session->signaled = false;
|
2016-12-08 13:06:19 -03:00
|
|
|
server_session->hle_handler = std::move(hle_handler);
|
2016-06-14 19:03:30 -04:00
|
|
|
|
|
|
|
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
|
2016-12-14 14:33:49 -03:00
|
|
|
// from its ClientSession, so wake up any threads that may be waiting on a svcReplyAndReceive or
|
|
|
|
// similar.
|
2016-12-01 00:50:13 -03:00
|
|
|
|
|
|
|
// If this ServerSession has an associated HLE handler, forward the request to it.
|
2016-12-09 14:52:12 -03:00
|
|
|
if (hle_handler != nullptr) {
|
|
|
|
// Attempt to translate the incoming request's command buffer.
|
|
|
|
ResultCode result = TranslateHLERequest(this);
|
|
|
|
if (result.IsError())
|
|
|
|
return result;
|
|
|
|
hle_handler->HandleSyncRequest(SharedPtr<ServerSession>(this));
|
|
|
|
// TODO(Subv): Translate the response command buffer.
|
|
|
|
}
|
2016-12-01 00:50:13 -03:00
|
|
|
|
2016-12-14 14:33:49 -03:00
|
|
|
// If this ServerSession does not have an HLE implementation, just wake up the threads waiting
|
|
|
|
// on it.
|
2016-06-14 19:03:30 -04:00
|
|
|
signaled = true;
|
|
|
|
WakeupAllWaitingThreads();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2016-12-14 14:33:49 -03:00
|
|
|
ServerSession::SessionPair ServerSession::CreateSessionPair(
|
|
|
|
const std::string& name, std::shared_ptr<Service::SessionRequestHandler> hle_handler) {
|
|
|
|
auto server_session =
|
|
|
|
ServerSession::Create(name + "_Server", std::move(hle_handler)).MoveFrom();
|
|
|
|
// We keep a non-owning pointer to the ServerSession in the ClientSession because we don't want
|
|
|
|
// to prevent the ServerSession's destructor from being called when the emulated
|
|
|
|
// application closes the last ServerSession handle.
|
2016-12-08 17:01:10 -03:00
|
|
|
auto client_session = ClientSession::Create(server_session.get(), name + "_Client").MoveFrom();
|
2016-06-14 19:03:30 -04:00
|
|
|
|
2016-12-05 13:02:08 -03:00
|
|
|
return std::make_tuple(std::move(server_session), std::move(client_session));
|
2016-06-14 19:03:30 -04:00
|
|
|
}
|
|
|
|
|
2016-12-09 14:52:12 -03:00
|
|
|
ResultCode TranslateHLERequest(ServerSession* server_session) {
|
|
|
|
// TODO(Subv): Implement this function once multiple concurrent processes are supported.
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
2016-06-14 19:03:30 -04:00
|
|
|
}
|