vulkan-wsi-layer/wsi/surface_properties.hpp
Iason Paraskevopoulos 3da272878d Reset query pool on creation
Makes use of the VK_QUERY_POOL_CREATE_RESET_BIT_KHR when creating the
query pool. It disables present timing when VK_KHR_maintenance9 is not
supported by the device.

Signed-off-by: Iason Paraskevopoulos <iason.paraskevopoulos@arm.com>
Change-Id: I9d364050f0cd4a2be74e1b1d44cf3f6a0795e6f4
2025-10-17 15:02:40 +00:00

329 lines
12 KiB
C++

/*
* Copyright (c) 2017-2019, 2021-2025 Arm Limited.
*
* SPDX-License-Identifier: MIT
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
/**
* @file surface_properties.hpp
*
* @brief Vulkan WSI surface query interfaces.
*/
#pragma once
#include <vulkan/vulkan.h>
#include "layer/wsi_layer_experimental.hpp"
#include "layer/private_data.hpp"
#include "util/custom_allocator.hpp"
#include "util/drm/drm_utils.hpp"
#include "util/extension_list.hpp"
#include "util/format_modifiers.hpp"
#include "util/log.hpp"
#include "util/macros.hpp"
namespace wsi
{
/**
* @brief The base surface property query interface.
*/
class surface_properties
{
public:
/**
* @brief Implementation of vkGetPhysicalDeviceSurfaceCapabilitiesKHR for the specific VkSurface type.
*/
virtual VkResult get_surface_capabilities(VkPhysicalDevice physical_device,
VkSurfaceCapabilitiesKHR *surface_capabilities) = 0;
/**
* @brief Implementation of vkGetPhysicalDeviceSurfaceCapabilities2KHR for the specific VkSurface type.
*/
virtual VkResult get_surface_capabilities(VkPhysicalDevice physical_device,
const VkPhysicalDeviceSurfaceInfo2KHR *pSurfaceInfo,
VkSurfaceCapabilities2KHR *surface_capabilities) = 0;
/**
* @brief Implementation of vkGetPhysicalDeviceSurfaceFormatsKHR for the specific VkSurface type.
*/
virtual VkResult get_surface_formats(VkPhysicalDevice physical_device, uint32_t *surface_formats_count,
VkSurfaceFormatKHR *surface_formats,
VkSurfaceFormat2KHR *extended_surface_formats = nullptr) = 0;
/**
* @brief Implementation of vkGetPhysicalDeviceSurfacePresentModesKHR for the specific VkSurface type.
*/
virtual VkResult get_surface_present_modes(VkPhysicalDevice physical_device, VkSurfaceKHR surface,
uint32_t *present_mode_count, VkPresentModeKHR *present_modes) = 0;
/**
* @brief Return the device extensions that this surface_properties implementation needs.
*/
virtual VkResult get_required_device_extensions(util::extension_list &extension_list, const uint32_t api_version)
{
/* Requires no additional extensions */
UNUSED(extension_list);
UNUSED(api_version);
return VK_SUCCESS;
}
/**
* @brief Return the instance extensions that this surface_properties implementation needs.
*/
virtual VkResult get_required_instance_extensions(util::extension_list &extension_list,
const uint32_t api_version) = 0;
/**
* @brief Implements vkGetProcAddr for entrypoints specific to the surface type.
*
* At least the specific VkSurface creation entrypoint must be intercepted.
*/
virtual PFN_vkVoidFunction get_proc_addr(const char *name) = 0;
/**
* @brief Check if the proper surface extension has been enabled for the specific VkSurface type.
*/
virtual bool is_surface_extension_enabled(const layer::instance_private_data &instance_data) = 0;
/* There is no maximum theoretically speaking however we choose 6 for practicality */
static constexpr uint32_t MAX_SWAPCHAIN_IMAGE_COUNT = 6;
/**
* @brief Get the scaling and gravity capabilities of the surface.
*/
virtual void get_surface_present_scaling_and_gravity(
VkSurfacePresentScalingCapabilitiesEXT *scaling_capabilities) = 0;
virtual bool is_compatible_present_modes(VkPresentModeKHR present_mode_a, VkPresentModeKHR present_mode_b) = 0;
#if VULKAN_WSI_LAYER_EXPERIMENTAL
/**
* @brief Get the present timing surface capabilities for the specific VkSurface type.
*/
VkResult get_present_timing_surface_caps(VkPhysicalDevice physical_device,
VkPresentTimingSurfaceCapabilitiesEXT *present_timing_surface_caps);
#endif /* VULKAN_WSI_LAYER_EXPERIMENTAL */
/**
* @brief Virtual destructor.
*/
virtual ~surface_properties() = default;
private:
/**
* @brief Set which presentation modes are compatible with each other for a particular surface
*/
virtual void populate_present_mode_compatibilities() = 0;
#if VULKAN_WSI_LAYER_EXPERIMENTAL
/**
* @brief Get the present timing surface capabilities for the specific VkSurface type.
*/
virtual VkResult get_present_timing_surface_caps_internal(
VkPhysicalDevice physical_device, VkPresentTimingSurfaceCapabilitiesEXT *present_timing_surface_caps) = 0;
#endif /* VULKAN_WSI_LAYER_EXPERIMENTAL */
};
class surface_format_properties
{
public:
surface_format_properties(VkFormat format)
: m_surface_format{ format, VK_COLORSPACE_SRGB_NONLINEAR_KHR }
, m_compression{ VK_STRUCTURE_TYPE_IMAGE_COMPRESSION_PROPERTIES_EXT, nullptr, VK_IMAGE_COMPRESSION_DEFAULT_EXT,
VK_IMAGE_COMPRESSION_FIXED_RATE_NONE_EXT }
{
}
surface_format_properties()
: surface_format_properties{ VK_FORMAT_UNDEFINED }
{
}
VkResult check_device_support(VkPhysicalDevice phys_dev, VkPhysicalDeviceImageFormatInfo2 image_format_info);
VkResult add_device_compression_support(VkPhysicalDevice phys_dev,
VkPhysicalDeviceImageFormatInfo2 image_format_info);
void fill_format_properties(VkSurfaceFormat2KHR &surf_format) const;
template <typename K>
static surface_format_properties &from_iterator(std::pair<const K, surface_format_properties> &iter)
{
return iter.second;
}
static surface_format_properties &from_iterator(surface_format_properties &iter)
{
return iter;
}
VkSurfaceFormatKHR m_surface_format;
private:
VkImageCompressionPropertiesEXT m_compression;
};
/**
* @brief Helper function for the vkGetPhysicalDeviceSurfaceFormatsKHR entrypoint.
*
* Implements the common logic, which is used by all the WSI backends for
* setting the supported formats by the surface.
*
* @param begin Beginning of an iterator with the supported @ref surface_format_properties
* @param end End of the iterator.
* @param surface_formats_count Pointer for setting the length of the supported
* formats.
* @param surface_formats The supported formats by the surface.
* @param extended_surface_formats Extended surface formats supported by the surface, it
* is being used when the vkGetPhysicalDeviceSurfaceFormats2KHR
* entrypoint is used.
*
* @return VK_SUCCESS on success, an appropriate error code otherwise.
*
*/
template <typename It>
VkResult surface_properties_formats_helper(It begin, It end, uint32_t *surface_formats_count,
VkSurfaceFormatKHR *surface_formats,
VkSurfaceFormat2KHR *extended_surface_formats)
{
assert(surface_formats_count != nullptr);
const uint32_t supported_formats_count = std::distance(begin, end);
if (surface_formats == nullptr && extended_surface_formats == nullptr)
{
*surface_formats_count = supported_formats_count;
return VK_SUCCESS;
}
VkResult res = VK_SUCCESS;
if (supported_formats_count > *surface_formats_count)
{
res = VK_INCOMPLETE;
}
*surface_formats_count = std::min(*surface_formats_count, supported_formats_count);
uint32_t format_count = 0;
for (auto it = begin; it != end; ++it)
{
const auto &props = surface_format_properties::from_iterator(*it);
if (format_count >= *surface_formats_count)
{
break;
}
if (extended_surface_formats == nullptr)
{
surface_formats[format_count] = props.m_surface_format;
}
else
{
props.fill_format_properties(extended_surface_formats[format_count]);
}
format_count++;
}
return res;
}
/**
* @brief Common function for handling VkSurfacePresentModeEXT.
*
* If VkSurfacePresentModeEXT is present in the pNext chain of VkPhysicalDeviceSurfaceInfo2KHR, then
* the presentation mode specified in the VkSurfacePresentModeEXT struct is checked to see if it is
* supported by the surface.
*
* @param surface_info Pointer to Vulkan surface info struct.
* @param modes Array of presentation modes supported by the surface.
*
* @return VK_SUCCESS on success, VK_ERROR_OUT_OF_HOST_MEMORY otherwise.
*/
template <std::size_t SIZE>
VkResult check_surface_present_mode_query_is_supported(const VkPhysicalDeviceSurfaceInfo2KHR *surface_info,
const std::array<VkPresentModeKHR, SIZE> &modes)
{
auto surface_present_mode =
util::find_extension<VkSurfacePresentModeEXT>(VK_STRUCTURE_TYPE_SURFACE_PRESENT_MODE_EXT, surface_info);
if (surface_present_mode != nullptr)
{
VkPresentModeKHR present_mode = surface_present_mode->presentMode;
if (std::find(modes.begin(), modes.end(), present_mode) == modes.end())
{
WSI_LOG_ERROR("Querying surface capability support for a present mode that is not supported by the surface");
return VK_ERROR_OUT_OF_HOST_MEMORY;
}
}
return VK_SUCCESS;
}
/**
* @brief Common function for the get_surface_capabilities.
*
* Initiates the different fields in surface_capabilities struct, also
* according to the physical_device.
*
* @param physical_device Vulkan physical_device.
* @param surface_capabilities address of Vulkan surface capabilities struct.
*
*/
void get_surface_capabilities_common(VkPhysicalDevice physical_device, VkSurfaceCapabilitiesKHR *surface_capabilities);
/**
* @brief Common function for the get_surface_present_modes.
*
* Preparing the present_modes array for get_surface_present_modes.
*
* @param present_mode_count address of counter for the present modes.
* @param present_modes array of present modes.
* @param modes array of modes
*
* @return VK_SUCCESS on success, VK_INCOMPLETE otherwise.
*
*/
template <std::size_t SIZE>
VkResult get_surface_present_modes_common(uint32_t *present_mode_count, VkPresentModeKHR *present_modes,
const std::array<VkPresentModeKHR, SIZE> &modes)
{
VkResult res = VK_SUCCESS;
assert(present_mode_count != nullptr);
if (nullptr == present_modes)
{
*present_mode_count = modes.size();
return VK_SUCCESS;
}
if (modes.size() > *present_mode_count)
{
res = VK_INCOMPLETE;
}
*present_mode_count = std::min(*present_mode_count, static_cast<uint32_t>(modes.size()));
for (uint32_t i = 0; i < *present_mode_count; ++i)
{
present_modes[i] = modes[i];
}
return res;
}
} /* namespace wsi */