Vulkan/Context: Fix query pool leak

This commit is contained in:
Stenzek 2023-02-05 13:02:31 +10:00
parent d84a02a2c3
commit 2dd374d2a7
2 changed files with 36 additions and 53 deletions

View File

@ -351,7 +351,8 @@ bool Vulkan::Context::Create(std::string_view gpu_name, const WindowInfo* wi,
// Attempt to create the device. // Attempt to create the device.
if (!g_vulkan_context->CreateDevice(surface, enable_validation_layer, nullptr, 0, nullptr, 0, nullptr) || if (!g_vulkan_context->CreateDevice(surface, enable_validation_layer, nullptr, 0, nullptr, 0, nullptr) ||
!g_vulkan_context->CreateAllocator() || !g_vulkan_context->CreateGlobalDescriptorPool() || !g_vulkan_context->CreateAllocator() || !g_vulkan_context->CreateGlobalDescriptorPool() ||
!g_vulkan_context->CreateCommandBuffers() || !g_vulkan_context->CreateTextureStreamBuffer() || !g_vulkan_context->CreateQueryPool() || !g_vulkan_context->CreateCommandBuffers() ||
!g_vulkan_context->CreateTextureStreamBuffer() ||
(enable_surface && (*out_swap_chain = SwapChain::Create(wi_copy, surface, vsync)) == nullptr)) (enable_surface && (*out_swap_chain = SwapChain::Create(wi_copy, surface, vsync)) == nullptr))
{ {
// Since we are destroying the instance, we're also responsible for destroying the surface. // Since we are destroying the instance, we're also responsible for destroying the surface.
@ -368,31 +369,6 @@ bool Vulkan::Context::Create(std::string_view gpu_name, const WindowInfo* wi,
return true; return true;
} }
bool Vulkan::Context::CreateFromExistingInstance(
VkInstance instance, VkPhysicalDevice gpu, VkSurfaceKHR surface, bool take_ownership, bool enable_validation_layer,
bool enable_debug_utils, const char** required_device_extensions /* = nullptr */,
u32 num_required_device_extensions /* = 0 */, const char** required_device_layers /* = nullptr */,
u32 num_required_device_layers /* = 0 */, const VkPhysicalDeviceFeatures* required_features /* = nullptr */)
{
g_vulkan_context.reset(new Context(instance, gpu, take_ownership));
// Enable debug utils if the "Host GPU" log category is enabled.
if (enable_debug_utils)
g_vulkan_context->EnableDebugUtils();
// Attempt to create the device.
if (!g_vulkan_context->CreateDevice(surface, enable_validation_layer, required_device_extensions,
num_required_device_extensions, required_device_layers,
num_required_device_layers, required_features) ||
!g_vulkan_context->CreateGlobalDescriptorPool() || !g_vulkan_context->CreateCommandBuffers())
{
g_vulkan_context.reset();
return false;
}
return true;
}
void Vulkan::Context::Destroy() void Vulkan::Context::Destroy()
{ {
AssertMsg(g_vulkan_context, "Has context"); AssertMsg(g_vulkan_context, "Has context");
@ -405,6 +381,7 @@ void Vulkan::Context::Destroy()
g_vulkan_context->m_texture_upload_buffer.Destroy(false); g_vulkan_context->m_texture_upload_buffer.Destroy(false);
g_vulkan_context->DestroyRenderPassCache(); g_vulkan_context->DestroyRenderPassCache();
g_vulkan_context->DestroyQueryPool();
g_vulkan_context->DestroyGlobalDescriptorPool(); g_vulkan_context->DestroyGlobalDescriptorPool();
g_vulkan_context->DestroyCommandBuffers(); g_vulkan_context->DestroyCommandBuffers();
g_vulkan_context->DestroyAllocator(); g_vulkan_context->DestroyAllocator();
@ -806,37 +783,50 @@ bool Vulkan::Context::CreateGlobalDescriptorPool()
static_cast<u32>(countof(pool_sizes)), static_cast<u32>(countof(pool_sizes)),
pool_sizes}; pool_sizes};
VkResult res = vkCreateDescriptorPool(m_device, &pool_create_info, nullptr, &m_global_descriptor_pool); const VkResult res = vkCreateDescriptorPool(m_device, &pool_create_info, nullptr, &m_global_descriptor_pool);
if (res != VK_SUCCESS) if (res != VK_SUCCESS)
{ {
LOG_VULKAN_ERROR(res, "vkCreateDescriptorPool failed: "); LOG_VULKAN_ERROR(res, "vkCreateDescriptorPool failed: ");
return false; return false;
} }
Vulkan::Util::SetObjectName(g_vulkan_context->GetDevice(), m_global_descriptor_pool, "Global Descriptor Pool"); Vulkan::Util::SetObjectName(g_vulkan_context->GetDevice(), m_global_descriptor_pool, "Global Descriptor Pool");
return true;
}
void Vulkan::Context::DestroyGlobalDescriptorPool()
{
if (m_global_descriptor_pool == VK_NULL_HANDLE)
return;
vkDestroyDescriptorPool(m_device, m_global_descriptor_pool, nullptr);
m_global_descriptor_pool = VK_NULL_HANDLE;
}
bool Vulkan::Context::CreateQueryPool()
{
if (!m_gpu_timing_supported)
return true;
if (m_gpu_timing_supported)
{
const VkQueryPoolCreateInfo query_create_info = { const VkQueryPoolCreateInfo query_create_info = {
VK_STRUCTURE_TYPE_QUERY_POOL_CREATE_INFO, nullptr, 0, VK_QUERY_TYPE_TIMESTAMP, NUM_COMMAND_BUFFERS * 2, 0}; VK_STRUCTURE_TYPE_QUERY_POOL_CREATE_INFO, nullptr, 0, VK_QUERY_TYPE_TIMESTAMP, NUM_COMMAND_BUFFERS * 2, 0};
res = vkCreateQueryPool(m_device, &query_create_info, nullptr, &m_timestamp_query_pool); const VkResult res = vkCreateQueryPool(m_device, &query_create_info, nullptr, &m_timestamp_query_pool);
if (res != VK_SUCCESS) if (res != VK_SUCCESS)
{ {
LOG_VULKAN_ERROR(res, "vkCreateQueryPool failed: "); LOG_VULKAN_ERROR(res, "vkCreateQueryPool failed: ");
m_gpu_timing_supported = false; m_gpu_timing_supported = false;
return false; return false;
} }
}
return true; return true;
} }
void Vulkan::Context::DestroyGlobalDescriptorPool() void Vulkan::Context::DestroyQueryPool()
{ {
if (m_global_descriptor_pool != VK_NULL_HANDLE) if (!m_gpu_timing_supported)
{ return;
vkDestroyDescriptorPool(m_device, m_global_descriptor_pool, nullptr);
m_global_descriptor_pool = VK_NULL_HANDLE; vkDestroyQueryPool(m_device, m_timestamp_query_pool, nullptr);
} m_timestamp_query_pool = VK_NULL_HANDLE;
} }
bool Vulkan::Context::CreateTextureStreamBuffer() bool Vulkan::Context::CreateTextureStreamBuffer()

View File

@ -55,15 +55,6 @@ public:
static bool Create(std::string_view gpu_name, const WindowInfo* wi, std::unique_ptr<SwapChain>* out_swap_chain, static bool Create(std::string_view gpu_name, const WindowInfo* wi, std::unique_ptr<SwapChain>* out_swap_chain,
bool threaded_presentation, bool enable_debug_utils, bool enable_validation_layer, bool vsync); bool threaded_presentation, bool enable_debug_utils, bool enable_validation_layer, bool vsync);
// Creates a new context from a pre-existing instance.
static bool CreateFromExistingInstance(VkInstance instance, VkPhysicalDevice gpu, VkSurfaceKHR surface,
bool take_ownership, bool enable_validation_layer, bool enable_debug_utils,
const char** required_device_extensions = nullptr,
u32 num_required_device_extensions = 0,
const char** required_device_layers = nullptr,
u32 num_required_device_layers = 0,
const VkPhysicalDeviceFeatures* required_features = nullptr);
// Destroys context. // Destroys context.
static void Destroy(); static void Destroy();
@ -214,6 +205,8 @@ private:
void DestroyCommandBuffers(); void DestroyCommandBuffers();
bool CreateGlobalDescriptorPool(); bool CreateGlobalDescriptorPool();
void DestroyGlobalDescriptorPool(); void DestroyGlobalDescriptorPool();
bool CreateQueryPool();
void DestroyQueryPool();
bool CreateTextureStreamBuffer(); bool CreateTextureStreamBuffer();
void DestroyRenderPassCache(); void DestroyRenderPassCache();