• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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