2016-06-15 01:03:30 +02: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 06:52:19 +02:00
|
|
|
#include "core/hle/kernel/client_port.h"
|
2016-06-15 01:03:30 +02:00
|
|
|
#include "core/hle/kernel/client_session.h"
|
2017-06-05 06:52:19 +02:00
|
|
|
#include "core/hle/kernel/hle_ipc.h"
|
2016-06-15 01:03:30 +02:00
|
|
|
#include "core/hle/kernel/server_session.h"
|
2017-06-05 06:52:19 +02:00
|
|
|
#include "core/hle/kernel/session.h"
|
2016-06-15 01:03:30 +02:00
|
|
|
#include "core/hle/kernel/thread.h"
|
|
|
|
|
|
|
|
namespace Kernel {
|
|
|
|
|
2019-02-02 21:18:01 +01:00
|
|
|
ServerSession::ServerSession(KernelSystem& kernel) : WaitObject(kernel), kernel(kernel) {}
|
2016-12-08 21:01:10 +01:00
|
|
|
ServerSession::~ServerSession() {
|
2016-12-14 18:33:49 +01:00
|
|
|
// This destructor will be called automatically when the last ServerSession handle is closed by
|
|
|
|
// the emulated application.
|
2017-01-05 05:23:17 +01:00
|
|
|
|
|
|
|
// Decrease the port's connection count.
|
|
|
|
if (parent->port)
|
2018-08-24 19:31:20 +02:00
|
|
|
parent->port->ConnectionClosed();
|
2017-01-05 05:23:17 +01:00
|
|
|
|
|
|
|
// TODO(Subv): Wake up all the ClientSession's waiting threads and set
|
|
|
|
// the SendSyncRequest result to 0xC920181A.
|
|
|
|
|
|
|
|
parent->server = nullptr;
|
2016-12-08 21:01:10 +01:00
|
|
|
}
|
2016-06-15 01:03:30 +02:00
|
|
|
|
2019-03-23 21:04:19 +01:00
|
|
|
ResultVal<std::shared_ptr<ServerSession>> ServerSession::Create(KernelSystem& kernel,
|
|
|
|
std::string name) {
|
|
|
|
auto server_session{std::make_shared<ServerSession>(kernel)};
|
2016-06-15 01:03:30 +02:00
|
|
|
|
|
|
|
server_session->name = std::move(name);
|
2017-01-05 05:23:17 +01:00
|
|
|
server_session->parent = nullptr;
|
2016-06-15 01:03:30 +02:00
|
|
|
|
2017-06-06 07:39:26 +02:00
|
|
|
return MakeResult(std::move(server_session));
|
2016-06-15 01:03:30 +02:00
|
|
|
}
|
|
|
|
|
2017-01-01 22:53:22 +01:00
|
|
|
bool ServerSession::ShouldWait(Thread* thread) const {
|
2017-06-21 00:33:28 +02: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-15 01:03:30 +02:00
|
|
|
}
|
|
|
|
|
2017-01-01 22:53:22 +01:00
|
|
|
void ServerSession::Acquire(Thread* thread) {
|
|
|
|
ASSERT_MSG(!ShouldWait(thread), "object unavailable!");
|
2017-11-07 05:07:08 +01:00
|
|
|
|
|
|
|
// If the client endpoint was closed, don't do anything. This ServerSession is now useless and
|
|
|
|
// will linger until its last handle is closed by the running application.
|
|
|
|
if (parent->client == nullptr)
|
|
|
|
return;
|
|
|
|
|
2017-06-21 00:33:28 +02:00
|
|
|
// We are now handling a request, pop it from the stack.
|
|
|
|
ASSERT(!pending_requesting_threads.empty());
|
|
|
|
currently_handling = pending_requesting_threads.back();
|
|
|
|
pending_requesting_threads.pop_back();
|
2016-06-15 01:03:30 +02:00
|
|
|
}
|
|
|
|
|
2019-03-23 21:04:19 +01:00
|
|
|
ResultCode ServerSession::HandleSyncRequest(std::shared_ptr<Thread> thread) {
|
2016-06-15 01:03:30 +02:00
|
|
|
// The ServerSession received a sync request, this means that there's new data available
|
2016-12-14 18:33:49 +01:00
|
|
|
// from its ClientSession, so wake up any threads that may be waiting on a svcReplyAndReceive or
|
|
|
|
// similar.
|
2016-12-01 04:50:13 +01:00
|
|
|
|
|
|
|
// If this ServerSession has an associated HLE handler, forward the request to it.
|
2016-12-09 18:52:12 +01:00
|
|
|
if (hle_handler != nullptr) {
|
2019-02-02 21:39:54 +01:00
|
|
|
std::array<u32_le, IPC::COMMAND_BUFFER_LENGTH + 2 * IPC::MAX_STATIC_BUFFERS> cmd_buf;
|
2019-02-02 21:18:01 +01:00
|
|
|
Kernel::Process* current_process = thread->owner_process;
|
2019-02-02 21:39:54 +01:00
|
|
|
kernel.memory.ReadBlock(*current_process, thread->GetCommandBufferAddress(), cmd_buf.data(),
|
|
|
|
cmd_buf.size() * sizeof(u32));
|
2019-02-02 21:18:01 +01:00
|
|
|
|
2019-04-02 18:30:03 +02:00
|
|
|
Kernel::HLERequestContext context(kernel, SharedFrom(this), thread.get());
|
2019-02-02 21:39:54 +01:00
|
|
|
context.PopulateFromIncomingCommandBuffer(cmd_buf.data(), *current_process);
|
2019-02-02 21:18:01 +01:00
|
|
|
|
|
|
|
hle_handler->HandleSyncRequest(context);
|
|
|
|
|
|
|
|
ASSERT(thread->status == Kernel::ThreadStatus::Running ||
|
|
|
|
thread->status == Kernel::ThreadStatus::WaitHleEvent);
|
|
|
|
// Only write the response immediately if the thread is still running. If the HLE handler
|
|
|
|
// put the thread to sleep then the writing of the command buffer will be deferred to the
|
|
|
|
// wakeup callback.
|
|
|
|
if (thread->status == Kernel::ThreadStatus::Running) {
|
2019-02-02 21:39:54 +01:00
|
|
|
context.WriteToOutgoingCommandBuffer(cmd_buf.data(), *current_process);
|
|
|
|
kernel.memory.WriteBlock(*current_process, thread->GetCommandBufferAddress(),
|
|
|
|
cmd_buf.data(), cmd_buf.size() * sizeof(u32));
|
2019-02-02 21:18:01 +01:00
|
|
|
}
|
2017-11-08 01:58:29 +01:00
|
|
|
}
|
|
|
|
|
2018-07-20 03:39:05 +02:00
|
|
|
if (thread->status == ThreadStatus::Running) {
|
2017-10-02 05:09:00 +02:00
|
|
|
// Put the thread to sleep until the server replies, it will be awoken in
|
2017-11-08 01:58:29 +01:00
|
|
|
// svcReplyAndReceive for LLE servers.
|
2018-07-20 03:39:05 +02:00
|
|
|
thread->status = ThreadStatus::WaitIPC;
|
2017-11-08 01:58:29 +01:00
|
|
|
|
|
|
|
if (hle_handler != nullptr) {
|
|
|
|
// For HLE services, we put the request threads to sleep for a short duration to
|
|
|
|
// simulate IPC overhead, but only if the HLE handler didn't put the thread to sleep for
|
|
|
|
// other reasons like an async callback. The IPC overhead is needed to prevent
|
|
|
|
// starvation when a thread only does sync requests to HLE services while a
|
|
|
|
// lower-priority thread is waiting to run.
|
|
|
|
|
2017-11-09 05:14:40 +01:00
|
|
|
// This delay was approximated in a homebrew application by measuring the average time
|
|
|
|
// it takes for svcSendSyncRequest to return when performing the SetLcdForceBlack IPC
|
|
|
|
// request to the GSP:GPU service in a n3DS with firmware 11.6. The measured values have
|
|
|
|
// a high variance and vary between models.
|
|
|
|
static constexpr u64 IPCDelayNanoseconds = 39000;
|
2017-11-08 01:58:29 +01:00
|
|
|
thread->WakeAfterDelay(IPCDelayNanoseconds);
|
|
|
|
} 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 18:52:12 +01:00
|
|
|
}
|
2016-12-01 04:50:13 +01:00
|
|
|
|
2016-12-14 18:33:49 +01:00
|
|
|
// If this ServerSession does not have an HLE implementation, just wake up the threads waiting
|
|
|
|
// on it.
|
2016-06-15 01:03:30 +02:00
|
|
|
WakeupAllWaitingThreads();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-03-23 21:04:19 +01:00
|
|
|
std::tuple<std::shared_ptr<ServerSession>, std::shared_ptr<ClientSession>>
|
|
|
|
KernelSystem::CreateSessionPair(const std::string& name, std::shared_ptr<ClientPort> port) {
|
2018-10-13 22:11:20 +02:00
|
|
|
auto server_session = ServerSession::Create(*this, name + "_Server").Unwrap();
|
2019-03-23 21:04:19 +01:00
|
|
|
auto client_session{std::make_shared<ClientSession>(*this)};
|
2017-05-22 01:52:42 +02:00
|
|
|
client_session->name = name + "_Client";
|
2017-01-05 05:23:17 +01: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-15 01:03:30 +02:00
|
|
|
|
2016-12-05 17:02:08 +01:00
|
|
|
return std::make_tuple(std::move(server_session), std::move(client_session));
|
2016-06-15 01:03:30 +02:00
|
|
|
}
|
|
|
|
|
2017-06-21 00:33:28 +02:00
|
|
|
} // namespace Kernel
|