| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | // Copyright 2020 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <condition_variable>
 | 
					
						
							|  |  |  | #include <functional>
 | 
					
						
							|  |  |  | #include <mutex>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | #include <queue>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/scope_exit.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  | #include "common/thread.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | #include "core/core.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/server_session.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/service_thread.h"
 | 
					
						
							|  |  |  | #include "core/hle/lock.h"
 | 
					
						
							|  |  |  | #include "video_core/renderer_base.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ServiceThread::Impl final { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  |     explicit Impl(KernelCore& kernel, std::size_t num_threads, const std::string& name); | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  |     ~Impl(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void QueueSyncRequest(ServerSession& session, std::shared_ptr<HLERequestContext>&& context); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     std::vector<std::thread> threads; | 
					
						
							|  |  |  |     std::queue<std::function<void()>> requests; | 
					
						
							|  |  |  |     std::mutex queue_mutex; | 
					
						
							|  |  |  |     std::condition_variable condition; | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  |     const std::string service_name; | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  |     bool stop{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  | ServiceThread::Impl::Impl(KernelCore& kernel, std::size_t num_threads, const std::string& name) | 
					
						
							|  |  |  |     : service_name{name} { | 
					
						
							| 
									
										
										
										
											2020-12-10 16:03:35 -08:00
										 |  |  |     for (std::size_t i = 0; i < num_threads; ++i) | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  |         threads.emplace_back([this, &kernel] { | 
					
						
							|  |  |  |             Common::SetCurrentThreadName(std::string{"Hle_" + service_name}.c_str()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  |             // Wait for first request before trying to acquire a render context
 | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 std::unique_lock lock{queue_mutex}; | 
					
						
							|  |  |  |                 condition.wait(lock, [this] { return stop || !requests.empty(); }); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             kernel.RegisterHostThread(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             while (true) { | 
					
						
							|  |  |  |                 std::function<void()> task; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     std::unique_lock lock{queue_mutex}; | 
					
						
							|  |  |  |                     condition.wait(lock, [this] { return stop || !requests.empty(); }); | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  |                     if (stop || requests.empty()) { | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  |                         return; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     task = std::move(requests.front()); | 
					
						
							|  |  |  |                     requests.pop(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 task(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ServiceThread::Impl::QueueSyncRequest(ServerSession& session, | 
					
						
							|  |  |  |                                            std::shared_ptr<HLERequestContext>&& context) { | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::unique_lock lock{queue_mutex}; | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // ServerSession owns the service thread, so we cannot caption a strong pointer here in the
 | 
					
						
							|  |  |  |         // event that the ServerSession is terminated.
 | 
					
						
							|  |  |  |         std::weak_ptr<ServerSession> weak_ptr{SharedFrom(&session)}; | 
					
						
							|  |  |  |         requests.emplace([weak_ptr, context{std::move(context)}]() { | 
					
						
							|  |  |  |             if (auto strong_ptr = weak_ptr.lock()) { | 
					
						
							|  |  |  |                 strong_ptr->CompleteSyncRequest(*context); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     condition.notify_one(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ServiceThread::Impl::~Impl() { | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::unique_lock lock{queue_mutex}; | 
					
						
							|  |  |  |         stop = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     condition.notify_all(); | 
					
						
							|  |  |  |     for (std::thread& thread : threads) { | 
					
						
							|  |  |  |         thread.join(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 01:06:39 -08:00
										 |  |  | ServiceThread::ServiceThread(KernelCore& kernel, std::size_t num_threads, const std::string& name) | 
					
						
							|  |  |  |     : impl{std::make_unique<Impl>(kernel, num_threads, name)} {} | 
					
						
							| 
									
										
										
										
											2020-12-09 21:27:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | ServiceThread::~ServiceThread() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ServiceThread::QueueSyncRequest(ServerSession& session, | 
					
						
							|  |  |  |                                      std::shared_ptr<HLERequestContext>&& context) { | 
					
						
							|  |  |  |     impl->QueueSyncRequest(session, std::move(context)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Kernel
 |