mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2026-01-30 19:13:09 +00:00
Yellow squiggly lines begone! Done automatically on .cpp files through `run-clang-tidy`, with manual corrections to the mistakes. If an import is directly used, but is technically unnecessary since it's recursively imported by something else, it is *not* removed. The tool doesn't touch .h files, so I did some of them by hand while fixing errors due to old recursive imports. Not everything is removed, but the cleanup should be substantial enough. Because this done on Linux, code that isn't used on it is mostly untouched. (Hopefully no open PR is depending on these imports...)
619 lines
22 KiB
C++
619 lines
22 KiB
C++
// Copyright 2016 Dolphin Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include "VideoBackends/Vulkan/VKSwapChain.h"
|
|
|
|
#include <algorithm>
|
|
#include <cstdint>
|
|
|
|
#include "Common/Assert.h"
|
|
#include "Common/Contains.h"
|
|
#include "Common/Logging/Log.h"
|
|
#include "Common/MsgHandler.h"
|
|
|
|
#include "VideoBackends/Vulkan/CommandBufferManager.h"
|
|
#include "VideoBackends/Vulkan/ObjectCache.h"
|
|
#include "VideoBackends/Vulkan/VKTexture.h"
|
|
#include "VideoBackends/Vulkan/VulkanContext.h"
|
|
#include "VideoCommon/Present.h"
|
|
|
|
#if defined(VK_USE_PLATFORM_XLIB_KHR)
|
|
#include <X11/Xlib.h>
|
|
#endif
|
|
|
|
namespace Vulkan
|
|
{
|
|
SwapChain::SwapChain(const WindowSystemInfo& wsi, VkSurfaceKHR surface, bool vsync)
|
|
: m_wsi(wsi), m_surface(surface), m_vsync_enabled(vsync),
|
|
m_fullscreen_supported(g_vulkan_context->SupportsExclusiveFullscreen(wsi, surface))
|
|
{
|
|
}
|
|
|
|
SwapChain::~SwapChain()
|
|
{
|
|
DestroySwapChainImages();
|
|
DestroySwapChain();
|
|
DestroySurface();
|
|
}
|
|
|
|
VkSurfaceKHR SwapChain::CreateVulkanSurface(VkInstance instance, const WindowSystemInfo& wsi)
|
|
{
|
|
#if defined(VK_USE_PLATFORM_WIN32_KHR)
|
|
if (wsi.type == WindowSystemType::Windows)
|
|
{
|
|
VkWin32SurfaceCreateInfoKHR surface_create_info = {
|
|
VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR, // VkStructureType sType
|
|
nullptr, // const void* pNext
|
|
0, // VkWin32SurfaceCreateFlagsKHR flags
|
|
nullptr, // HINSTANCE hinstance
|
|
static_cast<HWND>(wsi.render_surface) // HWND hwnd
|
|
};
|
|
|
|
VkSurfaceKHR surface;
|
|
VkResult res = vkCreateWin32SurfaceKHR(instance, &surface_create_info, nullptr, &surface);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkCreateWin32SurfaceKHR failed: ");
|
|
return VK_NULL_HANDLE;
|
|
}
|
|
|
|
return surface;
|
|
}
|
|
#endif
|
|
|
|
#if defined(VK_USE_PLATFORM_XLIB_KHR)
|
|
if (wsi.type == WindowSystemType::X11)
|
|
{
|
|
VkXlibSurfaceCreateInfoKHR surface_create_info = {
|
|
VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR, // VkStructureType sType
|
|
nullptr, // const void* pNext
|
|
0, // VkXlibSurfaceCreateFlagsKHR flags
|
|
static_cast<Display*>(wsi.display_connection), // Display* dpy
|
|
reinterpret_cast<Window>(wsi.render_surface) // Window window
|
|
};
|
|
|
|
VkSurfaceKHR surface;
|
|
VkResult res = vkCreateXlibSurfaceKHR(instance, &surface_create_info, nullptr, &surface);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkCreateXlibSurfaceKHR failed: ");
|
|
return VK_NULL_HANDLE;
|
|
}
|
|
|
|
return surface;
|
|
}
|
|
#endif
|
|
|
|
#if defined(VK_USE_PLATFORM_ANDROID_KHR)
|
|
if (wsi.type == WindowSystemType::Android)
|
|
{
|
|
VkAndroidSurfaceCreateInfoKHR surface_create_info = {
|
|
VK_STRUCTURE_TYPE_ANDROID_SURFACE_CREATE_INFO_KHR, // VkStructureType sType
|
|
nullptr, // const void* pNext
|
|
0, // VkAndroidSurfaceCreateFlagsKHR flags
|
|
reinterpret_cast<ANativeWindow*>(wsi.render_surface) // ANativeWindow* window
|
|
};
|
|
|
|
VkSurfaceKHR surface;
|
|
VkResult res = vkCreateAndroidSurfaceKHR(instance, &surface_create_info, nullptr, &surface);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkCreateAndroidSurfaceKHR failed: ");
|
|
return VK_NULL_HANDLE;
|
|
}
|
|
|
|
return surface;
|
|
}
|
|
#endif
|
|
|
|
#if defined(VK_USE_PLATFORM_METAL_EXT)
|
|
if (wsi.type == WindowSystemType::MacOS)
|
|
{
|
|
VkMetalSurfaceCreateInfoEXT surface_create_info = {
|
|
VK_STRUCTURE_TYPE_METAL_SURFACE_CREATE_INFO_EXT, nullptr, 0,
|
|
static_cast<const CAMetalLayer*>(wsi.render_surface)};
|
|
|
|
VkSurfaceKHR surface;
|
|
VkResult res = vkCreateMetalSurfaceEXT(instance, &surface_create_info, nullptr, &surface);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkCreateMetalSurfaceEXT failed: ");
|
|
return VK_NULL_HANDLE;
|
|
}
|
|
|
|
return surface;
|
|
}
|
|
#endif
|
|
|
|
return VK_NULL_HANDLE;
|
|
}
|
|
|
|
std::unique_ptr<SwapChain> SwapChain::Create(const WindowSystemInfo& wsi, VkSurfaceKHR surface,
|
|
bool vsync)
|
|
{
|
|
std::unique_ptr<SwapChain> swap_chain = std::make_unique<SwapChain>(wsi, surface, vsync);
|
|
if (!swap_chain->CreateSwapChain() || !swap_chain->SetupSwapChainImages())
|
|
return nullptr;
|
|
|
|
return swap_chain;
|
|
}
|
|
|
|
bool SwapChain::SelectSurfaceFormat()
|
|
{
|
|
u32 format_count;
|
|
VkResult res = vkGetPhysicalDeviceSurfaceFormatsKHR(g_vulkan_context->GetPhysicalDevice(),
|
|
m_surface, &format_count, nullptr);
|
|
if (res != VK_SUCCESS || format_count == 0)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkGetPhysicalDeviceSurfaceFormatsKHR failed: ");
|
|
return false;
|
|
}
|
|
|
|
std::vector<VkSurfaceFormatKHR> surface_formats(format_count);
|
|
res = vkGetPhysicalDeviceSurfaceFormatsKHR(g_vulkan_context->GetPhysicalDevice(), m_surface,
|
|
&format_count, surface_formats.data());
|
|
ASSERT(res == VK_SUCCESS);
|
|
|
|
// If there is a single undefined surface format, the device doesn't care, so we'll just use RGBA8
|
|
if (surface_formats[0].format == VK_FORMAT_UNDEFINED)
|
|
{
|
|
m_surface_format.format = VK_FORMAT_R8G8B8A8_UNORM;
|
|
m_surface_format.colorSpace = VK_COLOR_SPACE_SRGB_NONLINEAR_KHR;
|
|
return true;
|
|
}
|
|
|
|
const VkSurfaceFormatKHR* surface_format_RGBA8 = nullptr;
|
|
const VkSurfaceFormatKHR* surface_format_BGRA8 = nullptr;
|
|
const VkSurfaceFormatKHR* surface_format_RGB10_A2 = nullptr;
|
|
const VkSurfaceFormatKHR* surface_format_RGBA16F_scRGB = nullptr;
|
|
|
|
// Try to find all suitable formats.
|
|
for (const VkSurfaceFormatKHR& surface_format : surface_formats)
|
|
{
|
|
// Some drivers seem to return a RGBA8 SRGB format here (Intel Mesa).
|
|
// Some other drivers return both a RGBA8 SRGB and UNORM formats (Nvidia).
|
|
// This results in gamma correction when presenting to the screen, which we don't want,
|
|
// because we already apply gamma ourselves, and we might not use sRGB gamma.
|
|
// Force using a linear format instead, if this is the case.
|
|
VkFormat format = VKTexture::GetLinearFormat(surface_format.format);
|
|
if (surface_format.colorSpace == VK_COLOR_SPACE_SRGB_NONLINEAR_KHR)
|
|
{
|
|
if (format == VK_FORMAT_R8G8B8A8_UNORM)
|
|
surface_format_RGBA8 = &surface_format;
|
|
else if (format == VK_FORMAT_B8G8R8A8_UNORM)
|
|
surface_format_BGRA8 = &surface_format;
|
|
else if (format == VK_FORMAT_A2B10G10R10_UNORM_PACK32)
|
|
surface_format_RGB10_A2 = &surface_format;
|
|
}
|
|
else if (format == VK_FORMAT_R16G16B16A16_SFLOAT &&
|
|
surface_format.colorSpace == VK_COLOR_SPACE_EXTENDED_SRGB_LINEAR_EXT)
|
|
{
|
|
surface_format_RGBA16F_scRGB = &surface_format;
|
|
}
|
|
}
|
|
|
|
const VkSurfaceFormatKHR* surface_format = nullptr;
|
|
|
|
// Pick the best format.
|
|
// "g_ActiveConfig" might not have been been updated yet.
|
|
if (g_Config.bHDR && surface_format_RGBA16F_scRGB)
|
|
surface_format = surface_format_RGBA16F_scRGB;
|
|
else if (surface_format_RGB10_A2)
|
|
surface_format = surface_format_RGB10_A2;
|
|
else if (surface_format_RGBA8)
|
|
surface_format = surface_format_RGBA8;
|
|
else if (surface_format_BGRA8)
|
|
surface_format = surface_format_BGRA8;
|
|
|
|
if (surface_format)
|
|
{
|
|
const VkFormat format = VKTexture::GetLinearFormat(surface_format->format);
|
|
if (format == VK_FORMAT_R8G8B8A8_UNORM)
|
|
m_texture_format = AbstractTextureFormat::RGBA8;
|
|
else if (format == VK_FORMAT_B8G8R8A8_UNORM)
|
|
m_texture_format = AbstractTextureFormat::BGRA8;
|
|
else if (format == VK_FORMAT_A2B10G10R10_UNORM_PACK32)
|
|
m_texture_format = AbstractTextureFormat::RGB10_A2;
|
|
else if (format == VK_FORMAT_R16G16B16A16_SFLOAT)
|
|
m_texture_format = AbstractTextureFormat::RGBA16F;
|
|
|
|
m_surface_format.format = format;
|
|
m_surface_format.colorSpace = surface_format->colorSpace;
|
|
return true;
|
|
}
|
|
|
|
PanicAlertFmt("Failed to find a suitable format for swap chain buffers.");
|
|
return false;
|
|
}
|
|
|
|
bool SwapChain::SelectPresentMode()
|
|
{
|
|
VkResult res;
|
|
u32 mode_count;
|
|
res = vkGetPhysicalDeviceSurfacePresentModesKHR(g_vulkan_context->GetPhysicalDevice(), m_surface,
|
|
&mode_count, nullptr);
|
|
if (res != VK_SUCCESS || mode_count == 0)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkGetPhysicalDeviceSurfaceFormatsKHR failed: ");
|
|
return false;
|
|
}
|
|
|
|
std::vector<VkPresentModeKHR> present_modes(mode_count);
|
|
res = vkGetPhysicalDeviceSurfacePresentModesKHR(g_vulkan_context->GetPhysicalDevice(), m_surface,
|
|
&mode_count, present_modes.data());
|
|
ASSERT(res == VK_SUCCESS);
|
|
|
|
// If vsync is enabled, use VK_PRESENT_MODE_FIFO_KHR.
|
|
// This check should not fail with conforming drivers, as the FIFO present mode is mandated by
|
|
// the specification (VK_KHR_swapchain). In case it isn't though, fall through to any other mode.
|
|
if (m_vsync_enabled && Common::Contains(present_modes, VK_PRESENT_MODE_FIFO_KHR))
|
|
{
|
|
m_present_mode = VK_PRESENT_MODE_FIFO_KHR;
|
|
return true;
|
|
}
|
|
|
|
// Prefer screen-tearing, if possible, for lowest latency.
|
|
if (Common::Contains(present_modes, VK_PRESENT_MODE_IMMEDIATE_KHR))
|
|
{
|
|
m_present_mode = VK_PRESENT_MODE_IMMEDIATE_KHR;
|
|
return true;
|
|
}
|
|
|
|
// Use optimized-vsync above vsync.
|
|
if (Common::Contains(present_modes, VK_PRESENT_MODE_MAILBOX_KHR))
|
|
{
|
|
m_present_mode = VK_PRESENT_MODE_MAILBOX_KHR;
|
|
return true;
|
|
}
|
|
|
|
// Fall back to whatever is available.
|
|
m_present_mode = present_modes[0];
|
|
return true;
|
|
}
|
|
|
|
bool SwapChain::CreateSwapChain()
|
|
{
|
|
// Look up surface properties to determine image count and dimensions
|
|
VkSurfaceCapabilitiesKHR surface_capabilities;
|
|
VkResult res = vkGetPhysicalDeviceSurfaceCapabilitiesKHR(g_vulkan_context->GetPhysicalDevice(),
|
|
m_surface, &surface_capabilities);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkGetPhysicalDeviceSurfaceCapabilitiesKHR failed: ");
|
|
return false;
|
|
}
|
|
|
|
// Select swap chain format and present mode
|
|
if (!SelectSurfaceFormat() || !SelectPresentMode())
|
|
return false;
|
|
|
|
// Select number of images in swap chain, we prefer one buffer in the background to work on
|
|
uint32_t image_count = surface_capabilities.minImageCount + 1;
|
|
|
|
// maxImageCount can be zero, in which case there isn't an upper limit on the number of buffers.
|
|
if (surface_capabilities.maxImageCount > 0)
|
|
image_count = std::min(image_count, surface_capabilities.maxImageCount);
|
|
|
|
// Determine the dimensions of the swap chain. Values of -1 indicate the size we specify here
|
|
// determines window size?
|
|
VkExtent2D size = surface_capabilities.currentExtent;
|
|
if (size.width == UINT32_MAX)
|
|
{
|
|
size.width = std::max(g_presenter->GetBackbufferWidth(), 1);
|
|
size.height = std::max(g_presenter->GetBackbufferHeight(), 1);
|
|
}
|
|
size.width = std::clamp(size.width, surface_capabilities.minImageExtent.width,
|
|
surface_capabilities.maxImageExtent.width);
|
|
size.height = std::clamp(size.height, surface_capabilities.minImageExtent.height,
|
|
surface_capabilities.maxImageExtent.height);
|
|
|
|
// Prefer identity transform if possible
|
|
VkSurfaceTransformFlagBitsKHR transform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
|
|
if (!(surface_capabilities.supportedTransforms & VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR))
|
|
transform = surface_capabilities.currentTransform;
|
|
|
|
// Select swap chain flags, we only need a colour attachment
|
|
VkImageUsageFlags image_usage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
|
|
if (!(surface_capabilities.supportedUsageFlags & VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT))
|
|
{
|
|
ERROR_LOG_FMT(VIDEO, "Vulkan: Swap chain does not support usage as color attachment");
|
|
return false;
|
|
}
|
|
|
|
// Select the number of image layers for Quad-Buffered stereoscopy
|
|
uint32_t image_layers = g_ActiveConfig.stereo_mode == StereoMode::QuadBuffer ? 2 : 1;
|
|
|
|
// Store the old/current swap chain when recreating for resize
|
|
VkSwapchainKHR old_swap_chain = m_swap_chain;
|
|
m_swap_chain = VK_NULL_HANDLE;
|
|
|
|
// Now we can actually create the swap chain
|
|
VkSwapchainCreateInfoKHR swap_chain_info = {VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR,
|
|
nullptr,
|
|
0,
|
|
m_surface,
|
|
image_count,
|
|
m_surface_format.format,
|
|
m_surface_format.colorSpace,
|
|
size,
|
|
image_layers,
|
|
image_usage,
|
|
VK_SHARING_MODE_EXCLUSIVE,
|
|
0,
|
|
nullptr,
|
|
transform,
|
|
VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR,
|
|
m_present_mode,
|
|
VK_TRUE,
|
|
old_swap_chain};
|
|
std::array<uint32_t, 2> indices = {{
|
|
g_vulkan_context->GetGraphicsQueueFamilyIndex(),
|
|
g_vulkan_context->GetPresentQueueFamilyIndex(),
|
|
}};
|
|
if (g_vulkan_context->GetGraphicsQueueFamilyIndex() !=
|
|
g_vulkan_context->GetPresentQueueFamilyIndex())
|
|
{
|
|
swap_chain_info.imageSharingMode = VK_SHARING_MODE_CONCURRENT;
|
|
swap_chain_info.queueFamilyIndexCount = 2;
|
|
swap_chain_info.pQueueFamilyIndices = indices.data();
|
|
}
|
|
|
|
#ifdef SUPPORTS_VULKAN_EXCLUSIVE_FULLSCREEN
|
|
if (m_fullscreen_supported)
|
|
{
|
|
VkSurfaceFullScreenExclusiveInfoEXT fullscreen_support = {};
|
|
swap_chain_info.pNext = &fullscreen_support;
|
|
fullscreen_support.sType = VK_STRUCTURE_TYPE_SURFACE_FULL_SCREEN_EXCLUSIVE_INFO_EXT;
|
|
fullscreen_support.fullScreenExclusive = VK_FULL_SCREEN_EXCLUSIVE_APPLICATION_CONTROLLED_EXT;
|
|
|
|
auto platform_info = g_vulkan_context->GetPlatformExclusiveFullscreenInfo(m_wsi);
|
|
fullscreen_support.pNext = &platform_info;
|
|
|
|
res = vkCreateSwapchainKHR(g_vulkan_context->GetDevice(), &swap_chain_info, nullptr,
|
|
&m_swap_chain);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
// Try without exclusive fullscreen.
|
|
WARN_LOG_FMT(VIDEO, "Failed to create exclusive fullscreen swapchain, trying without.");
|
|
swap_chain_info.pNext = nullptr;
|
|
g_backend_info.bSupportsExclusiveFullscreen = false;
|
|
m_fullscreen_supported = false;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (m_swap_chain == VK_NULL_HANDLE)
|
|
{
|
|
res = vkCreateSwapchainKHR(g_vulkan_context->GetDevice(), &swap_chain_info, nullptr,
|
|
&m_swap_chain);
|
|
}
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkCreateSwapchainKHR failed: ");
|
|
return false;
|
|
}
|
|
|
|
// Now destroy the old swap chain, since it's been recreated.
|
|
// We can do this immediately since all work should have been completed before calling resize.
|
|
if (old_swap_chain != VK_NULL_HANDLE)
|
|
vkDestroySwapchainKHR(g_vulkan_context->GetDevice(), old_swap_chain, nullptr);
|
|
|
|
m_width = size.width;
|
|
m_height = size.height;
|
|
m_layers = image_layers;
|
|
return true;
|
|
}
|
|
|
|
bool SwapChain::SetupSwapChainImages()
|
|
{
|
|
ASSERT(m_swap_chain_images.empty());
|
|
|
|
uint32_t image_count;
|
|
VkResult res =
|
|
vkGetSwapchainImagesKHR(g_vulkan_context->GetDevice(), m_swap_chain, &image_count, nullptr);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkGetSwapchainImagesKHR failed: ");
|
|
return false;
|
|
}
|
|
|
|
std::vector<VkImage> images(image_count);
|
|
res = vkGetSwapchainImagesKHR(g_vulkan_context->GetDevice(), m_swap_chain, &image_count,
|
|
images.data());
|
|
ASSERT(res == VK_SUCCESS);
|
|
|
|
const TextureConfig texture_config(
|
|
TextureConfig(m_width, m_height, 1, m_layers, 1, m_texture_format,
|
|
AbstractTextureFlag_RenderTarget, AbstractTextureType::Texture_2DArray));
|
|
const VkRenderPass load_render_pass = g_object_cache->GetRenderPass(
|
|
m_surface_format.format, VK_FORMAT_UNDEFINED, 1, VK_ATTACHMENT_LOAD_OP_LOAD);
|
|
const VkRenderPass clear_render_pass = g_object_cache->GetRenderPass(
|
|
m_surface_format.format, VK_FORMAT_UNDEFINED, 1, VK_ATTACHMENT_LOAD_OP_CLEAR);
|
|
if (load_render_pass == VK_NULL_HANDLE || clear_render_pass == VK_NULL_HANDLE)
|
|
{
|
|
PanicAlertFmt("Failed to get swap chain render passes.");
|
|
return false;
|
|
}
|
|
|
|
m_swap_chain_images.reserve(image_count);
|
|
for (uint32_t i = 0; i < image_count; i++)
|
|
{
|
|
SwapChainImage image;
|
|
image.image = images[i];
|
|
|
|
// Create texture object, which creates a view of the backbuffer
|
|
image.texture =
|
|
VKTexture::CreateAdopted(texture_config, image.image,
|
|
m_layers > 1 ? VK_IMAGE_VIEW_TYPE_2D_ARRAY : VK_IMAGE_VIEW_TYPE_2D,
|
|
VK_IMAGE_LAYOUT_UNDEFINED);
|
|
if (!image.texture)
|
|
return false;
|
|
|
|
image.framebuffer = VKFramebuffer::Create(image.texture.get(), nullptr, {});
|
|
if (!image.framebuffer)
|
|
{
|
|
image.texture.reset();
|
|
return false;
|
|
}
|
|
|
|
m_swap_chain_images.emplace_back(std::move(image));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void SwapChain::DestroySwapChainImages()
|
|
{
|
|
for (auto& it : m_swap_chain_images)
|
|
{
|
|
// Images themselves are cleaned up by the swap chain object
|
|
it.framebuffer.reset();
|
|
it.texture.reset();
|
|
}
|
|
m_swap_chain_images.clear();
|
|
}
|
|
|
|
void SwapChain::DestroySwapChain()
|
|
{
|
|
if (m_swap_chain == VK_NULL_HANDLE)
|
|
return;
|
|
|
|
// Release exclusive fullscreen before destroying.
|
|
if (m_current_fullscreen_state)
|
|
SetFullscreenState(false);
|
|
|
|
vkDestroySwapchainKHR(g_vulkan_context->GetDevice(), m_swap_chain, nullptr);
|
|
m_swap_chain = VK_NULL_HANDLE;
|
|
}
|
|
|
|
VkResult SwapChain::AcquireNextImage()
|
|
{
|
|
VkResult res = vkAcquireNextImageKHR(g_vulkan_context->GetDevice(), m_swap_chain, UINT64_MAX,
|
|
g_command_buffer_mgr->GetCurrentCommandBufferSemaphore(),
|
|
VK_NULL_HANDLE, &m_current_swap_chain_image_index);
|
|
m_current_swap_chain_image_is_valid = res >= 0;
|
|
if (IsCurrentImageValid())
|
|
g_command_buffer_mgr->MarkCurrentCommandBufferSemaphoreUsed();
|
|
if (res != VK_SUCCESS && res != VK_ERROR_OUT_OF_DATE_KHR && res != VK_SUBOPTIMAL_KHR)
|
|
LOG_VULKAN_ERROR(res, "vkAcquireNextImageKHR failed: ");
|
|
|
|
return res;
|
|
}
|
|
|
|
bool SwapChain::ResizeSwapChain()
|
|
{
|
|
DestroySwapChainImages();
|
|
if (!CreateSwapChain() || !SetupSwapChainImages())
|
|
{
|
|
PanicAlertFmt("Failed to re-configure swap chain images, this is fatal (for now)");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SwapChain::RecreateSwapChain()
|
|
{
|
|
DestroySwapChainImages();
|
|
DestroySwapChain();
|
|
if (!CreateSwapChain() || !SetupSwapChainImages())
|
|
{
|
|
PanicAlertFmt("Failed to re-configure swap chain images, this is fatal (for now)");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SwapChain::SetVSync(bool enabled)
|
|
{
|
|
if (m_vsync_enabled == enabled)
|
|
return true;
|
|
|
|
// Recreate the swap chain with the new present mode.
|
|
m_vsync_enabled = enabled;
|
|
return RecreateSwapChain();
|
|
}
|
|
|
|
bool SwapChain::SetFullscreenState(bool state)
|
|
{
|
|
#ifdef SUPPORTS_VULKAN_EXCLUSIVE_FULLSCREEN
|
|
if (m_current_fullscreen_state == state)
|
|
return true;
|
|
|
|
if (state)
|
|
{
|
|
VkResult res = vkAcquireFullScreenExclusiveModeEXT(g_vulkan_context->GetDevice(), m_swap_chain);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkAcquireFullScreenExclusiveModeEXT failed:");
|
|
return false;
|
|
}
|
|
|
|
INFO_LOG_FMT(VIDEO, "Exclusive fullscreen acquired.");
|
|
}
|
|
else
|
|
{
|
|
VkResult res = vkReleaseFullScreenExclusiveModeEXT(g_vulkan_context->GetDevice(), m_swap_chain);
|
|
if (res != VK_SUCCESS)
|
|
LOG_VULKAN_ERROR(res, "vkReleaseFullScreenExclusiveModeEXT failed:");
|
|
|
|
INFO_LOG_FMT(VIDEO, "Exclusive fullscreen released.");
|
|
}
|
|
|
|
m_current_fullscreen_state = state;
|
|
return true;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
bool SwapChain::RecreateSurface(void* native_handle)
|
|
{
|
|
// Destroy the old swap chain, images, and surface.
|
|
DestroySwapChainImages();
|
|
DestroySwapChain();
|
|
DestroySurface();
|
|
|
|
// Re-create the surface with the new native handle
|
|
m_wsi.render_surface = native_handle;
|
|
m_surface = CreateVulkanSurface(g_vulkan_context->GetVulkanInstance(), m_wsi);
|
|
if (m_surface == VK_NULL_HANDLE)
|
|
return false;
|
|
|
|
// The validation layers get angry at us if we don't call this before creating the swapchain.
|
|
VkBool32 present_supported = VK_TRUE;
|
|
VkResult res = vkGetPhysicalDeviceSurfaceSupportKHR(
|
|
g_vulkan_context->GetPhysicalDevice(), g_vulkan_context->GetPresentQueueFamilyIndex(),
|
|
m_surface, &present_supported);
|
|
if (res != VK_SUCCESS)
|
|
{
|
|
LOG_VULKAN_ERROR(res, "vkGetPhysicalDeviceSurfaceSupportKHR failed: ");
|
|
return false;
|
|
}
|
|
if (!present_supported)
|
|
{
|
|
PanicAlertFmt("Recreated surface does not support presenting.");
|
|
return false;
|
|
}
|
|
|
|
// Update exclusive fullscreen support (unlikely to change).
|
|
m_fullscreen_supported = g_vulkan_context->SupportsExclusiveFullscreen(m_wsi, m_surface);
|
|
g_backend_info.bSupportsExclusiveFullscreen = m_fullscreen_supported;
|
|
m_current_fullscreen_state = false;
|
|
m_next_fullscreen_state = false;
|
|
|
|
// Finally re-create the swap chain
|
|
if (!CreateSwapChain() || !SetupSwapChainImages())
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void SwapChain::DestroySurface()
|
|
{
|
|
vkDestroySurfaceKHR(g_vulkan_context->GetVulkanInstance(), m_surface, nullptr);
|
|
m_surface = VK_NULL_HANDLE;
|
|
}
|
|
} // namespace Vulkan
|