mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-26 01:49:46 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			239 lines
		
	
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			239 lines
		
	
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2017 Dolphin Emulator Project
 | |
| // Licensed under GPLv2+
 | |
| // Refer to the license.txt file included.
 | |
| 
 | |
| #include "VideoCommon/AsyncShaderCompiler.h"
 | |
| #include <thread>
 | |
| #include "Common/Assert.h"
 | |
| #include "Common/Logging/Log.h"
 | |
| 
 | |
| namespace VideoCommon
 | |
| {
 | |
| AsyncShaderCompiler::AsyncShaderCompiler()
 | |
| {
 | |
| }
 | |
| 
 | |
| AsyncShaderCompiler::~AsyncShaderCompiler()
 | |
| {
 | |
|   // Pending work can be left at shutdown.
 | |
|   // The work item classes are expected to clean up after themselves.
 | |
|   ASSERT(!HasWorkerThreads());
 | |
| }
 | |
| 
 | |
| void AsyncShaderCompiler::QueueWorkItem(WorkItemPtr item, u32 priority)
 | |
| {
 | |
|   // If no worker threads are available, compile synchronously.
 | |
|   if (!HasWorkerThreads())
 | |
|   {
 | |
|     item->Compile();
 | |
|     m_completed_work.push_back(std::move(item));
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     std::lock_guard<std::mutex> guard(m_pending_work_lock);
 | |
|     m_pending_work.emplace(priority, std::move(item));
 | |
|     m_worker_thread_wake.notify_one();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AsyncShaderCompiler::RetrieveWorkItems()
 | |
| {
 | |
|   std::deque<WorkItemPtr> completed_work;
 | |
|   {
 | |
|     std::lock_guard<std::mutex> guard(m_completed_work_lock);
 | |
|     m_completed_work.swap(completed_work);
 | |
|   }
 | |
| 
 | |
|   while (!completed_work.empty())
 | |
|   {
 | |
|     completed_work.front()->Retrieve();
 | |
|     completed_work.pop_front();
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool AsyncShaderCompiler::HasPendingWork()
 | |
| {
 | |
|   std::lock_guard<std::mutex> guard(m_pending_work_lock);
 | |
|   return !m_pending_work.empty() || m_busy_workers.load() != 0;
 | |
| }
 | |
| 
 | |
| bool AsyncShaderCompiler::HasCompletedWork()
 | |
| {
 | |
|   std::lock_guard<std::mutex> guard(m_completed_work_lock);
 | |
|   return !m_completed_work.empty();
 | |
| }
 | |
| 
 | |
| void AsyncShaderCompiler::WaitUntilCompletion()
 | |
| {
 | |
|   while (HasPendingWork())
 | |
|     std::this_thread::sleep_for(std::chrono::milliseconds(1));
 | |
| }
 | |
| 
 | |
| void AsyncShaderCompiler::WaitUntilCompletion(
 | |
|     const std::function<void(size_t, size_t)>& progress_callback)
 | |
| {
 | |
|   if (!HasPendingWork())
 | |
|     return;
 | |
| 
 | |
|   // Wait a second before opening a progress dialog.
 | |
|   // This way, if the operation completes quickly, we don't annoy the user.
 | |
|   constexpr u32 CHECK_INTERVAL_MS = 50;
 | |
|   constexpr auto CHECK_INTERVAL = std::chrono::milliseconds(CHECK_INTERVAL_MS);
 | |
|   for (u32 i = 0; i < (1000 / CHECK_INTERVAL_MS); i++)
 | |
|   {
 | |
|     std::this_thread::sleep_for(std::chrono::milliseconds(CHECK_INTERVAL));
 | |
|     if (!HasPendingWork())
 | |
|       return;
 | |
|   }
 | |
| 
 | |
|   // Grab the number of pending items. We use this to work out how many are left.
 | |
|   size_t total_items = 0;
 | |
|   {
 | |
|     // Safe to hold both locks here, since nowhere else does.
 | |
|     std::lock_guard<std::mutex> pending_guard(m_pending_work_lock);
 | |
|     std::lock_guard<std::mutex> completed_guard(m_completed_work_lock);
 | |
|     total_items = m_completed_work.size() + m_pending_work.size() + m_busy_workers.load() + 1;
 | |
|   }
 | |
| 
 | |
|   // Update progress while the compiles complete.
 | |
|   for (;;)
 | |
|   {
 | |
|     size_t remaining_items;
 | |
|     {
 | |
|       std::lock_guard<std::mutex> pending_guard(m_pending_work_lock);
 | |
|       if (m_pending_work.empty() && !m_busy_workers.load())
 | |
|         break;
 | |
|       remaining_items = m_pending_work.size();
 | |
|     }
 | |
| 
 | |
|     progress_callback(total_items - remaining_items, total_items);
 | |
|     std::this_thread::sleep_for(CHECK_INTERVAL);
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool AsyncShaderCompiler::StartWorkerThreads(u32 num_worker_threads)
 | |
| {
 | |
|   if (num_worker_threads == 0)
 | |
|     return true;
 | |
| 
 | |
|   for (u32 i = 0; i < num_worker_threads; i++)
 | |
|   {
 | |
|     void* thread_param = nullptr;
 | |
|     if (!WorkerThreadInitMainThread(&thread_param))
 | |
|     {
 | |
|       WARN_LOG(VIDEO, "Failed to initialize shader compiler worker thread.");
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     m_worker_thread_start_result.store(false);
 | |
| 
 | |
|     std::thread thr(&AsyncShaderCompiler::WorkerThreadEntryPoint, this, thread_param);
 | |
|     m_init_event.Wait();
 | |
| 
 | |
|     if (!m_worker_thread_start_result.load())
 | |
|     {
 | |
|       WARN_LOG(VIDEO, "Failed to start shader compiler worker thread.");
 | |
|       thr.join();
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     m_worker_threads.push_back(std::move(thr));
 | |
|   }
 | |
| 
 | |
|   return HasWorkerThreads();
 | |
| }
 | |
| 
 | |
| bool AsyncShaderCompiler::ResizeWorkerThreads(u32 num_worker_threads)
 | |
| {
 | |
|   if (m_worker_threads.size() == num_worker_threads)
 | |
|     return true;
 | |
| 
 | |
|   StopWorkerThreads();
 | |
|   return StartWorkerThreads(num_worker_threads);
 | |
| }
 | |
| 
 | |
| bool AsyncShaderCompiler::HasWorkerThreads() const
 | |
| {
 | |
|   return !m_worker_threads.empty();
 | |
| }
 | |
| 
 | |
| void AsyncShaderCompiler::StopWorkerThreads()
 | |
| {
 | |
|   if (!HasWorkerThreads())
 | |
|     return;
 | |
| 
 | |
|   // Signal worker threads to stop, and wake all of them.
 | |
|   {
 | |
|     std::lock_guard<std::mutex> guard(m_pending_work_lock);
 | |
|     m_exit_flag.Set();
 | |
|     m_worker_thread_wake.notify_all();
 | |
|   }
 | |
| 
 | |
|   // Wait for worker threads to exit.
 | |
|   for (std::thread& thr : m_worker_threads)
 | |
|     thr.join();
 | |
|   m_worker_threads.clear();
 | |
|   m_exit_flag.Clear();
 | |
| }
 | |
| 
 | |
| bool AsyncShaderCompiler::WorkerThreadInitMainThread(void** param)
 | |
| {
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| bool AsyncShaderCompiler::WorkerThreadInitWorkerThread(void* param)
 | |
| {
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void AsyncShaderCompiler::WorkerThreadExit(void* param)
 | |
| {
 | |
| }
 | |
| 
 | |
| void AsyncShaderCompiler::WorkerThreadEntryPoint(void* param)
 | |
| {
 | |
|   // Initialize worker thread with backend-specific method.
 | |
|   if (!WorkerThreadInitWorkerThread(param))
 | |
|   {
 | |
|     WARN_LOG(VIDEO, "Failed to initialize shader compiler worker.");
 | |
|     m_worker_thread_start_result.store(false);
 | |
|     m_init_event.Set();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   m_worker_thread_start_result.store(true);
 | |
|   m_init_event.Set();
 | |
| 
 | |
|   WorkerThreadRun();
 | |
| 
 | |
|   WorkerThreadExit(param);
 | |
| }
 | |
| 
 | |
| void AsyncShaderCompiler::WorkerThreadRun()
 | |
| {
 | |
|   std::unique_lock<std::mutex> pending_lock(m_pending_work_lock);
 | |
|   while (!m_exit_flag.IsSet())
 | |
|   {
 | |
|     m_worker_thread_wake.wait(pending_lock);
 | |
| 
 | |
|     while (!m_pending_work.empty() && !m_exit_flag.IsSet())
 | |
|     {
 | |
|       m_busy_workers++;
 | |
|       auto iter = m_pending_work.begin();
 | |
|       WorkItemPtr item(std::move(iter->second));
 | |
|       m_pending_work.erase(iter);
 | |
|       pending_lock.unlock();
 | |
| 
 | |
|       if (item->Compile())
 | |
|       {
 | |
|         std::lock_guard<std::mutex> completed_guard(m_completed_work_lock);
 | |
|         m_completed_work.push_back(std::move(item));
 | |
|       }
 | |
| 
 | |
|       pending_lock.lock();
 | |
|       m_busy_workers--;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| }  // namespace VideoCommon
 |