1// Copyright 2022-2023 The Khronos Group Inc.
2//
3// SPDX-License-Identifier: CC-BY-4.0
4
5= VK_EXT_legacy_dithering
6:toc: left
7:refpage: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/
8:sectnums:
9
10This proposal regards layering OpenGL over Vulkan, and provides an efficient
11path to implement OpenGL dithering on Vulkan.
12
13== Problem Statement
14
15Dithering is an important technique necessary to remove color banding, which in
16itself is an artifact of low-frequency sampling (by our eyes) of a step
17function (jumps from one color value to another due to quantization).
18This technique is particularly useful to improve the visuals of gradients
19quantized over a low-precision format (for example
20`VK_FORMAT_R5G6B5_UNORM_PACK16`), or even the highest-precision formats when
21the two ends of the gradient are close in value (for example sky at late dusk).
22
23**Dithering is best done by the Vulkan application.**
24For example, an application rendering to a swapchain with `R5G6B5` format can
25render instead to an `R8G8B8A8` (or even `B10G11R11`) lazily-allocated image in
26one subpass, then apply dithering in a second subpass along with other
27necessary work such as color space transformations.
28
29In OpenGL and OpenGL ES, the `GL_DITHER` state is enabled by default and
30dithering is made the responsibility of the driver.
31However, the dithering algorithm itself is undefined, and explicitly allows "no
32dithering" as a possible algorithm.
33As such, some vendors do not provide dithering in OpenGL at all, while others
34do, especially on some tile-based hardware.
35Those that do provide dithering implement it differently, including using
36different dithering matrix sizes, or performing dithering at different times
37(e.g. fragment output vs render pass store operation).
38Most importantly, devices that apply dithering in OpenGL have specialized
39hardware to perform this efficiently.
40
41As far as OpenGL layering is concerned, no dithering is technically an
42acceptable algorithm.
43However, in practice, due to this state being on by default and implemented by
44a number of vendors, numerous Android applications have come to (accidentally
45or otherwise) rely on the visual improvements dithering brings.
46It is therefore necessary for an OpenGL layer over Vulkan to provide a
47dithering implementation.
48
49== Solution Space
50
51=== Emulation Through Per-Draw Dithering
52
53Dithering could be emulated by applying the Bayer matrix at the end of the
54fragment shader.
55This can potentially be constrained to low-bit formats to limit its effect on
56performance and shader size.
57
58Pros:
59
60- Easy to implement by adding code to the fragment shader during shader
61  compilation.
62
63Cons:
64
65- When enabled, the dithering cost is paid extra for each pixel that is
66  overdrawn.
67- As dithering is non-temporal per the OpenGL spec, overdraw with additive
68  blending can accentuate the dithering that is applied, making the image look
69  grainy.
70- Dithering alpha can create visual artifacts as the background is leaked into
71  an otherwise completely opaque shape when blending is enabled.
72  * Quantizing alpha in the shader itself can alleviate this.
73- The generated shader size is increased, with the usual caveats regarding
74  performance.
75
76=== Emulation Through Per-Subpass Dithering
77
78Dithering can also be applied at the end of the subpass, closely emulating what
79some hardware do.
80For this to work, a post-process subpass needs to be added that applies
81dithering.
82The original subpass needs to render to a lazily-allocated image with higher
83precision.
84
85Pros:
86
87- Performant when possible.
88- No issues with blending.
89
90Cons:
91
92- Costly on non-tiled-based hardware both in memory and performance.
93- Relatively complicated to code.
94- Currently, Vulkan provides no feedback to ensure this is done efficiently
95  (e.g. whether the tile-based Vulkan driver could successfully merge and
96   execute the two subpasses on the tile, without implicit splits such as due
97   to memory constraints).
98- The high and low precision versions of the attachment live on the tile memory
99  at the same time for the final subpass, increasing tile memory usage and
100  potentially degrading performance (if the tile size has to change because of
101  that).
102
103=== Exposing Dithering Hardware Through a Vulkan Extension
104
105For vendors that support dithering in hardware, exposing it in a Vulkan
106extension would enable the OpenGL layer to match the vendors' current OpenGL
107driver in behavior.
108
109Pros:
110
111- Trivial to use.
112- Efficient.
113
114Cons:
115
116- Hardware vendors may be planning on removing this feature.
117- The dithering algorithm is grossly underspecified in OpenGL, and vendors are
118  known to apply it incompletely or incorrectly in some situations.
119  A Vulkan extension would necessarily be almost as vague as the OpenGL spec,
120  which is not in the spirit of Vulkan.
121
122== Proposal
123
124Despite the caveats, a Vulkan extension represents the most efficient way
125currently to provide equivalent visuals in an OpenGL layer compared to the
126vendors' OpenGL implementations.
127Once the OpenGL layer replaces vendors' implementations, and once the hardware
128features are removed, an emulation path can be chosen for dithering.
129By that time, it is hoped that applications requiring implicit dithering are a
130relic of the past, and paying the extra cost on newer more efficient hardware
131would be negligible, or at least acceptable.
132
133=== Features
134
135[source,c]
136----
137typedef struct VkPhysicalDeviceLegacyDitheringFeaturesEXT {
138    VkStructureType    sType;
139    void*              pNext;
140    VkBool32           legacyDithering;
141} VkPhysicalDeviceLegacyDitheringFeaturesEXT;
142----
143
144- `legacyDithering` specifies that OpenGL dithering is available.
145
146=== Enabling Dithering
147
148Extending `VkSubpassDescriptionFlagBits`,
149`VK_SUBPASS_DESCRIPTION_ENABLE_LEGACY_DITHERING_BIT_EXT` will enable dithering
150for the subpass.
151Extending `VkRenderingFlagBits`, `VK_RENDERING_ENABLE_LEGACY_DITHERING_BIT_EXT`
152does the same when using dynamic rendering.
153
154The Vulkan implementation is expected to apply dithering equivalently to the
155vendor's OpenGL driver.
156
157=== Limitations
158
159The dithering applied through the use of this extension is unspecified, and
160it is possible that the implementation performs no dithering at all for some
161formats.
162However, it will be equivalent to the vendor's OpenGL driver given equivalent
163OpenGL API calls.
164The following limitations thus apply to OpenGL as well.
165
166- Only certain formats may actually be dithered.
167- The details of the dithering algorithm are unknown.
168- Correctness of the dithering algorithm with respect to sRGB are not
169  guaranteed.
170
171**It is strongly recommended that Vulkan applications implement dithering on
172their own if needed.**
173This extension is intended only for use by OpenGL emulation layers.
174