| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  | #include <map>
 | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | #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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     std::jthread m_host_thread; | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     std::mutex m_session_mutex; | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |     std::map<KServerSession*, std::shared_ptr<SessionRequestManager>> m_sessions; | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     KEvent* m_wakeup_event; | 
					
						
							|  |  |  |     KProcess* m_process; | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     KThread* m_thread; | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     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
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |         // Lock to get the set.
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         std::scoped_lock lk{m_session_mutex}; | 
					
						
							| 
									
										
										
										
											2021-11-28 13:05:18 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |         // Reserve the needed quantity.
 | 
					
						
							|  |  |  |         objs.reserve(m_sessions.size() + 1); | 
					
						
							|  |  |  |         managers.reserve(m_sessions.size()); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         // Copy to our local list.
 | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |         for (const auto& [session, manager] : m_sessions) { | 
					
						
							|  |  |  |             objs.push_back(session); | 
					
						
							|  |  |  |             managers.push_back(manager); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         // Insert the wakeup event at the end.
 | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |         objs.push_back(&m_wakeup_event->GetReadableEvent()); | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											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.
 | 
					
						
							| 
									
										
										
										
											2022-10-26 17:32:14 -04:00
										 |  |  |     rc = server_session->SendReplyHLE(); | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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-24 21:23:53 -04:00
										 |  |  |         // Lock to get the set.
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         std::scoped_lock lk{m_session_mutex}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |         // Erase the session.
 | 
					
						
							|  |  |  |         ASSERT(m_sessions.erase(server_session) == 1); | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											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-11-02 20:21:32 -04:00
										 |  |  |     kernel.RegisterHostThread(m_thread); | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |         // Lock to get the set.
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |         std::scoped_lock lk{m_session_mutex}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Insert the session and manager.
 | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |         m_sessions[server_session] = manager; | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 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(); | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     m_host_thread.join(); | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Lock mutex.
 | 
					
						
							|  |  |  |     m_session_mutex.lock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Close all remaining sessions.
 | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |     for (const auto& [server_session, manager] : m_sessions) { | 
					
						
							|  |  |  |         server_session->Close(); | 
					
						
							| 
									
										
										
										
											2021-12-05 23:42:58 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-24 21:23:53 -04:00
										 |  |  |     // Destroy remaining managers.
 | 
					
						
							|  |  |  |     m_sessions.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     // Close event.
 | 
					
						
							|  |  |  |     m_wakeup_event->GetReadableEvent().Close(); | 
					
						
							|  |  |  |     m_wakeup_event->Close(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     // Close thread.
 | 
					
						
							|  |  |  |     m_thread->Close(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     // 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
 | 
					
						
							| 
									
										
										
										
											2022-11-03 10:22:05 -04:00
										 |  |  |     KScopedResourceReservation event_reservation(m_process, LimitableResource::EventCountMax); | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  |     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(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     // Reserve a new thread from the process resource limit
 | 
					
						
							| 
									
										
										
										
											2022-11-03 10:22:05 -04:00
										 |  |  |     KScopedResourceReservation thread_reservation(m_process, LimitableResource::ThreadCountMax); | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     ASSERT(thread_reservation.Succeeded()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Initialize thread.
 | 
					
						
							|  |  |  |     m_thread = KThread::Create(kernel); | 
					
						
							|  |  |  |     ASSERT(KThread::InitializeDummyThread(m_thread, m_process).IsSuccess()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Commit the thread reservation.
 | 
					
						
							|  |  |  |     thread_reservation.Commit(); | 
					
						
							| 
									
										
										
										
											2022-10-16 01:53:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Start thread.
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     m_host_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
 |