// Copyright 2019, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Vulkan swapchain code. * @author Lubosz Sarnecki * @author Jakob Bornecrantz * @ingroup comp_client */ #include #include #include #include #include "comp_vk_swapchain.h" /* * * Pre declare functions. * */ static void vk_swapchain_create_image_view(struct vk_bundle *vk, VkImage image, VkFormat format, VkImageView *view); static void vk_swapchain_create_image_views(struct vk_swapchain *sc); static void vk_swapchain_destroy_old(struct vk_swapchain *sc, VkSwapchainKHR old); static VkExtent2D vk_swapchain_select_extent(struct vk_swapchain *sc, VkSurfaceCapabilitiesKHR caps, uint32_t width, uint32_t height); static bool _find_surface_format(struct vk_swapchain *sc, VkSurfaceKHR surface, VkSurfaceFormatKHR *format); static bool _check_surface_present_mode(struct vk_bundle *vk, VkSurfaceKHR surface, VkPresentModeKHR present_mode); /* * * Functions! * */ void vk_swapchain_init(struct vk_swapchain *sc, struct vk_bundle *vk, vk_swapchain_cb dimension_cb, void *priv) { sc->vk = vk; sc->cb_priv = priv; sc->dimension_cb = dimension_cb; } void vk_swapchain_create(struct vk_swapchain *sc, uint32_t width, uint32_t height, VkFormat color_format, VkColorSpaceKHR color_space, VkPresentModeKHR present_mode) { VkBool32 supported; VkResult ret; sc->image_count = 0; sc->swap_chain = VK_NULL_HANDLE; sc->color_format = color_format; sc->color_space = color_space; sc->present_mode = present_mode; // Sanity check. sc->vk->vkGetPhysicalDeviceSurfaceSupportKHR(sc->vk->physical_device, 0, sc->surface, &supported); if (!supported) { VK_ERROR(sc->vk, "vkGetPhysicalDeviceSurfaceSupportKHR: " "surface not supported!"); } // More sanity checks. if (!_check_surface_present_mode(sc->vk, sc->surface, sc->present_mode)) { return; } // Find the correct format. if (!_find_surface_format(sc, sc->surface, &sc->surface_format)) { return; } // Get the caps first. VkSurfaceCapabilitiesKHR surface_caps; ret = sc->vk->vkGetPhysicalDeviceSurfaceCapabilitiesKHR( sc->vk->physical_device, sc->surface, &surface_caps); if (ret != VK_SUCCESS) { VK_ERROR(sc->vk, "vkGetPhysicalDeviceSurfaceCapabilitiesKHR: %s", vk_result_string(ret)); return; } VkSwapchainKHR old_swap_chain = sc->swap_chain; VkExtent2D extent = vk_swapchain_select_extent(sc, surface_caps, width, height); VkSwapchainCreateInfoKHR swap_chain_info = { .sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR, .surface = sc->surface, .minImageCount = surface_caps.minImageCount, .imageFormat = sc->surface_format.format, .imageColorSpace = sc->surface_format.colorSpace, .imageExtent = { .width = extent.width, .height = extent.height, }, .imageArrayLayers = 1, .imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT, .imageSharingMode = VK_SHARING_MODE_EXCLUSIVE, .queueFamilyIndexCount = 0, .preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR, .compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR, .presentMode = sc->present_mode, .clipped = VK_TRUE, .oldSwapchain = old_swap_chain, }; ret = sc->vk->vkCreateSwapchainKHR(sc->vk->device, &swap_chain_info, NULL, &sc->swap_chain); if (ret != VK_SUCCESS) { VK_ERROR(sc->vk, "vkCreateSwapchainKHR: %s", vk_result_string(ret)); return; } if (old_swap_chain != VK_NULL_HANDLE) { vk_swapchain_destroy_old(sc, old_swap_chain); } vk_swapchain_create_image_views(sc); } static VkExtent2D vk_swapchain_select_extent(struct vk_swapchain *sc, VkSurfaceCapabilitiesKHR caps, uint32_t width, uint32_t height) { VkExtent2D extent; memset(&extent, 0, sizeof(extent)); // If width (and height) equals the special value 0xFFFFFFFF, // the size of the surface will be set by the swapchain if (caps.currentExtent.width == (uint32_t)-1) { extent.width = width; extent.height = height; } else { extent = caps.currentExtent; if (caps.currentExtent.width != width || caps.currentExtent.height != height) { VK_DEBUG(sc->vk, "Using swap chain extent dimensions %dx%d " "instead of requested %dx%d.", caps.currentExtent.width, caps.currentExtent.height, width, height); sc->dimension_cb(caps.currentExtent.width, caps.currentExtent.height, sc->cb_priv); } } return extent; } static void vk_swapchain_destroy_old(struct vk_swapchain *sc, VkSwapchainKHR old) { for (uint32_t i = 0; i < sc->image_count; i++) { sc->vk->vkDestroyImageView(sc->vk->device, sc->buffers[i].view, NULL); } sc->vk->vkDestroySwapchainKHR(sc->vk->device, old, NULL); } VkResult vk_swapchain_acquire_next_image(struct vk_swapchain *sc, VkSemaphore semaphore, uint32_t *index) { return sc->vk->vkAcquireNextImageKHR(sc->vk->device, sc->swap_chain, UINT64_MAX, semaphore, VK_NULL_HANDLE, index); } VkResult vk_swapchain_present(struct vk_swapchain *sc, VkQueue queue, uint32_t index, VkSemaphore semaphore) { VkPresentInfoKHR presentInfo = { .sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR, .waitSemaphoreCount = 1, .pWaitSemaphores = &semaphore, .swapchainCount = 1, .pSwapchains = &sc->swap_chain, .pImageIndices = &index, }; return sc->vk->vkQueuePresentKHR(queue, &presentInfo); } static bool _find_surface_format(struct vk_swapchain *sc, VkSurfaceKHR surface, VkSurfaceFormatKHR *format) { uint32_t num_formats; VkSurfaceFormatKHR *formats = NULL; sc->vk->vkGetPhysicalDeviceSurfaceFormatsKHR( sc->vk->physical_device, surface, &num_formats, NULL); if (num_formats != 0) { formats = (VkSurfaceFormatKHR *)malloc( sizeof(VkSurfaceFormatKHR) * num_formats); sc->vk->vkGetPhysicalDeviceSurfaceFormatsKHR( sc->vk->physical_device, surface, &num_formats, formats); } else { VK_ERROR(sc->vk, "Could not enumerate surface formats."); return false; } for (uint32_t i = 0; i < num_formats; i++) { if (formats[i].format == sc->color_format && formats[i].colorSpace == sc->color_space) { format->format = formats[i].format; format->colorSpace = formats[i].colorSpace; free(formats); return true; } } free(formats); VK_ERROR(sc->vk, "Requested format not supported."); return false; } static bool _check_surface_present_mode(struct vk_bundle *vk, VkSurfaceKHR surface, VkPresentModeKHR present_mode) { uint32_t num_present_modes; VkPresentModeKHR *present_modes; vk->vkGetPhysicalDeviceSurfacePresentModesKHR( vk->physical_device, surface, &num_present_modes, NULL); if (num_present_modes != 0) { present_modes = (VkPresentModeKHR *)malloc( sizeof(VkPresentModeKHR) * num_present_modes); vk->vkGetPhysicalDeviceSurfacePresentModesKHR( vk->physical_device, surface, &num_present_modes, present_modes); } else { VK_ERROR(vk, "Could not enumerate present modes."); return false; } for (uint32_t i = 0; i < num_present_modes; i++) { if (present_modes[i] == present_mode) { free(present_modes); return true; } } free(present_modes); VK_ERROR(vk, "Requested present mode not supported.\n"); return false; } static void vk_swapchain_create_image_views(struct vk_swapchain *sc) { sc->vk->vkGetSwapchainImagesKHR(sc->vk->device, sc->swap_chain, &sc->image_count, NULL); assert(sc->image_count > 0); VK_DEBUG(sc->vk, "Creating %d image views.", sc->image_count); VkImage *images = (VkImage *)malloc(sizeof(VkImage) * sc->image_count); sc->vk->vkGetSwapchainImagesKHR(sc->vk->device, sc->swap_chain, &sc->image_count, images); if (sc->buffers != NULL) { free(sc->buffers); } sc->buffers = (struct vk_swapchain_buffer *)malloc( sizeof(struct vk_swapchain_buffer) * sc->image_count); for (uint32_t i = 0; i < sc->image_count; i++) { sc->buffers[i].image = images[i]; vk_swapchain_create_image_view(sc->vk, sc->buffers[i].image, sc->surface_format.format, &sc->buffers[i].view); } free(images); } void vk_swapchain_cleanup(struct vk_swapchain *sc) { for (uint32_t i = 0; i < sc->image_count; i++) { if (sc->buffers[i].view) { sc->vk->vkDestroyImageView(sc->vk->device, sc->buffers[i].view, NULL); sc->buffers[i].view = VK_NULL_HANDLE; } } if (sc->swap_chain != VK_NULL_HANDLE) { sc->vk->vkDestroySwapchainKHR(sc->vk->device, sc->swap_chain, NULL); sc->swap_chain = VK_NULL_HANDLE; } if (sc->surface != VK_NULL_HANDLE) { sc->vk->vkDestroySurfaceKHR(sc->vk->instance, sc->surface, NULL); sc->swap_chain = VK_NULL_HANDLE; } } static void vk_swapchain_create_image_view(struct vk_bundle *vk, VkImage image, VkFormat format, VkImageView *view) { VkResult ret; VkImageViewCreateInfo view_create_info = { .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO, .image = image, .viewType = VK_IMAGE_VIEW_TYPE_2D, .format = format, .components = { .r = VK_COMPONENT_SWIZZLE_R, .g = VK_COMPONENT_SWIZZLE_G, .b = VK_COMPONENT_SWIZZLE_B, .a = VK_COMPONENT_SWIZZLE_A, }, .subresourceRange = { .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT, .baseMipLevel = 0, .levelCount = 1, .baseArrayLayer = 0, .layerCount = 1, }, }; ret = vk->vkCreateImageView(vk->device, &view_create_info, NULL, view); if (ret != VK_SUCCESS) { VK_ERROR(vk, "vkCreateImageView: %s", vk_result_string(ret)); } }