• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*-------------------------------------------------------------------------
2  * OpenGL Conformance Test Suite
3  * -----------------------------
4  *
5  * Copyright (c) 2017 The Khronos Group 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  gl4cShaderAtomicCounterOpsTests.cpp
21  * \brief Conformance tests for the ARB_shader_atomic_counter_ops functionality.
22  */ /*-------------------------------------------------------------------*/
23 
24 #include "gl4cShaderAtomicCounterOpsTests.hpp"
25 #include "gluContextInfo.hpp"
26 #include "gluDefs.hpp"
27 #include "gluDrawUtil.hpp"
28 #include "gluObjectWrapper.hpp"
29 #include "gluShaderProgram.hpp"
30 #include "glwEnums.hpp"
31 #include "glwFunctions.hpp"
32 #include "tcuRenderTarget.hpp"
33 
34 #include <algorithm>
35 #include <sstream>
36 #include <string>
37 
38 using namespace glw;
39 
40 namespace gl4cts
41 {
42 
ShaderPipeline(glu::ShaderType testedShader,AtomicOperation * newOp,bool contextGL46)43 ShaderAtomicCounterOpsTestBase::ShaderPipeline::ShaderPipeline(glu::ShaderType testedShader, AtomicOperation* newOp,
44 															   bool contextGL46)
45 	: m_program(NULL), m_programCompute(NULL), m_testedShader(testedShader), m_atomicOp(newOp)
46 {
47 	m_shaders[glu::SHADERTYPE_VERTEX] = "<version>\n"
48 										"<head>"
49 										"in highp vec2 inPosition;\n"
50 										"out highp vec3 vsPosition;\n"
51 										"out highp vec4 vsColor;\n"
52 										"void main()\n"
53 										"{\n"
54 										"	gl_Position = vec4(inPosition, 0.0, 1.0);\n"
55 										"	vsPosition = vec3(inPosition, 0.0);\n"
56 										"	vec4 outColor = vec4(1.0);\n"
57 										"<atomic_operation>"
58 										"	vsColor = outColor;\n"
59 										"}\n";
60 
61 	m_shaders[glu::SHADERTYPE_FRAGMENT] = "<version>\n"
62 										  "<head>"
63 										  "in highp vec4 gsColor;\n"
64 										  "out highp vec4 fsColor;\n"
65 										  "void main()\n"
66 										  "{\n"
67 										  "	vec4 outColor = gsColor; \n"
68 										  "<atomic_operation>"
69 										  "	fsColor = outColor;\n"
70 										  "}\n";
71 
72 	m_shaders[glu::SHADERTYPE_TESSELLATION_CONTROL] = "<version>\n"
73 													  "<head>"
74 													  "layout(vertices = 3) out;\n"
75 													  "in highp vec4 vsColor[];\n"
76 													  "in highp vec3 vsPosition[];\n"
77 													  "out highp vec3 tcsPosition[];\n"
78 													  "out highp vec4 tcsColor[];\n"
79 													  "void main()\n"
80 													  "{\n"
81 													  "	tcsPosition[gl_InvocationID] = vsPosition[gl_InvocationID];\n"
82 													  "	vec4 outColor = vsColor[gl_InvocationID];\n"
83 													  "<atomic_operation>"
84 													  "	tcsColor[gl_InvocationID] = outColor;\n"
85 													  "	gl_TessLevelInner[0] = 3;\n"
86 													  "	gl_TessLevelOuter[0] = 3;\n"
87 													  "	gl_TessLevelOuter[1] = 3;\n"
88 													  "	gl_TessLevelOuter[2] = 3;\n"
89 													  "}\n";
90 
91 	m_shaders[glu::SHADERTYPE_TESSELLATION_EVALUATION] = "<version>\n"
92 														 "<head>"
93 														 "layout(triangles, equal_spacing, cw) in;\n"
94 														 "in highp vec3 tcsPosition[];\n"
95 														 "in highp vec4 tcsColor[];\n"
96 														 "out highp vec4 tesColor;\n"
97 														 "void main()\n"
98 														 "{\n"
99 														 "	vec3 p0 = gl_TessCoord.x * tcsPosition[0];\n"
100 														 "	vec3 p1 = gl_TessCoord.y * tcsPosition[1];\n"
101 														 "	vec3 p2 = gl_TessCoord.z * tcsPosition[2];\n"
102 														 "	vec4 outColor = tcsColor[0];\n"
103 														 "<atomic_operation>"
104 														 "	tesColor = outColor;\n"
105 														 "	gl_Position = vec4(normalize(p0 + p1 + p2), 1.0);\n"
106 														 "}\n";
107 
108 	m_shaders[glu::SHADERTYPE_GEOMETRY] = "<version>\n"
109 										  "<head>"
110 										  "layout(triangles) in;\n"
111 										  "layout(triangle_strip, max_vertices = 3) out;\n"
112 										  "in highp vec4 tesColor[];\n"
113 										  "out highp vec4 gsColor;\n"
114 										  "void main()\n"
115 										  "{\n"
116 										  "	for (int i = 0; i<3; i++)\n"
117 										  "	{\n"
118 										  "		gl_Position = gl_in[i].gl_Position;\n"
119 										  "		vec4 outColor = tesColor[i];\n"
120 										  "<atomic_operation>"
121 										  "		gsColor = outColor;\n"
122 										  "		EmitVertex();\n"
123 										  "	}\n"
124 										  "	EndPrimitive();\n"
125 										  "}\n";
126 
127 	m_shaders[glu::SHADERTYPE_COMPUTE] = "<version>\n"
128 										 "<head>"
129 										 "layout(rgba32f, binding = 2) writeonly uniform highp image2D destImage;\n"
130 										 "layout (local_size_x = 16, local_size_y = 16) in;\n"
131 										 "void main (void)\n"
132 										 "{\n"
133 										 "	vec4 outColor = vec4(1.0);\n"
134 										 "<atomic_operation>"
135 										 "	imageStore(destImage, ivec2(gl_GlobalInvocationID.xy), outColor);\n"
136 										 "}\n";
137 
138 	// prepare shaders
139 
140 	std::string		  postfix(contextGL46 ? "" : "ARB");
141 	std::stringstream atomicOperationStream;
142 	atomicOperationStream << "uint returned = " << m_atomicOp->getFunction() + postfix + "(counter, ";
143 	if (m_atomicOp->getCompareValue() != 0)
144 	{
145 		atomicOperationStream << m_atomicOp->getCompareValue();
146 		atomicOperationStream << "u, ";
147 	}
148 	atomicOperationStream << m_atomicOp->getParamValue();
149 	atomicOperationStream << "u);\n";
150 	atomicOperationStream << "uint after = atomicCounter(counter);\n";
151 
152 	if (m_atomicOp->shouldTestReturnValue())
153 	{
154 		atomicOperationStream << "if(after == returned) outColor = vec4(0.0f);\n";
155 	}
156 
157 	atomicOperationStream << "atomicCounterIncrement(calls);\n";
158 
159 	std::string versionString;
160 	std::string headString;
161 	if (contextGL46)
162 	{
163 		versionString = "#version 460 core";
164 		headString	= "layout (binding=0) uniform atomic_uint counter;\n"
165 					 "layout (binding=1) uniform atomic_uint calls;\n";
166 	}
167 	else
168 	{
169 		versionString = "#version 450 core";
170 		headString	= "#extension GL_ARB_shader_atomic_counters: enable\n"
171 					 "#extension GL_ARB_shader_atomic_counter_ops: enable\n"
172 					 "layout (binding=0) uniform atomic_uint counter;\n"
173 					 "layout (binding=1) uniform atomic_uint calls;\n";
174 	}
175 
176 	for (unsigned int i = 0; i <= glu::SHADERTYPE_COMPUTE; ++i)
177 	{
178 		prepareShader(m_shaders[i], "<version>", versionString);
179 		prepareShader(m_shaders[i], "<head>", i == testedShader ? headString : "");
180 		prepareShader(m_shaders[i], "<atomic_operation>", i == testedShader ? atomicOperationStream.str() : "");
181 	}
182 }
183 
~ShaderPipeline()184 ShaderAtomicCounterOpsTestBase::ShaderPipeline::~ShaderPipeline()
185 {
186 	if (m_program)
187 	{
188 		delete m_program;
189 	}
190 
191 	if (m_programCompute)
192 	{
193 		delete m_programCompute;
194 	}
195 }
196 
prepareShader(std::string & shader,const std::string & tag,const std::string & value)197 void ShaderAtomicCounterOpsTestBase::ShaderPipeline::prepareShader(std::string& shader, const std::string& tag,
198 																   const std::string& value)
199 {
200 	size_t tagPos = shader.find(tag);
201 
202 	if (tagPos != std::string::npos)
203 		shader.replace(tagPos, tag.length(), value);
204 }
205 
create(deqp::Context & context)206 void ShaderAtomicCounterOpsTestBase::ShaderPipeline::create(deqp::Context& context)
207 {
208 	glu::ProgramSources sources;
209 	for (unsigned int i = 0; i < glu::SHADERTYPE_COMPUTE; ++i)
210 	{
211 		if (!m_shaders[i].empty())
212 		{
213 			sources.sources[i].push_back(m_shaders[i]);
214 		}
215 	}
216 	m_program = new glu::ShaderProgram(context.getRenderContext(), sources);
217 
218 	if (!m_program->isOk())
219 	{
220 		TCU_FAIL("Shader compilation failed");
221 	}
222 
223 	glu::ProgramSources sourcesCompute;
224 	sourcesCompute.sources[glu::SHADERTYPE_COMPUTE].push_back(m_shaders[glu::SHADERTYPE_COMPUTE]);
225 	m_programCompute = new glu::ShaderProgram(context.getRenderContext(), sourcesCompute);
226 
227 	if (!m_programCompute->isOk())
228 	{
229 		TCU_FAIL("Shader compilation failed");
230 	}
231 }
232 
use(deqp::Context & context)233 void ShaderAtomicCounterOpsTestBase::ShaderPipeline::use(deqp::Context& context)
234 {
235 	const glw::Functions& gl = context.getRenderContext().getFunctions();
236 	gl.useProgram(m_program->getProgram());
237 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram failed");
238 }
239 
test(deqp::Context & context)240 void ShaderAtomicCounterOpsTestBase::ShaderPipeline::test(deqp::Context& context)
241 {
242 	const glw::Functions& gl = context.getRenderContext().getFunctions();
243 
244 	gl.clearColor(0.5f, 0.5f, 0.5f, 1.0f);
245 	gl.clear(GL_COLOR_BUFFER_BIT);
246 
247 	if (m_testedShader == glu::SHADERTYPE_COMPUTE)
248 	{
249 		executeComputeShader(context);
250 	}
251 	else
252 	{
253 		renderQuad(context);
254 	}
255 
256 	gl.flush();
257 }
258 
renderQuad(deqp::Context & context)259 void ShaderAtomicCounterOpsTestBase::ShaderPipeline::renderQuad(deqp::Context& context)
260 {
261 	const glw::Functions& gl = context.getRenderContext().getFunctions();
262 
263 	deUint16 const quadIndices[] = { 0, 1, 2, 2, 1, 3 };
264 
265 	float const position[] = { -1.0f, -1.0f, -1.0f, 1.0f, 1.0f, -1.0f, 1.0f, 1.0f };
266 
267 	glu::VertexArrayBinding vertexArrays[] = { glu::va::Float("inPosition", 2, 4, 0, position) };
268 
269 	this->use(context);
270 
271 	glu::PrimitiveList primitiveList = glu::pr::Patches(DE_LENGTH_OF_ARRAY(quadIndices), quadIndices);
272 
273 	glu::draw(context.getRenderContext(), this->getShaderProgram()->getProgram(), DE_LENGTH_OF_ARRAY(vertexArrays),
274 			  vertexArrays, primitiveList);
275 
276 	GLU_EXPECT_NO_ERROR(gl.getError(), "glu::draw error");
277 
278 	gl.memoryBarrier(GL_BUFFER_UPDATE_BARRIER_BIT);
279 	GLU_EXPECT_NO_ERROR(gl.getError(), "glMemoryBarrier() error");
280 }
281 
executeComputeShader(deqp::Context & context)282 void ShaderAtomicCounterOpsTestBase::ShaderPipeline::executeComputeShader(deqp::Context& context)
283 {
284 	const glw::Functions& gl = context.getRenderContext().getFunctions();
285 
286 	const glu::Texture outputTexture(context.getRenderContext());
287 
288 	gl.useProgram(m_programCompute->getProgram());
289 
290 	// output image
291 	gl.bindTexture(GL_TEXTURE_2D, *outputTexture);
292 	gl.texStorage2D(GL_TEXTURE_2D, 1, GL_RGBA32UI, 16, 16);
293 	gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
294 	gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
295 	GLU_EXPECT_NO_ERROR(gl.getError(), "Uploading image data failed");
296 
297 	// bind image
298 	gl.bindImageTexture(2, *outputTexture, 0, GL_FALSE, 0, GL_READ_WRITE, GL_RGBA32F);
299 	GLU_EXPECT_NO_ERROR(gl.getError(), "Image setup failed");
300 
301 	// dispatch compute
302 	gl.dispatchCompute(1, 1, 1);
303 	GLU_EXPECT_NO_ERROR(gl.getError(), "glDispatchCompute() error");
304 	gl.memoryBarrier(GL_TEXTURE_FETCH_BARRIER_BIT);
305 	GLU_EXPECT_NO_ERROR(gl.getError(), "glMemoryBarrier() error");
306 
307 	// render output texture
308 
309 	std::string vs = "#version 450 core\n"
310 					 "in highp vec2 position;\n"
311 					 "in vec2 inTexcoord;\n"
312 					 "out vec2 texcoord;\n"
313 					 "void main()\n"
314 					 "{\n"
315 					 "	texcoord = inTexcoord;\n"
316 					 "	gl_Position = vec4(position, 0.0, 1.0);\n"
317 					 "}\n";
318 
319 	std::string fs = "#version 450 core\n"
320 					 "uniform sampler2D sampler;\n"
321 					 "in vec2 texcoord;\n"
322 					 "out vec4 color;\n"
323 					 "void main()\n"
324 					 "{\n"
325 					 "	color = texture(sampler, texcoord);\n"
326 					 "}\n";
327 
328 	glu::ProgramSources sources;
329 	sources.sources[glu::SHADERTYPE_VERTEX].push_back(vs);
330 	sources.sources[glu::SHADERTYPE_FRAGMENT].push_back(fs);
331 	glu::ShaderProgram renderShader(context.getRenderContext(), sources);
332 
333 	if (!m_program->isOk())
334 	{
335 		TCU_FAIL("Shader compilation failed");
336 	}
337 
338 	gl.bindTexture(GL_TEXTURE_2D, *outputTexture);
339 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindTexture() call failed.");
340 
341 	gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
342 	gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
343 	GLU_EXPECT_NO_ERROR(gl.getError(), "texParameteri failed");
344 
345 	gl.useProgram(renderShader.getProgram());
346 	GLU_EXPECT_NO_ERROR(gl.getError(), "useProgram failed");
347 
348 	gl.uniform1i(gl.getUniformLocation(renderShader.getProgram(), "sampler"), 0);
349 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUniform1i failed");
350 
351 	deUint16 const quadIndices[] = { 0, 1, 2, 2, 1, 3 };
352 
353 	float const position[] = { -1.0f, -1.0f, -1.0f, 1.0f, 1.0f, -1.0f, 1.0f, 1.0f };
354 
355 	float const texCoord[] = { 0.0f, 0.0f, 0.0f, 1.0f, 1.0f, 0.0f, 1.0f, 1.0f };
356 
357 	glu::VertexArrayBinding vertexArrays[] = { glu::va::Float("position", 2, 4, 0, position),
358 											   glu::va::Float("inTexcoord", 2, 4, 0, texCoord) };
359 
360 	glu::draw(context.getRenderContext(), renderShader.getProgram(), DE_LENGTH_OF_ARRAY(vertexArrays), vertexArrays,
361 			  glu::pr::TriangleStrip(DE_LENGTH_OF_ARRAY(quadIndices), quadIndices));
362 
363 	GLU_EXPECT_NO_ERROR(gl.getError(), "glu::draw error");
364 }
365 
fillAtomicCounterBuffer(AtomicOperation * atomicOp)366 void ShaderAtomicCounterOpsTestBase::fillAtomicCounterBuffer(AtomicOperation* atomicOp)
367 {
368 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
369 
370 	GLuint* dataPtr;
371 
372 	// fill values buffer
373 
374 	GLuint inputValue = atomicOp->getInputValue();
375 
376 	gl.bindBuffer(GL_ATOMIC_COUNTER_BUFFER, m_atomicCounterBuffer);
377 	GLU_EXPECT_NO_ERROR(gl.getError(), "bindBuffer() call failed.");
378 
379 	dataPtr = (GLuint*)gl.mapBufferRange(GL_ATOMIC_COUNTER_BUFFER, 0, sizeof(GLuint),
380 										 GL_MAP_WRITE_BIT | GL_MAP_INVALIDATE_BUFFER_BIT | GL_MAP_UNSYNCHRONIZED_BIT);
381 	GLU_EXPECT_NO_ERROR(gl.getError(), "mapBufferRange() call failed.");
382 
383 	*dataPtr = inputValue;
384 
385 	gl.unmapBuffer(GL_ATOMIC_COUNTER_BUFFER);
386 	GLU_EXPECT_NO_ERROR(gl.getError(), "unmapBuffer() call failed.");
387 
388 	// fill calls buffer
389 
390 	gl.bindBuffer(GL_ATOMIC_COUNTER_BUFFER, m_atomicCounterCallsBuffer);
391 	GLU_EXPECT_NO_ERROR(gl.getError(), "bindBuffer() call failed.");
392 
393 	dataPtr = (GLuint*)gl.mapBufferRange(GL_ATOMIC_COUNTER_BUFFER, 0, sizeof(GLuint),
394 										 GL_MAP_WRITE_BIT | GL_MAP_INVALIDATE_BUFFER_BIT | GL_MAP_UNSYNCHRONIZED_BIT);
395 	GLU_EXPECT_NO_ERROR(gl.getError(), "mapBufferRange() call failed.");
396 
397 	*dataPtr = 0;
398 
399 	gl.unmapBuffer(GL_ATOMIC_COUNTER_BUFFER);
400 	GLU_EXPECT_NO_ERROR(gl.getError(), "unmapBuffer() call failed.");
401 }
402 
checkAtomicCounterBuffer(AtomicOperation * atomicOp)403 bool ShaderAtomicCounterOpsTestBase::checkAtomicCounterBuffer(AtomicOperation* atomicOp)
404 {
405 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
406 
407 	GLuint* dataPtr;
408 
409 	// get value
410 
411 	gl.bindBuffer(GL_ATOMIC_COUNTER_BUFFER, m_atomicCounterBuffer);
412 	GLU_EXPECT_NO_ERROR(gl.getError(), "bindBuffer() call failed.");
413 
414 	dataPtr = (GLuint*)gl.mapBufferRange(GL_ATOMIC_COUNTER_BUFFER, 0, sizeof(GLuint), GL_MAP_READ_BIT);
415 	GLU_EXPECT_NO_ERROR(gl.getError(), "mapBufferRange() call failed.");
416 
417 	GLuint finalValue = *dataPtr;
418 
419 	gl.unmapBuffer(GL_ATOMIC_COUNTER_BUFFER);
420 	GLU_EXPECT_NO_ERROR(gl.getError(), "unmapBuffer() call failed.");
421 
422 	// get calls
423 
424 	gl.bindBuffer(GL_ATOMIC_COUNTER_BUFFER, m_atomicCounterCallsBuffer);
425 	GLU_EXPECT_NO_ERROR(gl.getError(), "bindBuffer() call failed.");
426 
427 	dataPtr = (GLuint*)gl.mapBufferRange(GL_ATOMIC_COUNTER_BUFFER, 0, sizeof(GLuint), GL_MAP_READ_BIT);
428 	GLU_EXPECT_NO_ERROR(gl.getError(), "mapBufferRange() call failed.");
429 
430 	GLuint numberOfCalls = *dataPtr;
431 
432 	gl.unmapBuffer(GL_ATOMIC_COUNTER_BUFFER);
433 	GLU_EXPECT_NO_ERROR(gl.getError(), "unmapBuffer() call failed.");
434 
435 	// validate
436 
437 	GLuint expectedValue = atomicOp->getResult(numberOfCalls);
438 
439 	return finalValue == expectedValue;
440 }
441 
bindBuffers()442 void ShaderAtomicCounterOpsTestBase::bindBuffers()
443 {
444 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
445 
446 	gl.bindBufferBase(GL_ATOMIC_COUNTER_BUFFER, 0, m_atomicCounterBuffer);
447 	GLU_EXPECT_NO_ERROR(gl.getError(), "bindBufferBase() call failed.");
448 
449 	gl.bindBufferBase(GL_ATOMIC_COUNTER_BUFFER, 1, m_atomicCounterCallsBuffer);
450 	GLU_EXPECT_NO_ERROR(gl.getError(), "bindBufferBase() call failed.");
451 }
452 
validateColor(tcu::Vec4 testedColor,tcu::Vec4 desiredColor)453 bool ShaderAtomicCounterOpsTestBase::validateColor(tcu::Vec4 testedColor, tcu::Vec4 desiredColor)
454 {
455 	const float epsilon = 1.1f / 31.0f; // Accommodate framebuffers with 5-bit channels.
456 	return de::abs(testedColor.x() - desiredColor.x()) < epsilon &&
457 		   de::abs(testedColor.y() - desiredColor.y()) < epsilon &&
458 		   de::abs(testedColor.z() - desiredColor.z()) < epsilon;
459 }
460 
validateScreenPixels(tcu::Vec4 desiredColor,tcu::Vec4 ignoredColor)461 bool ShaderAtomicCounterOpsTestBase::validateScreenPixels(tcu::Vec4 desiredColor, tcu::Vec4 ignoredColor)
462 {
463 	const glw::Functions&   gl			 = m_context.getRenderContext().getFunctions();
464 	const tcu::RenderTarget renderTarget = m_context.getRenderContext().getRenderTarget();
465 	tcu::IVec2				size(renderTarget.getWidth(), renderTarget.getHeight());
466 
467 	glw::GLfloat* pixels = new glw::GLfloat[size.x() * size.y() * 4];
468 
469 	// clear buffer
470 	for (int x = 0; x < size.x(); ++x)
471 	{
472 		for (int y = 0; y < size.y(); ++y)
473 		{
474 			int mappedPixelPosition = y * size.x() + x;
475 
476 			pixels[mappedPixelPosition * 4 + 0] = -1.0f;
477 			pixels[mappedPixelPosition * 4 + 1] = -1.0f;
478 			pixels[mappedPixelPosition * 4 + 2] = -1.0f;
479 			pixels[mappedPixelPosition * 4 + 3] = -1.0f;
480 		}
481 	}
482 
483 	// read pixels
484 	gl.readPixels(0, 0, size.x(), size.y(), GL_RGBA, GL_FLOAT, pixels);
485 
486 	// validate pixels
487 	bool rendered = false;
488 	for (int x = 0; x < size.x(); ++x)
489 	{
490 		for (int y = 0; y < size.y(); ++y)
491 		{
492 			int mappedPixelPosition = y * size.x() + x;
493 
494 			tcu::Vec4 color(pixels[mappedPixelPosition * 4 + 0], pixels[mappedPixelPosition * 4 + 1],
495 							pixels[mappedPixelPosition * 4 + 2], pixels[mappedPixelPosition * 4 + 3]);
496 
497 			if (!validateColor(color, ignoredColor))
498 			{
499 				rendered = true;
500 				if (!validateColor(color, desiredColor))
501 				{
502 					delete[] pixels;
503 					return false;
504 				}
505 			}
506 		}
507 	}
508 
509 	delete[] pixels;
510 
511 	return rendered;
512 }
513 
ShaderAtomicCounterOpsTestBase(deqp::Context & context,const char * name,const char * description)514 ShaderAtomicCounterOpsTestBase::ShaderAtomicCounterOpsTestBase(deqp::Context& context, const char* name,
515 															   const char* description)
516 	: TestCase(context, name, description), m_atomicCounterBuffer(0), m_atomicCounterCallsBuffer(0)
517 {
518 	glu::ContextType contextType = m_context.getRenderContext().getType();
519 	m_contextSupportsGL46		 = glu::contextSupports(contextType, glu::ApiType::core(4, 6));
520 }
521 
init()522 void ShaderAtomicCounterOpsTestBase::init()
523 {
524 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
525 
526 	// generate atomic counter buffer
527 
528 	gl.genBuffers(1, &m_atomicCounterBuffer);
529 	GLU_EXPECT_NO_ERROR(gl.getError(), "genBuffers() call failed.");
530 
531 	gl.bindBuffer(GL_ATOMIC_COUNTER_BUFFER, m_atomicCounterBuffer);
532 	GLU_EXPECT_NO_ERROR(gl.getError(), "bindBuffer() call failed.");
533 
534 	gl.bufferData(GL_ATOMIC_COUNTER_BUFFER, sizeof(GLuint), NULL, GL_DYNAMIC_DRAW);
535 	GLU_EXPECT_NO_ERROR(gl.getError(), "bufferData() call failed.");
536 
537 	// generate atomic counter calls buffer
538 
539 	gl.genBuffers(1, &m_atomicCounterCallsBuffer);
540 	GLU_EXPECT_NO_ERROR(gl.getError(), "genBuffers() call failed.");
541 
542 	gl.bindBuffer(GL_ATOMIC_COUNTER_BUFFER, m_atomicCounterCallsBuffer);
543 	GLU_EXPECT_NO_ERROR(gl.getError(), "bindBuffer() call failed.");
544 
545 	gl.bufferData(GL_ATOMIC_COUNTER_BUFFER, sizeof(GLuint), NULL, GL_DYNAMIC_DRAW);
546 	GLU_EXPECT_NO_ERROR(gl.getError(), "bufferData() call failed.");
547 
548 	// setup tested atomic operations
549 
550 	setOperations();
551 
552 	// setup shaders
553 
554 	for (ShaderPipelineIter iter = m_shaderPipelines.begin(); iter != m_shaderPipelines.end(); ++iter)
555 	{
556 		iter->create(m_context);
557 	}
558 }
559 
deinit()560 void ShaderAtomicCounterOpsTestBase::deinit()
561 {
562 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
563 
564 	// delete atomic counter buffer
565 
566 	gl.deleteBuffers(1, &m_atomicCounterBuffer);
567 	GLU_EXPECT_NO_ERROR(gl.getError(), "deleteBuffers() call failed.");
568 
569 	// delete atomic counter calls buffer
570 
571 	gl.deleteBuffers(1, &m_atomicCounterCallsBuffer);
572 	GLU_EXPECT_NO_ERROR(gl.getError(), "deleteBuffers() call failed.");
573 
574 	// delete operations
575 
576 	for (AtomicOperationIter iter = m_operations.begin(); iter != m_operations.end(); ++iter)
577 	{
578 		delete *iter;
579 	}
580 }
581 
iterate()582 tcu::TestNode::IterateResult ShaderAtomicCounterOpsTestBase::iterate()
583 {
584 	if (!m_contextSupportsGL46)
585 	{
586 		if (!m_context.getContextInfo().isExtensionSupported("GL_ARB_shader_atomic_counters") ||
587 			!m_context.getContextInfo().isExtensionSupported("GL_ARB_shader_atomic_counter_ops"))
588 		{
589 			m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not supported");
590 			return STOP;
591 		}
592 	}
593 
594 	for (ShaderPipelineIter iter = m_shaderPipelines.begin(); iter != m_shaderPipelines.end(); ++iter)
595 	{
596 		fillAtomicCounterBuffer(iter->getAtomicOperation());
597 		bindBuffers();
598 		iter->test(m_context);
599 
600 		bool		operationValueValid = checkAtomicCounterBuffer(iter->getAtomicOperation());
601 		std::string operationFailMsg	= "Result of atomic operation was different than expected (" +
602 									   iter->getAtomicOperation()->getFunction() + ").";
603 		TCU_CHECK_MSG(operationValueValid, operationFailMsg.c_str());
604 
605 		bool		returnValueValid = validateScreenPixels(tcu::Vec4(1.0f), tcu::Vec4(0.5f));
606 		std::string returnFailMsg	= "Result of atomic operation return value was different than expected (" +
607 									iter->getAtomicOperation()->getFunction() + ").";
608 		TCU_CHECK_MSG(returnValueValid, returnFailMsg.c_str());
609 	}
610 
611 	m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
612 	return STOP;
613 }
614 
615 /** Constructor.
616 *
617 *  @param context Rendering context
618 */
ShaderAtomicCounterOpsAdditionSubstractionTestCase(deqp::Context & context)619 ShaderAtomicCounterOpsAdditionSubstractionTestCase::ShaderAtomicCounterOpsAdditionSubstractionTestCase(
620 	deqp::Context& context)
621 	: ShaderAtomicCounterOpsTestBase(
622 		  context, "ShaderAtomicCounterOpsAdditionSubstractionTestCase",
623 		  "Implements verification of new built-in addition and substraction atomic counter operations")
624 {
625 }
626 
setOperations()627 void ShaderAtomicCounterOpsAdditionSubstractionTestCase::setOperations()
628 {
629 	glw::GLuint input = 12;
630 	glw::GLuint param = 4;
631 
632 	addOperation(new AtomicOperationAdd(input, param));
633 	addOperation(new AtomicOperationSubtract(input, param));
634 }
635 
636 /** Constructor.
637 *
638 *  @param context Rendering context
639 */
ShaderAtomicCounterOpsMinMaxTestCase(deqp::Context & context)640 ShaderAtomicCounterOpsMinMaxTestCase::ShaderAtomicCounterOpsMinMaxTestCase(deqp::Context& context)
641 	: ShaderAtomicCounterOpsTestBase(
642 		  context, "ShaderAtomicCounterOpsMinMaxTestCase",
643 		  "Implements verification of new built-in minimum and maximum atomic counter operations")
644 {
645 }
646 
setOperations()647 void ShaderAtomicCounterOpsMinMaxTestCase::setOperations()
648 {
649 	glw::GLuint input	= 12;
650 	glw::GLuint params[] = { 4, 16 };
651 
652 	addOperation(new AtomicOperationMin(input, params[0]));
653 	addOperation(new AtomicOperationMin(input, params[1]));
654 	addOperation(new AtomicOperationMax(input, params[0]));
655 	addOperation(new AtomicOperationMax(input, params[1]));
656 }
657 
658 /** Constructor.
659 *
660 *  @param context Rendering context
661 */
ShaderAtomicCounterOpsBitwiseTestCase(deqp::Context & context)662 ShaderAtomicCounterOpsBitwiseTestCase::ShaderAtomicCounterOpsBitwiseTestCase(deqp::Context& context)
663 	: ShaderAtomicCounterOpsTestBase(context, "ShaderAtomicCounterOpsBitwiseTestCase",
664 									 "Implements verification of new built-in bitwise atomic counter operations")
665 {
666 }
667 
setOperations()668 void ShaderAtomicCounterOpsBitwiseTestCase::setOperations()
669 {
670 	glw::GLuint input = 0x2ED; // 0b1011101101;
671 	glw::GLuint param = 0x3A9; // 0b1110101001;
672 
673 	addOperation(new AtomicOperationAnd(input, param));
674 	addOperation(new AtomicOperationOr(input, param));
675 	addOperation(new AtomicOperationXor(input, param));
676 }
677 
678 /** Constructor.
679 *
680 *  @param context Rendering context
681 */
ShaderAtomicCounterOpsExchangeTestCase(deqp::Context & context)682 ShaderAtomicCounterOpsExchangeTestCase::ShaderAtomicCounterOpsExchangeTestCase(deqp::Context& context)
683 	: ShaderAtomicCounterOpsTestBase(
684 		  context, "ShaderAtomicCounterOpsExchangeTestCase",
685 		  "Implements verification of new built-in exchange and swap atomic counter operations")
686 {
687 }
688 
setOperations()689 void ShaderAtomicCounterOpsExchangeTestCase::setOperations()
690 {
691 	glw::GLuint input	 = 5;
692 	glw::GLuint param	 = 10;
693 	glw::GLuint compare[] = { 5, 20 };
694 
695 	addOperation(new AtomicOperationExchange(input, param));
696 	addOperation(new AtomicOperationCompSwap(input, param, compare[0]));
697 	addOperation(new AtomicOperationCompSwap(input, param, compare[1]));
698 }
699 
700 /** Constructor.
701 *
702 *  @param context Rendering context.
703 */
ShaderAtomicCounterOps(deqp::Context & context)704 ShaderAtomicCounterOps::ShaderAtomicCounterOps(deqp::Context& context)
705 	: TestCaseGroup(context, "shader_atomic_counter_ops_tests",
706 					"Verify conformance of CTS_ARB_shader_atomic_counter_ops implementation")
707 {
708 }
709 
710 /** Initializes the test group contents. */
init()711 void ShaderAtomicCounterOps::init()
712 {
713 	addChild(new ShaderAtomicCounterOpsAdditionSubstractionTestCase(m_context));
714 	addChild(new ShaderAtomicCounterOpsMinMaxTestCase(m_context));
715 	addChild(new ShaderAtomicCounterOpsBitwiseTestCase(m_context));
716 	addChild(new ShaderAtomicCounterOpsExchangeTestCase(m_context));
717 }
718 } /* gl4cts namespace */
719