2016-06-14 23:03:30 +00:00
|
|
|
// Copyright 2016 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <tuple>
|
|
|
|
|
2017-06-05 04:52:19 +00:00
|
|
|
#include "core/hle/kernel/client_port.h"
|
2016-06-14 23:03:30 +00:00
|
|
|
#include "core/hle/kernel/client_session.h"
|
2017-12-29 05:36:22 +00:00
|
|
|
#include "core/hle/kernel/handle_table.h"
|
2017-06-05 04:52:19 +00:00
|
|
|
#include "core/hle/kernel/hle_ipc.h"
|
2017-12-29 05:36:22 +00:00
|
|
|
#include "core/hle/kernel/process.h"
|
2016-06-14 23:03:30 +00:00
|
|
|
#include "core/hle/kernel/server_session.h"
|
2017-06-05 04:52:19 +00:00
|
|
|
#include "core/hle/kernel/session.h"
|
2016-06-14 23:03:30 +00:00
|
|
|
#include "core/hle/kernel/thread.h"
|
|
|
|
|
|
|
|
namespace Kernel {
|
|
|
|
|
2016-12-03 03:58:02 +00:00
|
|
|
ServerSession::ServerSession() = default;
|
2016-12-08 20:01:10 +00:00
|
|
|
ServerSession::~ServerSession() {
|
2016-12-14 17:33:49 +00:00
|
|
|
// This destructor will be called automatically when the last ServerSession handle is closed by
|
|
|
|
// the emulated application.
|
2017-01-05 04:23:17 +00:00
|
|
|
|
|
|
|
// Decrease the port's connection count.
|
|
|
|
if (parent->port)
|
|
|
|
parent->port->active_sessions--;
|
|
|
|
|
|
|
|
// TODO(Subv): Wake up all the ClientSession's waiting threads and set
|
|
|
|
// the SendSyncRequest result to 0xC920181A.
|
|
|
|
|
|
|
|
parent->server = nullptr;
|
2016-12-08 20:01:10 +00:00
|
|
|
}
|
2016-06-14 23:03:30 +00:00
|
|
|
|
2017-06-06 05:39:26 +00:00
|
|
|
ResultVal<SharedPtr<ServerSession>> ServerSession::Create(std::string name) {
|
2016-06-14 23:03:30 +00:00
|
|
|
SharedPtr<ServerSession> server_session(new ServerSession);
|
|
|
|
|
|
|
|
server_session->name = std::move(name);
|
2017-01-05 04:23:17 +00:00
|
|
|
server_session->parent = nullptr;
|
2016-06-14 23:03:30 +00:00
|
|
|
|
2017-06-06 05:39:26 +00:00
|
|
|
return MakeResult(std::move(server_session));
|
2016-06-14 23:03:30 +00:00
|
|
|
}
|
|
|
|
|
2017-01-01 21:53:22 +00:00
|
|
|
bool ServerSession::ShouldWait(Thread* thread) const {
|
2017-06-20 22:33:28 +00:00
|
|
|
// Closed sessions should never wait, an error will be returned from svcReplyAndReceive.
|
|
|
|
if (parent->client == nullptr)
|
|
|
|
return false;
|
|
|
|
// Wait if we have no pending requests, or if we're currently handling a request.
|
|
|
|
return pending_requesting_threads.empty() || currently_handling != nullptr;
|
2016-06-14 23:03:30 +00:00
|
|
|
}
|
|
|
|
|
2017-01-01 21:53:22 +00:00
|
|
|
void ServerSession::Acquire(Thread* thread) {
|
|
|
|
ASSERT_MSG(!ShouldWait(thread), "object unavailable!");
|
2017-06-20 22:33:28 +00:00
|
|
|
// We are now handling a request, pop it from the stack.
|
|
|
|
// TODO(Subv): What happens if the client endpoint is closed before any requests are made?
|
|
|
|
ASSERT(!pending_requesting_threads.empty());
|
|
|
|
currently_handling = pending_requesting_threads.back();
|
|
|
|
pending_requesting_threads.pop_back();
|
2016-06-14 23:03:30 +00:00
|
|
|
}
|
|
|
|
|
2017-06-20 22:33:28 +00:00
|
|
|
ResultCode ServerSession::HandleSyncRequest(SharedPtr<Thread> thread) {
|
2016-06-14 23:03:30 +00:00
|
|
|
// The ServerSession received a sync request, this means that there's new data available
|
2016-12-14 17:33:49 +00:00
|
|
|
// from its ClientSession, so wake up any threads that may be waiting on a svcReplyAndReceive or
|
|
|
|
// similar.
|
2016-12-01 03:50:13 +00:00
|
|
|
|
|
|
|
// If this ServerSession has an associated HLE handler, forward the request to it.
|
2017-10-31 23:30:05 +00:00
|
|
|
ResultCode result{RESULT_SUCCESS};
|
2016-12-09 17:52:12 +00:00
|
|
|
if (hle_handler != nullptr) {
|
|
|
|
// Attempt to translate the incoming request's command buffer.
|
2017-10-31 23:30:05 +00:00
|
|
|
ResultCode translate_result = TranslateHLERequest(this);
|
|
|
|
if (translate_result.IsError())
|
|
|
|
return translate_result;
|
2017-12-29 05:36:22 +00:00
|
|
|
|
|
|
|
Kernel::HLERequestContext context(this);
|
|
|
|
u32* cmd_buf = (u32*)Memory::GetPointer(Kernel::GetCurrentThread()->GetTLSAddress());
|
|
|
|
context.PopulateFromIncomingCommandBuffer(cmd_buf, *Kernel::g_current_process,
|
|
|
|
Kernel::g_handle_table);
|
|
|
|
|
|
|
|
result = hle_handler->HandleSyncRequest(context);
|
2017-06-20 22:33:28 +00:00
|
|
|
} else {
|
|
|
|
// Add the thread to the list of threads that have issued a sync request with this
|
|
|
|
// server.
|
|
|
|
pending_requesting_threads.push_back(std::move(thread));
|
2016-12-09 17:52:12 +00:00
|
|
|
}
|
2016-12-01 03:50:13 +00:00
|
|
|
|
2016-12-14 17:33:49 +00:00
|
|
|
// If this ServerSession does not have an HLE implementation, just wake up the threads waiting
|
|
|
|
// on it.
|
2016-06-14 23:03:30 +00:00
|
|
|
WakeupAllWaitingThreads();
|
2017-10-31 23:30:05 +00:00
|
|
|
return result;
|
2016-06-14 23:03:30 +00:00
|
|
|
}
|
|
|
|
|
2017-06-06 05:39:26 +00:00
|
|
|
ServerSession::SessionPair ServerSession::CreateSessionPair(const std::string& name,
|
|
|
|
SharedPtr<ClientPort> port) {
|
2017-06-19 02:03:15 +00:00
|
|
|
auto server_session = ServerSession::Create(name + "_Server").Unwrap();
|
2017-05-21 23:52:42 +00:00
|
|
|
SharedPtr<ClientSession> client_session(new ClientSession);
|
|
|
|
client_session->name = name + "_Client";
|
2017-01-05 04:23:17 +00:00
|
|
|
|
|
|
|
std::shared_ptr<Session> parent(new Session);
|
|
|
|
parent->client = client_session.get();
|
|
|
|
parent->server = server_session.get();
|
|
|
|
parent->port = port;
|
|
|
|
|
|
|
|
client_session->parent = parent;
|
|
|
|
server_session->parent = parent;
|
2016-06-14 23:03:30 +00:00
|
|
|
|
2016-12-05 16:02:08 +00:00
|
|
|
return std::make_tuple(std::move(server_session), std::move(client_session));
|
2016-06-14 23:03:30 +00:00
|
|
|
}
|
|
|
|
|
2016-12-09 17:52:12 +00:00
|
|
|
ResultCode TranslateHLERequest(ServerSession* server_session) {
|
|
|
|
// TODO(Subv): Implement this function once multiple concurrent processes are supported.
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
2017-06-20 22:33:28 +00:00
|
|
|
} // namespace Kernel
|