2016-08-13 12:57:50 +00:00
|
|
|
// Copyright 2016 Dolphin Emulator Project
|
2021-07-05 01:22:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
#include "VideoBackends/Vulkan/VKGfx.h"
|
2021-12-10 02:22:16 +00:00
|
|
|
|
2017-09-08 09:42:56 +00:00
|
|
|
#include <algorithm>
|
2016-10-01 03:07:50 +00:00
|
|
|
#include <cstddef>
|
2016-08-13 12:57:50 +00:00
|
|
|
#include <cstdio>
|
|
|
|
#include <limits>
|
2016-10-01 03:07:50 +00:00
|
|
|
#include <string>
|
2017-04-09 19:05:24 +00:00
|
|
|
#include <tuple>
|
2016-10-01 03:07:50 +00:00
|
|
|
|
2018-01-21 13:13:25 +00:00
|
|
|
#include "Common/Assert.h"
|
2017-02-01 15:56:13 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
2023-06-17 05:08:07 +00:00
|
|
|
#include "Common/EnumUtils.h"
|
2016-10-01 03:07:50 +00:00
|
|
|
#include "Common/Logging/Log.h"
|
|
|
|
#include "Common/MsgHandler.h"
|
2016-08-13 12:57:50 +00:00
|
|
|
|
|
|
|
#include "VideoBackends/Vulkan/CommandBufferManager.h"
|
|
|
|
#include "VideoBackends/Vulkan/ObjectCache.h"
|
|
|
|
#include "VideoBackends/Vulkan/StateTracker.h"
|
2017-09-08 09:42:56 +00:00
|
|
|
#include "VideoBackends/Vulkan/VKPipeline.h"
|
|
|
|
#include "VideoBackends/Vulkan/VKShader.h"
|
2020-09-15 12:43:41 +00:00
|
|
|
#include "VideoBackends/Vulkan/VKSwapChain.h"
|
2017-04-23 04:44:34 +00:00
|
|
|
#include "VideoBackends/Vulkan/VKTexture.h"
|
2020-09-15 12:43:41 +00:00
|
|
|
#include "VideoBackends/Vulkan/VKVertexFormat.h"
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2017-05-25 13:59:23 +00:00
|
|
|
#include "VideoCommon/DriverDetails.h"
|
2019-02-15 01:59:50 +00:00
|
|
|
#include "VideoCommon/FramebufferManager.h"
|
2023-01-27 04:03:15 +00:00
|
|
|
#include "VideoCommon/Present.h"
|
2017-04-17 13:14:17 +00:00
|
|
|
#include "VideoCommon/RenderState.h"
|
2016-08-13 12:57:50 +00:00
|
|
|
#include "VideoCommon/VideoConfig.h"
|
|
|
|
|
|
|
|
namespace Vulkan
|
|
|
|
{
|
2023-01-28 02:12:28 +00:00
|
|
|
VKGfx::VKGfx(std::unique_ptr<SwapChain> swap_chain, float backbuffer_scale)
|
|
|
|
: m_swap_chain(std::move(swap_chain)), m_backbuffer_scale(backbuffer_scale)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2016-10-02 11:37:24 +00:00
|
|
|
UpdateActiveConfig();
|
2023-01-28 08:08:39 +00:00
|
|
|
for (SamplerState& sampler_state : m_sampler_states)
|
|
|
|
sampler_state = RenderState::GetPointSamplerState();
|
2018-02-09 10:52:25 +00:00
|
|
|
|
2016-08-13 12:57:50 +00:00
|
|
|
// Various initialization routines will have executed commands on the command buffer.
|
|
|
|
// Execute what we have done before beginning the first frame.
|
2019-02-15 01:59:50 +00:00
|
|
|
ExecuteCommandBuffer(true, false);
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
VKGfx::~VKGfx() = default;
|
|
|
|
|
|
|
|
bool VKGfx::IsHeadless() const
|
2018-02-09 10:52:25 +00:00
|
|
|
{
|
2023-01-28 02:12:28 +00:00
|
|
|
return m_swap_chain == nullptr;
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
std::unique_ptr<AbstractTexture> VKGfx::CreateTexture(const TextureConfig& config,
|
|
|
|
std::string_view name)
|
2017-09-30 06:25:36 +00:00
|
|
|
{
|
2021-08-28 05:30:05 +00:00
|
|
|
return VKTexture::Create(config, name);
|
2017-09-30 06:25:36 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
std::unique_ptr<AbstractStagingTexture> VKGfx::CreateStagingTexture(StagingTextureType type,
|
|
|
|
const TextureConfig& config)
|
2017-10-21 14:49:40 +00:00
|
|
|
{
|
|
|
|
return VKStagingTexture::Create(type, config);
|
|
|
|
}
|
|
|
|
|
2021-08-28 05:30:05 +00:00
|
|
|
std::unique_ptr<AbstractShader>
|
2023-01-28 02:12:28 +00:00
|
|
|
VKGfx::CreateShaderFromSource(ShaderStage stage, std::string_view source, std::string_view name)
|
2017-09-08 09:42:56 +00:00
|
|
|
{
|
2021-08-28 05:30:05 +00:00
|
|
|
return VKShader::CreateFromSource(stage, source, name);
|
2017-09-08 09:42:56 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
std::unique_ptr<AbstractShader> VKGfx::CreateShaderFromBinary(ShaderStage stage, const void* data,
|
|
|
|
size_t length, std::string_view name)
|
2017-09-08 09:42:56 +00:00
|
|
|
{
|
2021-08-28 05:30:05 +00:00
|
|
|
return VKShader::CreateFromBinary(stage, data, length, name);
|
2017-09-08 09:42:56 +00:00
|
|
|
}
|
|
|
|
|
2019-02-15 01:59:50 +00:00
|
|
|
std::unique_ptr<NativeVertexFormat>
|
2023-01-28 02:12:28 +00:00
|
|
|
VKGfx::CreateNativeVertexFormat(const PortableVertexDeclaration& vtx_decl)
|
2019-02-15 01:59:50 +00:00
|
|
|
{
|
|
|
|
return std::make_unique<VertexFormat>(vtx_decl);
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
std::unique_ptr<AbstractPipeline> VKGfx::CreatePipeline(const AbstractPipelineConfig& config,
|
|
|
|
const void* cache_data,
|
|
|
|
size_t cache_data_length)
|
2017-09-08 09:42:56 +00:00
|
|
|
{
|
|
|
|
return VKPipeline::Create(config);
|
|
|
|
}
|
|
|
|
|
2023-05-29 01:59:02 +00:00
|
|
|
std::unique_ptr<AbstractFramebuffer>
|
|
|
|
VKGfx::CreateFramebuffer(AbstractTexture* color_attachment, AbstractTexture* depth_attachment,
|
|
|
|
std::vector<AbstractTexture*> additional_color_attachments)
|
2018-01-21 10:22:45 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
return VKFramebuffer::Create(static_cast<VKTexture*>(color_attachment),
|
2023-05-29 01:59:02 +00:00
|
|
|
static_cast<VKTexture*>(depth_attachment),
|
|
|
|
std::move(additional_color_attachments));
|
2018-01-21 10:22:45 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetPipeline(const AbstractPipeline* pipeline)
|
2017-09-08 09:42:56 +00:00
|
|
|
{
|
2018-02-24 15:15:35 +00:00
|
|
|
StateTracker::GetInstance()->SetPipeline(static_cast<const VKPipeline*>(pipeline));
|
2017-09-08 09:42:56 +00:00
|
|
|
}
|
|
|
|
|
2023-01-31 02:44:38 +00:00
|
|
|
void VKGfx::ClearRegion(const MathUtil::Rectangle<int>& target_rc, bool color_enable,
|
2023-01-28 02:12:28 +00:00
|
|
|
bool alpha_enable, bool z_enable, u32 color, u32 z)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2016-09-11 06:37:41 +00:00
|
|
|
VkRect2D target_vk_rc = {
|
|
|
|
{target_rc.left, target_rc.top},
|
|
|
|
{static_cast<uint32_t>(target_rc.GetWidth()), static_cast<uint32_t>(target_rc.GetHeight())}};
|
|
|
|
|
2016-10-23 11:00:20 +00:00
|
|
|
// Convert RGBA8 -> floating-point values.
|
|
|
|
VkClearValue clear_color_value = {};
|
|
|
|
VkClearValue clear_depth_value = {};
|
|
|
|
clear_color_value.color.float32[0] = static_cast<float>((color >> 16) & 0xFF) / 255.0f;
|
|
|
|
clear_color_value.color.float32[1] = static_cast<float>((color >> 8) & 0xFF) / 255.0f;
|
|
|
|
clear_color_value.color.float32[2] = static_cast<float>((color >> 0) & 0xFF) / 255.0f;
|
|
|
|
clear_color_value.color.float32[3] = static_cast<float>((color >> 24) & 0xFF) / 255.0f;
|
2019-02-15 01:59:50 +00:00
|
|
|
clear_depth_value.depthStencil.depth = static_cast<float>(z & 0xFFFFFF) / 16777216.0f;
|
|
|
|
if (!g_ActiveConfig.backend_info.bSupportsReversedDepthRange)
|
|
|
|
clear_depth_value.depthStencil.depth = 1.0f - clear_depth_value.depthStencil.depth;
|
2016-10-23 11:00:20 +00:00
|
|
|
|
2016-09-11 06:37:41 +00:00
|
|
|
// If we're not in a render pass (start of the frame), we can use a clear render pass
|
|
|
|
// to discard the data, rather than loading and then clearing.
|
2017-09-03 04:04:14 +00:00
|
|
|
bool use_clear_attachments = (color_enable && alpha_enable) || z_enable;
|
|
|
|
bool use_clear_render_pass =
|
|
|
|
!StateTracker::GetInstance()->InRenderPass() && color_enable && alpha_enable && z_enable;
|
|
|
|
|
|
|
|
// The NVIDIA Vulkan driver causes the GPU to lock up, or throw exceptions if MSAA is enabled,
|
|
|
|
// a non-full clear rect is specified, and a clear loadop or vkCmdClearAttachments is used.
|
|
|
|
if (g_ActiveConfig.iMultisamples > 1 &&
|
|
|
|
DriverDetails::HasBug(DriverDetails::BUG_BROKEN_MSAA_CLEAR))
|
2017-06-20 01:46:17 +00:00
|
|
|
{
|
|
|
|
use_clear_render_pass = false;
|
2017-09-03 04:04:14 +00:00
|
|
|
use_clear_attachments = false;
|
2017-06-20 01:46:17 +00:00
|
|
|
}
|
2017-09-03 04:04:14 +00:00
|
|
|
|
|
|
|
// This path cannot be used if the driver implementation doesn't guarantee pixels with no drawn
|
|
|
|
// geometry in "this" renderpass won't be cleared
|
|
|
|
if (DriverDetails::HasBug(DriverDetails::BUG_BROKEN_CLEAR_LOADOP_RENDERPASS))
|
2016-09-11 06:37:41 +00:00
|
|
|
use_clear_render_pass = false;
|
|
|
|
|
2023-05-29 01:59:02 +00:00
|
|
|
auto* vk_frame_buffer = static_cast<VKFramebuffer*>(m_current_framebuffer);
|
|
|
|
|
2016-09-11 06:37:41 +00:00
|
|
|
// Fastest path: Use a render pass to clear the buffers.
|
|
|
|
if (use_clear_render_pass)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2023-05-29 01:59:02 +00:00
|
|
|
vk_frame_buffer->SetAndClear(target_vk_rc, clear_color_value, clear_depth_value);
|
2016-09-11 06:37:41 +00:00
|
|
|
return;
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2016-09-11 06:37:41 +00:00
|
|
|
// Fast path: Use vkCmdClearAttachments to clear the buffers within a render path
|
|
|
|
// We can't use this when preserving alpha but clearing color.
|
2017-09-03 04:04:14 +00:00
|
|
|
if (use_clear_attachments)
|
2016-09-11 06:37:41 +00:00
|
|
|
{
|
2023-05-29 01:59:02 +00:00
|
|
|
std::vector<VkClearAttachment> clear_attachments;
|
|
|
|
bool has_color = false;
|
2016-09-11 06:37:41 +00:00
|
|
|
if (color_enable && alpha_enable)
|
|
|
|
{
|
2023-05-29 01:59:02 +00:00
|
|
|
VkClearAttachment clear_attachment;
|
|
|
|
clear_attachment.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
clear_attachment.colorAttachment = 0;
|
|
|
|
clear_attachment.clearValue = clear_color_value;
|
|
|
|
clear_attachments.push_back(std::move(clear_attachment));
|
2016-09-11 06:37:41 +00:00
|
|
|
color_enable = false;
|
|
|
|
alpha_enable = false;
|
2023-05-29 01:59:02 +00:00
|
|
|
has_color = true;
|
2016-09-11 06:37:41 +00:00
|
|
|
}
|
|
|
|
if (z_enable)
|
|
|
|
{
|
2023-05-29 01:59:02 +00:00
|
|
|
VkClearAttachment clear_attachment;
|
|
|
|
clear_attachment.aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT;
|
|
|
|
clear_attachment.colorAttachment = 0;
|
|
|
|
clear_attachment.clearValue = clear_depth_value;
|
|
|
|
clear_attachments.push_back(std::move(clear_attachment));
|
2016-09-11 06:37:41 +00:00
|
|
|
z_enable = false;
|
|
|
|
}
|
2023-05-29 01:59:02 +00:00
|
|
|
if (has_color)
|
|
|
|
{
|
|
|
|
for (std::size_t i = 0; i < vk_frame_buffer->GetNumberOfAdditonalAttachments(); i++)
|
|
|
|
{
|
|
|
|
VkClearAttachment clear_attachment;
|
|
|
|
clear_attachment.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
clear_attachment.colorAttachment = 0;
|
|
|
|
clear_attachment.clearValue = clear_color_value;
|
|
|
|
clear_attachments.push_back(std::move(clear_attachment));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!clear_attachments.empty())
|
2016-09-11 06:37:41 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
VkClearRect vk_rect = {target_vk_rc, 0, g_framebuffer_manager->GetEFBLayers()};
|
2016-10-22 10:50:36 +00:00
|
|
|
if (!StateTracker::GetInstance()->IsWithinRenderArea(
|
|
|
|
target_vk_rc.offset.x, target_vk_rc.offset.y, target_vk_rc.extent.width,
|
|
|
|
target_vk_rc.extent.height))
|
2016-09-11 06:37:41 +00:00
|
|
|
{
|
2016-10-22 10:50:36 +00:00
|
|
|
StateTracker::GetInstance()->EndClearRenderPass();
|
2016-09-11 06:37:41 +00:00
|
|
|
}
|
2016-10-22 10:50:36 +00:00
|
|
|
StateTracker::GetInstance()->BeginRenderPass();
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2023-05-29 01:59:02 +00:00
|
|
|
vkCmdClearAttachments(g_command_buffer_mgr->GetCurrentCommandBuffer(),
|
|
|
|
static_cast<uint32_t>(clear_attachments.size()),
|
|
|
|
clear_attachments.data(), 1, &vk_rect);
|
2016-09-11 06:37:41 +00:00
|
|
|
}
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2016-09-11 06:37:41 +00:00
|
|
|
// Anything left over for the slow path?
|
2016-08-13 12:57:50 +00:00
|
|
|
if (!color_enable && !alpha_enable && !z_enable)
|
|
|
|
return;
|
|
|
|
|
2023-01-31 02:44:38 +00:00
|
|
|
AbstractGfx::ClearRegion(target_rc, color_enable, alpha_enable, z_enable, color, z);
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::Flush()
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
ExecuteCommandBuffer(true, false);
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::WaitForGPUIdle()
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
ExecuteCommandBuffer(false, true);
|
2018-11-28 04:30:47 +00:00
|
|
|
}
|
2017-05-29 22:02:09 +00:00
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::BindBackbuffer(const ClearColor& clear_color)
|
2018-11-28 04:30:47 +00:00
|
|
|
{
|
2016-10-22 10:50:36 +00:00
|
|
|
StateTracker::GetInstance()->EndRenderPass();
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2022-11-07 01:20:22 +00:00
|
|
|
if (!g_command_buffer_mgr->CheckLastPresentDone())
|
|
|
|
g_command_buffer_mgr->WaitForWorkerThreadIdle();
|
|
|
|
|
2018-01-26 06:23:24 +00:00
|
|
|
// Handle host window resizes.
|
|
|
|
CheckForSurfaceChange();
|
|
|
|
CheckForSurfaceResize();
|
|
|
|
|
2019-09-30 15:10:08 +00:00
|
|
|
// Check for exclusive fullscreen request.
|
|
|
|
if (m_swap_chain->GetCurrentFullscreenState() != m_swap_chain->GetNextFullscreenState() &&
|
|
|
|
!m_swap_chain->SetFullscreenState(m_swap_chain->GetNextFullscreenState()))
|
|
|
|
{
|
|
|
|
// if it fails, don't keep trying
|
|
|
|
m_swap_chain->SetNextFullscreenState(m_swap_chain->GetCurrentFullscreenState());
|
|
|
|
}
|
|
|
|
|
2021-04-11 08:10:38 +00:00
|
|
|
const bool present_fail = g_command_buffer_mgr->CheckLastPresentFail();
|
|
|
|
VkResult res = present_fail ? g_command_buffer_mgr->GetLastPresentResult() :
|
|
|
|
m_swap_chain->AcquireNextImage();
|
|
|
|
|
|
|
|
if (res == VK_ERROR_FULL_SCREEN_EXCLUSIVE_MODE_LOST_EXT &&
|
|
|
|
!m_swap_chain->GetCurrentFullscreenState())
|
|
|
|
{
|
|
|
|
// AMD's binary driver as of 21.3 seems to return exclusive fullscreen lost even when it was
|
|
|
|
// never requested, so long as the caller requested it to be application controlled. Handle
|
|
|
|
// this ignoring the lost result and just continuing as normal if we never acquired it.
|
|
|
|
res = VK_SUCCESS;
|
|
|
|
if (present_fail)
|
|
|
|
{
|
|
|
|
// We still need to acquire an image.
|
|
|
|
res = m_swap_chain->AcquireNextImage();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 15:10:08 +00:00
|
|
|
if (res != VK_SUCCESS)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
// Execute cmdbuffer before resizing, as the last frame could still be presenting.
|
|
|
|
ExecuteCommandBuffer(false, true);
|
2019-09-30 15:10:08 +00:00
|
|
|
|
|
|
|
// Was this a lost exclusive fullscreen?
|
|
|
|
if (res == VK_ERROR_FULL_SCREEN_EXCLUSIVE_MODE_LOST_EXT)
|
|
|
|
{
|
|
|
|
// The present keeps returning exclusive mode lost unless we re-create the swap chain.
|
2020-11-09 08:26:14 +00:00
|
|
|
INFO_LOG_FMT(VIDEO, "Lost exclusive fullscreen.");
|
2019-09-30 15:10:08 +00:00
|
|
|
m_swap_chain->RecreateSwapChain();
|
|
|
|
}
|
|
|
|
else if (res == VK_SUBOPTIMAL_KHR || res == VK_ERROR_OUT_OF_DATE_KHR)
|
|
|
|
{
|
2020-11-09 08:26:14 +00:00
|
|
|
INFO_LOG_FMT(VIDEO, "Resizing swap chain due to suboptimal/out-of-date");
|
2019-09-30 15:10:08 +00:00
|
|
|
m_swap_chain->ResizeSwapChain();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-10-11 21:23:38 +00:00
|
|
|
ERROR_LOG_FMT(VIDEO, "Unknown present error {:#010X} {}, please report.",
|
2023-06-17 05:08:07 +00:00
|
|
|
Common::ToUnderlying(res), VkResultToString(res));
|
2019-09-30 15:10:08 +00:00
|
|
|
m_swap_chain->RecreateSwapChain();
|
|
|
|
}
|
|
|
|
|
2019-01-27 02:59:57 +00:00
|
|
|
res = m_swap_chain->AcquireNextImage();
|
2021-04-11 08:10:38 +00:00
|
|
|
if (res != VK_SUCCESS)
|
2023-06-17 05:08:07 +00:00
|
|
|
PanicAlertFmt("Failed to grab image from swap chain: {:#010X} {}", Common::ToUnderlying(res),
|
2022-10-11 21:23:38 +00:00
|
|
|
VkResultToString(res));
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Transition from undefined (or present src, but it can be substituted) to
|
|
|
|
// color attachment ready for writing. These transitions must occur outside
|
|
|
|
// a render pass, unless the render pass declares a self-dependency.
|
2019-02-15 01:59:50 +00:00
|
|
|
m_swap_chain->GetCurrentTexture()->OverrideImageLayout(VK_IMAGE_LAYOUT_UNDEFINED);
|
|
|
|
m_swap_chain->GetCurrentTexture()->TransitionToLayout(
|
|
|
|
g_command_buffer_mgr->GetCurrentCommandBuffer(), VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
|
|
|
|
SetAndClearFramebuffer(m_swap_chain->GetCurrentFramebuffer(),
|
|
|
|
ClearColor{{0.0f, 0.0f, 0.0f, 1.0f}});
|
2018-11-28 04:30:47 +00:00
|
|
|
}
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::PresentBackbuffer()
|
2018-11-28 04:30:47 +00:00
|
|
|
{
|
2016-08-13 12:57:50 +00:00
|
|
|
// End drawing to backbuffer
|
2017-09-09 06:09:24 +00:00
|
|
|
StateTracker::GetInstance()->EndRenderPass();
|
2016-08-13 12:57:50 +00:00
|
|
|
|
|
|
|
// Transition the backbuffer to PRESENT_SRC to ensure all commands drawing
|
|
|
|
// to it have finished before present.
|
2019-01-27 02:59:57 +00:00
|
|
|
m_swap_chain->GetCurrentTexture()->TransitionToLayout(
|
|
|
|
g_command_buffer_mgr->GetCurrentCommandBuffer(), VK_IMAGE_LAYOUT_PRESENT_SRC_KHR);
|
2018-11-28 04:30:47 +00:00
|
|
|
|
|
|
|
// Submit the current command buffer, signaling rendering finished semaphore when it's done
|
|
|
|
// Because this final command buffer is rendering to the swap chain, we need to wait for
|
|
|
|
// the available semaphore to be signaled before executing the buffer. This final submission
|
|
|
|
// can happen off-thread in the background while we're preparing the next frame.
|
2019-03-17 05:59:22 +00:00
|
|
|
g_command_buffer_mgr->SubmitCommandBuffer(true, false, m_swap_chain->GetSwapChain(),
|
2019-01-27 02:59:57 +00:00
|
|
|
m_swap_chain->GetCurrentImageIndex());
|
2019-02-15 01:59:50 +00:00
|
|
|
|
|
|
|
// New cmdbuffer, so invalidate state.
|
|
|
|
StateTracker::GetInstance()->InvalidateCachedState();
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetFullscreen(bool enable_fullscreen)
|
2019-09-30 15:10:08 +00:00
|
|
|
{
|
|
|
|
if (!m_swap_chain->IsFullscreenSupported())
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_swap_chain->SetNextFullscreenState(enable_fullscreen);
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
bool VKGfx::IsFullscreen() const
|
2019-09-30 15:10:08 +00:00
|
|
|
{
|
|
|
|
return m_swap_chain && m_swap_chain->GetCurrentFullscreenState();
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::ExecuteCommandBuffer(bool submit_off_thread, bool wait_for_completion)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
StateTracker::GetInstance()->EndRenderPass();
|
2018-11-28 04:30:47 +00:00
|
|
|
|
2019-03-17 05:59:22 +00:00
|
|
|
g_command_buffer_mgr->SubmitCommandBuffer(submit_off_thread, wait_for_completion);
|
2018-11-28 04:30:47 +00:00
|
|
|
|
2019-02-15 01:59:50 +00:00
|
|
|
StateTracker::GetInstance()->InvalidateCachedState();
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::CheckForSurfaceChange()
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2023-01-27 04:03:15 +00:00
|
|
|
if (!g_presenter->SurfaceChangedTestAndClear() || !m_swap_chain)
|
2016-08-13 12:57:50 +00:00
|
|
|
return;
|
|
|
|
|
2018-01-26 06:23:24 +00:00
|
|
|
// Submit the current draws up until rendering the XFB.
|
2019-02-15 01:59:50 +00:00
|
|
|
ExecuteCommandBuffer(false, true);
|
2017-09-16 06:15:20 +00:00
|
|
|
|
|
|
|
// Clear the present failed flag, since we don't want to resize after recreating.
|
|
|
|
g_command_buffer_mgr->CheckLastPresentFail();
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2018-10-03 13:03:13 +00:00
|
|
|
// Recreate the surface. If this fails we're in trouble.
|
2023-01-27 04:03:15 +00:00
|
|
|
if (!m_swap_chain->RecreateSurface(g_presenter->GetNewSurfaceHandle()))
|
2020-12-02 18:17:27 +00:00
|
|
|
PanicAlertFmt("Failed to recreate Vulkan surface. Cannot continue.");
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2017-09-16 06:15:20 +00:00
|
|
|
// Handle case where the dimensions are now different.
|
|
|
|
OnSwapChainResized();
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::CheckForSurfaceResize()
|
2018-01-26 06:23:24 +00:00
|
|
|
{
|
2023-01-27 04:03:15 +00:00
|
|
|
if (!g_presenter->SurfaceResizedTestAndClear())
|
2018-01-26 06:23:24 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
// If we don't have a surface, how can we resize the swap chain?
|
|
|
|
// CheckForSurfaceChange should handle this case.
|
|
|
|
if (!m_swap_chain)
|
|
|
|
{
|
2020-11-09 08:26:14 +00:00
|
|
|
WARN_LOG_FMT(VIDEO, "Surface resize event received without active surface, ignoring");
|
2018-01-26 06:23:24 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for the GPU to catch up since we're going to destroy the swap chain.
|
2019-02-15 01:59:50 +00:00
|
|
|
ExecuteCommandBuffer(false, true);
|
2018-01-26 06:23:24 +00:00
|
|
|
|
|
|
|
// Clear the present failed flag, since we don't want to resize after recreating.
|
|
|
|
g_command_buffer_mgr->CheckLastPresentFail();
|
|
|
|
|
|
|
|
// Resize the swap chain.
|
|
|
|
m_swap_chain->RecreateSwapChain();
|
|
|
|
OnSwapChainResized();
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::OnConfigChanged(u32 bits)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2023-01-30 11:49:23 +00:00
|
|
|
AbstractGfx::OnConfigChanged(bits);
|
|
|
|
|
2019-02-15 01:59:50 +00:00
|
|
|
if (bits & CONFIG_CHANGE_BIT_HOST_CONFIG)
|
|
|
|
g_object_cache->ReloadPipelineCache();
|
2016-08-13 12:57:50 +00:00
|
|
|
|
|
|
|
// For vsync, we need to change the present mode, which means recreating the swap chain.
|
2018-11-28 04:30:47 +00:00
|
|
|
if (m_swap_chain && bits & CONFIG_CHANGE_BIT_VSYNC)
|
2016-10-02 12:09:19 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
ExecuteCommandBuffer(false, true);
|
2019-01-27 02:24:53 +00:00
|
|
|
m_swap_chain->SetVSync(g_ActiveConfig.bVSyncActive);
|
2016-10-02 12:09:19 +00:00
|
|
|
}
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2017-06-26 19:08:21 +00:00
|
|
|
// For quad-buffered stereo we need to change the layer count, so recreate the swap chain.
|
Video: implement color correction to match the NTSC and PAL color spaces (and gamma) that GC and Wii targeted.
To further increase the accuracy of the post process phase, I've added (scRGB) HDR support, which is necessary
to fully display the PAL and NTSC-J color spaces, and also to improve the quality of post process texture samplings and
do them in linear space instead of gamma space (which is very important when playing at low resolutions).
For SDR, the quality is also slightly increased, at least if any post process runs, as the buffer is now
R10G10B10A2 (on Vulkan, DX11 and DX12) if supported; previously it was R8G8B8A8 but the alpha bits were wasted.
Gamma correction is arguably the most important thing as Dolphin on Windows outputted in "sRGB" (implicitly)
as that's what Windows expects by default, though sRGB gamma is very different from the gamma commonly used
by video standards dating to the pre HDR era (roughly gamma 2.35).
Additionally, the addition of HDR support (which is pretty straight forward and minimal), added support for
our own custom AutoHDR shaders, which would allow us to achieve decent looking HDR in Dolphin games without
having to use SpecialK or Windows 11 AutoHDR. Both of which don't necessarily play nice with older games
with strongly different and simpler lighting. HDR should also be supported in Linux.
Development of my own AutoHDR shader is almost complete and will come next.
This has been carefully tested and there should be no regression in any of the different features that Dolphin
offers, like multisampling, stereo rendering, other post processes, etc etc.
Fixes: https://bugs.dolphin-emu.org/issues/8941
Co-authored-by: EndlesslyFlowering <EndlesslyFlowering@protonmail.com>
Co-authored-by: Dogway <lin_ares@hotmail.com>
2023-06-10 08:48:05 +00:00
|
|
|
if (m_swap_chain && (bits & CONFIG_CHANGE_BIT_STEREO_MODE) || (bits & CONFIG_CHANGE_BIT_HDR))
|
2017-09-16 06:15:20 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
ExecuteCommandBuffer(false, true);
|
2017-09-16 06:15:20 +00:00
|
|
|
m_swap_chain->RecreateSwapChain();
|
|
|
|
}
|
2017-06-26 19:08:21 +00:00
|
|
|
|
2016-08-13 12:57:50 +00:00
|
|
|
// Wipe sampler cache if force texture filtering or anisotropy changes.
|
2018-11-28 04:30:47 +00:00
|
|
|
if (bits & (CONFIG_CHANGE_BIT_ANISOTROPY | CONFIG_CHANGE_BIT_FORCE_TEXTURE_FILTERING))
|
2019-02-15 01:59:50 +00:00
|
|
|
{
|
|
|
|
ExecuteCommandBuffer(false, true);
|
2016-08-13 12:57:50 +00:00
|
|
|
ResetSamplerStates();
|
2019-02-15 01:59:50 +00:00
|
|
|
}
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::OnSwapChainResized()
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2023-01-27 04:03:15 +00:00
|
|
|
g_presenter->SetBackbuffer(m_swap_chain->GetWidth(), m_swap_chain->GetHeight());
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::BindFramebuffer(VKFramebuffer* fb)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2016-10-22 10:50:36 +00:00
|
|
|
StateTracker::GetInstance()->EndRenderPass();
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2019-02-15 01:59:50 +00:00
|
|
|
// Shouldn't be bound as a texture.
|
2023-05-29 01:59:02 +00:00
|
|
|
fb->Unbind();
|
2018-01-21 10:22:45 +00:00
|
|
|
|
|
|
|
fb->TransitionForRender();
|
2019-02-15 01:59:50 +00:00
|
|
|
StateTracker::GetInstance()->SetFramebuffer(fb);
|
2018-01-21 10:22:45 +00:00
|
|
|
m_current_framebuffer = fb;
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetFramebuffer(AbstractFramebuffer* framebuffer)
|
2018-01-21 10:22:45 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
if (m_current_framebuffer == framebuffer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VKFramebuffer* vkfb = static_cast<VKFramebuffer*>(framebuffer);
|
2018-01-21 10:22:45 +00:00
|
|
|
BindFramebuffer(vkfb);
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetAndDiscardFramebuffer(AbstractFramebuffer* framebuffer)
|
2018-01-21 10:22:45 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
if (m_current_framebuffer == framebuffer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VKFramebuffer* vkfb = static_cast<VKFramebuffer*>(framebuffer);
|
2018-01-21 10:22:45 +00:00
|
|
|
BindFramebuffer(vkfb);
|
|
|
|
|
|
|
|
// If we're discarding, begin the discard pass, then switch to a load pass.
|
|
|
|
// This way if the command buffer is flushed, we don't start another discard pass.
|
2019-02-15 01:59:50 +00:00
|
|
|
StateTracker::GetInstance()->BeginDiscardRenderPass();
|
2018-01-21 10:22:45 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetAndClearFramebuffer(AbstractFramebuffer* framebuffer, const ClearColor& color_value,
|
|
|
|
float depth_value)
|
2018-01-21 10:22:45 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
VKFramebuffer* vkfb = static_cast<VKFramebuffer*>(framebuffer);
|
2018-01-21 10:22:45 +00:00
|
|
|
BindFramebuffer(vkfb);
|
|
|
|
|
2023-05-29 01:59:02 +00:00
|
|
|
VkClearValue clear_color_value;
|
|
|
|
std::memcpy(clear_color_value.color.float32, color_value.data(),
|
|
|
|
sizeof(clear_color_value.color.float32));
|
|
|
|
VkClearValue clear_depth_value;
|
|
|
|
clear_depth_value.depthStencil.depth = depth_value;
|
|
|
|
clear_depth_value.depthStencil.stencil = 0;
|
|
|
|
vkfb->SetAndClear(vkfb->GetRect(), clear_color_value, clear_depth_value);
|
2018-01-21 10:22:45 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetTexture(u32 index, const AbstractTexture* texture)
|
2018-01-21 13:13:25 +00:00
|
|
|
{
|
|
|
|
// Texture should always be in SHADER_READ_ONLY layout prior to use.
|
|
|
|
// This is so we don't need to transition during render passes.
|
2019-02-15 01:59:50 +00:00
|
|
|
const VKTexture* tex = static_cast<const VKTexture*>(texture);
|
|
|
|
if (tex)
|
|
|
|
{
|
|
|
|
if (tex->GetLayout() != VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL)
|
|
|
|
{
|
|
|
|
if (StateTracker::GetInstance()->InRenderPass())
|
|
|
|
{
|
2023-01-28 02:12:28 +00:00
|
|
|
WARN_LOG_FMT(VIDEO, "Transitioning image in render pass in VKGfx::SetTexture()");
|
2019-02-15 01:59:50 +00:00
|
|
|
StateTracker::GetInstance()->EndRenderPass();
|
|
|
|
}
|
|
|
|
|
|
|
|
tex->TransitionToLayout(g_command_buffer_mgr->GetCurrentCommandBuffer(),
|
|
|
|
VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
StateTracker::GetInstance()->SetTexture(index, tex->GetView());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
StateTracker::GetInstance()->SetTexture(0, VK_NULL_HANDLE);
|
|
|
|
}
|
2018-01-21 13:13:25 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetSamplerState(u32 index, const SamplerState& state)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
|
|
|
// Skip lookup if the state hasn't changed.
|
2021-08-09 04:11:50 +00:00
|
|
|
if (m_sampler_states[index] == state)
|
2016-08-13 12:57:50 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
// Look up new state and replace in state tracker.
|
2017-09-09 08:30:15 +00:00
|
|
|
VkSampler sampler = g_object_cache->GetSampler(state);
|
2016-08-13 12:57:50 +00:00
|
|
|
if (sampler == VK_NULL_HANDLE)
|
|
|
|
{
|
2020-11-09 08:26:14 +00:00
|
|
|
ERROR_LOG_FMT(VIDEO, "Failed to create sampler");
|
2016-08-13 12:57:50 +00:00
|
|
|
sampler = g_object_cache->GetPointSampler();
|
|
|
|
}
|
|
|
|
|
2017-09-09 08:30:15 +00:00
|
|
|
StateTracker::GetInstance()->SetSampler(index, sampler);
|
2021-08-09 04:11:50 +00:00
|
|
|
m_sampler_states[index] = state;
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetComputeImageTexture(AbstractTexture* texture, bool read, bool write)
|
2019-02-15 01:59:50 +00:00
|
|
|
{
|
|
|
|
VKTexture* vk_texture = static_cast<VKTexture*>(texture);
|
|
|
|
if (vk_texture)
|
|
|
|
{
|
|
|
|
StateTracker::GetInstance()->EndRenderPass();
|
|
|
|
StateTracker::GetInstance()->SetImageTexture(vk_texture->GetView());
|
|
|
|
vk_texture->TransitionToLayout(g_command_buffer_mgr->GetCurrentCommandBuffer(),
|
|
|
|
read ? (write ? VKTexture::ComputeImageLayout::ReadWrite :
|
|
|
|
VKTexture::ComputeImageLayout::ReadOnly) :
|
|
|
|
VKTexture::ComputeImageLayout::WriteOnly);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
StateTracker::GetInstance()->SetImageTexture(VK_NULL_HANDLE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::UnbindTexture(const AbstractTexture* texture)
|
2018-01-21 13:13:25 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
StateTracker::GetInstance()->UnbindTexture(static_cast<const VKTexture*>(texture)->GetView());
|
2018-01-21 13:13:25 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::ResetSamplerStates()
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
|
|
|
// Invalidate all sampler states, next draw will re-initialize them.
|
2019-02-15 01:59:50 +00:00
|
|
|
for (u32 i = 0; i < m_sampler_states.size(); i++)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2021-08-09 04:11:50 +00:00
|
|
|
m_sampler_states[i] = RenderState::GetPointSamplerState();
|
2016-10-22 10:50:36 +00:00
|
|
|
StateTracker::GetInstance()->SetSampler(i, g_object_cache->GetPointSampler());
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Invalidate all sampler objects (some will be unused now).
|
|
|
|
g_object_cache->ClearSamplerCache();
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetScissorRect(const MathUtil::Rectangle<int>& rc)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2018-01-21 12:12:32 +00:00
|
|
|
VkRect2D scissor = {{rc.left, rc.top},
|
|
|
|
{static_cast<u32>(rc.GetWidth()), static_cast<u32>(rc.GetHeight())}};
|
2019-04-28 06:01:07 +00:00
|
|
|
|
|
|
|
// See Vulkan spec for vkCmdSetScissor:
|
|
|
|
// The x and y members of offset must be greater than or equal to 0.
|
|
|
|
if (scissor.offset.x < 0)
|
|
|
|
{
|
|
|
|
scissor.extent.width -= -scissor.offset.x;
|
|
|
|
scissor.offset.x = 0;
|
|
|
|
}
|
|
|
|
if (scissor.offset.y < 0)
|
|
|
|
{
|
|
|
|
scissor.extent.height -= -scissor.offset.y;
|
|
|
|
scissor.offset.y = 0;
|
|
|
|
}
|
2016-10-22 10:50:36 +00:00
|
|
|
StateTracker::GetInstance()->SetScissor(scissor);
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::SetViewport(float x, float y, float width, float height, float near_depth,
|
|
|
|
float far_depth)
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
VkViewport viewport = {x, y, width, height, near_depth, far_depth};
|
2016-10-22 10:50:36 +00:00
|
|
|
StateTracker::GetInstance()->SetViewport(viewport);
|
2016-08-13 12:57:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::Draw(u32 base_vertex, u32 num_vertices)
|
2018-11-27 07:16:53 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
if (!StateTracker::GetInstance()->Bind())
|
2018-11-27 07:16:53 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
vkCmdDraw(g_command_buffer_mgr->GetCurrentCommandBuffer(), num_vertices, 1, base_vertex, 0);
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::DrawIndexed(u32 base_index, u32 num_indices, u32 base_vertex)
|
2018-11-27 07:16:53 +00:00
|
|
|
{
|
|
|
|
if (!StateTracker::GetInstance()->Bind())
|
|
|
|
return;
|
|
|
|
|
|
|
|
vkCmdDrawIndexed(g_command_buffer_mgr->GetCurrentCommandBuffer(), num_indices, 1, base_index,
|
|
|
|
base_vertex, 0);
|
|
|
|
}
|
2019-02-15 01:59:50 +00:00
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
void VKGfx::DispatchComputeShader(const AbstractShader* shader, u32 groupsize_x, u32 groupsize_y,
|
|
|
|
u32 groupsize_z, u32 groups_x, u32 groups_y, u32 groups_z)
|
2019-02-15 01:59:50 +00:00
|
|
|
{
|
|
|
|
StateTracker::GetInstance()->SetComputeShader(static_cast<const VKShader*>(shader));
|
|
|
|
if (StateTracker::GetInstance()->BindCompute())
|
|
|
|
vkCmdDispatch(g_command_buffer_mgr->GetCurrentCommandBuffer(), groups_x, groups_y, groups_z);
|
|
|
|
}
|
|
|
|
|
2023-01-28 02:12:28 +00:00
|
|
|
SurfaceInfo VKGfx::GetSurfaceInfo() const
|
|
|
|
{
|
|
|
|
return {m_swap_chain ? m_swap_chain->GetWidth() : 1u,
|
|
|
|
m_swap_chain ? m_swap_chain->GetHeight() : 0u, m_backbuffer_scale,
|
|
|
|
m_swap_chain ? m_swap_chain->GetTextureFormat() : AbstractTextureFormat::Undefined};
|
|
|
|
}
|
|
|
|
|
2016-08-13 12:57:50 +00:00
|
|
|
} // namespace Vulkan
|