1// Copyright 2022-2023 The Khronos Group Inc.
2//
3// SPDX-License-Identifier: CC-BY-4.0
4
5= VK_EXT_surface_maintenance1
6:toc: left
7:refpage: https://registry.khronos.org/vulkan/specs/1.3-extensions/man/html/
8:sectnums:
9
10This proposal investigates and addresses a number of practical issues with the
11`VK_KHR_surface` specification.
12
13== Problem Statement
14
15The following is a list of issues considered in this proposal:
16
17  * Min and max image count as reported in `VkSurfaceCapabilitiesKHR` depend on
18    present mode.
19  * The scaling behavior of the surface is undefined when the swapchain and
20    the surface dimensions do not match.
21  * Switching between some present modes technically should not require
22    recreating the swapchain
23
24=== Image Count Query Per Present Mode
25
26The implementation may require a different number of images, as reported in
27`VkSurfaceCapabilitiesKHR::minImageCount` and
28`VkSurfaceCapabilitiesKHR::maxImageCount` depending on the desired present mode.
29However, there is currently no way to specify the present mode when calling
30`vkGetPhysicalDeviceSurfaceCapabilitiesKHR`.
31
32=== Undefined Scaling Behavior
33
34Currently, a few of the surface extensions (such as `VK_KHR_xcb_surface` and
35`VK_KHR_win32_surface`) require the swapchain's extents to match the surface's.
36A mismatch between the swapchain and the surface dimensions, as the window is
37resized, would result in `VK_ERROR_OUT_OF_DATE_KHR`, which can be undesirable
38behavior.
39On other surfaces where scaling happens, it is undefined how the image is
40scaled.
41The image may stretch, scale with fixed aspect ratio, snap to a corner or be
42centered.
43
44=== Switching Present Modes
45
46Certain present modes are compatible, such that a swapchain operating under one
47mode can be easily switched to another.
48In such a case, recreating the swapchain is undesirable.
49
50See the proposal for `VK_EXT_swapchain_maintenance1` for details.
51
52== Solution Space
53
54=== Image Count Query Per Present Mode
55
56This issue can be rectified by providing the desired present mode when querying
57surface capabilities.
58
59=== Undefined Scaling Behavior
60
61This issue can be resolved by the application querying and opting in to a
62specific supported scaling behavior.  Potential behaviors are:
63
64  * Stretch the swapchain image to the dimensions of the surface
65  * Stretch the swapchain image while maintaining the aspect ratio to fit
66    within the surface
67  * No scaling applied, and swapchain pixels are mapped one-to-one to the
68    surface
69
70When the resulting image does not cover the entire surface, additional behavior
71can be specified for each of the X and Y axes:
72
73  * Gravitate to left/top
74  * Gravitate to right/bottom
75  * Center
76
77This extension allows the supported behavior to be queried, while
78`VK_EXT_swapchain_maintenance1` would be used to select the desired behavior at
79swapchain creation time.
80
81=== Switching Present Modes
82
83To support the `VK_EXT_swapchain_maintenance1` extension and allow creating
84swapchains that may change present modes, it should be possible to query the
85set of compatible present modes for this purpose.
86
87This can be done by providing one present mode and querying the list of
88compatible present modes from the surface.
89
90== Proposal
91
92Introduced by this API are:
93
94=== Image Count Query Per Present Mode
95
96To query the min and max image counts for a specific present mode, chain the
97following to `VkPhysicalDeviceSurfaceInfo2KHR` when calling
98`vkGetPhysicalDeviceSurfaceCapabilities2KHR`:
99
100[source,c]
101----
102typedef struct VkSurfacePresentModeEXT {
103    VkStructureType    sType;
104    void*              pNext;
105    VkPresentModeKHR   presentMode;
106} VkSurfacePresentModeEXT;
107----
108
109The returned values in `VkSurfaceCapabilitiesKHR::minImageCount` and
110`VkSurfaceCapabilitiesKHR::maxImageCount` would then be specific to the given
111present mode.
112
113=== Undefined Scaling Behavior
114
115To query supported scaling behavior, chain `VkSurfacePresentModeEXT` to
116`VkPhysicalDeviceSurfaceInfo2KHR` and chain the following to
117`VkSurfaceCapabilities2KHR`:
118
119[source,c]
120----
121typedef struct VkSurfacePresentScalingCapabilitiesEXT {
122    VkStructureType           sType;
123    void*                     pNext;
124    VkPresentScalingFlagsEXT  supportedPresentScaling;
125    VkPresentGravityFlagsEXT  supportedPresentGravityX;
126    VkPresentGravityFlagsEXT  supportedPresentGravityY;
127    VkExtent2D                minScaledImageExtent;
128    VkExtent2D                maxScaledImageExtent;
129} VkSurfacePresentScalingCapabilitiesEXT;
130----
131
132Where `VkPresentScalingFlagsEXT` is defined as:
133
134[source,c]
135----
136typedef enum VkPresentScalingFlagBitsEXT {
137    VK_PRESENT_SCALING_ONE_TO_ONE_BIT_EXT = 0x00000001,
138    VK_PRESENT_SCALING_ASPECT_RATIO_STRETCH_BIT_EXT = 0x00000002,
139    VK_PRESENT_SCALING_STRETCH_BIT_EXT = 0x00000004,
140} VkPresentScalingFlagBitsEXT;
141
142typedef VkFlags VkPresentScalingFlagsEXT;
143----
144
145And `VkPresentGravityFlagsEXT` is defined as:
146
147[source,c]
148----
149typedef enum VkPresentGravityFlagBitsEXT {
150    VK_PRESENT_GRAVITY_MIN_BIT_EXT = 0x00000001,
151    VK_PRESENT_GRAVITY_MAX_BIT_EXT = 0x00000002,
152    VK_PRESENT_GRAVITY_CENTERED_BIT_EXT = 0x00000004,
153} VkPresentGravityFlagBitsEXT;
154
155typedef VkFlags VkPresentGravityFlagsEXT;
156----
157
158Note that scaling may not be supported for certain present modes.
159
160The `VK_EXT_swapchain_maintenance1` extension must be used to create a
161swapchain with a supported behavior.
162Swapchains created with scaling enabled, must use image extents that are
163bounded by `minScaledImageExtent` and `maxScaledImageExtent`.
164
165=== Switching Present Modes
166
167To query the list of compatible present modes with a given present mode for the
168purposes of mode switching, chain `VkSurfacePresentModeEXT` to
169`VkPhysicalDeviceSurfaceInfo2KHR` and chain the following to
170`VkSurfaceCapabilities2KHR`:
171
172[source,c]
173----
174typedef struct VkSurfacePresentModeCompatibilityEXT {
175    VkStructureType    sType;
176    void*              pNext;
177    uint32_t           presentModeCount;
178    VkPresentMode*     pPresentModes;
179} VkSurfacePresentModeCompatibilityEXT;
180----
181
182The implementation will return the count of compatible present modes in
183`presentModeCount`, and if provided, the list of them in `pPresentModes`.
184
185== Issues
186