1 /*-------------------------------------------------------------------------
2 * Vulkan CTS Framework
3 * --------------------
4 *
5 * Copyright (c) 2016 Google Inc.
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 *
19 *//*!
20 * \file
21 * \brief Windowing System Integration (WSI) Utilities.
22 *//*--------------------------------------------------------------------*/
23
24 #include "vkDeviceUtil.hpp"
25 #include "vkRefUtil.hpp"
26 #include "vkTypeUtil.hpp"
27 #include "vkObjUtil.hpp"
28 #include "vkCmdUtil.hpp"
29 #include "vkQueryUtil.hpp"
30 #include "vkWsiUtil.hpp"
31 #include "vkBarrierUtil.hpp"
32
33 #include "deArrayUtil.hpp"
34 #include "deMemory.h"
35
36 #include <limits>
37 #include <vector>
38
39 using std::vector;
40
41 #if defined (DEQP_SUPPORT_X11)
42 # include <X11/Xlib.h>
43 # if defined (DEQP_SUPPORT_XCB)
44 # include <xcb/xcb.h>
45 # endif // DEQP_SUPPORT_XCB
46 #endif // DEQP_SUPPORT_X11
47
48 #if defined (DEQP_SUPPORT_WAYLAND)
49 # include "tcuLnxWayland.hpp"
50 # define WAYLAND_DISPLAY DE_NULL
51 #endif // DEQP_SUPPORT_WAYLAND
52
53 #if ( DE_OS == DE_OS_WIN32 )
54 #define NOMINMAX
55 #define WIN32_LEAN_AND_MEAN
56 #include <windows.h>
57 #endif
58
59 namespace vk
60 {
61 namespace wsi
62 {
63
64 //! Get canonical WSI name that should be used for example in test case and group names.
getName(Type wsiType)65 const char* getName (Type wsiType)
66 {
67 static const char* const s_names[] =
68 {
69 "xlib",
70 "xcb",
71 "wayland",
72 "android",
73 "win32",
74 "macos",
75 "headless",
76 "ohos",
77 "direct_drm",
78 };
79 return de::getSizedArrayElement<TYPE_LAST>(s_names, wsiType);
80 }
81
getExtensionName(Type wsiType)82 const char* getExtensionName (Type wsiType)
83 {
84 static const char* const s_extNames[] =
85 {
86 "VK_KHR_xlib_surface",
87 "VK_KHR_xcb_surface",
88 "VK_KHR_wayland_surface",
89 "VK_KHR_android_surface",
90 "VK_KHR_win32_surface",
91 "VK_MVK_macos_surface",
92 "VK_EXT_headless_surface",
93 "VK_OpenHarmony_OHOS_surface",
94 "VK_EXT_acquire_drm_display",
95 };
96 return de::getSizedArrayElement<TYPE_LAST>(s_extNames, wsiType);
97 }
98
getPlatformProperties(Type wsiType)99 const PlatformProperties& getPlatformProperties (Type wsiType)
100 {
101 // \note These are declared here (rather than queried through vk::Platform for example)
102 // on purpose. The behavior of a platform is partly defined by the platform spec,
103 // and partly by WSI extensions, and platform ports should not need to override
104 // that definition.
105
106 const deUint32 noDisplayLimit = std::numeric_limits<deUint32>::max();
107 const deUint32 noWindowLimit = std::numeric_limits<deUint32>::max();
108
109 static const PlatformProperties s_properties[] =
110 {
111 // VK_KHR_xlib_surface
112 {
113 PlatformProperties::FEATURE_INITIAL_WINDOW_SIZE|PlatformProperties::FEATURE_RESIZE_WINDOW,
114 PlatformProperties::SWAPCHAIN_EXTENT_MUST_MATCH_WINDOW_SIZE,
115 noDisplayLimit,
116 noWindowLimit,
117 },
118 // VK_KHR_xcb_surface
119 {
120 PlatformProperties::FEATURE_INITIAL_WINDOW_SIZE|PlatformProperties::FEATURE_RESIZE_WINDOW,
121 PlatformProperties::SWAPCHAIN_EXTENT_MUST_MATCH_WINDOW_SIZE,
122 noDisplayLimit,
123 noWindowLimit,
124 },
125 // VK_KHR_wayland_surface
126 {
127 0u,
128 PlatformProperties::SWAPCHAIN_EXTENT_SETS_WINDOW_SIZE,
129 noDisplayLimit,
130 noWindowLimit,
131 },
132 // VK_KHR_android_surface
133 {
134 PlatformProperties::FEATURE_INITIAL_WINDOW_SIZE,
135 PlatformProperties::SWAPCHAIN_EXTENT_SCALED_TO_WINDOW_SIZE,
136 1u,
137 1u, // Only one window available
138 },
139 // VK_KHR_win32_surface
140 {
141 PlatformProperties::FEATURE_INITIAL_WINDOW_SIZE|PlatformProperties::FEATURE_RESIZE_WINDOW,
142 PlatformProperties::SWAPCHAIN_EXTENT_MUST_MATCH_WINDOW_SIZE,
143 noDisplayLimit,
144 noWindowLimit,
145 },
146 // VK_MVK_macos_surface
147 {
148 PlatformProperties::FEATURE_INITIAL_WINDOW_SIZE|PlatformProperties::FEATURE_RESIZE_WINDOW,
149 PlatformProperties::SWAPCHAIN_EXTENT_MUST_MATCH_WINDOW_SIZE,
150 noDisplayLimit,
151 noWindowLimit,
152 },
153 // VK_EXT_headless_surface
154 {
155 0u,
156 PlatformProperties::SWAPCHAIN_EXTENT_SETS_WINDOW_SIZE,
157 noDisplayLimit,
158 noWindowLimit,
159 },
160 //VK_OpenHarmony_OHOS_surface
161 {
162 PlatformProperties::FEATURE_INITIAL_WINDOW_SIZE,
163 PlatformProperties::SWAPCHAIN_EXTENT_SCALED_TO_WINDOW_SIZE,
164 1u,
165 1u, // Only one window available
166 },
167 // VK_EXT_acquire_drm_display
168 {
169 0u,
170 PlatformProperties::SWAPCHAIN_EXTENT_MUST_MATCH_WINDOW_SIZE,
171 1u,
172 1u,
173 },
174 };
175
176 return de::getSizedArrayElement<TYPE_LAST>(s_properties, wsiType);
177 }
178
179 #ifndef CTS_USES_VULKANSC
createDisplaySurface(const InstanceInterface & vki,VkInstance instance,VkDisplayKHR display,const tcu::CommandLine & cmdLine,const VkAllocationCallbacks * pAllocator,VkSurfaceKHR * pSurface)180 static VkResult createDisplaySurface (const InstanceInterface& vki,
181 VkInstance instance,
182 VkDisplayKHR display,
183 const tcu::CommandLine& cmdLine,
184 const VkAllocationCallbacks* pAllocator,
185 VkSurfaceKHR* pSurface)
186 {
187 VkPhysicalDevice physDevice = chooseDevice(vki, instance, cmdLine);
188
189 vector<VkDisplayPlanePropertiesKHR> planeProperties;
190 deUint32 planeCount = 0u;
191 deUint32 planeIndex = 0u;
192 bool planeFound = false;
193 VK_CHECK_SUPPORTED(vki.getPhysicalDeviceDisplayPlanePropertiesKHR(physDevice, &planeCount, DE_NULL));
194
195 planeProperties.resize(planeCount);
196 VK_CHECK_SUPPORTED(vki.getPhysicalDeviceDisplayPlanePropertiesKHR(physDevice, &planeCount, &planeProperties[0]));
197
198 for (deUint32 i = 0; i < planeCount; ++i)
199 {
200 vector<VkDisplayKHR> supportedDisplays;
201 deUint32 supportedDisplayCount = 0u;
202 VK_CHECK_SUPPORTED(vki.getDisplayPlaneSupportedDisplaysKHR(physDevice, i, &supportedDisplayCount, DE_NULL));
203
204 supportedDisplays.resize(supportedDisplayCount);
205 VK_CHECK_SUPPORTED(vki.getDisplayPlaneSupportedDisplaysKHR(physDevice, i, &supportedDisplayCount, &supportedDisplays[0]));
206
207 for (deUint32 j = 0; j < supportedDisplayCount; ++j)
208 {
209 if (display == supportedDisplays[i])
210 {
211 planeIndex = i;
212 planeFound = true;
213 break;
214 }
215 }
216
217 if (planeFound)
218 break;
219 }
220 if (!planeFound)
221 TCU_THROW(NotSupportedError, "No supported displays for planes.");
222
223 vector<VkDisplayModePropertiesKHR> displayModeProperties;
224 deUint32 displayModeCount = 0u;
225 VK_CHECK_SUPPORTED(vki.getDisplayModePropertiesKHR(physDevice, display, &displayModeCount, DE_NULL));
226 if (displayModeCount < 1)
227 TCU_THROW(NotSupportedError, "No display modes defined.");
228
229 displayModeProperties.resize(displayModeCount);
230 VK_CHECK_SUPPORTED(vki.getDisplayModePropertiesKHR(physDevice, display, &displayModeCount, &displayModeProperties[0]));
231
232 const VkDisplaySurfaceCreateInfoKHR createInfo = {
233 VK_STRUCTURE_TYPE_DISPLAY_SURFACE_CREATE_INFO_KHR, // VkStructureType sType
234 DE_NULL, // const void* pNext
235 0, // VkDisplaySurfaceCreateFlagsKHR flags
236 displayModeProperties[0].displayMode, // VkDisplayModeKHR displayMode
237 planeIndex, // uint32_t planeIndex
238 planeProperties[planeIndex].currentStackIndex, // uint32_t planeStackIndex
239 VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR, // VkSurfaceTransformFlagBitsKHR transform
240 1.0f, // float globalAlpha
241 VK_DISPLAY_PLANE_ALPHA_OPAQUE_BIT_KHR, // VkDisplayPlaneAlphaFlagBitsKHR alphaMode
242 displayModeProperties[0].parameters.visibleRegion, // VkExtent2D imageExtent
243 };
244
245 return vki.createDisplayPlaneSurfaceKHR(instance, &createInfo, pAllocator, pSurface);
246 }
247 #endif // CTS_USES_VULKANSC
248
createSurface(const InstanceInterface & vki,VkInstance instance,Type wsiType,const Display & nativeDisplay,const Window & nativeWindow,const tcu::CommandLine & cmdLine,const VkAllocationCallbacks * pAllocator,VkSurfaceKHR * pSurface)249 VkResult createSurface (const InstanceInterface& vki,
250 VkInstance instance,
251 Type wsiType,
252 const Display& nativeDisplay,
253 const Window& nativeWindow,
254 const tcu::CommandLine& cmdLine,
255 const VkAllocationCallbacks* pAllocator,
256 VkSurfaceKHR* pSurface)
257 {
258 // Update this function if you add more WSI implementations
259 DE_STATIC_ASSERT(TYPE_LAST <= (1<<7));
260
261 #ifdef CTS_USES_VULKANSC
262 DE_UNREF(vki);
263 DE_UNREF(instance);
264 DE_UNREF(wsiType);
265 DE_UNREF(nativeDisplay);
266 DE_UNREF(nativeWindow);
267 DE_UNREF(cmdLine);
268 DE_UNREF(pAllocator);
269 DE_UNREF(pSurface);
270
271 TCU_THROW(NotSupportedError, "Vulkan SC does not support createSurface");
272 #else // CTS_USES_VULKANSC
273 switch (wsiType)
274 {
275 case TYPE_XLIB:
276 {
277 const XlibDisplayInterface& xlibDisplay = dynamic_cast<const XlibDisplayInterface&>(nativeDisplay);
278 const XlibWindowInterface& xlibWindow = dynamic_cast<const XlibWindowInterface&>(nativeWindow);
279 const VkXlibSurfaceCreateInfoKHR createInfo =
280 {
281 VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR,
282 DE_NULL,
283 (VkXlibSurfaceCreateFlagsKHR)0,
284 xlibDisplay.getNative(),
285 xlibWindow.getNative()
286 };
287
288 return vki.createXlibSurfaceKHR(instance, &createInfo, pAllocator, pSurface);
289 }
290
291 case TYPE_XCB:
292 {
293 const XcbDisplayInterface& xcbDisplay = dynamic_cast<const XcbDisplayInterface&>(nativeDisplay);
294 const XcbWindowInterface& xcbWindow = dynamic_cast<const XcbWindowInterface&>(nativeWindow);
295 const VkXcbSurfaceCreateInfoKHR createInfo =
296 {
297 VK_STRUCTURE_TYPE_XCB_SURFACE_CREATE_INFO_KHR,
298 DE_NULL,
299 (VkXcbSurfaceCreateFlagsKHR)0,
300 xcbDisplay.getNative(),
301 xcbWindow.getNative()
302 };
303
304 return vki.createXcbSurfaceKHR(instance, &createInfo, pAllocator, pSurface);
305 }
306
307 case TYPE_WAYLAND:
308 {
309 const WaylandDisplayInterface& waylandDisplay = dynamic_cast<const WaylandDisplayInterface&>(nativeDisplay);
310 const WaylandWindowInterface& waylandWindow = dynamic_cast<const WaylandWindowInterface&>(nativeWindow);
311 const VkWaylandSurfaceCreateInfoKHR createInfo =
312 {
313 VK_STRUCTURE_TYPE_WAYLAND_SURFACE_CREATE_INFO_KHR,
314 DE_NULL,
315 (VkWaylandSurfaceCreateFlagsKHR)0,
316 waylandDisplay.getNative(),
317 waylandWindow.getNative()
318 };
319
320 return vki.createWaylandSurfaceKHR(instance, &createInfo, pAllocator, pSurface);
321 }
322
323 case TYPE_ANDROID:
324 {
325 const AndroidWindowInterface& androidWindow = dynamic_cast<const AndroidWindowInterface&>(nativeWindow);
326 const VkAndroidSurfaceCreateInfoKHR createInfo =
327 {
328 VK_STRUCTURE_TYPE_ANDROID_SURFACE_CREATE_INFO_KHR,
329 DE_NULL,
330 (VkAndroidSurfaceCreateFlagsKHR)0,
331 androidWindow.getNative()
332 };
333
334 return vki.createAndroidSurfaceKHR(instance, &createInfo, pAllocator, pSurface);
335 }
336
337 case TYPE_WIN32:
338 {
339 const Win32DisplayInterface& win32Display = dynamic_cast<const Win32DisplayInterface&>(nativeDisplay);
340 const Win32WindowInterface& win32Window = dynamic_cast<const Win32WindowInterface&>(nativeWindow);
341 const VkWin32SurfaceCreateInfoKHR createInfo =
342 {
343 VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR,
344 DE_NULL,
345 (VkWin32SurfaceCreateFlagsKHR)0,
346 win32Display.getNative(),
347 win32Window.getNative()
348 };
349
350 return vki.createWin32SurfaceKHR(instance, &createInfo, pAllocator, pSurface);
351 }
352
353 case TYPE_MACOS:
354 {
355 const MacOSWindowInterface& macOSWindow = dynamic_cast<const MacOSWindowInterface&>(nativeWindow);
356 const VkMacOSSurfaceCreateInfoMVK createInfo =
357 {
358 VK_STRUCTURE_TYPE_MACOS_SURFACE_CREATE_INFO_MVK,
359 DE_NULL,
360 (VkMacOSSurfaceCreateFlagsMVK)0,
361 macOSWindow.getNative()
362 };
363
364 return vki.createMacOSSurfaceMVK(instance, &createInfo, pAllocator, pSurface);
365 }
366
367 case TYPE_HEADLESS:
368 {
369 const VkHeadlessSurfaceCreateInfoEXT createInfo =
370 {
371 VK_STRUCTURE_TYPE_HEADLESS_SURFACE_CREATE_INFO_EXT,
372 DE_NULL,
373 (VkHeadlessSurfaceCreateFlagsEXT)0
374 };
375
376 return vki.createHeadlessSurfaceEXT(instance, &createInfo, pAllocator, pSurface);
377 }
378
379 case TYPE_DIRECT_DRM:
380 {
381 DirectDrmDisplayInterface &drmDisplay = dynamic_cast<DirectDrmDisplayInterface &>(const_cast<Display&>(nativeDisplay));
382 drmDisplay.initializeDisplay(vki, instance, cmdLine);
383 return createDisplaySurface(vki, instance, drmDisplay.getNative(), cmdLine, pAllocator, pSurface);
384 }
385
386 case TYPE_OHOS:
387 {//cxunmz
388 const OhosWindowInterface& ohosWindow = dynamic_cast<const OhosWindowInterface&>(nativeWindow);
389 const VkOHOSSurfaceCreateInfoOpenHarmony createInfo =
390 {
391 VK_STRUCTURE_TYPE_OHOS_SURFACE_CREATE_INFO_OPENHARMONY,
392 DE_NULL,
393 (VkOHOSSurfaceCreateFlagsOpenHarmony)0,
394 ohosWindow.getNative()
395 };
396 return vki.createOHOSSurfaceOpenHarmony(instance, &createInfo, pAllocator, pSurface);
397 }
398
399 default:
400 DE_FATAL("Unknown WSI type");
401 return VK_ERROR_SURFACE_LOST_KHR;
402 }
403 #endif // CTS_USES_VULKANSC
404 return VK_ERROR_SURFACE_LOST_KHR;
405 }
406
createSurface(const InstanceInterface & vki,VkInstance instance,Type wsiType,const Display & nativeDisplay,const Window & nativeWindow,const tcu::CommandLine & cmdLine,const VkAllocationCallbacks * pAllocator)407 Move<VkSurfaceKHR> createSurface (const InstanceInterface& vki,
408 VkInstance instance,
409 Type wsiType,
410 const Display& nativeDisplay,
411 const Window& nativeWindow,
412 const tcu::CommandLine& cmdLine,
413 const VkAllocationCallbacks* pAllocator)
414 {
415 VkSurfaceKHR object = 0;
416 VK_CHECK(createSurface(vki, instance, wsiType, nativeDisplay, nativeWindow, cmdLine, pAllocator, &object));
417 return Move<VkSurfaceKHR>(check<VkSurfaceKHR>(object), Deleter<VkSurfaceKHR>(vki, instance, pAllocator));
418 }
419
getPhysicalDeviceSurfaceSupport(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,deUint32 queueFamilyIndex,VkSurfaceKHR surface)420 VkBool32 getPhysicalDeviceSurfaceSupport (const InstanceInterface& vki,
421 VkPhysicalDevice physicalDevice,
422 deUint32 queueFamilyIndex,
423 VkSurfaceKHR surface)
424 {
425 VkBool32 result = 0;
426
427 VK_CHECK(vki.getPhysicalDeviceSurfaceSupportKHR(physicalDevice, queueFamilyIndex, surface, &result));
428
429 return result;
430 }
431
getPhysicalDevicePresentationSupport(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,deUint32 queueFamilyIndex,Type wsiType,const Display & nativeDisplay)432 VkBool32 getPhysicalDevicePresentationSupport (const InstanceInterface& vki,
433 VkPhysicalDevice physicalDevice,
434 deUint32 queueFamilyIndex,
435 Type wsiType,
436 const Display& nativeDisplay)
437 {
438 #ifdef CTS_USES_VULKANSC
439 DE_UNREF(vki);
440 DE_UNREF(physicalDevice);
441 DE_UNREF(queueFamilyIndex);
442 DE_UNREF(wsiType);
443 DE_UNREF(nativeDisplay);
444 TCU_THROW(NotSupportedError, "Vulkan SC does not support getPhysicalDevicePresentationSupport");
445 #else // CTS_USES_VULKANSC
446 switch (wsiType)
447 {
448 case TYPE_XLIB:
449 {
450 const XlibDisplayInterface& xlibDisplay = dynamic_cast<const XlibDisplayInterface&>(nativeDisplay);
451 pt::XlibVisualID visualID (0U);
452 #if defined (DEQP_SUPPORT_X11)
453 ::Display* displayPtr = (::Display*)(xlibDisplay.getNative().internal);
454 visualID.internal = (deUint32)(::XDefaultVisual(displayPtr,0)->visualid);
455 #endif
456 return vki.getPhysicalDeviceXlibPresentationSupportKHR(physicalDevice, queueFamilyIndex, xlibDisplay.getNative(), visualID);
457 }
458 case TYPE_XCB:
459 {
460 const XcbDisplayInterface& xcbDisplay = dynamic_cast<const XcbDisplayInterface&>(nativeDisplay);
461 pt::XcbVisualid visualID (0U);
462 #if defined (DEQP_SUPPORT_XCB)
463 xcb_connection_t* connPtr = (xcb_connection_t*)(xcbDisplay.getNative().internal);
464 xcb_screen_t* screen = xcb_setup_roots_iterator(xcb_get_setup(connPtr)).data;
465 visualID.internal = (deUint32)(screen->root_visual);
466 #endif
467 return vki.getPhysicalDeviceXcbPresentationSupportKHR(physicalDevice, queueFamilyIndex, xcbDisplay.getNative(), visualID);
468 }
469 case TYPE_WAYLAND:
470 {
471 const WaylandDisplayInterface& waylandDisplay = dynamic_cast<const WaylandDisplayInterface&>(nativeDisplay);
472 return vki.getPhysicalDeviceWaylandPresentationSupportKHR(physicalDevice, queueFamilyIndex, waylandDisplay.getNative());
473 }
474 case TYPE_WIN32:
475 {
476 return vki.getPhysicalDeviceWin32PresentationSupportKHR(physicalDevice, queueFamilyIndex);
477 }
478 case TYPE_HEADLESS:
479 case TYPE_ANDROID:
480 case TYPE_MACOS:
481 case TYPE_OHOS://cxunmz
482 case TYPE_DIRECT_DRM:
483 {
484 return 1;
485 }
486 default:
487 DE_FATAL("Unknown WSI type");
488 return 0;
489 }
490 #endif // CTS_USES_VULKANSC
491 return 1;
492 }
493
getPhysicalDeviceSurfaceCapabilities(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,VkSurfaceKHR surface)494 VkSurfaceCapabilitiesKHR getPhysicalDeviceSurfaceCapabilities (const InstanceInterface& vki,
495 VkPhysicalDevice physicalDevice,
496 VkSurfaceKHR surface)
497 {
498 VkSurfaceCapabilitiesKHR capabilities;
499
500 deMemset(&capabilities, 0, sizeof(capabilities));
501
502 VK_CHECK(vki.getPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, surface, &capabilities));
503
504 return capabilities;
505 }
506
getPhysicalDeviceSurfaceCapabilities2EXT(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,VkSurfaceKHR surface)507 VkSurfaceCapabilities2EXT getPhysicalDeviceSurfaceCapabilities2EXT (const InstanceInterface& vki,
508 VkPhysicalDevice physicalDevice,
509 VkSurfaceKHR surface)
510 {
511 VkSurfaceCapabilities2EXT capabilities;
512
513 deMemset(&capabilities, 0, sizeof(capabilities));
514 capabilities.sType = VK_STRUCTURE_TYPE_SURFACE_CAPABILITIES_2_EXT;
515
516 VK_CHECK(vki.getPhysicalDeviceSurfaceCapabilities2EXT(physicalDevice, surface, &capabilities));
517
518 return capabilities;
519 }
520
sameSurfaceCapabilities(const VkSurfaceCapabilitiesKHR & khr,const VkSurfaceCapabilities2EXT & ext)521 bool sameSurfaceCapabilities (const VkSurfaceCapabilitiesKHR& khr,
522 const VkSurfaceCapabilities2EXT& ext)
523 {
524 return ( khr.minImageCount == ext.minImageCount &&
525 khr.maxImageCount == ext.maxImageCount &&
526 khr.currentExtent.width == ext.currentExtent.width &&
527 khr.currentExtent.height == ext.currentExtent.height &&
528 khr.minImageExtent.width == ext.minImageExtent.width &&
529 khr.minImageExtent.height == ext.minImageExtent.height &&
530 khr.maxImageExtent.width == ext.maxImageExtent.width &&
531 khr.maxImageExtent.height == ext.maxImageExtent.height &&
532 khr.maxImageArrayLayers == ext.maxImageArrayLayers &&
533 khr.supportedTransforms == ext.supportedTransforms &&
534 khr.currentTransform == ext.currentTransform &&
535 khr.supportedCompositeAlpha == ext.supportedCompositeAlpha &&
536 khr.supportedUsageFlags == ext.supportedUsageFlags );
537 }
538
getPhysicalDeviceSurfaceFormats(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,VkSurfaceKHR surface)539 std::vector<VkSurfaceFormatKHR> getPhysicalDeviceSurfaceFormats (const InstanceInterface& vki,
540 VkPhysicalDevice physicalDevice,
541 VkSurfaceKHR surface)
542 {
543 deUint32 numFormats = 0;
544
545 VK_CHECK(vki.getPhysicalDeviceSurfaceFormatsKHR(physicalDevice, surface, &numFormats, DE_NULL));
546
547 if (numFormats > 0)
548 {
549 std::vector<VkSurfaceFormatKHR> formats (numFormats);
550
551 VK_CHECK(vki.getPhysicalDeviceSurfaceFormatsKHR(physicalDevice, surface, &numFormats, &formats[0]));
552
553 return formats;
554 }
555 else
556 return std::vector<VkSurfaceFormatKHR>();
557 }
558
getPhysicalDeviceSurfacePresentModes(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,VkSurfaceKHR surface)559 std::vector<VkPresentModeKHR> getPhysicalDeviceSurfacePresentModes (const InstanceInterface& vki,
560 VkPhysicalDevice physicalDevice,
561 VkSurfaceKHR surface)
562 {
563 deUint32 numModes = 0;
564
565 VK_CHECK(vki.getPhysicalDeviceSurfacePresentModesKHR(physicalDevice, surface, &numModes, DE_NULL));
566
567 if (numModes > 0)
568 {
569 std::vector<VkPresentModeKHR> modes (numModes);
570
571 VK_CHECK(vki.getPhysicalDeviceSurfacePresentModesKHR(physicalDevice, surface, &numModes, &modes[0]));
572
573 return modes;
574 }
575 else
576 return std::vector<VkPresentModeKHR>();
577 }
578
getSwapchainImages(const DeviceInterface & vkd,VkDevice device,VkSwapchainKHR swapchain)579 std::vector<VkImage> getSwapchainImages (const DeviceInterface& vkd,
580 VkDevice device,
581 VkSwapchainKHR swapchain)
582 {
583 deUint32 numImages = 0;
584
585 VK_CHECK(vkd.getSwapchainImagesKHR(device, swapchain, &numImages, DE_NULL));
586
587 if (numImages > 0)
588 {
589 std::vector<VkImage> images (numImages);
590
591 VK_CHECK(vkd.getSwapchainImagesKHR(device, swapchain, &numImages, &images[0]));
592
593 return images;
594 }
595 else
596 return std::vector<VkImage>();
597 }
598
599 namespace
600 {
601
getSupportedQueueFamilyIndices(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,VkSurfaceKHR surface)602 std::vector<deUint32> getSupportedQueueFamilyIndices (const InstanceInterface& vki, VkPhysicalDevice physicalDevice, VkSurfaceKHR surface)
603 {
604 deUint32 numTotalFamilyIndices;
605 vki.getPhysicalDeviceQueueFamilyProperties(physicalDevice, &numTotalFamilyIndices, DE_NULL);
606
607 std::vector<VkQueueFamilyProperties> queueFamilyProperties(numTotalFamilyIndices);
608 vki.getPhysicalDeviceQueueFamilyProperties(physicalDevice, &numTotalFamilyIndices, &queueFamilyProperties[0]);
609
610 std::vector<deUint32> supportedFamilyIndices;
611 for (deUint32 queueFamilyNdx = 0; queueFamilyNdx < numTotalFamilyIndices; ++queueFamilyNdx)
612 {
613 if (getPhysicalDeviceSurfaceSupport(vki, physicalDevice, queueFamilyNdx, surface) != VK_FALSE)
614 supportedFamilyIndices.push_back(queueFamilyNdx);
615 }
616
617 return supportedFamilyIndices;
618 }
619
getSortedSupportedQueueFamilyIndices(const vk::InstanceInterface & vki,vk::VkPhysicalDevice physicalDevice,vk::VkSurfaceKHR surface)620 std::vector<deUint32> getSortedSupportedQueueFamilyIndices (const vk::InstanceInterface& vki, vk::VkPhysicalDevice physicalDevice, vk::VkSurfaceKHR surface)
621 {
622 std::vector<deUint32> indices = getSupportedQueueFamilyIndices(vki, physicalDevice, surface);
623 std::sort(begin(indices), end(indices));
624 return indices;
625 }
626
627 } // anonymous
628
chooseQueueFamilyIndex(const vk::InstanceInterface & vki,vk::VkPhysicalDevice physicalDevice,const std::vector<vk::VkSurfaceKHR> & surfaces)629 deUint32 chooseQueueFamilyIndex (const vk::InstanceInterface& vki, vk::VkPhysicalDevice physicalDevice, const std::vector<vk::VkSurfaceKHR>& surfaces)
630 {
631 auto indices = getCompatibleQueueFamilyIndices(vki, physicalDevice, surfaces);
632
633 if (indices.empty())
634 TCU_THROW(NotSupportedError, "Device does not support presentation to the given surfaces");
635
636 return indices[0];
637 }
638
chooseQueueFamilyIndex(const vk::InstanceInterface & vki,vk::VkPhysicalDevice physicalDevice,vk::VkSurfaceKHR surface)639 deUint32 chooseQueueFamilyIndex (const vk::InstanceInterface& vki, vk::VkPhysicalDevice physicalDevice, vk::VkSurfaceKHR surface)
640 {
641 return chooseQueueFamilyIndex(vki, physicalDevice, std::vector<vk::VkSurfaceKHR>(1u, surface));
642 }
643
getCompatibleQueueFamilyIndices(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,const std::vector<VkSurfaceKHR> & surfaces)644 std::vector<deUint32> getCompatibleQueueFamilyIndices (const InstanceInterface& vki, VkPhysicalDevice physicalDevice, const std::vector<VkSurfaceKHR>& surfaces)
645 {
646 DE_ASSERT(!surfaces.empty());
647
648 auto indices = getSortedSupportedQueueFamilyIndices(vki, physicalDevice, surfaces[0]);
649
650 for (size_t i = 1; i < surfaces.size(); ++i)
651 {
652 auto newIndices = getSortedSupportedQueueFamilyIndices(vki, physicalDevice, surfaces[i]);
653
654 // Set intersection and overwrite.
655 decltype(indices) intersection;
656 std::set_intersection(begin(indices), end(indices), begin(newIndices), end(newIndices), std::back_inserter(intersection));
657 indices = std::move(intersection);
658 }
659
660 return indices;
661 }
662
getFullScreenSize(const vk::wsi::Type wsiType,const vk::wsi::Display & display,const tcu::UVec2 & fallbackSize)663 tcu::UVec2 getFullScreenSize (const vk::wsi::Type wsiType, const vk::wsi::Display& display, const tcu::UVec2& fallbackSize)
664 {
665 tcu::UVec2 result = fallbackSize;
666
667 switch (wsiType)
668 {
669 case TYPE_XLIB:
670 {
671 #if defined (DEQP_SUPPORT_X11)
672 const XlibDisplayInterface& xlibDisplay = dynamic_cast<const XlibDisplayInterface&>(display);
673 ::Display* displayPtr = (::Display*)(xlibDisplay.getNative().internal);
674 const Screen* screen = ScreenOfDisplay(displayPtr, 0);
675 result.x() = deUint32(screen->width);
676 result.y() = deUint32(screen->height);
677 #endif
678 break;
679 }
680 case TYPE_XCB:
681 {
682 #if defined (DEQP_SUPPORT_XCB)
683 // const XcbDisplayInterface& xcbDisplay = dynamic_cast<const XcbDisplayInterface&>(display);
684 // xcb_connection_t* connPtr = (xcb_connection_t*)(xcbDisplay.getNative().internal);
685 // xcb_screen_t* screen = xcb_setup_roots_iterator(xcb_get_setup(connPtr)).data;
686 // result.x() = deUint32(screen->width_in_pixels);
687 // result.y() = deUint32(screen->height_in_pixels);
688 #endif
689 break;
690 }
691 case TYPE_WAYLAND:
692 {
693 #if defined (DEQP_SUPPORT_WAYLAND)
694 #endif
695 break;
696 }
697 case TYPE_ANDROID:
698 {
699 #if ( DE_OS == DE_OS_ANDROID )
700 #endif
701 break;
702 }
703 case TYPE_WIN32:
704 {
705 #if ( DE_OS == DE_OS_WIN32 )
706 de::MovePtr<Window> nullWindow (display.createWindow(tcu::Nothing));
707 const Win32WindowInterface& win32Window = dynamic_cast<const Win32WindowInterface&>(*nullWindow);
708 HMONITOR hMonitor = (HMONITOR)MonitorFromWindow((HWND)win32Window.getNative().internal, MONITOR_DEFAULTTONEAREST);
709 MONITORINFO monitorInfo;
710 monitorInfo.cbSize = sizeof(MONITORINFO);
711 GetMonitorInfo(hMonitor, &monitorInfo);
712 result.x() = deUint32(abs(monitorInfo.rcMonitor.right - monitorInfo.rcMonitor.left));
713 result.y() = deUint32(abs(monitorInfo.rcMonitor.top - monitorInfo.rcMonitor.bottom));
714 #endif
715 break;
716 }
717
718 case TYPE_MACOS:
719 {
720 #if ( DE_OS == DE_OS_OSX )
721 #endif
722 break;
723 }
724
725 case TYPE_OHOS://cxunmz
726 {
727 break;
728 }
729
730 default:
731 DE_FATAL("Unknown WSI type");
732 break;
733 }
734
735 DE_UNREF(display);
736 return result;
737 }
738
isDisplaySurface(Type wsiType)739 VkBool32 isDisplaySurface (Type wsiType)
740 {
741 switch (wsiType)
742 {
743 case TYPE_XLIB:
744 case TYPE_XCB:
745 case TYPE_WAYLAND:
746 case TYPE_ANDROID:
747 case TYPE_WIN32:
748 case TYPE_MACOS:
749 case TYPE_HEADLESS:
750 case TYPE_OHOS:
751 return 0;
752 case TYPE_DIRECT_DRM:
753 return 1;
754 default:
755 DE_FATAL("Unknown WSI type");
756 return 0;
757 }
758 }
759
createRenderPass(const DeviceInterface & vkd,const VkDevice device,const VkFormat colorAttachmentFormat,const bool explicitLayoutTransitions)760 Move<VkRenderPass> WsiTriangleRenderer::createRenderPass (const DeviceInterface& vkd,
761 const VkDevice device,
762 const VkFormat colorAttachmentFormat,
763 const bool explicitLayoutTransitions)
764 {
765 const VkAttachmentDescription colorAttDesc =
766 {
767 (VkAttachmentDescriptionFlags)0,
768 colorAttachmentFormat,
769 VK_SAMPLE_COUNT_1_BIT,
770 VK_ATTACHMENT_LOAD_OP_CLEAR,
771 VK_ATTACHMENT_STORE_OP_STORE,
772 VK_ATTACHMENT_LOAD_OP_DONT_CARE,
773 VK_ATTACHMENT_STORE_OP_DONT_CARE,
774 (explicitLayoutTransitions) ? VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL : VK_IMAGE_LAYOUT_UNDEFINED,
775 (explicitLayoutTransitions) ? VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL : VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
776 };
777 const VkAttachmentReference colorAttRef =
778 {
779 0u,
780 VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
781 };
782 const VkSubpassDescription subpassDesc =
783 {
784 (VkSubpassDescriptionFlags)0u,
785 VK_PIPELINE_BIND_POINT_GRAPHICS,
786 0u, // inputAttachmentCount
787 DE_NULL, // pInputAttachments
788 1u, // colorAttachmentCount
789 &colorAttRef, // pColorAttachments
790 DE_NULL, // pResolveAttachments
791 DE_NULL, // depthStencilAttachment
792 0u, // preserveAttachmentCount
793 DE_NULL, // pPreserveAttachments
794 };
795 const VkSubpassDependency dependencies[] =
796 {
797 {
798 VK_SUBPASS_EXTERNAL, // srcSubpass
799 0u, // dstSubpass
800 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT,
801 VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT,
802 VK_ACCESS_MEMORY_READ_BIT,
803 (VK_ACCESS_COLOR_ATTACHMENT_READ_BIT|
804 VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT),
805 VK_DEPENDENCY_BY_REGION_BIT
806 },
807 {
808 0u, // srcSubpass
809 VK_SUBPASS_EXTERNAL, // dstSubpass
810 VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT,
811 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT,
812 (VK_ACCESS_COLOR_ATTACHMENT_READ_BIT|
813 VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT),
814 VK_ACCESS_MEMORY_READ_BIT,
815 VK_DEPENDENCY_BY_REGION_BIT
816 },
817 };
818 const VkRenderPassCreateInfo renderPassParams =
819 {
820 VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO,
821 DE_NULL,
822 (VkRenderPassCreateFlags)0,
823 1u,
824 &colorAttDesc,
825 1u,
826 &subpassDesc,
827 DE_LENGTH_OF_ARRAY(dependencies),
828 dependencies,
829 };
830
831 return vk::createRenderPass(vkd, device, &renderPassParams);
832 }
833
createPipelineLayout(const DeviceInterface & vkd,const VkDevice device)834 Move<VkPipelineLayout> WsiTriangleRenderer::createPipelineLayout (const DeviceInterface& vkd,
835 const VkDevice device)
836 {
837 const VkPushConstantRange pushConstantRange =
838 {
839 VK_SHADER_STAGE_VERTEX_BIT,
840 0u, // offset
841 (deUint32)sizeof(deUint32), // size
842 };
843 const VkPipelineLayoutCreateInfo pipelineLayoutParams =
844 {
845 VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
846 DE_NULL,
847 (vk::VkPipelineLayoutCreateFlags)0,
848 0u, // setLayoutCount
849 DE_NULL, // pSetLayouts
850 1u,
851 &pushConstantRange,
852 };
853
854 return vk::createPipelineLayout(vkd, device, &pipelineLayoutParams);
855 }
856
createPipeline(const DeviceInterface & vkd,const VkDevice device,const VkRenderPass renderPass,const VkPipelineLayout pipelineLayout,const BinaryCollection & binaryCollection,const tcu::UVec2 & renderSize)857 Move<VkPipeline> WsiTriangleRenderer::createPipeline (const DeviceInterface& vkd,
858 const VkDevice device,
859 const VkRenderPass renderPass,
860 const VkPipelineLayout pipelineLayout,
861 const BinaryCollection& binaryCollection,
862 const tcu::UVec2& renderSize)
863 {
864 // \note VkShaderModules are fully consumed by vkCreateGraphicsPipelines()
865 // and can be deleted immediately following that call.
866 const Unique<VkShaderModule> vertShaderModule (createShaderModule(vkd, device, binaryCollection.get("tri-vert"), 0));
867 const Unique<VkShaderModule> fragShaderModule (createShaderModule(vkd, device, binaryCollection.get("tri-frag"), 0));
868 const std::vector<VkViewport> viewports (1, makeViewport(renderSize));
869 const std::vector<VkRect2D> scissors (1, makeRect2D(renderSize));
870
871 return vk::makeGraphicsPipeline(vkd, // const DeviceInterface& vk
872 device, // const VkDevice device
873 pipelineLayout, // const VkPipelineLayout pipelineLayout
874 *vertShaderModule, // const VkShaderModule vertexShaderModule
875 DE_NULL, // const VkShaderModule tessellationControlShaderModule
876 DE_NULL, // const VkShaderModule tessellationEvalShaderModule
877 DE_NULL, // const VkShaderModule geometryShaderModule
878 *fragShaderModule, // const VkShaderModule fragmentShaderModule
879 renderPass, // const VkRenderPass renderPass
880 viewports, // const std::vector<VkViewport>& viewports
881 scissors); // const std::vector<VkRect2D>& scissors
882 }
883
createAttachmentView(const DeviceInterface & vkd,const VkDevice device,const VkImage image,const VkFormat format)884 Move<VkImageView> WsiTriangleRenderer::createAttachmentView (const DeviceInterface& vkd,
885 const VkDevice device,
886 const VkImage image,
887 const VkFormat format)
888 {
889 const VkImageViewCreateInfo viewParams =
890 {
891 VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
892 DE_NULL,
893 (VkImageViewCreateFlags)0,
894 image,
895 VK_IMAGE_VIEW_TYPE_2D,
896 format,
897 vk::makeComponentMappingRGBA(),
898 {
899 VK_IMAGE_ASPECT_COLOR_BIT,
900 0u, // baseMipLevel
901 1u, // levelCount
902 0u, // baseArrayLayer
903 1u, // layerCount
904 },
905 };
906
907 return vk::createImageView(vkd, device, &viewParams);
908 }
909
createFramebuffer(const DeviceInterface & vkd,const VkDevice device,const VkRenderPass renderPass,const VkImageView colorAttachment,const tcu::UVec2 & renderSize)910 Move<VkFramebuffer> WsiTriangleRenderer::createFramebuffer (const DeviceInterface& vkd,
911 const VkDevice device,
912 const VkRenderPass renderPass,
913 const VkImageView colorAttachment,
914 const tcu::UVec2& renderSize)
915 {
916 const VkFramebufferCreateInfo framebufferParams =
917 {
918 VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO,
919 DE_NULL,
920 (VkFramebufferCreateFlags)0,
921 renderPass,
922 1u,
923 &colorAttachment,
924 renderSize.x(),
925 renderSize.y(),
926 1u, // layers
927 };
928
929 return vk::createFramebuffer(vkd, device, &framebufferParams);
930 }
931
createBuffer(const DeviceInterface & vkd,VkDevice device,VkDeviceSize size,VkBufferUsageFlags usage)932 Move<VkBuffer> WsiTriangleRenderer::createBuffer (const DeviceInterface& vkd,
933 VkDevice device,
934 VkDeviceSize size,
935 VkBufferUsageFlags usage)
936 {
937 const VkBufferCreateInfo bufferParams =
938 {
939 VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO,
940 DE_NULL,
941 (VkBufferCreateFlags)0,
942 size,
943 usage,
944 VK_SHARING_MODE_EXCLUSIVE,
945 0,
946 DE_NULL
947 };
948
949 return vk::createBuffer(vkd, device, &bufferParams);
950 }
951
WsiTriangleRenderer(const DeviceInterface & vkd,const VkDevice device,Allocator & allocator,const BinaryCollection & binaryRegistry,bool explicitLayoutTransitions,const vector<VkImage> swapchainImages,const vector<VkImage> aliasImages,const VkFormat framebufferFormat,const tcu::UVec2 & renderSize)952 WsiTriangleRenderer::WsiTriangleRenderer (const DeviceInterface& vkd,
953 const VkDevice device,
954 Allocator& allocator,
955 const BinaryCollection& binaryRegistry,
956 bool explicitLayoutTransitions,
957 const vector<VkImage> swapchainImages,
958 const vector<VkImage> aliasImages,
959 const VkFormat framebufferFormat,
960 const tcu::UVec2& renderSize)
961 : m_vkd (vkd)
962 , m_explicitLayoutTransitions (explicitLayoutTransitions)
963 , m_swapchainImages (swapchainImages)
964 , m_aliasImages (aliasImages)
965 , m_renderSize (renderSize)
966 , m_renderPass (createRenderPass(vkd, device, framebufferFormat, m_explicitLayoutTransitions))
967 , m_pipelineLayout (createPipelineLayout(vkd, device))
968 , m_pipeline (createPipeline(vkd, device, *m_renderPass, *m_pipelineLayout, binaryRegistry, renderSize))
969 , m_vertexBuffer (createBuffer(vkd, device, (VkDeviceSize)(sizeof(float)*4*3), VK_BUFFER_USAGE_VERTEX_BUFFER_BIT))
970 , m_vertexBufferMemory (allocator.allocate(getBufferMemoryRequirements(vkd, device, *m_vertexBuffer),
971 MemoryRequirement::HostVisible))
972 {
973 m_attachmentViews.resize(swapchainImages.size());
974 m_attachmentLayouts.resize(swapchainImages.size());
975 m_framebuffers.resize(swapchainImages.size());
976
977 for (size_t imageNdx = 0; imageNdx < swapchainImages.size(); ++imageNdx)
978 {
979 m_attachmentViews[imageNdx] = ImageViewSp(new Unique<VkImageView>(createAttachmentView(vkd, device, swapchainImages[imageNdx], framebufferFormat)));
980 m_attachmentLayouts[imageNdx] = VK_IMAGE_LAYOUT_UNDEFINED;
981 m_framebuffers[imageNdx] = FramebufferSp(new Unique<VkFramebuffer>(createFramebuffer(vkd, device, *m_renderPass, **m_attachmentViews[imageNdx], renderSize)));
982 }
983
984 VK_CHECK(vkd.bindBufferMemory(device, *m_vertexBuffer, m_vertexBufferMemory->getMemory(), m_vertexBufferMemory->getOffset()));
985
986 {
987 const VkMappedMemoryRange memRange =
988 {
989 VK_STRUCTURE_TYPE_MAPPED_MEMORY_RANGE,
990 DE_NULL,
991 m_vertexBufferMemory->getMemory(),
992 m_vertexBufferMemory->getOffset(),
993 VK_WHOLE_SIZE
994 };
995 const tcu::Vec4 vertices[] =
996 {
997 tcu::Vec4(-0.5f, -0.5f, 0.0f, 1.0f),
998 tcu::Vec4(+0.5f, -0.5f, 0.0f, 1.0f),
999 tcu::Vec4( 0.0f, +0.5f, 0.0f, 1.0f)
1000 };
1001 DE_STATIC_ASSERT(sizeof(vertices) == sizeof(float)*4*3);
1002
1003 deMemcpy(m_vertexBufferMemory->getHostPtr(), &vertices[0], sizeof(vertices));
1004 VK_CHECK(vkd.flushMappedMemoryRanges(device, 1u, &memRange));
1005 }
1006 }
1007
WsiTriangleRenderer(WsiTriangleRenderer && other)1008 WsiTriangleRenderer::WsiTriangleRenderer (WsiTriangleRenderer&& other)
1009 : m_vkd (other.m_vkd)
1010 , m_explicitLayoutTransitions (other.m_explicitLayoutTransitions)
1011 , m_swapchainImages (other.m_swapchainImages)
1012 , m_aliasImages (other.m_aliasImages)
1013 , m_renderSize (other.m_renderSize)
1014 , m_renderPass (other.m_renderPass)
1015 , m_pipelineLayout (other.m_pipelineLayout)
1016 , m_pipeline (other.m_pipeline)
1017 , m_vertexBuffer (other.m_vertexBuffer)
1018 , m_vertexBufferMemory (other.m_vertexBufferMemory)
1019 , m_attachmentViews (other.m_attachmentViews)
1020 , m_attachmentLayouts (other.m_attachmentLayouts)
1021 , m_framebuffers (other.m_framebuffers)
1022 {
1023 }
1024
~WsiTriangleRenderer(void)1025 WsiTriangleRenderer::~WsiTriangleRenderer (void)
1026 {
1027 }
1028
recordFrame(VkCommandBuffer cmdBuffer,deUint32 imageNdx,deUint32 frameNdx) const1029 void WsiTriangleRenderer::recordFrame (VkCommandBuffer cmdBuffer,
1030 deUint32 imageNdx,
1031 deUint32 frameNdx) const
1032 {
1033 const VkFramebuffer curFramebuffer = **m_framebuffers[imageNdx];
1034
1035 beginCommandBuffer(m_vkd, cmdBuffer, 0u);
1036
1037 if (m_explicitLayoutTransitions || m_attachmentLayouts[imageNdx] == VK_IMAGE_LAYOUT_UNDEFINED)
1038 {
1039 const auto range = makeImageSubresourceRange(VK_IMAGE_ASPECT_COLOR_BIT, 0u, 1u, 0u, 1u);
1040 const auto newLayout = (m_explicitLayoutTransitions ? VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL : VK_IMAGE_LAYOUT_PRESENT_SRC_KHR);
1041 const auto srcStage = VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT;
1042 const auto srcMask = 0u;
1043 const auto dstStage = (m_explicitLayoutTransitions ? VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT : VK_PIPELINE_STAGE_ALL_COMMANDS_BIT);
1044 const auto dstMask = (m_explicitLayoutTransitions ? VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT : 0);
1045
1046 const auto barrier = makeImageMemoryBarrier(srcMask, dstMask, m_attachmentLayouts[imageNdx], newLayout, m_aliasImages[imageNdx], range);
1047 m_vkd.cmdPipelineBarrier(cmdBuffer, srcStage, dstStage, 0u, 0u, nullptr, 0u, nullptr, 1u, &barrier);
1048
1049 m_attachmentLayouts[imageNdx] = newLayout;
1050 }
1051
1052 beginRenderPass(m_vkd, cmdBuffer, *m_renderPass, curFramebuffer, makeRect2D(0, 0, m_renderSize.x(), m_renderSize.y()), tcu::Vec4(0.125f, 0.25f, 0.75f, 1.0f));
1053
1054 m_vkd.cmdBindPipeline(cmdBuffer, VK_PIPELINE_BIND_POINT_GRAPHICS, *m_pipeline);
1055
1056 {
1057 const VkDeviceSize bindingOffset = 0;
1058 m_vkd.cmdBindVertexBuffers(cmdBuffer, 0u, 1u, &m_vertexBuffer.get(), &bindingOffset);
1059 }
1060
1061 m_vkd.cmdPushConstants(cmdBuffer, *m_pipelineLayout, VK_SHADER_STAGE_VERTEX_BIT, 0u, (deUint32)sizeof(deUint32), &frameNdx);
1062 m_vkd.cmdDraw(cmdBuffer, 3u, 1u, 0u, 0u);
1063 endRenderPass(m_vkd, cmdBuffer);
1064
1065 if (m_explicitLayoutTransitions)
1066 {
1067 VkImageSubresourceRange range = { VK_IMAGE_ASPECT_COLOR_BIT, 0, 1, 0, 1 };
1068 const VkImageMemoryBarrier barrier = makeImageMemoryBarrier (VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT, 0,
1069 m_attachmentLayouts[imageNdx], VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
1070 m_aliasImages[imageNdx], range);
1071 m_vkd.cmdPipelineBarrier(cmdBuffer, VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT, VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, 0u, 0u, DE_NULL, 0u, DE_NULL, 1u, &barrier);
1072 m_attachmentLayouts[imageNdx] = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR;
1073 }
1074
1075 endCommandBuffer(m_vkd, cmdBuffer);
1076 }
1077
recordDeviceGroupFrame(VkCommandBuffer cmdBuffer,deUint32 firstDeviceID,deUint32 secondDeviceID,deUint32 devicesCount,deUint32 imageNdx,deUint32 frameNdx) const1078 void WsiTriangleRenderer::recordDeviceGroupFrame (VkCommandBuffer cmdBuffer,
1079 deUint32 firstDeviceID,
1080 deUint32 secondDeviceID,
1081 deUint32 devicesCount,
1082 deUint32 imageNdx,
1083 deUint32 frameNdx) const
1084 {
1085 const VkFramebuffer curFramebuffer = **m_framebuffers[imageNdx];
1086
1087 beginCommandBuffer(m_vkd, cmdBuffer, 0u);
1088
1089 if (m_explicitLayoutTransitions || m_attachmentLayouts[imageNdx] == VK_IMAGE_LAYOUT_UNDEFINED)
1090 {
1091 const auto range = makeImageSubresourceRange(VK_IMAGE_ASPECT_COLOR_BIT, 0u, 1u, 0u, 1u);
1092 const auto newLayout = (m_explicitLayoutTransitions ? VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL : VK_IMAGE_LAYOUT_PRESENT_SRC_KHR);
1093 const auto srcStage = VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT;
1094 const auto srcMask = 0u;
1095 const auto dstStage = (m_explicitLayoutTransitions ? VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT : VK_PIPELINE_STAGE_ALL_COMMANDS_BIT);
1096 const auto dstMask = (m_explicitLayoutTransitions ? VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT : 0);
1097
1098 const auto barrier = makeImageMemoryBarrier(srcMask, dstMask, m_attachmentLayouts[imageNdx], newLayout, m_aliasImages[imageNdx], range);
1099 m_vkd.cmdPipelineBarrier(cmdBuffer, srcStage, dstStage, 0u, 0u, nullptr, 0u, nullptr, 1u, &barrier);
1100
1101 m_attachmentLayouts[imageNdx] = newLayout;
1102 }
1103
1104 // begin renderpass
1105 {
1106 const VkClearValue clearValue = makeClearValueColorF32(0.125f, 0.25f, 0.75f, 1.0f);
1107
1108 VkRect2D zeroRect = { { 0, 0, },{ 0, 0, } };
1109 vector<VkRect2D> renderAreas;
1110 for (deUint32 i = 0; i < devicesCount; i++)
1111 renderAreas.push_back(zeroRect);
1112
1113 // Render completely if there is only 1 device
1114 if (devicesCount == 1u)
1115 {
1116 renderAreas[0].extent.width = (deInt32)m_renderSize.x();
1117 renderAreas[0].extent.height = (deInt32)m_renderSize.y();
1118 }
1119 else
1120 {
1121 // Split into 2 vertical halves
1122 renderAreas[firstDeviceID].extent.width = (deInt32)m_renderSize.x() / 2;
1123 renderAreas[firstDeviceID].extent.height = (deInt32)m_renderSize.y();
1124 renderAreas[secondDeviceID] = renderAreas[firstDeviceID];
1125 renderAreas[secondDeviceID].offset.x = (deInt32)m_renderSize.x() / 2;
1126 }
1127
1128 const VkDeviceGroupRenderPassBeginInfo deviceGroupRPBeginInfo =
1129 {
1130 VK_STRUCTURE_TYPE_DEVICE_GROUP_RENDER_PASS_BEGIN_INFO,
1131 DE_NULL,
1132 (deUint32)((1 << devicesCount) - 1),
1133 devicesCount,
1134 &renderAreas[0]
1135 };
1136
1137 const VkRenderPassBeginInfo passBeginParams =
1138 {
1139 VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO, // sType
1140 &deviceGroupRPBeginInfo, // pNext
1141 *m_renderPass, // renderPass
1142 curFramebuffer, // framebuffer
1143 {
1144 { 0, 0 },
1145 { m_renderSize.x(), m_renderSize.y() }
1146 }, // renderArea
1147 1u, // clearValueCount
1148 &clearValue, // pClearValues
1149 };
1150 m_vkd.cmdBeginRenderPass(cmdBuffer, &passBeginParams, VK_SUBPASS_CONTENTS_INLINE);
1151 }
1152
1153 m_vkd.cmdBindPipeline(cmdBuffer, VK_PIPELINE_BIND_POINT_GRAPHICS, *m_pipeline);
1154
1155 {
1156 const VkDeviceSize bindingOffset = 0;
1157 m_vkd.cmdBindVertexBuffers(cmdBuffer, 0u, 1u, &m_vertexBuffer.get(), &bindingOffset);
1158 }
1159
1160 m_vkd.cmdPushConstants(cmdBuffer, *m_pipelineLayout, VK_SHADER_STAGE_VERTEX_BIT, 0u, (deUint32)sizeof(deUint32), &frameNdx);
1161 m_vkd.cmdDraw(cmdBuffer, 3u, 1u, 0u, 0u);
1162 endRenderPass(m_vkd, cmdBuffer);
1163
1164 if (m_explicitLayoutTransitions)
1165 {
1166 VkImageSubresourceRange range = { VK_IMAGE_ASPECT_COLOR_BIT, 0, 1, 0, 1 };
1167 const VkImageMemoryBarrier barrier = makeImageMemoryBarrier (VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT, 0,
1168 m_attachmentLayouts[imageNdx], VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
1169 m_aliasImages[imageNdx], range);
1170 m_vkd.cmdPipelineBarrier(cmdBuffer, VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT, VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, 0u, 0u, DE_NULL, 0u, DE_NULL, 1u, &barrier);
1171 m_attachmentLayouts[imageNdx] = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR;
1172 }
1173
1174 endCommandBuffer(m_vkd, cmdBuffer);
1175 }
1176
getPrograms(SourceCollections & dst)1177 void WsiTriangleRenderer::getPrograms (SourceCollections& dst)
1178 {
1179 dst.glslSources.add("tri-vert") << glu::VertexSource(
1180 "#version 310 es\n"
1181 "layout(location = 0) in highp vec4 a_position;\n"
1182 "layout(push_constant) uniform FrameData\n"
1183 "{\n"
1184 " highp uint frameNdx;\n"
1185 "} frameData;\n"
1186 "void main (void)\n"
1187 "{\n"
1188 " highp float angle = float(frameData.frameNdx) / 100.0;\n"
1189 " highp float c = cos(angle);\n"
1190 " highp float s = sin(angle);\n"
1191 " highp mat4 t = mat4( c, -s, 0, 0,\n"
1192 " s, c, 0, 0,\n"
1193 " 0, 0, 1, 0,\n"
1194 " 0, 0, 0, 1);\n"
1195 " gl_Position = t * a_position;\n"
1196 "}\n");
1197 dst.glslSources.add("tri-frag") << glu::FragmentSource(
1198 "#version 310 es\n"
1199 "layout(location = 0) out lowp vec4 o_color;\n"
1200 "void main (void) { o_color = vec4(1.0, 0.0, 1.0, 1.0); }\n");
1201 }
1202
1203 } // wsi
1204 } // vk
1205