5#include <QVarLengthArray>
10 : m_instance(instance)
17 qDebug(
"Creating Vulkan instance for VK_KHR_display");
25 qWarning(
"Failed to enable VK_KHR_display extension");
30 m_getPhysicalDeviceDisplayPropertiesKHR = (PFN_vkGetPhysicalDeviceDisplayPropertiesKHR)
32 m_getDisplayModePropertiesKHR = (PFN_vkGetDisplayModePropertiesKHR)
34 m_getPhysicalDeviceDisplayPlanePropertiesKHR = (PFN_vkGetPhysicalDeviceDisplayPlanePropertiesKHR)
37 m_getDisplayPlaneSupportedDisplaysKHR = (PFN_vkGetDisplayPlaneSupportedDisplaysKHR)
39 m_getDisplayPlaneCapabilitiesKHR = (PFN_vkGetDisplayPlaneCapabilitiesKHR)
42 m_createDisplayPlaneSurfaceKHR = (PFN_vkCreateDisplayPlaneSurfaceKHR)
46 m_enumeratePhysicalDevices = (PFN_vkEnumeratePhysicalDevices)
49 m_getPhysicalDeviceSurfaceSupportKHR =
reinterpret_cast<PFN_vkGetPhysicalDeviceSurfaceSupportKHR
>(
55 uint32_t physDevCount = 0;
56 m_enumeratePhysicalDevices(
m_vkInst, &physDevCount,
nullptr);
62 VkResult err = m_enumeratePhysicalDevices(
m_vkInst, &physDevCount, physDevs.
data());
63 if (err != VK_SUCCESS || !physDevCount) {
64 qWarning(
"Failed to enumerate physical devices: %d", err);
70 if (requestedPhysDevIndex >= 0 && uint32_t(requestedPhysDevIndex) < physDevCount)
71 m_physDev = physDevs[requestedPhysDevIndex];
74 if (m_physDev == VK_NULL_HANDLE)
75 m_physDev = physDevs[0];
77 if (chooseDisplay()) {
78 if (m_createdCallback)
79 m_createdCallback(
this, m_createdCallbackUserData);
84 uint32_t queueFamilyIndex,
87 if (!m_getPhysicalDeviceSurfaceSupportKHR)
90 VkSurfaceKHR surface = QVulkanInstance::surfaceForWindow(
window);
91 VkBool32 supported =
false;
92 m_getPhysicalDeviceSurfaceSupportKHR(physicalDevice, queueFamilyIndex, surface, &supported);
97bool QVkKhrDisplayVulkanInstance::chooseDisplay()
100 uint32_t displayCount = 0;
101 VkResult err = m_getPhysicalDeviceDisplayPropertiesKHR(m_physDev, &displayCount,
nullptr);
102 if (err != VK_SUCCESS) {
103 qWarning(
"Failed to get display properties: %d", err);
107 qDebug(
"Display count: %u", displayCount);
110 m_getPhysicalDeviceDisplayPropertiesKHR(m_physDev, &displayCount, displayProps.data());
112 m_display = VK_NULL_HANDLE;
113 m_displayMode = VK_NULL_HANDLE;
116 uint32_t wantedDisplayIndex = 0;
117 uint32_t wantedModeIndex = 0;
123 for (uint32_t
i = 0;
i < displayCount; ++
i) {
124 const VkDisplayPropertiesKHR &disp(displayProps[
i]);
125 qDebug(
"Display #%u:\n display: %p\n name: %s\n dimensions: %ux%u\n resolution: %ux%u",
126 i, (
void *) disp.display, disp.displayName,
127 disp.physicalDimensions.width, disp.physicalDimensions.height,
128 disp.physicalResolution.width, disp.physicalResolution.height);
130 if (
i == wantedDisplayIndex)
131 m_display = disp.display;
133 uint32_t modeCount = 0;
134 if (m_getDisplayModePropertiesKHR(m_physDev, disp.display, &modeCount,
nullptr) != VK_SUCCESS) {
135 qWarning(
"Failed to get modes for display");
139 m_getDisplayModePropertiesKHR(m_physDev, disp.display, &modeCount, modeProps.data());
140 for (uint32_t
j = 0;
j < modeCount; ++
j) {
141 const VkDisplayModePropertiesKHR &
mode(modeProps[
j]);
142 qDebug(
" Mode #%u:\n mode: %p\n visibleRegion: %ux%u\n refreshRate: %u",
143 j, (
void *)
mode.displayMode,
144 mode.parameters.visibleRegion.width,
mode.parameters.visibleRegion.height,
145 mode.parameters.refreshRate);
146 if (
j == wantedModeIndex) {
147 m_displayMode =
mode.displayMode;
148 m_width =
mode.parameters.visibleRegion.width;
149 m_height =
mode.parameters.visibleRegion.height;
154 if (m_display == VK_NULL_HANDLE || m_displayMode == VK_NULL_HANDLE) {
155 qWarning(
"Failed to choose display and mode");
159 qDebug(
"Using display #%u with mode #%u", wantedDisplayIndex, wantedModeIndex);
161 uint32_t planeCount = 0;
162 err = m_getPhysicalDeviceDisplayPlanePropertiesKHR(m_physDev, &planeCount,
nullptr);
163 if (err != VK_SUCCESS) {
164 qWarning(
"Failed to get plane properties: %d", err);
168 qDebug(
"Plane count: %u", planeCount);
171 m_getPhysicalDeviceDisplayPlanePropertiesKHR(m_physDev, &planeCount, planeProps.data());
173 m_planeIndex = UINT_MAX;
174 for (uint32_t
i = 0;
i < planeCount; ++
i) {
175 uint32_t supportedDisplayCount = 0;
176 err = m_getDisplayPlaneSupportedDisplaysKHR(m_physDev,
i, &supportedDisplayCount,
nullptr);
177 if (err != VK_SUCCESS) {
178 qWarning(
"Failed to query supported displays for plane: %d", err);
183 m_getDisplayPlaneSupportedDisplaysKHR(m_physDev,
i, &supportedDisplayCount, supportedDisplays.data());
184 qDebug(
"Plane #%u supports %u displays, currently bound to display %p",
185 i, supportedDisplayCount, (
void *) planeProps[
i].currentDisplay);
187 VkDisplayPlaneCapabilitiesKHR caps;
188 err = m_getDisplayPlaneCapabilitiesKHR(m_physDev, m_displayMode,
i, &caps);
189 if (err != VK_SUCCESS) {
190 qWarning(
"Failed to query plane capabilities: %d", err);
194 qDebug(
" supportedAlpha: %d (1=no, 2=global, 4=per pixel, 8=per pixel premul)\n"
195 " minSrc=%d, %d %ux%u\n"
196 " maxSrc=%d, %d %ux%u\n"
197 " minDst=%d, %d %ux%u\n"
198 " maxDst=%d, %d %ux%u",
199 int(caps.supportedAlpha),
200 caps.minSrcPosition.x, caps.minSrcPosition.y, caps.minSrcExtent.width, caps.minSrcExtent.height,
201 caps.maxSrcPosition.x, caps.maxSrcPosition.y, caps.maxSrcExtent.width, caps.maxSrcExtent.height,
202 caps.minDstPosition.x, caps.minDstPosition.y, caps.minDstExtent.width, caps.minDstExtent.height,
203 caps.maxDstPosition.x, caps.maxDstPosition.y, caps.maxDstExtent.width, caps.maxDstExtent.height);
206 if (supportedDisplays.contains(m_display)
207 && (planeProps[
i].currentDisplay == VK_NULL_HANDLE || planeProps[
i].currentDisplay == m_display))
210 m_planeStackIndex = planeProps[
i].currentStackIndex;
214 if (m_planeIndex == UINT_MAX) {
215 qWarning(
"Failed to find a suitable plane");
219 qDebug(
"Using plane #%u", m_planeIndex);
229 qDebug(
"Creating VkSurfaceKHR via VK_KHR_display for window %p", (
void *)
window);
232 qWarning(
"No physical device, cannot create surface");
233 return VK_NULL_HANDLE;
235 if (!m_display || !m_displayMode) {
236 qWarning(
"No display mode chosen, cannot create surface");
237 return VK_NULL_HANDLE;
240 VkDisplaySurfaceCreateInfoKHR surfaceCreateInfo = {};
241 surfaceCreateInfo.sType = VK_STRUCTURE_TYPE_DISPLAY_SURFACE_CREATE_INFO_KHR;
242 surfaceCreateInfo.displayMode = m_displayMode;
243 surfaceCreateInfo.planeIndex = m_planeIndex;
244 surfaceCreateInfo.planeStackIndex = m_planeStackIndex;
245 surfaceCreateInfo.transform = VK_SURFACE_TRANSFORM_INHERIT_BIT_KHR;
246 surfaceCreateInfo.globalAlpha = 1.0f;
247 surfaceCreateInfo.alphaMode = VK_DISPLAY_PLANE_ALPHA_OPAQUE_BIT_KHR;
248 surfaceCreateInfo.imageExtent = { m_width, m_height };
250 VkSurfaceKHR surface = VK_NULL_HANDLE;
251 VkResult err = m_createDisplayPlaneSurfaceKHR(
m_vkInst, &surfaceCreateInfo,
nullptr, &surface);
252 if (err != VK_SUCCESS || surface == VK_NULL_HANDLE) {
253 qWarning(
"Failed to create surface: %d", err);
254 return VK_NULL_HANDLE;
257 qDebug(
"Created surface %p", (
void *) surface);
262 return VK_NULL_HANDLE;
VkSurfaceKHR createSurface(QWindow *window)
bool supportsPresent(VkPhysicalDevice physicalDevice, uint32_t queueFamilyIndex, QWindow *window) override
void presentAboutToBeQueued(QWindow *window) override
void createOrAdoptInstance() override
QVkKhrDisplayVulkanInstance(QVulkanInstance *instance)
The QVulkanInstance class represents a native Vulkan instance, enabling Vulkan rendering onto a QSurf...
Combined button and popup list for selecting options.
#define QByteArrayLiteral(str)
static bool contains(const QJsonArray &haystack, unsigned needle)
#define QStringLiteral(str)
Q_CORE_EXPORT bool qEnvironmentVariableIsSet(const char *varName) noexcept
Q_CORE_EXPORT int qEnvironmentVariableIntValue(const char *varName, bool *ok=nullptr) noexcept