1 /*------------------------------------------------------------------------
2  * Vulkan Conformance Tests
3  * ------------------------
4  *
5  * Copyright (c) 2014 The Android Open Source Project
6  * Copyright (c) 2016 The Khronos Group Inc.
7  *
8  * Licensed under the Apache License, Version 2.0 (the "License");
9  * you may not use this file except in compliance with the License.
10  * You may obtain a copy of the License at
11  *
12  *      http://www.apache.org/licenses/LICENSE-2.0
13  *
14  * Unless required by applicable law or agreed to in writing, software
15  * distributed under the License is distributed on an "AS IS" BASIS,
16  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17  * See the License for the specific language governing permissions and
18  * limitations under the License.
19  *
20  *//*!
21  * \file
22 * \brief Tessellation Geometry Interaction - Point Size
23 *//*--------------------------------------------------------------------*/
24 
25 #include "vktTessellationGeometryPassthroughTests.hpp"
26 #include "vktTestCaseUtil.hpp"
27 #include "vktTessellationUtil.hpp"
28 
29 #include "tcuTestLog.hpp"
30 
31 #include "vkDefs.hpp"
32 #include "vkBarrierUtil.hpp"
33 #include "vkQueryUtil.hpp"
34 #include "vkBuilderUtil.hpp"
35 #include "vkTypeUtil.hpp"
36 #include "vkImageUtil.hpp"
37 #include "vkCmdUtil.hpp"
38 #include "vkObjUtil.hpp"
39 
40 #include "deUniquePtr.hpp"
41 
42 #include <string>
43 #include <vector>
44 
45 namespace vkt
46 {
47 namespace tessellation
48 {
49 
50 using namespace vk;
51 
52 namespace
53 {
54 
55 enum Constants
56 {
57 	RENDER_SIZE = 32,
58 };
59 
60 enum FlagBits
61 {
62 	FLAG_VERTEX_SET						= 1u << 0,		// !< set gl_PointSize in vertex shader
63 	FLAG_TESSELLATION_EVALUATION_SET	= 1u << 1,		// !< set gl_PointSize in tessellation evaluation shader
64 	FLAG_TESSELLATION_ADD				= 1u << 2,		// !< read and add to gl_PointSize in tessellation shader pair
65 	FLAG_GEOMETRY_SET					= 1u << 3,		// !< set gl_PointSize in geometry shader
66 	FLAG_GEOMETRY_ADD					= 1u << 4,		// !< read and add to gl_PointSize in geometry shader
67 };
68 typedef deUint32 Flags;
69 
checkPointSizeRequirements(const InstanceInterface & vki,const VkPhysicalDevice physDevice,const int maxPointSize)70 void checkPointSizeRequirements (const InstanceInterface& vki, const VkPhysicalDevice physDevice, const int maxPointSize)
71 {
72 	const VkPhysicalDeviceProperties properties = getPhysicalDeviceProperties(vki, physDevice);
73 	if (maxPointSize > static_cast<int>(properties.limits.pointSizeRange[1]))
74 		throw tcu::NotSupportedError("Test requires point size " + de::toString(maxPointSize));
75 	// Point size granularity must be 1.0 at most, so no need to check it for this test.
76 }
77 
getExpectedPointSize(const Flags flags)78 int getExpectedPointSize (const Flags flags)
79 {
80 	int addition = 0;
81 
82 	// geometry
83 	if (flags & FLAG_GEOMETRY_SET)
84 		return 6;
85 	else if (flags & FLAG_GEOMETRY_ADD)
86 		addition += 2;
87 
88 	// tessellation
89 	if (flags & FLAG_TESSELLATION_EVALUATION_SET)
90 		return 4 + addition;
91 	else if (flags & FLAG_TESSELLATION_ADD)
92 		addition += 2;
93 
94 	// vertex
95 	if (flags & FLAG_VERTEX_SET)
96 		return 2 + addition;
97 
98 	// undefined
99 	DE_ASSERT(false);
100 	return -1;
101 }
102 
isTessellationStage(const Flags flags)103 inline bool isTessellationStage (const Flags flags)
104 {
105 	return (flags & (FLAG_TESSELLATION_EVALUATION_SET | FLAG_TESSELLATION_ADD)) != 0;
106 }
107 
isGeometryStage(const Flags flags)108 inline bool isGeometryStage (const Flags flags)
109 {
110 	return (flags & (FLAG_GEOMETRY_SET | FLAG_GEOMETRY_ADD)) != 0;
111 }
112 
verifyImage(tcu::TestLog & log,const tcu::ConstPixelBufferAccess image,const int expectedSize)113 bool verifyImage (tcu::TestLog& log, const tcu::ConstPixelBufferAccess image, const int expectedSize)
114 {
115 	log << tcu::TestLog::Message << "Verifying rendered point size. Expecting " << expectedSize << " pixels." << tcu::TestLog::EndMessage;
116 
117 	bool			resultAreaFound	= false;
118 	tcu::IVec4		resultArea;
119 	const tcu::Vec4	black(0.0, 0.0, 0.0, 1.0);
120 
121 	// Find rasterization output area
122 
123 	for (int y = 0; y < image.getHeight(); ++y)
124 	for (int x = 0; x < image.getWidth();  ++x)
125 		if (image.getPixel(x, y) != black)
126 		{
127 			if (!resultAreaFound)
128 			{
129 				// first fragment
130 				resultArea = tcu::IVec4(x, y, x + 1, y + 1);
131 				resultAreaFound = true;
132 			}
133 			else
134 			{
135 				// union area
136 				resultArea.x() = de::min(resultArea.x(), x);
137 				resultArea.y() = de::min(resultArea.y(), y);
138 				resultArea.z() = de::max(resultArea.z(), x+1);
139 				resultArea.w() = de::max(resultArea.w(), y+1);
140 			}
141 		}
142 
143 	if (!resultAreaFound)
144 	{
145 		log << tcu::TestLog::Message << "Verification failed, could not find any point fragments." << tcu::TestLog::EndMessage;
146 		return false;
147 	}
148 
149 	const tcu::IVec2 pointSize = resultArea.swizzle(2,3) - resultArea.swizzle(0, 1);
150 
151 	if (pointSize.x() != pointSize.y())
152 	{
153 		log << tcu::TestLog::Message << "ERROR! Rasterized point is not a square. Point size was " << pointSize << tcu::TestLog::EndMessage;
154 		return false;
155 	}
156 
157 	if (pointSize.x() != expectedSize)
158 	{
159 		log << tcu::TestLog::Message << "ERROR! Point size invalid, expected " << expectedSize << ", got " << pointSize.x() << tcu::TestLog::EndMessage;
160 		return false;
161 	}
162 
163 	return true;
164 }
165 
initPrograms(vk::SourceCollections & programCollection,const Flags flags)166 void initPrograms (vk::SourceCollections& programCollection, const Flags flags)
167 {
168 	// Vertex shader
169 	{
170 		std::ostringstream src;
171 		src << glu::getGLSLVersionDeclaration(glu::GLSL_VERSION_310_ES) << "\n"
172 			<< "\n"
173 			<< "void main (void)\n"
174 			<< "{\n"
175 			<< "    gl_Position = vec4(0.0, 0.0, 0.0, 1.0);\n";
176 
177 		if (flags & FLAG_VERTEX_SET)
178 			src << "    gl_PointSize = 2.0;\n";
179 
180 		src << "}\n";
181 
182 		programCollection.glslSources.add("vert") << glu::VertexSource(src.str());
183 	}
184 
185 	// Fragment shader
186 	{
187 		std::ostringstream src;
188 		src << glu::getGLSLVersionDeclaration(glu::GLSL_VERSION_310_ES) << "\n"
189 			<< "layout(location = 0) out mediump vec4 fragColor;\n"
190 			<< "\n"
191 			<< "void main (void)\n"
192 			<< "{\n"
193 			<< "    fragColor = vec4(1.0);\n"
194 			<< "}\n";
195 
196 		programCollection.glslSources.add("frag") << glu::FragmentSource(src.str());
197 	}
198 
199 	if (isTessellationStage(flags))
200 	{
201 		// Tessellation control shader
202 		{
203 			std::ostringstream src;
204 			src << glu::getGLSLVersionDeclaration(glu::GLSL_VERSION_310_ES) << "\n"
205 				<< "#extension GL_EXT_tessellation_shader : require\n"
206 				<< "#extension GL_EXT_tessellation_point_size : require\n"
207 				<< "layout(vertices = 1) out;\n"
208 				<< "\n"
209 				<< "void main (void)\n"
210 				<< "{\n"
211 				<< "    gl_TessLevelOuter[0] = 3.0;\n"
212 				<< "    gl_TessLevelOuter[1] = 3.0;\n"
213 				<< "    gl_TessLevelOuter[2] = 3.0;\n"
214 				<< "    gl_TessLevelInner[0] = 3.0;\n"
215 				<< "\n"
216 				<< "    gl_out[gl_InvocationID].gl_Position = gl_in[gl_InvocationID].gl_Position;\n";
217 
218 			if (flags & FLAG_TESSELLATION_ADD)
219 				src << "    // pass as is to eval\n"
220 					<< "    gl_out[gl_InvocationID].gl_PointSize = gl_in[gl_InvocationID].gl_PointSize;\n";
221 
222 			src << "}\n";
223 
224 			programCollection.glslSources.add("tesc") << glu::TessellationControlSource(src.str());
225 		}
226 
227 		// Tessellation evaluation shader
228 		{
229 			std::ostringstream src;
230 			src << glu::getGLSLVersionDeclaration(glu::GLSL_VERSION_310_ES) << "\n"
231 				<< "#extension GL_EXT_tessellation_shader : require\n"
232 				<< "#extension GL_EXT_tessellation_point_size : require\n"
233 				<< "layout(triangles, point_mode) in;\n"
234 				<< "\n"
235 				<< "void main (void)\n"
236 				<< "{\n"
237 				<< "    // hide all but one vertex\n"
238 				<< "    if (gl_TessCoord.x < 0.99)\n"
239 				<< "        gl_Position = vec4(-2.0, 0.0, 0.0, 1.0);\n"
240 				<< "    else\n"
241 				<< "        gl_Position = gl_in[0].gl_Position;\n";
242 
243 			if (flags & FLAG_TESSELLATION_ADD)
244 				src << "\n"
245 					<< "    // add to point size\n"
246 					<< "    gl_PointSize = gl_in[0].gl_PointSize + 2.0;\n";
247 			else if (flags & FLAG_TESSELLATION_EVALUATION_SET)
248 				src << "\n"
249 					<< "    // set point size\n"
250 					<< "    gl_PointSize = 4.0;\n";
251 
252 			src << "}\n";
253 
254 			programCollection.glslSources.add("tese") << glu::TessellationEvaluationSource(src.str());
255 		}
256 	}
257 
258 	if (isGeometryStage(flags))
259 	{
260 		// Geometry shader
261 		std::ostringstream src;
262 		src << glu::getGLSLVersionDeclaration(glu::GLSL_VERSION_310_ES) << "\n"
263 			<< "#extension GL_EXT_geometry_shader : require\n"
264 			<< "#extension GL_EXT_geometry_point_size : require\n"
265 			<< "layout(points) in;\n"
266 			<< "layout(points, max_vertices = 1) out;\n"
267 			<< "\n"
268 			<< "void main (void)\n"
269 			<< "{\n"
270 			<< "    gl_Position  = gl_in[0].gl_Position;\n";
271 
272 		if (flags & FLAG_GEOMETRY_SET)
273 			src << "    gl_PointSize = 6.0;\n";
274 		else if (flags & FLAG_GEOMETRY_ADD)
275 			src << "    gl_PointSize = gl_in[0].gl_PointSize + 2.0;\n";
276 
277 		src << "\n"
278 			<< "    EmitVertex();\n"
279 			<< "}\n";
280 
281 		programCollection.glslSources.add("geom") << glu::GeometrySource(src.str());
282 	}
283 }
284 
test(Context & context,const Flags flags)285 tcu::TestStatus test (Context& context, const Flags flags)
286 {
287 	const int expectedPointSize = getExpectedPointSize(flags);
288 	{
289 		const InstanceInterface& vki        = context.getInstanceInterface();
290 		const VkPhysicalDevice   physDevice = context.getPhysicalDevice();
291 
292 		requireFeatures           (vki, physDevice, FEATURE_TESSELLATION_SHADER | FEATURE_GEOMETRY_SHADER | FEATURE_SHADER_TESSELLATION_AND_GEOMETRY_POINT_SIZE);
293 		checkPointSizeRequirements(vki, physDevice, expectedPointSize);
294 	}
295 	{
296 		tcu::TestLog& log = context.getTestContext().getLog();
297 
298 		if (flags & FLAG_VERTEX_SET)
299 			log << tcu::TestLog::Message << "Setting point size in vertex shader to 2.0." << tcu::TestLog::EndMessage;
300 		if (flags & FLAG_TESSELLATION_EVALUATION_SET)
301 			log << tcu::TestLog::Message << "Setting point size in tessellation evaluation shader to 4.0." << tcu::TestLog::EndMessage;
302 		if (flags & FLAG_TESSELLATION_ADD)
303 			log << tcu::TestLog::Message << "Reading point size in tessellation control shader and adding 2.0 to it in evaluation." << tcu::TestLog::EndMessage;
304 		if (flags & FLAG_GEOMETRY_SET)
305 			log << tcu::TestLog::Message << "Setting point size in geometry shader to 6.0." << tcu::TestLog::EndMessage;
306 		if (flags & FLAG_GEOMETRY_ADD)
307 			log << tcu::TestLog::Message << "Reading point size in geometry shader and adding 2.0." << tcu::TestLog::EndMessage;
308 	}
309 
310 	const DeviceInterface&	vk					= context.getDeviceInterface();
311 	const VkDevice			device				= context.getDevice();
312 	const VkQueue			queue				= context.getUniversalQueue();
313 	const deUint32			queueFamilyIndex	= context.getUniversalQueueFamilyIndex();
314 	Allocator&				allocator			= context.getDefaultAllocator();
315 
316 	// Color attachment
317 
318 	const tcu::IVec2			  renderSize				 = tcu::IVec2(RENDER_SIZE, RENDER_SIZE);
319 	const VkFormat				  colorFormat				 = VK_FORMAT_R8G8B8A8_UNORM;
320 	const VkImageSubresourceRange colorImageSubresourceRange = makeImageSubresourceRange(VK_IMAGE_ASPECT_COLOR_BIT, 0u, 1u, 0u, 1u);
321 	const Image					  colorAttachmentImage		 (vk, device, allocator,
322 															 makeImageCreateInfo(renderSize, colorFormat, VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT, 1u),
323 															 MemoryRequirement::Any);
324 
325 	// Color output buffer
326 
327 	const VkDeviceSize	colorBufferSizeBytes = renderSize.x()*renderSize.y() * tcu::getPixelSize(mapVkFormat(colorFormat));
328 	const Buffer		colorBuffer          (vk, device, allocator, makeBufferCreateInfo(colorBufferSizeBytes, VK_BUFFER_USAGE_TRANSFER_DST_BIT), MemoryRequirement::HostVisible);
329 
330 	// Pipeline
331 
332 	const Unique<VkImageView>		colorAttachmentView(makeImageView						(vk, device, *colorAttachmentImage, VK_IMAGE_VIEW_TYPE_2D, colorFormat, colorImageSubresourceRange));
333 	const Unique<VkRenderPass>		renderPass		   (makeRenderPass						(vk, device, colorFormat));
334 	const Unique<VkFramebuffer>		framebuffer		   (makeFramebuffer						(vk, device, *renderPass, *colorAttachmentView, renderSize.x(), renderSize.y(), 1u));
335 	const Unique<VkPipelineLayout>	pipelineLayout	   (makePipelineLayoutWithoutDescriptors(vk, device));
336 	const Unique<VkCommandPool>		cmdPool			   (makeCommandPool						(vk, device, queueFamilyIndex));
337 	const Unique<VkCommandBuffer>	cmdBuffer		   (allocateCommandBuffer				(vk, device, *cmdPool, VK_COMMAND_BUFFER_LEVEL_PRIMARY));
338 
339 	GraphicsPipelineBuilder			pipelineBuilder;
340 
341 	pipelineBuilder
342 		.setPrimitiveTopology		  (VK_PRIMITIVE_TOPOLOGY_POINT_LIST)
343 		.setRenderSize				  (renderSize)
344 		.setPatchControlPoints		  (1)
345 		.setShader					  (vk, device, VK_SHADER_STAGE_VERTEX_BIT,					context.getBinaryCollection().get("vert"), DE_NULL)
346 		.setShader					  (vk, device, VK_SHADER_STAGE_FRAGMENT_BIT,				context.getBinaryCollection().get("frag"), DE_NULL);
347 
348 	if (isTessellationStage(flags))
349 		pipelineBuilder
350 			.setShader				  (vk, device, VK_SHADER_STAGE_TESSELLATION_CONTROL_BIT,	context.getBinaryCollection().get("tesc"), DE_NULL)
351 			.setShader				  (vk, device, VK_SHADER_STAGE_TESSELLATION_EVALUATION_BIT, context.getBinaryCollection().get("tese"), DE_NULL);
352 
353 	if (isGeometryStage(flags))
354 		pipelineBuilder
355 			.setShader				  (vk, device, VK_SHADER_STAGE_GEOMETRY_BIT,				context.getBinaryCollection().get("geom"), DE_NULL);
356 
357 	const Unique<VkPipeline> pipeline(pipelineBuilder.build(vk, device, *pipelineLayout, *renderPass));
358 
359 	// Draw commands
360 
361 	beginCommandBuffer(vk, *cmdBuffer);
362 
363 	{
364 		const VkImageMemoryBarrier colorAttachmentLayoutBarrier = makeImageMemoryBarrier(
365 			(VkAccessFlags)0, VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT,
366 			VK_IMAGE_LAYOUT_UNDEFINED, VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
367 			*colorAttachmentImage, colorImageSubresourceRange);
368 
369 		vk.cmdPipelineBarrier(*cmdBuffer, VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT, VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT, 0u,
370 			0u, DE_NULL, 0u, DE_NULL, 1u, &colorAttachmentLayoutBarrier);
371 	}
372 
373 	// Begin render pass
374 	{
375 		const VkRect2D	renderArea	= makeRect2D(renderSize);
376 		const tcu::Vec4	clearColor	(0.0f, 0.0f, 0.0f, 1.0f);
377 
378 		beginRenderPass(vk, *cmdBuffer, *renderPass, *framebuffer, renderArea, clearColor);
379 	}
380 
381 	vk.cmdBindPipeline(*cmdBuffer, VK_PIPELINE_BIND_POINT_GRAPHICS, *pipeline);
382 
383 	vk.cmdDraw(*cmdBuffer, 1u, 1u, 0u, 0u);
384 	endRenderPass(vk, *cmdBuffer);
385 
386 	// Copy render result to a host-visible buffer
387 	copyImageToBuffer(vk, *cmdBuffer, *colorAttachmentImage, *colorBuffer, renderSize);
388 
389 	endCommandBuffer(vk, *cmdBuffer);
390 	submitCommandsAndWait(vk, device, queue, *cmdBuffer);
391 
392 	// Verify results
393 	{
394 		const Allocation& alloc = colorBuffer.getAllocation();
395 		invalidateMappedMemoryRange(vk, device, alloc.getMemory(), alloc.getOffset(), colorBufferSizeBytes);
396 		tcu::ConstPixelBufferAccess image(mapVkFormat(colorFormat), renderSize.x(), renderSize.y(), 1, alloc.getHostPtr());
397 
398 		tcu::TestLog& log = context.getTestContext().getLog();
399 		log << tcu::LogImage("color0", "", image);
400 
401 		if (verifyImage(log, image, expectedPointSize))
402 			return tcu::TestStatus::pass("OK");
403 		else
404 			return tcu::TestStatus::fail("Didn't render expected point");
405 	}
406 }
407 
getTestCaseName(const Flags flags)408 std::string getTestCaseName (const Flags flags)
409 {
410 	std::ostringstream buf;
411 
412 	// join per-bit descriptions into a single string with '_' separator
413 	if (flags & FLAG_VERTEX_SET)					buf																		<< "vertex_set";
414 	if (flags & FLAG_TESSELLATION_EVALUATION_SET)	buf << ((flags & (FLAG_TESSELLATION_EVALUATION_SET-1))	? ("_") : (""))	<< "evaluation_set";
415 	if (flags & FLAG_TESSELLATION_ADD)				buf << ((flags & (FLAG_TESSELLATION_ADD-1))				? ("_") : (""))	<< "control_pass_eval_add";
416 	if (flags & FLAG_GEOMETRY_SET)					buf << ((flags & (FLAG_GEOMETRY_SET-1))					? ("_") : (""))	<< "geometry_set";
417 	if (flags & FLAG_GEOMETRY_ADD)					buf << ((flags & (FLAG_GEOMETRY_ADD-1))					? ("_") : (""))	<< "geometry_add";
418 
419 	return buf.str();
420 }
421 
getTestCaseDescription(const Flags flags)422 std::string getTestCaseDescription (const Flags flags)
423 {
424 	std::ostringstream buf;
425 
426 	// join per-bit descriptions into a single string with ", " separator
427 	if (flags & FLAG_VERTEX_SET)					buf																			<< "set point size in vertex shader";
428 	if (flags & FLAG_TESSELLATION_EVALUATION_SET)	buf << ((flags & (FLAG_TESSELLATION_EVALUATION_SET-1))	? (", ") : (""))	<< "set point size in tessellation evaluation shader";
429 	if (flags & FLAG_TESSELLATION_ADD)				buf << ((flags & (FLAG_TESSELLATION_ADD-1))				? (", ") : (""))	<< "add to point size in tessellation shader";
430 	if (flags & FLAG_GEOMETRY_SET)					buf << ((flags & (FLAG_GEOMETRY_SET-1))					? (", ") : (""))	<< "set point size in geometry shader";
431 	if (flags & FLAG_GEOMETRY_ADD)					buf << ((flags & (FLAG_GEOMETRY_ADD-1))					? (", ") : (""))	<< "add to point size in geometry shader";
432 
433 	return buf.str();
434 }
435 
436 } // anonymous
437 
438 //! Ported from dEQP-GLES31.functional.tessellation_geometry_interaction.point_size.*
439 //! with the exception of the default 1.0 point size cases (not valid in Vulkan).
createGeometryPointSizeTests(tcu::TestContext & testCtx)440 tcu::TestCaseGroup* createGeometryPointSizeTests (tcu::TestContext& testCtx)
441 {
442 	de::MovePtr<tcu::TestCaseGroup> group (new tcu::TestCaseGroup(testCtx, "point_size", "Test point size"));
443 
444 	static const Flags caseFlags[] =
445 	{
446 		FLAG_VERTEX_SET,
447 							FLAG_TESSELLATION_EVALUATION_SET,
448 																	FLAG_GEOMETRY_SET,
449 		FLAG_VERTEX_SET	|	FLAG_TESSELLATION_EVALUATION_SET,
450 		FLAG_VERTEX_SET |											FLAG_GEOMETRY_SET,
451 		FLAG_VERTEX_SET	|	FLAG_TESSELLATION_EVALUATION_SET	|	FLAG_GEOMETRY_SET,
452 		FLAG_VERTEX_SET	|	FLAG_TESSELLATION_ADD				|	FLAG_GEOMETRY_ADD,
453 	};
454 
455 	for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(caseFlags); ++ndx)
456 	{
457 		const std::string name = getTestCaseName       (caseFlags[ndx]);
458 		const std::string desc = getTestCaseDescription(caseFlags[ndx]);
459 
460 		addFunctionCaseWithPrograms(group.get(), name, desc, initPrograms, test, caseFlags[ndx]);
461 	}
462 
463 	return group.release();
464 }
465 
466 } // tessellation
467 } // vkt
468