| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | // Copyright 2019 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/microprofile.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"
 | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | static void RunThread(VideoCore::RendererBase& renderer, Core::Frontend::GraphicsContext& context, | 
					
						
							|  |  |  |                       Tegra::DmaPusher& dma_pusher, SynchState& state) { | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  |     MicroProfileOnThreadCreate("GpuThread"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Wait for first GPU command before acquiring the window context
 | 
					
						
							| 
									
										
										
										
											2019-06-18 16:58:29 -04:00
										 |  |  |     while (state.queue.Empty()) | 
					
						
							|  |  |  |         ; | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // If emulation was stopped during disk shader loading, abort before trying to acquire context
 | 
					
						
							|  |  |  |     if (!state.is_running) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     auto current_context = context.Acquire(); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 20:58:32 -05:00
										 |  |  |     CommandDataContainer next; | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  |     while (state.is_running) { | 
					
						
							| 
									
										
										
										
											2019-11-23 15:17:28 -05:00
										 |  |  |         next = state.queue.PopWait(); | 
					
						
							|  |  |  |         if (const auto submit_list = std::get_if<SubmitListCommand>(&next.data)) { | 
					
						
							|  |  |  |             dma_pusher.Push(std::move(submit_list->entries)); | 
					
						
							|  |  |  |             dma_pusher.DispatchCalls(); | 
					
						
							|  |  |  |         } else if (const auto data = std::get_if<SwapBuffersCommand>(&next.data)) { | 
					
						
							|  |  |  |             renderer.SwapBuffers(data->framebuffer ? &*data->framebuffer : nullptr); | 
					
						
							| 
									
										
										
										
											2020-02-17 18:10:23 -04:00
										 |  |  |         } else if (const auto data = std::get_if<OnCommandListEndCommand>(&next.data)) { | 
					
						
							|  |  |  |             renderer.Rasterizer().ReleaseFences(); | 
					
						
							| 
									
										
										
										
											2019-11-23 15:17:28 -05:00
										 |  |  |         } else if (const auto data = std::get_if<FlushRegionCommand>(&next.data)) { | 
					
						
							|  |  |  |             renderer.Rasterizer().FlushRegion(data->addr, data->size); | 
					
						
							|  |  |  |         } else if (const auto data = std::get_if<InvalidateRegionCommand>(&next.data)) { | 
					
						
							| 
									
										
										
										
											2020-02-16 09:51:37 -04:00
										 |  |  |             renderer.Rasterizer().OnCPUWrite(data->addr, data->size); | 
					
						
							| 
									
										
										
										
											2019-11-23 15:17:28 -05:00
										 |  |  |         } else if (std::holds_alternative<EndProcessingCommand>(next.data)) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-11-23 15:17:28 -05:00
										 |  |  |         state.signaled_fence.store(next.fence); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-09 14:02:00 -04:00
										 |  |  | ThreadManager::ThreadManager(Core::System& system) : system{system} {} | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | ThreadManager::~ThreadManager() { | 
					
						
							| 
									
										
										
										
											2019-04-09 14:02:00 -04:00
										 |  |  |     if (!thread.joinable()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 20:58:32 -05:00
										 |  |  |     // Notify GPU thread that a shutdown is pending
 | 
					
						
							|  |  |  |     PushCommand(EndProcessingCommand()); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  |     thread.join(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | void ThreadManager::StartThread(VideoCore::RendererBase& renderer, | 
					
						
							|  |  |  |                                 Core::Frontend::GraphicsContext& context, | 
					
						
							|  |  |  |                                 Tegra::DmaPusher& dma_pusher) { | 
					
						
							|  |  |  |     thread = std::thread{RunThread, std::ref(renderer), std::ref(context), std::ref(dma_pusher), | 
					
						
							|  |  |  |                          std::ref(state)}; | 
					
						
							| 
									
										
										
										
											2019-04-09 14:02:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | void ThreadManager::SubmitList(Tegra::CommandList&& entries) { | 
					
						
							| 
									
										
										
										
											2019-09-25 19:43:23 -04:00
										 |  |  |     PushCommand(SubmitListCommand(std::move(entries))); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05: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-02-16 16:24:37 -04:00
										 |  |  |     PushCommand(FlushRegionCommand(addr, size)); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  | void ThreadManager::InvalidateRegion(VAddr addr, u64 size) { | 
					
						
							| 
									
										
										
										
											2020-02-16 09:51:37 -04:00
										 |  |  |     system.Renderer().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
 | 
					
						
							| 
									
										
										
										
											2020-02-16 09:51:37 -04:00
										 |  |  |     system.Renderer().Rasterizer().OnCPUWrite(addr, size); | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 19:08:22 -04:00
										 |  |  | void ThreadManager::WaitIdle() const { | 
					
						
							| 
									
										
										
										
											2019-10-11 13:41:15 -04:00
										 |  |  |     while (state.last_fence > state.signaled_fence.load(std::memory_order_relaxed)) { | 
					
						
							| 
									
										
										
										
											2019-09-26 19:08:22 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 18:10:23 -04:00
										 |  |  | void ThreadManager::OnCommandListEnd() { | 
					
						
							|  |  |  |     PushCommand(OnCommandListEndCommand()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 20:08:09 -04:00
										 |  |  | u64 ThreadManager::PushCommand(CommandData&& command_data) { | 
					
						
							|  |  |  |     const u64 fence{++state.last_fence}; | 
					
						
							|  |  |  |     state.queue.Push(CommandDataContainer(std::move(command_data), fence)); | 
					
						
							|  |  |  |     return fence; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-23 22:17:55 -05:00
										 |  |  | } // namespace VideoCommon::GPUThread
 |