mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 17:39:09 +00:00 
			
		
		
		
	Fixes validation errors. See https://docs.vulkan.org/guide/latest/swapchain_semaphore_reuse.html
		
			
				
	
	
		
			165 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2016 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <array>
 | |
| #include <functional>
 | |
| #include <memory>
 | |
| #include <vector>
 | |
| 
 | |
| #include <Common/WorkQueueThread.h>
 | |
| #include "Common/Flag.h"
 | |
| 
 | |
| #include "VideoBackends/Vulkan/Constants.h"
 | |
| 
 | |
| namespace Vulkan
 | |
| {
 | |
| class CommandBufferManager
 | |
| {
 | |
| public:
 | |
|   explicit CommandBufferManager(bool use_threaded_submission);
 | |
|   ~CommandBufferManager();
 | |
| 
 | |
|   bool Initialize(size_t swapchain_image_count);
 | |
| 
 | |
|   // These command buffers are allocated per-frame. They are valid until the command buffer
 | |
|   // is submitted, after that you should call these functions again.
 | |
|   VkCommandBuffer GetCurrentInitCommandBuffer()
 | |
|   {
 | |
|     CmdBufferResources& cmd_buffer_resources = GetCurrentCmdBufferResources();
 | |
|     cmd_buffer_resources.init_command_buffer_used = true;
 | |
|     return cmd_buffer_resources.command_buffers[0];
 | |
|   }
 | |
|   VkCommandBuffer GetCurrentCommandBuffer() const
 | |
|   {
 | |
|     const CmdBufferResources& cmd_buffer_resources = m_command_buffers[m_current_cmd_buffer];
 | |
|     return cmd_buffer_resources.command_buffers[1];
 | |
|   }
 | |
|   // Allocates a descriptors set from the pool reserved for the current frame.
 | |
|   VkDescriptorSet AllocateDescriptorSet(VkDescriptorSetLayout set_layout);
 | |
| 
 | |
|   // Fence "counters" are used to track which commands have been completed by the GPU.
 | |
|   // If the last completed fence counter is greater or equal to N, it means that the work
 | |
|   // associated counter N has been completed by the GPU. The value of N to associate with
 | |
|   // commands can be retreived by calling GetCurrentFenceCounter().
 | |
|   u64 GetCompletedFenceCounter() const { return m_completed_fence_counter; }
 | |
| 
 | |
|   // Gets the fence that will be signaled when the currently executing command buffer is
 | |
|   // queued and executed. Do not wait for this fence before the buffer is executed.
 | |
|   u64 GetCurrentFenceCounter() const
 | |
|   {
 | |
|     auto& resources = m_command_buffers[m_current_cmd_buffer];
 | |
|     return resources.fence_counter;
 | |
|   }
 | |
| 
 | |
|   // Returns the semaphore for the current command buffer, which can be used to ensure the
 | |
|   // swap chain image is ready before the command buffer executes.
 | |
|   // Once you've confirmed that the semaphore will be signalled this frame, call
 | |
|   // `MarkCurrentCommandBufferSemaphoreUsed`.
 | |
|   VkSemaphore GetCurrentCommandBufferSemaphore()
 | |
|   {
 | |
|     return m_command_buffers[m_current_cmd_buffer].semaphore;
 | |
|   }
 | |
| 
 | |
|   // Marks the current command buffer's semaphore as used, so we'll wait on it in the next
 | |
|   // command buffer submission.
 | |
|   void MarkCurrentCommandBufferSemaphoreUsed()
 | |
|   {
 | |
|     m_command_buffers[m_current_cmd_buffer].semaphore_used = true;
 | |
|   }
 | |
| 
 | |
|   // Ensure that the worker thread has submitted any previous command buffers and is idle.
 | |
|   void WaitForWorkerThreadIdle();
 | |
| 
 | |
|   // Wait for a fence to be completed.
 | |
|   // Also invokes callbacks for completion.
 | |
|   void WaitForFenceCounter(u64 fence_counter);
 | |
| 
 | |
|   void SubmitCommandBuffer(bool submit_on_worker_thread, bool wait_for_completion,
 | |
|                            bool advance_to_next_frame = false,
 | |
|                            VkSwapchainKHR present_swap_chain = VK_NULL_HANDLE,
 | |
|                            uint32_t present_image_index = 0xFFFFFFFF);
 | |
| 
 | |
|   // Was the last present submitted to the queue a failure? If so, we must recreate our swapchain.
 | |
|   bool CheckLastPresentFail() { return m_last_present_failed.TestAndClear(); }
 | |
|   VkResult GetLastPresentResult() const { return m_last_present_result; }
 | |
| 
 | |
|   // Schedule a vulkan resource for destruction later on. This will occur when the command buffer
 | |
|   // is next re-used, and the GPU has finished working with the specified resource.
 | |
|   void DeferBufferViewDestruction(VkBufferView object);
 | |
|   void DeferBufferDestruction(VkBuffer buffer, VmaAllocation alloc);
 | |
|   void DeferFramebufferDestruction(VkFramebuffer object);
 | |
|   void DeferImageDestruction(VkImage object, VmaAllocation alloc);
 | |
|   void DeferImageViewDestruction(VkImageView object);
 | |
| 
 | |
| private:
 | |
|   bool CreateCommandBuffers(size_t swapchain_image_count);
 | |
|   void DestroyCommandBuffers();
 | |
| 
 | |
|   bool CreateSubmitThread();
 | |
| 
 | |
|   void WaitForCommandBufferCompletion(u32 command_buffer_index);
 | |
|   void SubmitCommandBuffer(u32 command_buffer_index, VkSwapchainKHR present_swap_chain,
 | |
|                            u32 present_image_index);
 | |
|   void BeginCommandBuffer();
 | |
| 
 | |
|   VkDescriptorPool CreateDescriptorPool(u32 descriptor_sizes);
 | |
| 
 | |
|   const u32 DESCRIPTOR_SETS_PER_POOL = 1024;
 | |
| 
 | |
|   struct CmdBufferResources
 | |
|   {
 | |
|     // [0] - Init (upload) command buffer, [1] - draw command buffer
 | |
|     VkCommandPool command_pool = VK_NULL_HANDLE;
 | |
|     std::array<VkCommandBuffer, 2> command_buffers = {};
 | |
|     VkFence fence = VK_NULL_HANDLE;
 | |
|     VkSemaphore semaphore = VK_NULL_HANDLE;
 | |
|     u64 fence_counter = 0;
 | |
|     bool init_command_buffer_used = false;
 | |
|     bool semaphore_used = false;
 | |
|     std::atomic<bool> waiting_for_submit{false};
 | |
|     u32 frame_index = 0;
 | |
| 
 | |
|     std::vector<std::function<void()>> cleanup_resources;
 | |
|   };
 | |
| 
 | |
|   struct FrameResources
 | |
|   {
 | |
|     std::vector<VkDescriptorPool> descriptor_pools;
 | |
|     u32 current_descriptor_pool_index = 0;
 | |
|   };
 | |
| 
 | |
|   FrameResources& GetCurrentFrameResources() { return m_frame_resources[m_current_frame]; }
 | |
| 
 | |
|   CmdBufferResources& GetCurrentCmdBufferResources()
 | |
|   {
 | |
|     return m_command_buffers[m_current_cmd_buffer];
 | |
|   }
 | |
| 
 | |
|   u64 m_next_fence_counter = 1;
 | |
|   u64 m_completed_fence_counter = 0;
 | |
| 
 | |
|   std::array<FrameResources, NUM_FRAMES_IN_FLIGHT> m_frame_resources;
 | |
|   std::array<CmdBufferResources, NUM_COMMAND_BUFFERS> m_command_buffers;
 | |
|   u32 m_current_frame = 0;
 | |
|   u32 m_current_cmd_buffer = 0;
 | |
| 
 | |
|   // Threaded command buffer execution
 | |
|   struct PendingCommandBufferSubmit
 | |
|   {
 | |
|     VkSwapchainKHR present_swap_chain;
 | |
|     u32 present_image_index;
 | |
|     u32 command_buffer_index;
 | |
|   };
 | |
|   Common::WorkQueueThreadSP<PendingCommandBufferSubmit> m_submit_thread;
 | |
|   std::vector<VkSemaphore> m_present_semaphores = {};
 | |
|   Common::Flag m_last_present_failed;
 | |
|   VkResult m_last_present_result = VK_SUCCESS;
 | |
|   bool m_use_threaded_submission = false;
 | |
|   u32 m_descriptor_set_count = DESCRIPTOR_SETS_PER_POOL;
 | |
| };
 | |
| 
 | |
| extern std::unique_ptr<CommandBufferManager> g_command_buffer_mgr;
 | |
| 
 | |
| }  // namespace Vulkan
 |