monado/src/xrt/compositor/main/comp_target_swapchain.c

509 lines
14 KiB
C
Raw Normal View History

// Copyright 2019-2020, Collabora, Ltd.
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
* @brief Target Vulkan swapchain code.
* @author Lubosz Sarnecki <lubosz.sarnecki@collabora.com>
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup comp_main
*/
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <string.h>
#include "util/u_misc.h"
#include "comp_target_swapchain.h"
/*
*
* 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 vk_bundle *vk,
VkSurfaceKHR surface,
VkPresentModeKHR present_mode);
/*
*
* Functions!
*
*/
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;
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.
cts->vk->vkGetPhysicalDeviceSurfaceSupportKHR(
cts->vk->physical_device, 0, cts->surface.handle, &supported);
if (!supported) {
VK_ERROR(cts->vk,
"vkGetPhysicalDeviceSurfaceSupportKHR: "
"surface not supported!");
}
// More sanity checks.
if (!_check_surface_present_mode(cts->vk, 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 = cts->vk->vkGetPhysicalDeviceSurfaceCapabilitiesKHR(
cts->vk->physical_device, cts->surface.handle, &surface_caps);
if (ret != VK_SUCCESS) {
VK_ERROR(cts->vk,
"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);
// 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 = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR,
.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR,
.presentMode = cts->present_mode,
.clipped = VK_TRUE,
.oldSwapchain = old_swapchain_handle,
};
ret = cts->vk->vkCreateSwapchainKHR(cts->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) {
VK_ERROR(cts->vk, "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;
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) {
VK_DEBUG(cts->vk,
"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)
{
if (old != VK_NULL_HANDLE) {
cts->vk->vkDestroySwapchainKHR(cts->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;
return cts->vk->vkAcquireNextImageKHR( //
cts->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;
VkPresentInfoKHR presentInfo = {
.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR,
.waitSemaphoreCount = 1,
.pWaitSemaphores = &semaphore,
.swapchainCount = 1,
.pSwapchains = &cts->swapchain.handle,
.pImageIndices = &index,
};
return cts->vk->vkQueuePresentKHR(queue, &presentInfo);
}
static bool
_find_surface_format(struct comp_target_swapchain *cts,
VkSurfaceKHR surface,
VkSurfaceFormatKHR *format)
{
uint32_t num_formats;
VkSurfaceFormatKHR *formats = NULL;
cts->vk->vkGetPhysicalDeviceSurfaceFormatsKHR(
cts->vk->physical_device, surface, &num_formats, NULL);
if (num_formats != 0) {
formats = U_TYPED_ARRAY_CALLOC(VkSurfaceFormatKHR, num_formats);
cts->vk->vkGetPhysicalDeviceSurfaceFormatsKHR(
cts->vk->physical_device, surface, &num_formats, formats);
} else {
VK_ERROR(cts->vk, "Could not enumerate surface formats.");
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];
VK_ERROR(cts->vk, "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];
VK_ERROR(
cts->vk,
"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];
VK_ERROR(cts->vk,
"Returning fallback format! cue up some Kenny "
"Loggins, cos we're in the DANGER ZONE!");
goto cleanup;
}
VK_ERROR(cts->vk, "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 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 =
U_TYPED_ARRAY_CALLOC(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
comp_target_swapchain_destroy_image_views(struct comp_target_swapchain *cts)
{
if (cts->base.images == NULL) {
return;
}
for (uint32_t i = 0; i < cts->base.num_images; i++) {
if (cts->base.images[i].view == VK_NULL_HANDLE) {
continue;
}
cts->vk->vkDestroyImageView(cts->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)
{
cts->vk->vkGetSwapchainImagesKHR( //
cts->vk->device, // device
cts->swapchain.handle, // swapchain
&cts->base.num_images, // pSwapchainImageCount
NULL); // pSwapchainImages
assert(cts->base.num_images > 0);
VK_DEBUG(cts->vk, "Creating %d image views.", cts->base.num_images);
VkImage *images = U_TYPED_ARRAY_CALLOC(VkImage, cts->base.num_images);
cts->vk->vkGetSwapchainImagesKHR( //
cts->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(cts->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)
{
comp_target_swapchain_destroy_image_views(cts);
if (cts->swapchain.handle != VK_NULL_HANDLE) {
cts->vk->vkDestroySwapchainKHR( //
cts->vk->device, // device
cts->swapchain.handle, // swapchain
NULL); //
cts->swapchain.handle = VK_NULL_HANDLE;
}
if (cts->surface.handle != VK_NULL_HANDLE) {
cts->vk->vkDestroySurfaceKHR( //
cts->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;
}
void
comp_target_swapchain_init_post_vulkan(struct comp_target_swapchain *cts,
struct vk_bundle *vk)
{
cts->vk = vk;
}