// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) #pragma once #include "vulkan_loader.h" #include "vulkan_texture.h" #include "window_info.h" #include "common/types.h" #include #include #include class VulkanSwapChain { public: ~VulkanSwapChain(); // Creates a vulkan-renderable surface for the specified window handle. static VkSurfaceKHR CreateVulkanSurface(VkInstance instance, VkPhysicalDevice physical_device, WindowInfo* wi); // Destroys a previously-created surface. static void DestroyVulkanSurface(VkInstance instance, WindowInfo* wi, VkSurfaceKHR surface); // Create a new swap chain from a pre-existing surface. static std::unique_ptr Create(const WindowInfo& wi, VkSurfaceKHR surface, bool vsync, std::optional exclusive_fullscreen_control); ALWAYS_INLINE VkSurfaceKHR GetSurface() const { return m_surface; } ALWAYS_INLINE VkSwapchainKHR GetSwapChain() const { return m_swap_chain; } ALWAYS_INLINE const VkSwapchainKHR* GetSwapChainPtr() const { return &m_swap_chain; } ALWAYS_INLINE const WindowInfo& GetWindowInfo() const { return m_window_info; } ALWAYS_INLINE u32 GetWidth() const { return m_window_info.surface_width; } ALWAYS_INLINE u32 GetHeight() const { return m_window_info.surface_height; } ALWAYS_INLINE float GetScale() const { return m_window_info.surface_scale; } ALWAYS_INLINE u32 GetCurrentImageIndex() const { return m_current_image; } ALWAYS_INLINE const u32* GetCurrentImageIndexPtr() const { return &m_current_image; } ALWAYS_INLINE u32 GetImageCount() const { return static_cast(m_images.size()); } ALWAYS_INLINE VkFormat GetImageFormat() const { return m_format; } ALWAYS_INLINE VkImage GetCurrentImage() const { return m_images[m_current_image].image; } ALWAYS_INLINE VkImageView GetCurrentImageView() const { return m_images[m_current_image].view; } ALWAYS_INLINE VkFramebuffer GetCurrentFramebuffer() const { return m_images[m_current_image].framebuffer; } ALWAYS_INLINE VkSemaphore GetImageAvailableSemaphore() const { return m_semaphores[m_current_semaphore].available_semaphore; } ALWAYS_INLINE const VkSemaphore* GetImageAvailableSemaphorePtr() const { return &m_semaphores[m_current_semaphore].available_semaphore; } ALWAYS_INLINE VkSemaphore GetRenderingFinishedSemaphore() const { return m_semaphores[m_current_semaphore].rendering_finished_semaphore; } ALWAYS_INLINE const VkSemaphore* GetRenderingFinishedSemaphorePtr() const { return &m_semaphores[m_current_semaphore].rendering_finished_semaphore; } // Returns true if the current present mode is synchronizing (adaptive or hard). ALWAYS_INLINE bool IsPresentModeSynchronizing() const { return (m_vsync_mode /*!= VsyncMode::Off*/); } VkRenderPass GetRenderPass(VkAttachmentLoadOp load_op) const; VkResult AcquireNextImage(); void ReleaseCurrentImage(); bool RecreateSurface(const WindowInfo& new_wi); bool ResizeSwapChain(u32 new_width = 0, u32 new_height = 0, float new_scale = 1.0f); // Change vsync enabled state. This may fail as it causes a swapchain recreation. bool SetVSync(bool mode); private: VulkanSwapChain(const WindowInfo& wi, VkSurfaceKHR surface, bool vsync, std::optional exclusive_fullscreen_control); static std::optional SelectSurfaceFormat(VkSurfaceKHR surface); static std::optional SelectPresentMode(VkSurfaceKHR surface, bool vsync); bool CreateSwapChain(); void DestroySwapChain(); bool SetupSwapChainImages(); void DestroySwapChainImages(); void DestroySurface(); struct Image { VkImage image; VkImageView view; VkFramebuffer framebuffer; }; struct ImageSemaphores { VkSemaphore available_semaphore; VkSemaphore rendering_finished_semaphore; }; WindowInfo m_window_info; VkSurfaceKHR m_surface = VK_NULL_HANDLE; VkSwapchainKHR m_swap_chain = VK_NULL_HANDLE; std::vector m_images; std::vector m_semaphores; VkFormat m_format = VK_FORMAT_UNDEFINED; bool m_vsync_mode = false; u32 m_current_image = 0; u32 m_current_semaphore = 0; std::optional m_image_acquire_result; std::optional m_exclusive_fullscreen_control; };