mirror of
https://github.com/seekrs/MacroLibX.git
synced 2026-01-12 07:03:34 +00:00
152 lines
6.2 KiB
C++
152 lines
6.2 KiB
C++
/* ************************************************************************** */
|
|
/* */
|
|
/* ::: :::::::: */
|
|
/* vk_swapchain.cpp :+: :+: :+: */
|
|
/* +:+ +:+ +:+ */
|
|
/* By: maldavid <kbz_8.dev@akel-engine.com> +#+ +:+ +#+ */
|
|
/* +#+#+#+#+#+ +#+ */
|
|
/* Created: 2022/10/06 18:22:28 by maldavid #+# #+# */
|
|
/* Updated: 2023/12/15 21:49:19 by maldavid ### ########.fr */
|
|
/* */
|
|
/* ************************************************************************** */
|
|
|
|
#include <renderer/core/render_core.h>
|
|
#include <renderer/renderer.h>
|
|
#include <platform/window.h>
|
|
#include <SDL2/SDL_vulkan.h>
|
|
#include <algorithm>
|
|
|
|
namespace mlx
|
|
{
|
|
void SwapChain::init(Renderer* renderer)
|
|
{
|
|
VkDevice device = Render_Core::get().getDevice().get();
|
|
|
|
_renderer = renderer;
|
|
_swapChainSupport = querySwapChainSupport(Render_Core::get().getDevice().getPhysicalDevice());
|
|
|
|
VkSurfaceFormatKHR surfaceFormat = renderer->getSurface().chooseSwapSurfaceFormat(_swapChainSupport.formats);
|
|
VkPresentModeKHR presentMode = chooseSwapPresentMode(_swapChainSupport.presentModes);
|
|
_extent = chooseSwapExtent(_swapChainSupport.capabilities);
|
|
|
|
uint32_t imageCount = _swapChainSupport.capabilities.minImageCount + 1;
|
|
if(_swapChainSupport.capabilities.maxImageCount > 0 && imageCount > _swapChainSupport.capabilities.maxImageCount)
|
|
imageCount = _swapChainSupport.capabilities.maxImageCount;
|
|
|
|
Queues::QueueFamilyIndices indices = Render_Core::get().getQueue().findQueueFamilies(Render_Core::get().getDevice().getPhysicalDevice(), renderer->getSurface().get());
|
|
uint32_t queueFamilyIndices[] = { indices.graphicsFamily.value(), indices.presentFamily.value() };
|
|
|
|
VkSwapchainCreateInfoKHR createInfo{};
|
|
createInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
|
|
createInfo.surface = renderer->getSurface().get();
|
|
createInfo.minImageCount = imageCount;
|
|
createInfo.imageFormat = surfaceFormat.format;
|
|
createInfo.imageColorSpace = surfaceFormat.colorSpace;
|
|
createInfo.imageExtent = _extent;
|
|
createInfo.imageArrayLayers = 1;
|
|
createInfo.imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
|
|
createInfo.preTransform = _swapChainSupport.capabilities.currentTransform;
|
|
createInfo.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
|
|
createInfo.presentMode = presentMode;
|
|
createInfo.clipped = VK_TRUE;
|
|
createInfo.oldSwapchain = VK_NULL_HANDLE;
|
|
if(indices.graphicsFamily != indices.presentFamily)
|
|
{
|
|
createInfo.imageSharingMode = VK_SHARING_MODE_CONCURRENT;
|
|
createInfo.queueFamilyIndexCount = 2;
|
|
createInfo.pQueueFamilyIndices = queueFamilyIndices;
|
|
}
|
|
else
|
|
createInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
|
|
|
|
if(vkCreateSwapchainKHR(device, &createInfo, nullptr, &_swapChain) != VK_SUCCESS)
|
|
core::error::report(e_kind::fatal_error, "Vulkan : failed to create the swapchain");
|
|
|
|
std::vector<VkImage> tmp;
|
|
vkGetSwapchainImagesKHR(device, _swapChain, &imageCount, nullptr);
|
|
_images.resize(imageCount);
|
|
tmp.resize(imageCount);
|
|
vkGetSwapchainImagesKHR(device, _swapChain, &imageCount, tmp.data());
|
|
|
|
for(std::size_t i = 0; i < imageCount; i++)
|
|
{
|
|
_images[i].create(tmp[i], surfaceFormat.format, _extent.width, _extent.height);
|
|
_images[i].transitionLayout(VK_IMAGE_LAYOUT_PRESENT_SRC_KHR);
|
|
_images[i].createImageView(VK_IMAGE_VIEW_TYPE_2D, VK_IMAGE_ASPECT_COLOR_BIT);
|
|
}
|
|
|
|
_swapChainImageFormat = surfaceFormat.format;
|
|
#ifdef DEBUG
|
|
core::error::report(e_kind::message, "Vulkan : created new swapchain");
|
|
#endif
|
|
}
|
|
|
|
SwapChain::SwapChainSupportDetails SwapChain::querySwapChainSupport(VkPhysicalDevice device)
|
|
{
|
|
SwapChain::SwapChainSupportDetails details;
|
|
VkSurfaceKHR surface = _renderer->getSurface().get();
|
|
|
|
if(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(device, surface, &details.capabilities) != VK_SUCCESS)
|
|
core::error::report(e_kind::fatal_error, "Vulkan : unable to retrieve surface capabilities");
|
|
|
|
uint32_t formatCount = 0;
|
|
vkGetPhysicalDeviceSurfaceFormatsKHR(device, surface, &formatCount, nullptr);
|
|
|
|
if(formatCount != 0)
|
|
{
|
|
details.formats.resize(formatCount);
|
|
vkGetPhysicalDeviceSurfaceFormatsKHR(device, surface, &formatCount, details.formats.data());
|
|
}
|
|
|
|
uint32_t presentModeCount;
|
|
vkGetPhysicalDeviceSurfacePresentModesKHR(device, surface, &presentModeCount, nullptr);
|
|
|
|
if(presentModeCount != 0)
|
|
{
|
|
details.presentModes.resize(presentModeCount);
|
|
vkGetPhysicalDeviceSurfacePresentModesKHR(device, surface, &presentModeCount, details.presentModes.data());
|
|
}
|
|
|
|
return details;
|
|
}
|
|
|
|
VkPresentModeKHR SwapChain::chooseSwapPresentMode([[maybe_unused]] const std::vector<VkPresentModeKHR>& availablePresentModes)
|
|
{
|
|
// in the future, you may choose to activate vsync or not
|
|
return VK_PRESENT_MODE_IMMEDIATE_KHR;
|
|
}
|
|
|
|
VkExtent2D SwapChain::chooseSwapExtent(const VkSurfaceCapabilitiesKHR& capabilities)
|
|
{
|
|
if(capabilities.currentExtent.width != std::numeric_limits<uint32_t>::max())
|
|
return capabilities.currentExtent;
|
|
|
|
int width, height;
|
|
SDL_Vulkan_GetDrawableSize(_renderer->getWindow()->getNativeWindow(), &width, &height);
|
|
|
|
VkExtent2D actualExtent = { static_cast<uint32_t>(width), static_cast<uint32_t>(height) };
|
|
|
|
actualExtent.width = std::clamp(actualExtent.width, capabilities.minImageExtent.width, capabilities.maxImageExtent.width);
|
|
actualExtent.height = std::clamp(actualExtent.height, capabilities.minImageExtent.height, capabilities.maxImageExtent.height);
|
|
|
|
return actualExtent;
|
|
}
|
|
|
|
void SwapChain::recreate()
|
|
{
|
|
destroy();
|
|
init(_renderer);
|
|
}
|
|
|
|
void SwapChain::destroy() noexcept
|
|
{
|
|
if(_swapChain == VK_NULL_HANDLE)
|
|
return;
|
|
vkDeviceWaitIdle(Render_Core::get().getDevice().get());
|
|
vkDestroySwapchainKHR(Render_Core::get().getDevice().get(), _swapChain, nullptr);
|
|
_swapChain = VK_NULL_HANDLE;
|
|
for(Image& img : _images)
|
|
img.destroyImageView();
|
|
}
|
|
}
|