| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2019 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/microprofile.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-24 23:28:46 -08:00
										 |  |  | #include "common/scope_exit.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-14 16:07:40 -07:00
										 |  |  | #include "common/settings.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-25 11:12:46 -04:00
										 |  |  | #include "common/thread.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-30 20:08:09 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | #include "core/frontend/emu_window.h"
 | 
					
						
							| 
									
										
										
										
											2021-11-05 15:52:31 +01:00
										 |  |  | #include "video_core/control/scheduler.h"
 | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | #include "video_core/dma_pusher.h"
 | 
					
						
							|  |  |  | #include "video_core/gpu.h"
 | 
					
						
							|  |  |  | #include "video_core/gpu_thread.h"
 | 
					
						
							|  |  |  | #include "video_core/renderer_base.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace VideoCommon::GPUThread { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Runs the GPU thread
 | 
					
						
							| 
									
										
										
										
											2021-09-15 20:32:54 -04:00
										 |  |  | static void RunThread(std::stop_token stop_token, Core::System& system, | 
					
						
							|  |  |  |                       VideoCore::RendererBase& renderer, Core::Frontend::GraphicsContext& context, | 
					
						
							| 
									
										
										
										
											2021-11-05 15:52:31 +01:00
										 |  |  |                       Tegra::Control::Scheduler& scheduler, SynchState& state) { | 
					
						
							| 
									
										
										
										
											2022-10-03 18:43:56 -04:00
										 |  |  |     std::string name = "GPU"; | 
					
						
							| 
									
										
										
										
											2020-02-25 11:12:46 -04:00
										 |  |  |     MicroProfileOnThreadCreate(name.c_str()); | 
					
						
							| 
									
										
										
										
											2020-12-24 23:28:46 -08:00
										 |  |  |     SCOPE_EXIT({ MicroProfileOnThreadExit(); }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-25 11:12:46 -04:00
										 |  |  |     Common::SetCurrentThreadName(name.c_str()); | 
					
						
							| 
									
										
										
										
											2020-04-05 09:48:53 -04:00
										 |  |  |     Common::SetCurrentThreadPriority(Common::ThreadPriority::High); | 
					
						
							| 
									
										
										
										
											2020-02-25 11:12:46 -04:00
										 |  |  |     system.RegisterHostThread(); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     auto current_context = context.Acquire(); | 
					
						
							| 
									
										
										
										
											2021-01-05 04:09:39 -03:00
										 |  |  |     VideoCore::RasterizerInterface* const rasterizer = renderer.ReadRasterizer(); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 20:32:54 -04:00
										 |  |  |     while (!stop_token.stop_requested()) { | 
					
						
							| 
									
										
										
										
											2022-07-06 01:33:17 -04:00
										 |  |  |         CommandDataContainer next = state.queue.PopWait(stop_token); | 
					
						
							| 
									
										
										
										
											2021-09-15 20:32:54 -04:00
										 |  |  |         if (stop_token.stop_requested()) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-04 14:39:12 -05:00
										 |  |  |         if (auto* submit_list = std::get_if<SubmitListCommand>(&next.data)) { | 
					
						
							| 
									
										
										
										
											2021-11-05 15:52:31 +01:00
										 |  |  |             scheduler.Push(submit_list->channel, std::move(submit_list->entries)); | 
					
						
							| 
									
										
										
										
											2021-01-07 15:56:15 -05:00
										 |  |  |         } else if (const auto* data = std::get_if<SwapBuffersCommand>(&next.data)) { | 
					
						
							| 
									
										
										
										
											2019-11-23 15:17:28 -05:00
										 |  |  |             renderer.SwapBuffers(data->framebuffer ? &*data->framebuffer : nullptr); | 
					
						
							| 
									
										
										
										
											2020-07-21 00:52:27 -04:00
										 |  |  |         } else if (std::holds_alternative<GPUTickCommand>(next.data)) { | 
					
						
							| 
									
										
										
										
											2020-02-20 11:55:32 -04:00
										 |  |  |             system.GPU().TickWork(); | 
					
						
							| 
									
										
										
										
											2020-12-04 14:39:12 -05:00
										 |  |  |         } else if (const auto* flush = std::get_if<FlushRegionCommand>(&next.data)) { | 
					
						
							| 
									
										
										
										
											2021-01-05 04:09:39 -03:00
										 |  |  |             rasterizer->FlushRegion(flush->addr, flush->size); | 
					
						
							| 
									
										
										
										
											2020-12-04 14:39:12 -05:00
										 |  |  |         } else if (const auto* invalidate = std::get_if<InvalidateRegionCommand>(&next.data)) { | 
					
						
							| 
									
										
										
										
											2021-01-05 04:09:39 -03:00
										 |  |  |             rasterizer->OnCPUWrite(invalidate->addr, invalidate->size); | 
					
						
							| 
									
										
										
										
											2019-11-23 15:17:28 -05:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2022-06-07 17:02:29 -04:00
										 |  |  |             ASSERT(false); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-11-23 15:17:28 -05:00
										 |  |  |         state.signaled_fence.store(next.fence); | 
					
						
							| 
									
										
										
										
											2021-04-07 11:41:31 +02:00
										 |  |  |         if (next.block) { | 
					
						
							|  |  |  |             // We have to lock the write_lock to ensure that the condition_variable wait not get a
 | 
					
						
							|  |  |  |             // race between the check and the lock itself.
 | 
					
						
							| 
									
										
										
										
											2022-04-07 19:32:40 +01:00
										 |  |  |             std::scoped_lock lk{state.write_lock}; | 
					
						
							| 
									
										
										
										
											2021-04-07 11:41:31 +02:00
										 |  |  |             state.cv.notify_all(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-12 00:24:33 -08:00
										 |  |  | ThreadManager::ThreadManager(Core::System& system_, bool is_async_) | 
					
						
							|  |  |  |     : system{system_}, is_async{is_async_} {} | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 20:32:54 -04:00
										 |  |  | ThreadManager::~ThreadManager() = default; | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | void ThreadManager::StartThread(VideoCore::RendererBase& renderer, | 
					
						
							|  |  |  |                                 Core::Frontend::GraphicsContext& context, | 
					
						
							| 
									
										
										
										
											2021-11-05 15:52:31 +01:00
										 |  |  |                                 Tegra::Control::Scheduler& scheduler) { | 
					
						
							| 
									
										
										
										
											2021-01-05 04:09:39 -03:00
										 |  |  |     rasterizer = renderer.ReadRasterizer(); | 
					
						
							| 
									
										
										
										
											2021-09-15 20:32:54 -04:00
										 |  |  |     thread = std::jthread(RunThread, std::ref(system), std::ref(renderer), std::ref(context), | 
					
						
							| 
									
										
										
										
											2021-11-05 15:52:31 +01:00
										 |  |  |                           std::ref(scheduler), std::ref(state)); | 
					
						
							| 
									
										
										
										
											2019-04-09 14:02:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-05 15:52:31 +01:00
										 |  |  | void ThreadManager::SubmitList(s32 channel, Tegra::CommandList&& entries) { | 
					
						
							|  |  |  |     PushCommand(SubmitListCommand(channel, std::move(entries))); | 
					
						
							| 
									
										
										
										
											2020-10-26 23:07:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-21 01:55:25 -03:00
										 |  |  | void ThreadManager::SwapBuffers(const Tegra::FramebufferConfig* framebuffer) { | 
					
						
							| 
									
										
										
										
											2019-11-27 17:46:07 -05:00
										 |  |  |     PushCommand(SwapBuffersCommand(framebuffer ? std::make_optional(*framebuffer) : std::nullopt)); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  | void ThreadManager::FlushRegion(VAddr addr, u64 size) { | 
					
						
							| 
									
										
										
										
											2020-12-12 00:24:33 -08:00
										 |  |  |     if (!is_async) { | 
					
						
							|  |  |  |         // Always flush with synchronous GPU mode
 | 
					
						
							| 
									
										
										
										
											2020-04-19 13:47:45 -04:00
										 |  |  |         PushCommand(FlushRegionCommand(addr, size)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-07 02:04:35 +02:00
										 |  |  |     if (!Settings::IsGPULevelExtreme()) { | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2020-02-17 22:29:04 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-07 02:04:35 +02:00
										 |  |  |     auto& gpu = system.GPU(); | 
					
						
							|  |  |  |     u64 fence = gpu.RequestFlush(addr, size); | 
					
						
							| 
									
										
										
										
											2022-01-30 10:31:13 +01:00
										 |  |  |     TickGPU(); | 
					
						
							|  |  |  |     gpu.WaitForSyncOperation(fence); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ThreadManager::TickGPU() { | 
					
						
							| 
									
										
										
										
											2022-02-06 01:16:11 +01:00
										 |  |  |     PushCommand(GPUTickCommand()); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  | void ThreadManager::InvalidateRegion(VAddr addr, u64 size) { | 
					
						
							| 
									
										
										
										
											2021-01-05 04:09:39 -03:00
										 |  |  |     rasterizer->OnCPUWrite(addr, size); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  | void ThreadManager::FlushAndInvalidateRegion(VAddr addr, u64 size) { | 
					
						
							| 
									
										
										
										
											2019-02-18 20:58:32 -05:00
										 |  |  |     // Skip flush on asynch mode, as FlushAndInvalidateRegion is not used for anything too important
 | 
					
						
							| 
									
										
										
										
											2021-01-05 04:09:39 -03:00
										 |  |  |     rasterizer->OnCPUWrite(addr, size); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 11:41:31 +02:00
										 |  |  | u64 ThreadManager::PushCommand(CommandData&& command_data, bool block) { | 
					
						
							|  |  |  |     if (!is_async) { | 
					
						
							|  |  |  |         // In synchronous GPU mode, block the caller until the command has executed
 | 
					
						
							|  |  |  |         block = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 13:57:49 +02:00
										 |  |  |     std::unique_lock lk(state.write_lock); | 
					
						
							| 
									
										
										
										
											2019-03-30 20:08:09 -04:00
										 |  |  |     const u64 fence{++state.last_fence}; | 
					
						
							| 
									
										
										
										
											2021-04-07 11:41:31 +02:00
										 |  |  |     state.queue.Push(CommandDataContainer(std::move(command_data), fence, block)); | 
					
						
							| 
									
										
										
										
											2020-12-12 00:24:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 11:41:31 +02:00
										 |  |  |     if (block) { | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |         Common::CondvarWait(state.cv, lk, thread.get_stop_token(), [this, fence] { | 
					
						
							| 
									
										
										
										
											2021-09-15 20:32:54 -04:00
										 |  |  |             return fence <= state.signaled_fence.load(std::memory_order_relaxed); | 
					
						
							| 
									
										
										
										
											2021-04-07 11:41:31 +02:00
										 |  |  |         }); | 
					
						
							| 
									
										
										
										
											2020-12-12 00:24:33 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 20:08:09 -04:00
										 |  |  |     return fence; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | } // namespace VideoCommon::GPUThread
 |