Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qvkkhrdisplayvulkaninstance.cpp
Go to the documentation of this file.
1// Copyright (C) 2021 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant reason:default
4
6#include <QVarLengthArray>
7
9
10QVkKhrDisplayVulkanInstance::QVkKhrDisplayVulkanInstance(QVulkanInstance *instance)
11 : m_instance(instance)
12{
13 loadVulkanLibrary(QStringLiteral("vulkan"), 1);
14}
15
16void QVkKhrDisplayVulkanInstance::createOrAdoptInstance()
17{
18 qDebug("Creating Vulkan instance for VK_KHR_display");
19
20 const QByteArray extName = QByteArrayLiteral("VK_KHR_display");
21 initInstance(m_instance, { extName });
22 if (!m_vkInst)
23 return;
24
25 if (!enabledExtensions().contains(extName)) {
26 qWarning("Failed to enable VK_KHR_display extension");
27 return;
28 }
29
30#if VK_KHR_display
31 m_getPhysicalDeviceDisplayPropertiesKHR = (PFN_vkGetPhysicalDeviceDisplayPropertiesKHR)
32 m_vkGetInstanceProcAddr(m_vkInst, "vkGetPhysicalDeviceDisplayPropertiesKHR");
33 m_getDisplayModePropertiesKHR = (PFN_vkGetDisplayModePropertiesKHR)
34 m_vkGetInstanceProcAddr(m_vkInst, "vkGetDisplayModePropertiesKHR");
35 m_getPhysicalDeviceDisplayPlanePropertiesKHR = (PFN_vkGetPhysicalDeviceDisplayPlanePropertiesKHR)
36 m_vkGetInstanceProcAddr(m_vkInst, "vkGetPhysicalDeviceDisplayPlanePropertiesKHR");
37
38 m_getDisplayPlaneSupportedDisplaysKHR = (PFN_vkGetDisplayPlaneSupportedDisplaysKHR)
39 m_vkGetInstanceProcAddr(m_vkInst, "vkGetDisplayPlaneSupportedDisplaysKHR");
40 m_getDisplayPlaneCapabilitiesKHR = (PFN_vkGetDisplayPlaneCapabilitiesKHR)
41 m_vkGetInstanceProcAddr(m_vkInst, "vkGetDisplayPlaneCapabilitiesKHR");
42
43 m_createDisplayPlaneSurfaceKHR = (PFN_vkCreateDisplayPlaneSurfaceKHR)
44 m_vkGetInstanceProcAddr(m_vkInst, "vkCreateDisplayPlaneSurfaceKHR");
45#endif
46
47 m_enumeratePhysicalDevices = (PFN_vkEnumeratePhysicalDevices)
48 m_vkGetInstanceProcAddr(m_vkInst, "vkEnumeratePhysicalDevices");
49
50 m_getPhysicalDeviceSurfaceSupportKHR = reinterpret_cast<PFN_vkGetPhysicalDeviceSurfaceSupportKHR>(
51 m_vkGetInstanceProcAddr(m_vkInst, "vkGetPhysicalDeviceSurfaceSupportKHR"));
52
53 // Use for first physical device, unless overridden by QT_VK_PHYSICAL_DEVICE_INDEX.
54 // This behavior matches what the Vulkan backend of QRhi would do.
55
56 uint32_t physDevCount = 0;
57 m_enumeratePhysicalDevices(m_vkInst, &physDevCount, nullptr);
58 if (!physDevCount) {
59 qWarning("No physical devices");
60 return;
61 }
62 QVarLengthArray<VkPhysicalDevice, 4> physDevs(physDevCount);
63 VkResult err = m_enumeratePhysicalDevices(m_vkInst, &physDevCount, physDevs.data());
64 if (err != VK_SUCCESS || !physDevCount) {
65 qWarning("Failed to enumerate physical devices: %d", err);
66 return;
67 }
68
69 if (qEnvironmentVariableIsSet("QT_VK_PHYSICAL_DEVICE_INDEX")) {
70 int requestedPhysDevIndex = qEnvironmentVariableIntValue("QT_VK_PHYSICAL_DEVICE_INDEX");
71 if (requestedPhysDevIndex >= 0 && uint32_t(requestedPhysDevIndex) < physDevCount)
72 m_physDev = physDevs[requestedPhysDevIndex];
73 }
74
75 if (m_physDev == VK_NULL_HANDLE)
76 m_physDev = physDevs[0];
77
78 if (chooseDisplay()) {
79 if (m_createdCallback)
80 m_createdCallback(this, m_createdCallbackUserData);
81 }
82}
83
84bool QVkKhrDisplayVulkanInstance::supportsPresent(VkPhysicalDevice physicalDevice,
85 uint32_t queueFamilyIndex,
86 QWindow *window)
87{
88 if (!m_getPhysicalDeviceSurfaceSupportKHR)
89 return true;
90
91 VkSurfaceKHR surface = QVulkanInstance::surfaceForWindow(window);
92 VkBool32 supported = false;
93 m_getPhysicalDeviceSurfaceSupportKHR(physicalDevice, queueFamilyIndex, surface, &supported);
94
95 return supported;
96}
97
98bool QVkKhrDisplayVulkanInstance::chooseDisplay()
99{
100#if VK_KHR_display
101 uint32_t displayCount = 0;
102 VkResult err = m_getPhysicalDeviceDisplayPropertiesKHR(m_physDev, &displayCount, nullptr);
103 if (err != VK_SUCCESS) {
104 qWarning("Failed to get display properties: %d", err);
105 return false;
106 }
107
108 qDebug("Display count: %u", displayCount);
109
110 QVarLengthArray<VkDisplayPropertiesKHR, 4> displayProps(displayCount);
111 m_getPhysicalDeviceDisplayPropertiesKHR(m_physDev, &displayCount, displayProps.data());
112
113 m_display = VK_NULL_HANDLE;
114 m_displayMode = VK_NULL_HANDLE;
115
116 // Pick the first display and the first mode, unless specified via env.vars.
117 uint32_t wantedDisplayIndex = 0;
118 uint32_t wantedModeIndex = 0;
119 if (qEnvironmentVariableIsSet("QT_VK_DISPLAY_INDEX"))
120 wantedDisplayIndex = uint32_t(qEnvironmentVariableIntValue("QT_VK_DISPLAY_INDEX"));
121 if (qEnvironmentVariableIsSet("QT_VK_MODE_INDEX"))
122 wantedModeIndex = uint32_t(qEnvironmentVariableIntValue("QT_VK_MODE_INDEX"));
123
124 for (uint32_t i = 0; i < displayCount; ++i) {
125 const VkDisplayPropertiesKHR &disp(displayProps[i]);
126 qDebug("Display #%u:\n display: %p\n name: %s\n dimensions: %ux%u\n resolution: %ux%u",
127 i, (void *) disp.display, disp.displayName,
128 disp.physicalDimensions.width, disp.physicalDimensions.height,
129 disp.physicalResolution.width, disp.physicalResolution.height);
130
131 if (i == wantedDisplayIndex)
132 m_display = disp.display;
133
134 uint32_t modeCount = 0;
135 if (m_getDisplayModePropertiesKHR(m_physDev, disp.display, &modeCount, nullptr) != VK_SUCCESS) {
136 qWarning("Failed to get modes for display");
137 continue;
138 }
139 QVarLengthArray<VkDisplayModePropertiesKHR, 16> modeProps(modeCount);
140 m_getDisplayModePropertiesKHR(m_physDev, disp.display, &modeCount, modeProps.data());
141 for (uint32_t j = 0; j < modeCount; ++j) {
142 const VkDisplayModePropertiesKHR &mode(modeProps[j]);
143 qDebug(" Mode #%u:\n mode: %p\n visibleRegion: %ux%u\n refreshRate: %u",
144 j, (void *) mode.displayMode,
145 mode.parameters.visibleRegion.width, mode.parameters.visibleRegion.height,
146 mode.parameters.refreshRate);
147 if (j == wantedModeIndex && i == wantedDisplayIndex) {
148 m_displayMode = mode.displayMode;
149 m_width = mode.parameters.visibleRegion.width;
150 m_height = mode.parameters.visibleRegion.height;
151 }
152 }
153 }
154
155 if (m_display == VK_NULL_HANDLE || m_displayMode == VK_NULL_HANDLE) {
156 qWarning("Failed to choose display and mode");
157 return false;
158 }
159
160 qDebug("Using display #%u with mode #%u", wantedDisplayIndex, wantedModeIndex);
161
162 uint32_t planeCount = 0;
163 err = m_getPhysicalDeviceDisplayPlanePropertiesKHR(m_physDev, &planeCount, nullptr);
164 if (err != VK_SUCCESS) {
165 qWarning("Failed to get plane properties: %d", err);
166 return false;
167 }
168
169 qDebug("Plane count: %u", planeCount);
170
171 QVarLengthArray<VkDisplayPlanePropertiesKHR, 4> planeProps(planeCount);
172 m_getPhysicalDeviceDisplayPlanePropertiesKHR(m_physDev, &planeCount, planeProps.data());
173
174 m_planeIndex = UINT_MAX;
175 for (uint32_t i = 0; i < planeCount; ++i) {
176 uint32_t supportedDisplayCount = 0;
177 err = m_getDisplayPlaneSupportedDisplaysKHR(m_physDev, i, &supportedDisplayCount, nullptr);
178 if (err != VK_SUCCESS) {
179 qWarning("Failed to query supported displays for plane: %d", err);
180 return false;
181 }
182
183 QVarLengthArray<VkDisplayKHR, 4> supportedDisplays(supportedDisplayCount);
184 m_getDisplayPlaneSupportedDisplaysKHR(m_physDev, i, &supportedDisplayCount, supportedDisplays.data());
185 qDebug("Plane #%u supports %u displays, currently bound to display %p",
186 i, supportedDisplayCount, (void *) planeProps[i].currentDisplay);
187
188 VkDisplayPlaneCapabilitiesKHR caps;
189 err = m_getDisplayPlaneCapabilitiesKHR(m_physDev, m_displayMode, i, &caps);
190 if (err != VK_SUCCESS) {
191 qWarning("Failed to query plane capabilities: %d", err);
192 return false;
193 }
194
195 qDebug(" supportedAlpha: %d (1=no, 2=global, 4=per pixel, 8=per pixel premul)\n"
196 " minSrc=%d, %d %ux%u\n"
197 " maxSrc=%d, %d %ux%u\n"
198 " minDst=%d, %d %ux%u\n"
199 " maxDst=%d, %d %ux%u",
200 int(caps.supportedAlpha),
201 caps.minSrcPosition.x, caps.minSrcPosition.y, caps.minSrcExtent.width, caps.minSrcExtent.height,
202 caps.maxSrcPosition.x, caps.maxSrcPosition.y, caps.maxSrcExtent.width, caps.maxSrcExtent.height,
203 caps.minDstPosition.x, caps.minDstPosition.y, caps.minDstExtent.width, caps.minDstExtent.height,
204 caps.maxDstPosition.x, caps.maxDstPosition.y, caps.maxDstExtent.width, caps.maxDstExtent.height);
205
206 // if the plane is not in use and supports our chosen display, use that plane
207 if (supportedDisplays.contains(m_display)
208 && (planeProps[i].currentDisplay == VK_NULL_HANDLE || planeProps[i].currentDisplay == m_display))
209 {
210 m_planeIndex = i;
211 m_planeStackIndex = planeProps[i].currentStackIndex;
212 }
213 }
214
215 if (m_planeIndex == UINT_MAX) {
216 qWarning("Failed to find a suitable plane");
217 return false;
218 }
219
220 qDebug("Using plane #%u", m_planeIndex);
221 return true;
222#else
223 return false;
224#endif
225}
226
227VkSurfaceKHR QVkKhrDisplayVulkanInstance::createSurface(QWindow *window)
228{
229#if VK_KHR_display
230 qDebug("Creating VkSurfaceKHR via VK_KHR_display for window %p", (void *) window);
231
232 if (!m_physDev) {
233 qWarning("No physical device, cannot create surface");
234 return VK_NULL_HANDLE;
235 }
236 if (!m_display || !m_displayMode) {
237 qWarning("No display mode chosen, cannot create surface");
238 return VK_NULL_HANDLE;
239 }
240
241 VkDisplaySurfaceCreateInfoKHR surfaceCreateInfo = {};
242 surfaceCreateInfo.sType = VK_STRUCTURE_TYPE_DISPLAY_SURFACE_CREATE_INFO_KHR;
243 surfaceCreateInfo.displayMode = m_displayMode;
244 surfaceCreateInfo.planeIndex = m_planeIndex;
245 surfaceCreateInfo.planeStackIndex = m_planeStackIndex;
246 surfaceCreateInfo.transform = VK_SURFACE_TRANSFORM_INHERIT_BIT_KHR;
247 surfaceCreateInfo.globalAlpha = 1.0f;
248 surfaceCreateInfo.alphaMode = VK_DISPLAY_PLANE_ALPHA_OPAQUE_BIT_KHR;
249 surfaceCreateInfo.imageExtent = { m_width, m_height };
250
251 VkSurfaceKHR surface = VK_NULL_HANDLE;
252 VkResult err = m_createDisplayPlaneSurfaceKHR(m_vkInst, &surfaceCreateInfo, nullptr, &surface);
253 if (err != VK_SUCCESS || surface == VK_NULL_HANDLE) {
254 qWarning("Failed to create surface: %d", err);
255 return VK_NULL_HANDLE;
256 }
257
258 qDebug("Created surface %p", (void *) surface);
259
260 return surface;
261#else
262 Q_UNUSED(window);
263 return VK_NULL_HANDLE;
264#endif
265}
266
267void QVkKhrDisplayVulkanInstance::presentAboutToBeQueued(QWindow *window)
268{
269 Q_UNUSED(window);
270}
271
272QT_END_NAMESPACE
Combined button and popup list for selecting options.