// Copyright 2019-2020, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Target Vulkan swapchain code. * @author Lubosz Sarnecki * @author Jakob Bornecrantz * @ingroup comp_main */ #include "util/u_misc.h" #include "main/comp_compositor.h" #include "main/comp_target_swapchain.h" #include #include #include #include /* * * Types, defines and data. * */ /*! * These formats will be 'preferred' - in future we may wish to give preference * to higher bit depths if they are available, but most display devices we are * interested in should support one these. */ static VkFormat preferred_color_formats[] = { VK_FORMAT_R8G8B8A8_SRGB, VK_FORMAT_R8G8B8A8_UNORM, VK_FORMAT_B8G8R8A8_UNORM, VK_FORMAT_A8B8G8R8_UNORM_PACK32, }; /* * * Pre declare functions. * */ static void comp_target_swapchain_create_image_views(struct comp_target_swapchain *cts); static void comp_target_swapchain_destroy_image_views(struct comp_target_swapchain *cts); static void comp_target_swapchain_destroy_old(struct comp_target_swapchain *cts, VkSwapchainKHR old); static VkExtent2D comp_target_swapchain_select_extent(struct comp_target_swapchain *cts, VkSurfaceCapabilitiesKHR caps, uint32_t width, uint32_t height); static bool _find_surface_format(struct comp_target_swapchain *cts, VkSurfaceKHR surface, VkSurfaceFormatKHR *format); static bool _check_surface_present_mode(struct comp_target_swapchain *cts, VkSurfaceKHR surface, VkPresentModeKHR present_mode); /* * * Functions! * */ static inline struct vk_bundle * get_vk(struct comp_target_swapchain *cts) { return &cts->base.c->vk; } void comp_target_swapchain_create_images(struct comp_target *ct, uint32_t width, uint32_t height, VkFormat color_format, VkColorSpaceKHR color_space, VkPresentModeKHR present_mode) { struct comp_target_swapchain *cts = (struct comp_target_swapchain *)ct; struct vk_bundle *vk = get_vk(cts); VkBool32 supported; VkResult ret; // Free old image views. comp_target_swapchain_destroy_image_views(cts); VkSwapchainKHR old_swapchain_handle = cts->swapchain.handle; cts->base.num_images = 0; cts->swapchain.handle = VK_NULL_HANDLE; cts->present_mode = present_mode; cts->preferred.color_format = color_format; cts->preferred.color_space = color_space; // Sanity check. ret = vk->vkGetPhysicalDeviceSurfaceSupportKHR( vk->physical_device, 0, cts->surface.handle, &supported); if (!supported) { COMP_ERROR(ct->c, "vkGetPhysicalDeviceSurfaceSupportKHR: surface not " "supported! '%s'", vk_result_string(ret)); } // More sanity checks. if (!_check_surface_present_mode(cts, cts->surface.handle, cts->present_mode)) { // Free old. comp_target_swapchain_destroy_old(cts, old_swapchain_handle); return; } // Find the correct format. if (!_find_surface_format(cts, cts->surface.handle, &cts->surface.format)) { // Free old. comp_target_swapchain_destroy_old(cts, old_swapchain_handle); return; } // Get the caps first. VkSurfaceCapabilitiesKHR surface_caps; ret = vk->vkGetPhysicalDeviceSurfaceCapabilitiesKHR( vk->physical_device, cts->surface.handle, &surface_caps); if (ret != VK_SUCCESS) { COMP_ERROR(ct->c, "vkGetPhysicalDeviceSurfaceCapabilitiesKHR: %s", vk_result_string(ret)); // Free old. comp_target_swapchain_destroy_old(cts, old_swapchain_handle); return; } // Get the extents of the swapchain. VkExtent2D extent = comp_target_swapchain_select_extent( cts, surface_caps, width, height); if (surface_caps.currentTransform & VK_SURFACE_TRANSFORM_ROTATE_90_BIT_KHR || surface_caps.currentTransform & VK_SURFACE_TRANSFORM_ROTATE_270_BIT_KHR) { COMP_DEBUG(ct->c, "Swapping width and height," "since we are going to pre rotate"); uint32_t w2 = extent.width; uint32_t h2 = extent.height; extent.width = h2; extent.height = w2; } // Create the swapchain now. VkSwapchainCreateInfoKHR swapchain_info = { .sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR, .surface = cts->surface.handle, .minImageCount = surface_caps.minImageCount, .imageFormat = cts->surface.format.format, .imageColorSpace = cts->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 = surface_caps.currentTransform, .compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR, .presentMode = cts->present_mode, .clipped = VK_TRUE, .oldSwapchain = old_swapchain_handle, }; ret = vk->vkCreateSwapchainKHR(vk->device, &swapchain_info, NULL, &cts->swapchain.handle); // Always destroy the old. comp_target_swapchain_destroy_old(cts, old_swapchain_handle); if (ret != VK_SUCCESS) { COMP_ERROR(ct->c, "vkCreateSwapchainKHR: %s", vk_result_string(ret)); return; } /* * Set target info. */ cts->base.width = extent.width; cts->base.height = extent.height; cts->base.format = cts->surface.format.format; cts->base.surface_transform = surface_caps.currentTransform; comp_target_swapchain_create_image_views(cts); } static VkExtent2D comp_target_swapchain_select_extent(struct comp_target_swapchain *cts, VkSurfaceCapabilitiesKHR caps, uint32_t width, uint32_t height) { // 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) { VkExtent2D extent = { .width = width, .height = height, }; return extent; } if (caps.currentExtent.width != width || caps.currentExtent.height != height) { COMP_DEBUG(cts->base.c, "Using swap chain extent dimensions %dx%d instead " "of requested %dx%d.", caps.currentExtent.width, caps.currentExtent.height, width, height); } return caps.currentExtent; } static void comp_target_swapchain_destroy_old(struct comp_target_swapchain *cts, VkSwapchainKHR old) { struct vk_bundle *vk = get_vk(cts); if (old != VK_NULL_HANDLE) { vk->vkDestroySwapchainKHR(vk->device, old, NULL); } } VkResult comp_target_swapchain_acquire_next_image(struct comp_target *ct, VkSemaphore semaphore, uint32_t *out_index) { struct comp_target_swapchain *cts = (struct comp_target_swapchain *)ct; struct vk_bundle *vk = get_vk(cts); return vk->vkAcquireNextImageKHR( // vk->device, // device cts->swapchain.handle, // timeout UINT64_MAX, // timeout semaphore, // semaphore VK_NULL_HANDLE, // fence out_index); // pImageIndex } VkResult comp_target_swapchain_present(struct comp_target *ct, VkQueue queue, uint32_t index, VkSemaphore semaphore) { struct comp_target_swapchain *cts = (struct comp_target_swapchain *)ct; struct vk_bundle *vk = get_vk(cts); VkPresentInfoKHR presentInfo = { .sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR, .waitSemaphoreCount = 1, .pWaitSemaphores = &semaphore, .swapchainCount = 1, .pSwapchains = &cts->swapchain.handle, .pImageIndices = &index, }; return vk->vkQueuePresentKHR(queue, &presentInfo); } static bool _find_surface_format(struct comp_target_swapchain *cts, VkSurfaceKHR surface, VkSurfaceFormatKHR *format) { struct vk_bundle *vk = get_vk(cts); uint32_t num_formats; VkSurfaceFormatKHR *formats = NULL; VkResult ret; ret = vk->vkGetPhysicalDeviceSurfaceFormatsKHR( vk->physical_device, surface, &num_formats, NULL); if (num_formats != 0) { formats = U_TYPED_ARRAY_CALLOC(VkSurfaceFormatKHR, num_formats); vk->vkGetPhysicalDeviceSurfaceFormatsKHR( vk->physical_device, surface, &num_formats, formats); } else { COMP_ERROR(cts->base.c, "Could not enumerate surface formats. '%s'", vk_result_string(ret)); return false; } VkSurfaceFormatKHR *formats_for_colorspace = NULL; formats_for_colorspace = U_TYPED_ARRAY_CALLOC(VkSurfaceFormatKHR, num_formats); uint32_t num_formats_cs = 0; uint32_t num_pref_formats = ARRAY_SIZE(preferred_color_formats); // Gather formats that match our color space, we will select // from these in preference to others. for (uint32_t i = 0; i < num_formats; i++) { if (formats[i].colorSpace == cts->preferred.color_space) { formats_for_colorspace[num_formats_cs] = formats[i]; num_formats_cs++; } } if (num_formats_cs > 0) { // we have at least one format with our preferred colorspace // if we have one that is on our preferred formats list, use it for (uint32_t i = 0; i < num_formats_cs; i++) { if (formats_for_colorspace[i].format == cts->preferred.color_format) { // perfect match. *format = formats_for_colorspace[i]; goto cleanup; } } // we don't have our swapchain default format and colorspace, // but we may have at least one preferred format with the // correct colorspace. for (uint32_t i = 0; i < num_formats_cs; i++) { for (uint32_t j = 0; j < num_pref_formats; j++) { if (formats_for_colorspace[i].format == preferred_color_formats[j]) { *format = formats_for_colorspace[i]; goto cleanup; } } } // are we still here? this means we have a format with our // preferred colorspace but we have no preferred color format - // maybe we only have 10/12 bpc or 15/16bpp format. return the // first one we have, at least its in the right color space. *format = formats_for_colorspace[0]; COMP_ERROR(cts->base.c, "Returning unknown color format"); goto cleanup; } else { // we have nothing with the preferred colorspace? we can try to // return a preferred format at least for (uint32_t i = 0; i < num_formats; i++) { for (uint32_t j = 0; j < num_pref_formats; j++) { if (formats[i].format == preferred_color_formats[j]) { *format = formats_for_colorspace[i]; COMP_ERROR( cts->base.c, "Returning known-wrong color " "space! Color shift may occur."); goto cleanup; } } } // if we are still here, we should just return the first format // we have. we know its the wrong colorspace, and its not on our // list of preferred formats, but its something. *format = formats[0]; COMP_ERROR(cts->base.c, "Returning fallback format! cue up some Kenny " "Loggins, cos we're in the DANGER ZONE!"); goto cleanup; } COMP_ERROR(cts->base.c, "We should not be here"); goto error; cleanup: free(formats_for_colorspace); free(formats); return true; error: free(formats_for_colorspace); free(formats); return false; } static bool _check_surface_present_mode(struct comp_target_swapchain *cts, VkSurfaceKHR surface, VkPresentModeKHR present_mode) { struct vk_bundle *vk = get_vk(cts); VkResult ret; uint32_t num_present_modes; VkPresentModeKHR *present_modes; ret = vk->vkGetPhysicalDeviceSurfacePresentModesKHR( vk->physical_device, surface, &num_present_modes, NULL); if (num_present_modes != 0) { present_modes = U_TYPED_ARRAY_CALLOC(VkPresentModeKHR, num_present_modes); vk->vkGetPhysicalDeviceSurfacePresentModesKHR( vk->physical_device, surface, &num_present_modes, present_modes); } else { COMP_ERROR(cts->base.c, "Could not enumerate present modes. '%s'", vk_result_string(ret)); 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); COMP_ERROR(cts->base.c, "Requested present mode not supported.\n"); return false; } static void comp_target_swapchain_destroy_image_views(struct comp_target_swapchain *cts) { if (cts->base.images == NULL) { return; } struct vk_bundle *vk = get_vk(cts); for (uint32_t i = 0; i < cts->base.num_images; i++) { if (cts->base.images[i].view == VK_NULL_HANDLE) { continue; } vk->vkDestroyImageView(vk->device, cts->base.images[i].view, NULL); cts->base.images[i].view = VK_NULL_HANDLE; } free(cts->base.images); cts->base.images = NULL; } static void comp_target_swapchain_create_image_views(struct comp_target_swapchain *cts) { struct vk_bundle *vk = get_vk(cts); vk->vkGetSwapchainImagesKHR( // vk->device, // device cts->swapchain.handle, // swapchain &cts->base.num_images, // pSwapchainImageCount NULL); // pSwapchainImages assert(cts->base.num_images > 0); COMP_DEBUG(cts->base.c, "Creating %d image views.", cts->base.num_images); VkImage *images = U_TYPED_ARRAY_CALLOC(VkImage, cts->base.num_images); vk->vkGetSwapchainImagesKHR( // vk->device, // device cts->swapchain.handle, // swapchain &cts->base.num_images, // pSwapchainImageCount images); // pSwapchainImages comp_target_swapchain_destroy_image_views(cts); cts->base.images = U_TYPED_ARRAY_CALLOC(struct comp_target_image, cts->base.num_images); VkImageSubresourceRange subresource_range = { .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT, .baseMipLevel = 0, .levelCount = 1, .baseArrayLayer = 0, .layerCount = 1, }; for (uint32_t i = 0; i < cts->base.num_images; i++) { cts->base.images[i].handle = images[i]; vk_create_view(vk, cts->base.images[i].handle, cts->surface.format.format, subresource_range, &cts->base.images[i].view); } free(images); } void comp_target_swapchain_cleanup(struct comp_target_swapchain *cts) { struct vk_bundle *vk = get_vk(cts); comp_target_swapchain_destroy_image_views(cts); if (cts->swapchain.handle != VK_NULL_HANDLE) { vk->vkDestroySwapchainKHR( // vk->device, // device cts->swapchain.handle, // swapchain NULL); // cts->swapchain.handle = VK_NULL_HANDLE; } if (cts->surface.handle != VK_NULL_HANDLE) { vk->vkDestroySurfaceKHR( // vk->instance, // instance cts->surface.handle, // surface NULL); // cts->swapchain.handle = VK_NULL_HANDLE; } } void comp_target_swapchain_init_set_fnptrs(struct comp_target_swapchain *cts) { cts->base.create_images = comp_target_swapchain_create_images; cts->base.acquire = comp_target_swapchain_acquire_next_image; cts->base.present = comp_target_swapchain_present; }