| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
 | 
					
						
							| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <functional>
 | 
					
						
							|  |  |  | #include <mutex>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/scope_exit.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  | #include "common/thread.h"
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | #include "core/hle/ipc_helpers.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/hle_ipc.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/k_event.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/k_scoped_resource_reservation.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  | #include "core/hle/kernel/k_session.h"
 | 
					
						
							| 
									
										
										
										
											2022-01-17 18:06:33 -08:00
										 |  |  | #include "core/hle/kernel/k_thread.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/service_thread.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ServiceThread::Impl final { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     explicit Impl(KernelCore& kernel, const std::string& service_name); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  |     ~Impl(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     void WaitAndProcessImpl(); | 
					
						
							|  |  |  |     void SessionClosed(KServerSession* server_session, | 
					
						
							|  |  |  |                        std::shared_ptr<SessionRequestManager> manager); | 
					
						
							|  |  |  |     void LoopProcess(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void RegisterServerSession(KServerSession* session, | 
					
						
							|  |  |  |                                std::shared_ptr<SessionRequestManager> manager); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     KernelCore& kernel; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::jthread m_thread; | 
					
						
							|  |  |  |     std::mutex m_session_mutex; | 
					
						
							|  |  |  |     std::vector<KServerSession*> m_sessions; | 
					
						
							|  |  |  |     std::vector<std::shared_ptr<SessionRequestManager>> m_managers; | 
					
						
							|  |  |  |     KEvent* m_wakeup_event; | 
					
						
							|  |  |  |     KProcess* m_process; | 
					
						
							|  |  |  |     std::atomic<bool> m_shutdown_requested; | 
					
						
							|  |  |  |     const std::string m_service_name; | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | void ServiceThread::Impl::WaitAndProcessImpl() { | 
					
						
							|  |  |  |     // Create local list of waitable sessions.
 | 
					
						
							|  |  |  |     std::vector<KSynchronizationObject*> objs; | 
					
						
							|  |  |  |     std::vector<std::shared_ptr<SessionRequestManager>> managers; | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |         // Lock to get the list.
 | 
					
						
							|  |  |  |         std::scoped_lock lk{m_session_mutex}; | 
					
						
							| 
									
										
										
										
											2021-11-28 13:05:18 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         // Resize to the needed quantity.
 | 
					
						
							|  |  |  |         objs.resize(m_sessions.size() + 1); | 
					
						
							|  |  |  |         managers.resize(m_managers.size()); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         // Copy to our local list.
 | 
					
						
							|  |  |  |         std::copy(m_sessions.begin(), m_sessions.end(), objs.begin()); | 
					
						
							|  |  |  |         std::copy(m_managers.begin(), m_managers.end(), managers.begin()); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         // Insert the wakeup event at the end.
 | 
					
						
							|  |  |  |         objs.back() = &m_wakeup_event->GetReadableEvent(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     // Wait on the list of sessions.
 | 
					
						
							|  |  |  |     s32 index{-1}; | 
					
						
							|  |  |  |     Result rc = KSynchronizationObject::Wait(kernel, &index, objs.data(), | 
					
						
							|  |  |  |                                              static_cast<s32>(objs.size()), -1); | 
					
						
							|  |  |  |     ASSERT(!rc.IsFailure()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If this was the wakeup event, clear it and finish.
 | 
					
						
							|  |  |  |     if (index >= static_cast<s64>(objs.size() - 1)) { | 
					
						
							|  |  |  |         m_wakeup_event->Clear(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-11-28 13:05:18 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     // This event is from a server session.
 | 
					
						
							|  |  |  |     auto* server_session = static_cast<KServerSession*>(objs[index]); | 
					
						
							|  |  |  |     auto& manager = managers[index]; | 
					
						
							| 
									
										
										
										
											2021-11-28 13:05:18 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     // Fetch the HLE request context.
 | 
					
						
							|  |  |  |     std::shared_ptr<HLERequestContext> context; | 
					
						
							|  |  |  |     rc = server_session->ReceiveRequest(&context, manager); | 
					
						
							| 
									
										
										
										
											2021-11-28 13:05:18 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     // If the session was closed, handle that.
 | 
					
						
							|  |  |  |     if (rc == ResultSessionClosed) { | 
					
						
							|  |  |  |         SessionClosed(server_session, manager); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         // Finish.
 | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2021-11-28 13:05:18 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // TODO: handle other cases
 | 
					
						
							|  |  |  |     ASSERT(rc == ResultSuccess); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Perform the request.
 | 
					
						
							|  |  |  |     Result service_rc = manager->CompleteSyncRequest(server_session, *context); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Reply to the client.
 | 
					
						
							|  |  |  |     rc = server_session->SendReply(true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (rc == ResultSessionClosed || service_rc == IPC::ERR_REMOTE_PROCESS_DEAD) { | 
					
						
							|  |  |  |         SessionClosed(server_session, manager); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // TODO: handle other cases
 | 
					
						
							|  |  |  |     ASSERT(rc == ResultSuccess); | 
					
						
							|  |  |  |     ASSERT(service_rc == ResultSuccess); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | void ServiceThread::Impl::SessionClosed(KServerSession* server_session, | 
					
						
							|  |  |  |                                         std::shared_ptr<SessionRequestManager> manager) { | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         // Lock to get the list.
 | 
					
						
							|  |  |  |         std::scoped_lock lk{m_session_mutex}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Get the index of the session.
 | 
					
						
							|  |  |  |         const auto index = | 
					
						
							|  |  |  |             std::find(m_sessions.begin(), m_sessions.end(), server_session) - m_sessions.begin(); | 
					
						
							|  |  |  |         ASSERT(index < static_cast<s64>(m_sessions.size())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Remove the session and its manager.
 | 
					
						
							|  |  |  |         m_sessions.erase(m_sessions.begin() + index); | 
					
						
							|  |  |  |         m_managers.erase(m_managers.begin() + index); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     // Close our reference to the server session.
 | 
					
						
							|  |  |  |     server_session->Close(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-20 22:39:44 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | void ServiceThread::Impl::LoopProcess() { | 
					
						
							|  |  |  |     Common::SetCurrentThreadName(m_service_name.c_str()); | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     kernel.RegisterHostThread(); | 
					
						
							| 
									
										
										
										
											2021-04-13 17:48:37 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     while (!m_shutdown_requested.load()) { | 
					
						
							|  |  |  |         WaitAndProcessImpl(); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ServiceThread::Impl::RegisterServerSession(KServerSession* server_session, | 
					
						
							|  |  |  |                                                 std::shared_ptr<SessionRequestManager> manager) { | 
					
						
							|  |  |  |     // Open the server session.
 | 
					
						
							|  |  |  |     server_session->Open(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Lock to get the list.
 | 
					
						
							|  |  |  |         std::scoped_lock lk{m_session_mutex}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Insert the session and manager.
 | 
					
						
							|  |  |  |         m_sessions.push_back(server_session); | 
					
						
							|  |  |  |         m_managers.push_back(manager); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Signal the wakeup event.
 | 
					
						
							|  |  |  |     m_wakeup_event->Signal(); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-05 23:42:58 -08:00
										 |  |  | ServiceThread::Impl::~Impl() { | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     // Shut down the processing thread.
 | 
					
						
							|  |  |  |     m_shutdown_requested.store(true); | 
					
						
							|  |  |  |     m_wakeup_event->Signal(); | 
					
						
							|  |  |  |     m_thread.join(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Lock mutex.
 | 
					
						
							|  |  |  |     m_session_mutex.lock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Close all remaining sessions.
 | 
					
						
							|  |  |  |     for (size_t i = 0; i < m_sessions.size(); i++) { | 
					
						
							|  |  |  |         m_sessions[i]->Close(); | 
					
						
							| 
									
										
										
										
											2021-12-05 23:42:58 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Close event.
 | 
					
						
							|  |  |  |     m_wakeup_event->GetReadableEvent().Close(); | 
					
						
							|  |  |  |     m_wakeup_event->Close(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Close process.
 | 
					
						
							|  |  |  |     m_process->Close(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ServiceThread::Impl::Impl(KernelCore& kernel_, const std::string& service_name) | 
					
						
							|  |  |  |     : kernel{kernel_}, m_service_name{service_name} { | 
					
						
							|  |  |  |     // Initialize process.
 | 
					
						
							|  |  |  |     m_process = KProcess::Create(kernel); | 
					
						
							|  |  |  |     KProcess::Initialize(m_process, kernel.System(), service_name, | 
					
						
							|  |  |  |                          KProcess::ProcessType::KernelInternal, kernel.GetSystemResourceLimit()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Reserve a new event from the process resource limit
 | 
					
						
							|  |  |  |     KScopedResourceReservation event_reservation(m_process, LimitableResource::Events); | 
					
						
							|  |  |  |     ASSERT(event_reservation.Succeeded()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Initialize event.
 | 
					
						
							|  |  |  |     m_wakeup_event = KEvent::Create(kernel); | 
					
						
							|  |  |  |     m_wakeup_event->Initialize(m_process); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Commit the event reservation.
 | 
					
						
							|  |  |  |     event_reservation.Commit(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Register the event.
 | 
					
						
							|  |  |  |     KEvent::Register(kernel, m_wakeup_event); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Start thread.
 | 
					
						
							|  |  |  |     m_thread = std::jthread([this] { LoopProcess(); }); | 
					
						
							| 
									
										
										
										
											2021-12-05 23:42:58 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | ServiceThread::ServiceThread(KernelCore& kernel, const std::string& name) | 
					
						
							|  |  |  |     : impl{std::make_unique<Impl>(kernel, name)} {} | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | ServiceThread::~ServiceThread() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | void ServiceThread::RegisterServerSession(KServerSession* session, | 
					
						
							|  |  |  |                                           std::shared_ptr<SessionRequestManager> manager) { | 
					
						
							|  |  |  |     impl->RegisterServerSession(session, manager); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Kernel
 |