1 /*-------------------------------------------------------------------------
2  * OpenGL Conformance Test Suite
3  * -----------------------------
4  *
5  * Copyright (c) 2015-2016 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
21  * \brief
22  */ /*-------------------------------------------------------------------*/
23 
24 /**
25  */ /*!
26  * \file  gl3cCullDistanceTests.cpp
27  * \brief Cull Distance Test Suite Implementation
28  */ /*-------------------------------------------------------------------*/
29 
30 #include "gl3cCullDistanceTests.hpp"
31 #include "gluContextInfo.hpp"
32 #include "gluDefs.hpp"
33 #include "gluStrUtil.hpp"
34 #include "glwEnums.hpp"
35 #include "glwFunctions.hpp"
36 #include "tcuRenderTarget.hpp"
37 #include "tcuTestLog.hpp"
38 
39 #include <cmath>
40 #include <sstream>
41 #include <string>
42 #include <vector>
43 
44 #ifndef GL_MAX_CULL_DISTANCES
45 #define GL_MAX_CULL_DISTANCES (0x82F9)
46 #endif
47 #ifndef GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES
48 #define GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES (0x82FA)
49 #endif
50 
51 namespace glcts
52 {
53 /** @brief Build OpenGL program
54  *
55  *  @param [in]  gl             OpenGL function bindings
56  *  @param [in]  testCtx        Context
57  *  @param [in]  cs_body        Compute shader source code
58  *  @param [in]  fs_body        Fragment shader source code
59  *  @param [in]  gs_body        Geometric shader source code
60  *  @param [in]  tc_body        Tessellation control shader source code
61  *  @param [in]  te_body        Tessellation evaluation shader source code
62  *  @param [in]  vs_body        Vertex shader source code
63  *  @param [in]  n_tf_varyings  Number of transform feedback varyings
64  *  @param [in]  tf_varyings    Transform feedback varyings names
65  *
66  *  @param [out] out_program    If succeeded output program GL handle, 0 otherwise.
67  */
buildProgram(const glw::Functions & gl,tcu::TestContext & testCtx,const glw::GLchar * cs_body,const glw::GLchar * fs_body,const glw::GLchar * gs_body,const glw::GLchar * tc_body,const glw::GLchar * te_body,const glw::GLchar * vs_body,const glw::GLuint & n_tf_varyings,const glw::GLchar ** tf_varyings,glw::GLuint * out_program)68 void CullDistance::Utilities::buildProgram(const glw::Functions& gl, tcu::TestContext& testCtx,
69 										   const glw::GLchar* cs_body, const glw::GLchar* fs_body,
70 										   const glw::GLchar* gs_body, const glw::GLchar* tc_body,
71 										   const glw::GLchar* te_body, const glw::GLchar* vs_body,
72 										   const glw::GLuint& n_tf_varyings, const glw::GLchar** tf_varyings,
73 										   glw::GLuint* out_program)
74 {
75 	glw::GLuint po_id = 0;
76 
77 	struct _shaders_configuration
78 	{
79 		glw::GLenum		   type;
80 		const glw::GLchar* body;
81 		glw::GLuint		   id;
82 	} shaders_configuration[] = { { GL_COMPUTE_SHADER, cs_body, 0 },		 { GL_FRAGMENT_SHADER, fs_body, 0 },
83 								  { GL_GEOMETRY_SHADER, gs_body, 0 },		 { GL_TESS_CONTROL_SHADER, tc_body, 0 },
84 								  { GL_TESS_EVALUATION_SHADER, te_body, 0 }, { GL_VERTEX_SHADER, vs_body, 0 } };
85 
86 	const glw::GLuint n_shaders_configuration = sizeof(shaders_configuration) / sizeof(shaders_configuration[0]);
87 
88 	/* Guard allocated OpenGL resources */
89 	try
90 	{
91 		/* Create needed programs */
92 		po_id = gl.createProgram();
93 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram() call failed.");
94 
95 		for (glw::GLuint n_shader_index = 0; n_shader_index < n_shaders_configuration; n_shader_index++)
96 		{
97 			if (shaders_configuration[n_shader_index].body != DE_NULL)
98 			{
99 				/* Generate shader object */
100 				shaders_configuration[n_shader_index].id = gl.createShader(shaders_configuration[n_shader_index].type);
101 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateShader() call failed");
102 
103 				glw::GLint		  compile_status = GL_FALSE;
104 				const glw::GLuint so_id			 = shaders_configuration[n_shader_index].id;
105 
106 				/* Assign shader source code */
107 				gl.shaderSource(shaders_configuration[n_shader_index].id, 1,		   /* count */
108 								&shaders_configuration[n_shader_index].body, DE_NULL); /* length */
109 				GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource() call failed");
110 
111 				gl.compileShader(so_id);
112 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader() call failed.");
113 
114 				gl.getShaderiv(so_id, GL_COMPILE_STATUS, &compile_status);
115 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv() call failed.");
116 
117 				if (compile_status == GL_FALSE)
118 				{
119 					std::vector<glw::GLchar> log_array(1);
120 					glw::GLint				 log_length = 0;
121 					std::string				 log_string("Failed to retrieve log");
122 
123 					/* Retrive compilation log length */
124 					gl.getShaderiv(so_id, GL_INFO_LOG_LENGTH, &log_length);
125 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv() call failed.");
126 
127 					log_array.resize(log_length + 1, 0);
128 
129 					gl.getShaderInfoLog(so_id, log_length, DE_NULL, &log_array[0]);
130 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog() call failed.");
131 
132 					log_string = std::string(&log_array[0]);
133 
134 					testCtx.getLog() << tcu::TestLog::Message << "Shader compilation has failed.\n"
135 									 << "Shader type: " << shaders_configuration[n_shader_index].type << "\n"
136 									 << "Shader compilation error log:\n"
137 									 << log_string << "\n"
138 									 << "Shader source code:\n"
139 									 << shaders_configuration[n_shader_index].body << "\n"
140 									 << tcu::TestLog::EndMessage;
141 
142 					TCU_FAIL("Shader compilation has failed.");
143 				}
144 
145 				/* Also attach the shader to the corresponding program object */
146 				gl.attachShader(po_id, so_id);
147 
148 				GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader() call failed");
149 			} /* if (shaders_configuration[n_shader_index].body != DE_NULL) */
150 		}	 /* for (all shader object IDs) */
151 
152 		/* Set transform feedback if requested */
153 		if (n_tf_varyings > 0)
154 		{
155 			gl.transformFeedbackVaryings(po_id, n_tf_varyings, tf_varyings, GL_INTERLEAVED_ATTRIBS);
156 			GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings() call failed");
157 		}
158 
159 		/* Try to link the program objects */
160 		if (po_id != 0)
161 		{
162 			glw::GLint link_status = GL_FALSE;
163 
164 			gl.linkProgram(po_id);
165 			GLU_EXPECT_NO_ERROR(gl.getError(), "glLinkProgram() call failed");
166 
167 			gl.getProgramiv(po_id, GL_LINK_STATUS, &link_status);
168 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv() call failed");
169 
170 			if (link_status == GL_FALSE)
171 			{
172 				std::vector<glw::GLchar> log_array(1);
173 				glw::GLsizei			 log_length = 0;
174 				std::string				 log_string;
175 
176 				/* Retreive compilation log length */
177 				gl.getProgramiv(po_id, GL_INFO_LOG_LENGTH, &log_length);
178 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv() call failed.");
179 
180 				log_array.resize(log_length + 1, 0);
181 
182 				/* Retreive compilation log */
183 				gl.getProgramInfoLog(po_id, log_length, DE_NULL, &log_array[0]);
184 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog() call failed.");
185 
186 				log_string = std::string(&log_array[0]);
187 
188 				/* Log linking error message */
189 				testCtx.getLog() << tcu::TestLog::Message << "Program linking has failed.\n"
190 								 << "Linking error log:\n"
191 								 << log_string << "\n"
192 								 << tcu::TestLog::EndMessage;
193 
194 				/* Log shader source code of shaders involved */
195 				for (glw::GLuint n_shader_index = 0; n_shader_index < n_shaders_configuration; n_shader_index++)
196 				{
197 					if (shaders_configuration[n_shader_index].body != DE_NULL)
198 					{
199 						testCtx.getLog() << tcu::TestLog::Message << "Shader source code of type "
200 										 << shaders_configuration[n_shader_index].type << " follows:\n"
201 										 << shaders_configuration[n_shader_index].body << "\n"
202 										 << tcu::TestLog::EndMessage;
203 					}
204 				}
205 
206 				TCU_FAIL("Program linking failed");
207 			}
208 		} /* if (po_id != 0) */
209 
210 		/* Delete all shaders we've created */
211 		for (glw::GLuint n_shader_index = 0; n_shader_index < n_shaders_configuration; n_shader_index++)
212 		{
213 			const glw::GLuint so_id = shaders_configuration[n_shader_index].id;
214 
215 			if (so_id != 0)
216 			{
217 				gl.deleteShader(so_id);
218 
219 				shaders_configuration[n_shader_index].id = 0;
220 
221 				GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteShader() call failed.");
222 			}
223 		}
224 
225 		/* Store the result progrtam IDs */
226 		*out_program = po_id;
227 	}
228 	catch (...)
229 	{
230 		/* Delete all shaders we've created */
231 		for (glw::GLuint n_shader_index = 0; n_shader_index < n_shaders_configuration; n_shader_index++)
232 		{
233 			const glw::GLuint so_id = shaders_configuration[n_shader_index].id;
234 
235 			if (so_id != 0)
236 			{
237 				gl.deleteShader(so_id);
238 
239 				shaders_configuration[n_shader_index].id = 0;
240 			}
241 		}
242 
243 		/* Delete the program object */
244 		if (po_id != 0)
245 		{
246 			gl.deleteProgram(po_id);
247 
248 			po_id = 0;
249 		}
250 
251 		/* Rethrow */
252 		throw;
253 	}
254 }
255 
256 /** @brief Replace all occurences of a substring in a string by a substring
257  *
258  *  @param [in,out] str    string to be edited
259  *  @param [in]     from   substring to be replaced
260  *  @param [out]    to     new substring
261  */
replaceAll(std::string & str,const std::string & from,const std::string & to)262 void CullDistance::Utilities::replaceAll(std::string& str, const std::string& from, const std::string& to)
263 {
264 	for (size_t start_pos = str.find(from, 0); start_pos != std::string::npos; start_pos = str.find(from, start_pos))
265 	{
266 		str.replace(start_pos, from.length(), to);
267 
268 		start_pos += to.length();
269 	}
270 
271 	return;
272 }
273 
274 /** @brief Convert integer to string representation
275  *
276  *  @param [in] integer     input integer to be converted
277  *
278  *  @return String representation of integer
279  */
intToString(glw::GLint integer)280 std::string CullDistance::Utilities::intToString(glw::GLint integer)
281 {
282 	std::stringstream temp_sstream;
283 
284 	temp_sstream << integer;
285 
286 	return temp_sstream.str();
287 }
288 
289 /** Constructor.
290  *
291  *  @param context Rendering context handle.
292  **/
APICoverageTest(deqp::Context & context)293 CullDistance::APICoverageTest::APICoverageTest(deqp::Context& context)
294 	: TestCase(context, "coverage", "Cull Distance API Coverage Test")
295 	, m_bo_id(0)
296 	, m_cs_id(0)
297 	, m_cs_to_id(0)
298 	, m_fbo_draw_id(0)
299 	, m_fbo_draw_to_id(0)
300 	, m_fbo_read_id(0)
301 	, m_fs_id(0)
302 	, m_gs_id(0)
303 	, m_po_id(0)
304 	, m_tc_id(0)
305 	, m_te_id(0)
306 	, m_vao_id(0)
307 	, m_vs_id(0)
308 {
309 	/* Left blank on purpose */
310 }
311 
312 /** @brief Cull Distance API Coverage Test deinitialization */
deinit()313 void CullDistance::APICoverageTest::deinit()
314 {
315 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
316 
317 	if (m_bo_id != 0)
318 	{
319 		gl.deleteBuffers(1, &m_bo_id);
320 
321 		m_bo_id = 0;
322 	}
323 
324 	if (m_cs_id != 0)
325 	{
326 		gl.deleteShader(m_cs_id);
327 
328 		m_cs_id = 0;
329 	}
330 
331 	if (m_cs_to_id != 0)
332 	{
333 		gl.deleteTextures(1, &m_cs_to_id);
334 
335 		m_cs_to_id = 0;
336 	}
337 
338 	if (m_fbo_draw_id != 0)
339 	{
340 		gl.deleteFramebuffers(1, &m_fbo_draw_id);
341 
342 		m_fbo_draw_id = 0;
343 	}
344 
345 	if (m_fbo_draw_to_id != 0)
346 	{
347 		gl.deleteTextures(1, &m_fbo_draw_to_id);
348 
349 		m_fbo_draw_to_id = 0;
350 	}
351 
352 	if (m_fbo_read_id != 0)
353 	{
354 		gl.deleteFramebuffers(1, &m_fbo_read_id);
355 
356 		m_fbo_read_id = 0;
357 	}
358 
359 	if (m_fs_id != 0)
360 	{
361 		gl.deleteShader(m_fs_id);
362 
363 		m_fs_id = 0;
364 	}
365 
366 	if (m_gs_id != 0)
367 	{
368 		gl.deleteShader(m_gs_id);
369 
370 		m_gs_id = 0;
371 	}
372 
373 	if (m_po_id != 0)
374 	{
375 		gl.deleteProgram(m_po_id);
376 
377 		m_po_id = 0;
378 	}
379 
380 	if (m_tc_id != 0)
381 	{
382 		gl.deleteShader(m_tc_id);
383 
384 		m_tc_id = 0;
385 	}
386 
387 	if (m_te_id != 0)
388 	{
389 		gl.deleteShader(m_te_id);
390 
391 		m_te_id = 0;
392 	}
393 
394 	if (m_vao_id != 0)
395 	{
396 		gl.deleteVertexArrays(1, &m_vao_id);
397 
398 		m_vao_id = 0;
399 	}
400 
401 	if (m_vs_id != 0)
402 	{
403 		gl.deleteShader(m_vs_id);
404 
405 		m_vs_id = 0;
406 	}
407 
408 	/* Restore default pack alignment value */
409 	gl.pixelStorei(GL_PACK_ALIGNMENT, 4);
410 }
411 
412 /** Executes test iteration.
413  *
414  *  @return Returns STOP when test has finished executing, CONTINUE if more iterations are needed.
415  */
iterate()416 tcu::TestNode::IterateResult CullDistance::APICoverageTest::iterate()
417 {
418 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
419 
420 	/* This test should only be executed if ARB_cull_distance is supported, or if
421 	 * we're running a GL4.5 context
422 	 */
423 	if (!m_context.getContextInfo().isExtensionSupported("GL_ARB_cull_distance") &&
424 		!glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)))
425 	{
426 		throw tcu::NotSupportedError("GL_ARB_cull_distance is not supported");
427 	}
428 
429 	/* Check that calling GetIntegerv with MAX_CULL_DISTANCES doesn't generate
430 	 * any errors and returns a value at least 8.
431 	 *
432 	 * Check that calling GetIntegerv with MAX_COMBINED_CLIP_AND_CULL_DISTANCES
433 	 * doesn't generate any errors and returns a value at least 8.
434 	 *
435 	 */
436 	glw::GLint error_code									 = GL_NO_ERROR;
437 	glw::GLint gl_max_cull_distances_value					 = 0;
438 	glw::GLint gl_max_combined_clip_and_cull_distances_value = 0;
439 
440 	gl.getIntegerv(GL_MAX_CULL_DISTANCES, &gl_max_cull_distances_value);
441 
442 	error_code = gl.getError();
443 	if (error_code != GL_NO_ERROR)
444 	{
445 		m_testCtx.getLog() << tcu::TestLog::Message << "glGetIntegerv() returned error code "
446 						   << "[" << glu::getErrorStr(error_code) << "] for GL_MAX_CULL_DISTANCES"
447 																	 " query instead of GL_NO_ERROR"
448 						   << tcu::TestLog::EndMessage;
449 
450 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
451 
452 		return STOP;
453 	}
454 
455 	gl.getIntegerv(GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES, &gl_max_combined_clip_and_cull_distances_value);
456 
457 	error_code = gl.getError();
458 	if (error_code != GL_NO_ERROR)
459 	{
460 		m_testCtx.getLog() << tcu::TestLog::Message << "glGetIntegerv() returned error code "
461 						   << "[" << glu::getErrorStr(error_code) << "] for "
462 																	 "GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES query "
463 																	 "instead of GL_NO_ERROR"
464 						   << tcu::TestLog::EndMessage;
465 
466 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
467 
468 		return STOP;
469 	}
470 
471 	/* Before we proceed with the two other tests, initialize a buffer & a texture
472 	 * object we will need to capture data from the programs */
473 	static const glw::GLuint bo_size = sizeof(int) * 4 /* components */ * 4 /* result points */;
474 
475 	gl.genBuffers(1, &m_bo_id);
476 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers() call failed.");
477 
478 	gl.genFramebuffers(1, &m_fbo_draw_id);
479 	gl.genFramebuffers(1, &m_fbo_read_id);
480 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenFramebuffers() call(s) failed.");
481 
482 	gl.genTextures(1, &m_cs_to_id);
483 	gl.genTextures(1, &m_fbo_draw_to_id);
484 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() call failed.");
485 
486 	gl.genVertexArrays(1, &m_vao_id);
487 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays() call failed.");
488 
489 	gl.bindVertexArray(m_vao_id);
490 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray() call failed.");
491 
492 	gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, m_bo_id);
493 	gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, /* index */
494 					  m_bo_id);
495 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer() or glBindBufferBase() call(s) failed.");
496 
497 	gl.bufferData(GL_TRANSFORM_FEEDBACK_BUFFER, bo_size, DE_NULL, GL_STATIC_DRAW);
498 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData() call failed.");
499 
500 	for (glw::GLuint n_to_id = 0; n_to_id < 2; /* CS, FBO */ ++n_to_id)
501 	{
502 		gl.bindTexture(GL_TEXTURE_2D, (n_to_id == 0) ? m_cs_to_id : m_fbo_draw_to_id);
503 		GLU_EXPECT_NO_ERROR(gl.getError(), "glBindTexture() call failed.");
504 
505 		gl.texStorage2D(GL_TEXTURE_2D, 1, /* levels */
506 						GL_R32I, 1,		  /* width */
507 						1);				  /* height */
508 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTexStorage2D() call failed.");
509 	}
510 
511 	if (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 3)) ||
512 	    m_context.getContextInfo().isExtensionSupported("GL_ARB_compute_shader")) {
513 		gl.bindImageTexture(0,			   /* unit */
514 				    m_cs_to_id, 0, /* level */
515 				    GL_FALSE,	  /* layered */
516 				    0,			   /* layer */
517 				    GL_WRITE_ONLY, GL_R32I);
518                 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindImageTexture() call failed.");
519 	}
520 
521 	gl.bindFramebuffer(GL_DRAW_FRAMEBUFFER, m_fbo_draw_id);
522 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindFramebuffer() call failed.");
523 
524 	gl.framebufferTexture2D(GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, m_fbo_draw_to_id, 0); /* level */
525 	GLU_EXPECT_NO_ERROR(gl.getError(), "glFramebufferTexture2D() call failed.");
526 
527 	gl.bindFramebuffer(GL_READ_FRAMEBUFFER, m_fbo_read_id);
528 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindFramebuffer() call failed.");
529 
530 	gl.viewport(0,  /* x */
531 				0,  /* y */
532 				1,  /* width */
533 				1); /* height */
534 	GLU_EXPECT_NO_ERROR(gl.getError(), "glViewport() call failed.");
535 
536 	gl.pixelStorei(GL_PACK_ALIGNMENT, 1);
537 	GLU_EXPECT_NO_ERROR(gl.getError(), "glPixelStorei() call failed.");
538 
539 	/* There are two new GL constants, where value we need to verify */
540 	struct _run
541 	{
542 		const glw::GLchar* essl_token_value;
543 		glw::GLenum		   gl_enum;
544 		glw::GLint		   gl_value;
545 		glw::GLint		   min_value;
546 		const glw::GLchar* name;
547 	} runs[] = { { "gl_MaxCullDistances", GL_MAX_CULL_DISTANCES, gl_max_cull_distances_value, 8 /*minimum required */,
548 				   "GL_MAX_CULL_DISTANCES" },
549 				 { "gl_MaxCombinedClipAndCullDistances", GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES,
550 				   gl_max_combined_clip_and_cull_distances_value, 8 /*minimum required */,
551 				   "GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES" } };
552 
553 	static const glw::GLuint n_runs = sizeof(runs) / sizeof(runs[0]);
554 
555 	for (glw::GLuint n_run = 0; n_run < n_runs; ++n_run)
556 	{
557 		_run& current_run = runs[n_run];
558 
559 		static const struct _stage
560 		{
561 			bool use_cs;
562 			bool use_fs;
563 			bool use_gs;
564 			bool use_tc;
565 			bool use_te;
566 			bool use_vs;
567 
568 			const glw::GLchar* fs_input;
569 			const glw::GLchar* gs_input;
570 			const glw::GLchar* tc_input;
571 			const glw::GLchar* te_input;
572 
573 			const glw::GLchar* tf_output_name;
574 			const glw::GLenum  tf_mode;
575 
576 			glw::GLenum draw_call_mode;
577 			glw::GLuint n_draw_call_vertices;
578 		} stages[] = { /* CS only test */
579 					   {
580 						   /* use_cs|use_fs|use_gs|use_tc|use_te|use_vs */
581 						   true, false, false, false, false, false,
582 
583 						   NULL,	/* fs_input             */
584 						   NULL,	/* gs_input             */
585 						   NULL,	/* tc_input             */
586 						   NULL,	/* te_input             */
587 						   NULL,	/* tf_output_name       */
588 						   GL_NONE, /* tf_mode              */
589 						   GL_NONE, /* draw_call_mode       */
590 						   0,		/* n_draw_call_vertices */
591 					   },
592 					   /* VS+GS+TC+TE+FS test */
593 					   {
594 						   /* use_cs|use_fs|use_gs|use_tc|use_te|use_vs */
595 						   false, true, true, true, true, true,
596 
597 						   "out_gs",	 /* fs_input             */
598 						   "out_te",	 /* gs_input             */
599 						   "out_vs",	 /* tc_input             */
600 						   "out_tc",	 /* te_input             */
601 						   "out_gs",	 /* tf_output_name       */
602 						   GL_TRIANGLES, /* tf_mode              */
603 						   GL_PATCHES,   /* draw_call_mode       */
604 						   3,			 /* n_draw_call_vertices */
605 					   },
606 					   /* VS+GS+FS test */
607 					   {
608 						   /* use_cs|use_fs|use_gs|use_tc|use_te|use_vs */
609 						   false, true, true, false, false, true,
610 
611 						   "out_gs",	 /* fs_input             */
612 						   "out_vs",	 /* gs_input             */
613 						   NULL,		 /* tc_input             */
614 						   NULL,		 /* te_input             */
615 						   "out_gs",	 /* tf_output_name       */
616 						   GL_TRIANGLES, /* tf_mode              */
617 						   GL_POINTS,	/* draw_call_mode       */
618 						   1,			 /* n_draw_call_vertices */
619 					   },
620 					   /* VS+TC+TE+FS test */
621 					   {
622 						   /* use_cs|use_fs|use_gs|use_tc|use_te|use_vs */
623 						   false, true, false, true, true, true,
624 
625 						   "out_te",   /* fs_input             */
626 						   NULL,	   /* gs_input             */
627 						   "out_vs",   /* tc_input             */
628 						   "out_tc",   /* te_input             */
629 						   "out_te",   /* tf_output_name       */
630 						   GL_POINTS,  /* tf_mode              */
631 						   GL_PATCHES, /* draw_call_mode       */
632 						   3		   /* n_draw_call_vertices */
633 					   },
634 					   /* VS test */
635 					   {
636 						   /* use_cs|use_fs|use_gs|use_tc|use_te|use_vs */
637 						   false, false, false, false, false, true,
638 
639 						   "out_vs",  /* fs_input             */
640 						   NULL,	  /* gs_input             */
641 						   NULL,	  /* tc_input             */
642 						   NULL,	  /* te_input             */
643 						   "out_vs",  /* tf_output_name       */
644 						   GL_POINTS, /* tf_mode              */
645 						   GL_POINTS, /* draw_call_mode       */
646 						   1		  /* n_draw_call_vertices */
647 					   }
648 		};
649 		const glw::GLuint n_stages = sizeof(stages) / sizeof(stages[0]);
650 
651 		/* Run through all test stages */
652 		for (glw::GLuint n_stage = 0; n_stage < n_stages; ++n_stage)
653 		{
654 			/* Check for OpenGL feature support */
655 			if (stages[n_stage].use_cs)
656 			{
657 				if (!glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 3)) &&
658 					!m_context.getContextInfo().isExtensionSupported("GL_ARB_compute_shader"))
659 				{
660 					continue; // no compute shader support
661 				}
662 			}
663 			if (stages[n_stage].use_tc || stages[n_stage].use_te)
664 			{
665 				if (!glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 0)) &&
666 					!m_context.getContextInfo().isExtensionSupported("GL_ARB_tessellation_shader"))
667 				{
668 					continue; // no tessellation shader support
669 				}
670 			}
671 
672 			/* Check that use of the GLSL built-in constant gl_MaxCullDistance in any
673 			 * shader stage (including compute shader) does not affect the shader
674 			 * compilation & program linking process.
675 			 */
676 			static const glw::GLchar* cs_body_template =
677 				"#version 150\n"
678 				"\n"
679 				"#extension GL_ARB_compute_shader          : require\n"
680 				"#extension GL_ARB_cull_distance           : require\n"
681 				"#extension GL_ARB_shader_image_load_store : require\n"
682 				"\n"
683 				"layout(local_size_x = 1, local_size_y = 1, local_size_z = 1) in;\n"
684 				"\n"
685 				"layout(r32i) uniform writeonly iimage2D result;\n"
686 				"\n"
687 				"void main()\n"
688 				"{\n"
689 				"    imageStore(result, ivec2(0),ivec4(TOKEN) );\n"
690 				"}\n";
691 			std::string cs_body = cs_body_template;
692 
693 			static const glw::GLchar* fs_body_template = "#version 150\n"
694 														 "\n"
695 														 "#extension GL_ARB_cull_distance : require\n"
696 														 "\n"
697 														 "flat in  int INPUT_FS_NAME;\n"
698 														 "out int out_fs;\n"
699 														 "\n"
700 														 "void main()\n"
701 														 "{\n"
702 														 "    if (INPUT_FS_NAME == TOKEN)\n"
703 														 "    {\n"
704 														 "        out_fs = TOKEN;\n"
705 														 "    }\n"
706 														 "    else\n"
707 														 "    {\n"
708 														 "        out_fs = -1;\n"
709 														 "    }\n"
710 														 "}\n";
711 			std::string fs_body = fs_body_template;
712 
713 			static const glw::GLchar* gs_body_template =
714 				"#version 150\n"
715 				"\n"
716 				"#extension GL_ARB_cull_distance : require\n"
717 				"\n"
718 				"flat in  int INPUT_GS_NAME[];\n"
719 				"flat out int out_gs;\n"
720 				"\n"
721 				"layout(points)                           in;\n"
722 				"layout(triangle_strip, max_vertices = 4) out;\n"
723 				"\n"
724 				"void main()\n"
725 				"{\n"
726 				"    int result_value = (INPUT_GS_NAME[0] == TOKEN) ? TOKEN : -1;\n"
727 				"\n"
728 				/* Draw a full-screen quad */
729 				"    gl_Position = vec4(-1.0, 1.0, 0.0, 1.0);\n"
730 				"    out_gs      = result_value;\n"
731 				"    EmitVertex();\n"
732 				"\n"
733 				"    gl_Position = vec4(-1.0, -1.0, 0.0, 1.0);\n"
734 				"    out_gs      = result_value;\n"
735 				"    EmitVertex();\n"
736 				"\n"
737 				"    gl_Position = vec4(1.0, 1.0, 0.0, 1.0);\n"
738 				"    out_gs      = result_value;\n"
739 				"    EmitVertex();\n"
740 				"\n"
741 				"    gl_Position = vec4(1.0, -1.0, 0.0, 1.0);\n"
742 				"    out_gs      = result_value;\n"
743 				"    EmitVertex();\n"
744 				"    EndPrimitive();\n"
745 				"}\n";
746 			std::string gs_body = gs_body_template;
747 
748 			static const glw::GLchar* tc_body_template =
749 				"#version 150\n"
750 				"\n"
751 				"#extension GL_ARB_cull_distance : require\n"
752 				"#extension GL_ARB_tessellation_shader : require\n"
753 				"\n"
754 				"layout(vertices = 1) out;\n"
755 				"\n"
756 				"flat in  int INPUT_TC_NAME[];\n"
757 				"flat out int out_tc       [];\n"
758 				"\n"
759 				"void main()\n"
760 				"{\n"
761 				"    int result_value = (INPUT_TC_NAME[0] == TOKEN) ? TOKEN : -1;\n"
762 				"\n"
763 				"    out_tc[gl_InvocationID]             = result_value;\n"
764 				"    gl_out[gl_InvocationID].gl_Position = gl_in[gl_InvocationID].gl_Position;\n"
765 				"    gl_TessLevelInner[0]                = 1.0;\n"
766 				"    gl_TessLevelInner[1]                = 1.0;\n"
767 				"    gl_TessLevelOuter[0]                = 1.0;\n"
768 				"    gl_TessLevelOuter[1]                = 1.0;\n"
769 				"    gl_TessLevelOuter[2]                = 1.0;\n"
770 				"    gl_TessLevelOuter[3]                = 1.0;\n"
771 				"}\n";
772 			std::string tc_body = tc_body_template;
773 
774 			static const glw::GLchar* te_body_template =
775 				"#version 150\n"
776 				"\n"
777 				"#extension GL_ARB_cull_distance : require\n"
778 				"#extension GL_ARB_tessellation_shader : require\n"
779 				"\n"
780 				"flat in  int INPUT_TE_NAME[];\n"
781 				"flat out int out_te;\n"
782 				"\n"
783 				"layout(isolines, point_mode) in;\n"
784 				"\n"
785 				"void main()\n"
786 				"{\n"
787 				"    int result_value = (INPUT_TE_NAME[0] == TOKEN) ? TOKEN : 0;\n"
788 				"\n"
789 				"    out_te = result_value;\n"
790 				"\n"
791 				"    gl_Position = vec4(0.0, 0.0, 0.0, 1.);\n"
792 				"}\n";
793 			std::string te_body = te_body_template;
794 
795 			static const glw::GLchar* vs_body_template = "#version 150\n"
796 														 "\n"
797 														 "#extension GL_ARB_cull_distance : require\n"
798 														 "\n"
799 														 "flat out int out_vs;\n"
800 														 "\n"
801 														 "void main()\n"
802 														 "{\n"
803 														 "    gl_Position = vec4(0.0, 0.0, 0.0, 1.0);\n"
804 														 "    out_vs      = TOKEN;\n"
805 														 "}\n";
806 			std::string vs_body = vs_body_template;
807 
808 			const _stage& current_stage = stages[n_stage];
809 
810 			/* Build shader bodies */
811 			struct _shader_body
812 			{
813 				std::string* body_ptr;
814 				glw::GLenum  gl_type;
815 			} shader_bodies[] = { { &cs_body, GL_COMPUTE_SHADER },		   { &fs_body, GL_FRAGMENT_SHADER },
816 								  { &gs_body, GL_GEOMETRY_SHADER },		   { &tc_body, GL_TESS_CONTROL_SHADER },
817 								  { &te_body, GL_TESS_EVALUATION_SHADER }, { &vs_body, GL_VERTEX_SHADER } };
818 			static const glw::GLchar* input_fs_token_string = "INPUT_FS_NAME";
819 			static const glw::GLchar* input_gs_token_string = "INPUT_GS_NAME";
820 			static const glw::GLchar* input_te_token_string = "INPUT_TE_NAME";
821 			static const glw::GLchar* input_tc_token_string = "INPUT_TC_NAME";
822 			static const glw::GLuint  n_shader_bodies		= sizeof(shader_bodies) / sizeof(shader_bodies[0]);
823 
824 			std::size_t				  token_position = std::string::npos;
825 			static const glw::GLchar* token_string   = "TOKEN";
826 
827 			for (glw::GLuint n_shader_body = 0; n_shader_body < n_shader_bodies; ++n_shader_body)
828 			{
829 				_shader_body& current_body = shader_bodies[n_shader_body];
830 
831 				/* Is this stage actually used? */
832 				if (((current_body.gl_type == GL_COMPUTE_SHADER) && (!current_stage.use_cs)) ||
833 					((current_body.gl_type == GL_FRAGMENT_SHADER) && (!current_stage.use_fs)) ||
834 					((current_body.gl_type == GL_TESS_CONTROL_SHADER) && (!current_stage.use_tc)) ||
835 					((current_body.gl_type == GL_TESS_EVALUATION_SHADER) && (!current_stage.use_te)) ||
836 					((current_body.gl_type == GL_VERTEX_SHADER) && (!current_stage.use_vs)))
837 				{
838 					/* Skip the iteration. */
839 					continue;
840 				}
841 
842 				/* Iterate over all token and replace them with stage-specific values */
843 				struct _token_value_pair
844 				{
845 					const glw::GLchar* token;
846 					const glw::GLchar* value;
847 				} token_value_pairs[] = {
848 					/* NOTE: The last entry is filled by the switch() block below */
849 					{ token_string, current_run.essl_token_value },
850 					{ NULL, NULL },
851 				};
852 
853 				const size_t n_token_value_pairs = sizeof(token_value_pairs) / sizeof(token_value_pairs[0]);
854 
855 				switch (current_body.gl_type)
856 				{
857 				case GL_COMPUTE_SHADER:
858 				case GL_VERTEX_SHADER:
859 					break;
860 
861 				case GL_FRAGMENT_SHADER:
862 				{
863 					token_value_pairs[1].token = input_fs_token_string;
864 					token_value_pairs[1].value = current_stage.fs_input;
865 
866 					break;
867 				}
868 
869 				case GL_GEOMETRY_SHADER:
870 				{
871 					token_value_pairs[1].token = input_gs_token_string;
872 					token_value_pairs[1].value = current_stage.gs_input;
873 
874 					break;
875 				}
876 
877 				case GL_TESS_CONTROL_SHADER:
878 				{
879 					token_value_pairs[1].token = input_tc_token_string;
880 					token_value_pairs[1].value = current_stage.tc_input;
881 
882 					break;
883 				}
884 
885 				case GL_TESS_EVALUATION_SHADER:
886 				{
887 					token_value_pairs[1].token = input_te_token_string;
888 					token_value_pairs[1].value = current_stage.te_input;
889 
890 					break;
891 				}
892 
893 				default:
894 					TCU_FAIL("Unrecognized shader body type");
895 				}
896 
897 				for (glw::GLuint n_pair = 0; n_pair < n_token_value_pairs; ++n_pair)
898 				{
899 					const _token_value_pair& current_pair = token_value_pairs[n_pair];
900 
901 					if (current_pair.token == NULL || current_pair.value == NULL)
902 					{
903 						continue;
904 					}
905 
906 					while ((token_position = current_body.body_ptr->find(current_pair.token)) != std::string::npos)
907 					{
908 						current_body.body_ptr->replace(token_position, strlen(current_pair.token), current_pair.value);
909 					}
910 				} /* for (all token+value pairs) */
911 			}	 /* for (all sader bodies) */
912 
913 			/* Build the test program */
914 			CullDistance::Utilities::buildProgram(
915 				gl, m_testCtx, current_stage.use_cs ? cs_body.c_str() : DE_NULL,
916 				current_stage.use_fs ? fs_body.c_str() : DE_NULL, current_stage.use_gs ? gs_body.c_str() : DE_NULL,
917 				current_stage.use_tc ? tc_body.c_str() : DE_NULL, current_stage.use_te ? te_body.c_str() : DE_NULL,
918 				current_stage.use_vs ? vs_body.c_str() : DE_NULL, (current_stage.tf_output_name != NULL) ? 1 : 0,
919 				(const glw::GLchar**)&current_stage.tf_output_name, &m_po_id);
920 
921 			/* Bind the test program */
922 			DE_ASSERT(m_po_id != 0);
923 
924 			gl.useProgram(m_po_id);
925 			GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram() call failed.");
926 
927 			/* Execute the draw call. Transform Feed-back should be enabled for all iterations
928 			 * par the CS one, since we use a different tool to capture the result data in the
929 			 * latter case.
930 			 */
931 			if (!current_stage.use_cs)
932 			{
933 				gl.beginTransformFeedback(current_stage.tf_mode);
934 				GLU_EXPECT_NO_ERROR(gl.getError(), "glBeginTransformFeedback() call failed.");
935 
936 				gl.drawArrays(current_stage.draw_call_mode, 0,	 /* first */
937 							  current_stage.n_draw_call_vertices); /* count */
938 				GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawArrays() call failed.");
939 
940 				gl.endTransformFeedback();
941 				GLU_EXPECT_NO_ERROR(gl.getError(), "glEndTransformFeedback() call failed.");
942 			} /* if (uses_tf) */
943 			else
944 			{
945 				gl.dispatchCompute(1,  /* num_groups_x */
946 								   1,  /* num_groups_y */
947 								   1); /* num_groups_z */
948 				GLU_EXPECT_NO_ERROR(gl.getError(), "glDispatchCompute() call failed.");
949 			}
950 
951 			/* Verify the result values */
952 			if (!current_stage.use_cs)
953 			{
954 				glw::GLint* result_data_ptr = DE_NULL;
955 
956 				/* Retrieve the data captured by Transform Feedback */
957 				result_data_ptr = (glw::GLint*)gl.mapBufferRange(GL_TRANSFORM_FEEDBACK_BUFFER, 0, /* offset */
958 																 sizeof(unsigned int) * 1, GL_MAP_READ_BIT);
959 				GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBufferRange() call failed.");
960 
961 				if (*result_data_ptr != current_run.gl_value)
962 				{
963 					m_testCtx.getLog() << tcu::TestLog::Message << current_run.name << " value "
964 																					   "["
965 									   << *result_data_ptr << "]"
966 															  " does not match the one reported by glGetIntegerv() "
967 															  "["
968 									   << current_run.gl_value << "]" << tcu::TestLog::EndMessage;
969 
970 					TCU_FAIL("GL constant value does not match the ES SL equivalent");
971 				}
972 
973 				if (*result_data_ptr < current_run.min_value)
974 				{
975 					m_testCtx.getLog() << tcu::TestLog::Message << current_run.name << " value "
976 																					   "["
977 									   << *result_data_ptr << "]"
978 															  " does not meet the minimum specification requirements "
979 															  "["
980 									   << current_run.min_value << "]" << tcu::TestLog::EndMessage;
981 
982 					TCU_FAIL("GL constant value does not meet minimum specification requirements");
983 				}
984 
985 				gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
986 				GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer() call failed.");
987 			}
988 
989 			for (glw::GLuint n_stage_internal = 0; n_stage_internal < 2; /* CS, FS write to separate textures */
990 				 ++n_stage_internal)
991 			{
992 				glw::GLuint to_id = (n_stage_internal == 0) ? m_cs_to_id : m_fbo_draw_to_id;
993 
994 				if (((n_stage_internal == 0) && (!current_stage.use_cs)) ||
995 					((n_stage_internal == 1) && (!current_stage.use_fs)))
996 				{
997 					/* Skip the iteration */
998 					continue;
999 				}
1000 
1001 				/* Check the image data the test CS / FS should have written */
1002 				glw::GLint result_value = 0;
1003 
1004 				gl.framebufferTexture2D(GL_READ_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, to_id, 0); /* level */
1005 				GLU_EXPECT_NO_ERROR(gl.getError(), "glFramebufferTexture2D() call failed.");
1006 
1007 				/* NOTE: We're using our custom read framebuffer here, so we'll be reading
1008 				 *       from the texture, that the writes have been issued to earlier. */
1009 				gl.finish();
1010 				GLU_EXPECT_NO_ERROR(gl.getError(), "glMemoryBarrier() call failed.");
1011 
1012 				gl.readPixels(0, /* x */
1013 							  0, /* y */
1014 							  1, /* width */
1015 							  1, /* height */
1016 							  GL_RED_INTEGER, GL_INT, &result_value);
1017 				GLU_EXPECT_NO_ERROR(gl.getError(), "glReadPixels() call failed.");
1018 
1019 				if (result_value != current_run.gl_value)
1020 				{
1021 					m_testCtx.getLog() << tcu::TestLog::Message << current_run.name
1022 									   << " value accessible to the compute / fragment shader "
1023 										  "["
1024 									   << result_value << "]"
1025 														  " does not match the one reported by glGetIntegerv() "
1026 														  "["
1027 									   << current_run.gl_value << "]" << tcu::TestLog::EndMessage;
1028 
1029 					TCU_FAIL("GL constant value does not match the ES SL equivalent");
1030 				}
1031 
1032 				if (result_value < current_run.min_value)
1033 				{
1034 					m_testCtx.getLog() << tcu::TestLog::Message << current_run.name
1035 									   << " value accessible to the compute / fragment shader "
1036 										  "["
1037 									   << result_value << "]"
1038 														  " does not meet the minimum specification requirements "
1039 														  "["
1040 									   << current_run.min_value << "]" << tcu::TestLog::EndMessage;
1041 
1042 					TCU_FAIL("GL constant value does not meet minimum specification requirements");
1043 				}
1044 			}
1045 
1046 			/* Clear the data buffer before we continue */
1047 			static const glw::GLubyte bo_clear_data[bo_size] = { 0 };
1048 
1049 			gl.bufferSubData(GL_TRANSFORM_FEEDBACK_BUFFER, 0, /* offset */
1050 							 bo_size, bo_clear_data);
1051 			GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferSubData() call failed.");
1052 
1053 			/* Clear the texture mip-map before we continue */
1054 			glw::GLint clear_values[4] = { 0, 0, 0, 0 };
1055 
1056 			gl.clearBufferiv(GL_COLOR, 0, /* drawbuffer */
1057 							 clear_values);
1058 			GLU_EXPECT_NO_ERROR(gl.getError(), "glClearBufferiv() call failed.");
1059 
1060 			/* Release program before we move on to the next iteration */
1061 			if (m_po_id != 0)
1062 			{
1063 				gl.deleteProgram(m_po_id);
1064 
1065 				m_po_id = 0;
1066 			}
1067 		} /* for (all stages) */
1068 	}	 /* for (both runs) */
1069 
1070 	/* All done */
1071 	m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
1072 
1073 	return STOP;
1074 }
1075 
1076 /** Constructor.
1077  *
1078  *  @param context Rendering context handle.
1079  **/
FunctionalTest(deqp::Context & context)1080 CullDistance::FunctionalTest::FunctionalTest(deqp::Context& context)
1081 	: TestCase(context, "functional", "Cull Distance Functional Test")
1082 	, m_bo_data()
1083 	, m_bo_id(0)
1084 	, m_fbo_id(0)
1085 	, m_po_id(0)
1086 	, m_render_primitives(0)
1087 	, m_render_vertices(0)
1088 	, m_sub_grid_cell_size(0)
1089 	, m_to_id(0)
1090 	, m_vao_id(0)
1091 	, m_to_height(512)
1092 	, m_to_width(512)
1093 	, m_to_pixel_data_cache()
1094 {
1095 	/* Left blank on purpose */
1096 }
1097 
1098 /** @brief Build OpenGL program for functional tests
1099  *
1100  *  @param [in]  clipdistances_array_size   use size of gl_ClipDistance array
1101  *  @param [in]  culldistances_array_size   use size of gl_CullDistance array
1102  *  @param [in]  dynamic_index_writes       use dunamic indexing for setting  the gl_ClipDistance and gl_CullDistance arrays
1103  *  @param [in]  primitive_mode             primitive_mode will be used for rendering
1104  *  @param [in]  redeclare_clipdistances    redeclare gl_ClipDistance
1105  *  @param [in]  redeclare_culldistances    redeclare gl_CullDistance
1106  *  @param [in]  use_core_functionality     use core OpenGL functionality
1107  *  @param [in]  use_gs                     use geometry shader
1108  *  @param [in]  use_ts                     use tessellation shader
1109  *  @param [in]  fetch_culldistance_from_fs fetch check sum of gl_ClipDistance and gl_CullDistance from fragment shader
1110  */
buildPO(glw::GLuint clipdistances_array_size,glw::GLuint culldistances_array_size,bool dynamic_index_writes,_primitive_mode primitive_mode,bool redeclare_clipdistances,bool redeclare_culldistances,bool use_core_functionality,bool use_gs,bool use_ts,bool fetch_culldistance_from_fs)1111 void CullDistance::FunctionalTest::buildPO(glw::GLuint clipdistances_array_size, glw::GLuint culldistances_array_size,
1112 										   bool dynamic_index_writes, _primitive_mode primitive_mode,
1113 										   bool redeclare_clipdistances, bool redeclare_culldistances,
1114 										   bool use_core_functionality, bool use_gs, bool use_ts,
1115 										   bool fetch_culldistance_from_fs)
1116 {
1117 	deinitPO();
1118 
1119 	/* Form the vertex shader */
1120 	glw::GLuint clipdistances_input_size =
1121 		clipdistances_array_size > 0 ? clipdistances_array_size : 1; /* Avoid zero-sized array compilation error */
1122 	glw::GLuint culldistances_input_size =
1123 		culldistances_array_size > 0 ? culldistances_array_size : 1; /* Avoid zero-sized array compilation error */
1124 	static const glw::GLchar* dynamic_array_setters =
1125 		"\n"
1126 		"#if TEMPLATE_N_GL_CLIPDISTANCE_ENTRIES\n"
1127 		"	 for (int n_clipdistance_entry = 0;\n"
1128 		"		  n_clipdistance_entry < TEMPLATE_N_GL_CLIPDISTANCE_ENTRIES;\n"
1129 		"		++n_clipdistance_entry)\n"
1130 		"	 {\n"
1131 		"	     ASSIGN_CLIP_DISTANCE(n_clipdistance_entry);\n"
1132 		"	 }\n"
1133 		"#endif"
1134 		"\n"
1135 		"#if TEMPLATE_N_GL_CULLDISTANCE_ENTRIES \n"
1136 		"	 for (int n_culldistance_entry = 0;\n"
1137 		"		  n_culldistance_entry < TEMPLATE_N_GL_CULLDISTANCE_ENTRIES;\n"
1138 		"		++n_culldistance_entry)\n"
1139 		"	 {\n"
1140 		"	     ASSIGN_CULL_DISTANCE(n_culldistance_entry);\n"
1141 		"	 }\n"
1142 		"#endif\n";
1143 
1144 	static const glw::GLchar* core_functionality = "#version 450\n";
1145 
1146 	static const glw::GLchar* extention_functionality = "#version 150\n"
1147 														"\n"
1148 														"#extension GL_ARB_cull_distance : require\n"
1149 														"TEMPLATE_EXTENSIONS\n"
1150 														"\n"
1151 														"#ifndef GL_ARB_cull_distance\n"
1152 														"    #error GL_ARB_cull_distance is undefined\n"
1153 														"#endif\n";
1154 
1155 	static const glw::GLchar* fetch_function = "highp float fetch()\n"
1156 											   "{\n"
1157 											   "    highp float sum = 0.0;\n"
1158 											   "\n"
1159 											   "TEMPLATE_SUM_SETTER"
1160 											   "\n"
1161 											   "    return sum / TEMPLATE_SUM_DIVIDER;\n"
1162 											   "}\n"
1163 											   "\n"
1164 											   "#define ASSIGN_RETURN_VALUE fetch()";
1165 
1166 	static const glw::GLchar* fs_template = "TEMPLATE_HEADER_DECLARATION\n"
1167 											"\n"
1168 											"TEMPLATE_REDECLARE_CLIPDISTANCE\n"
1169 											"TEMPLATE_REDECLARE_CULLDISTANCE\n"
1170 											"\n"
1171 											"TEMPLATE_ASSIGN_RETURN_VALUE\n"
1172 											"\n"
1173 											"out vec4 out_fs;\n"
1174 											"\n"
1175 											"/* Fragment shader main function */\n"
1176 											"void main()\n"
1177 											"{\n"
1178 											"    out_fs = vec4(ASSIGN_RETURN_VALUE, 1.0, 1.0, 1.0);\n"
1179 											"}\n";
1180 
1181 	static const glw::GLchar* gs_template = "TEMPLATE_HEADER_DECLARATION\n"
1182 											"\n"
1183 											"TEMPLATE_LAYOUT_IN\n"
1184 											"TEMPLATE_LAYOUT_OUT\n"
1185 											"\n"
1186 											"TEMPLATE_REDECLARE_CLIPDISTANCE\n"
1187 											"TEMPLATE_REDECLARE_CULLDISTANCE\n"
1188 											"\n"
1189 											"#define ASSIGN_CLIP_DISTANCE(IDX) TEMPLATE_ASSIGN_CLIP_DISTANCE\n"
1190 											"#define ASSIGN_CULL_DISTANCE(IDX) TEMPLATE_ASSIGN_CULL_DISTANCE\n"
1191 											"\n"
1192 											"/* Geometry shader (passthrough) main function */\n"
1193 											"void main()\n"
1194 											"{\n"
1195 											"    for (int n_vertex_index = 0;\n"
1196 											"             n_vertex_index < gl_in.length();\n"
1197 											"             n_vertex_index ++)\n"
1198 											"    {\n"
1199 											"        gl_Position = gl_in[n_vertex_index].gl_Position;\n"
1200 											"\n"
1201 											"        TEMPLATE_ARRAY_SETTERS\n"
1202 											"\n"
1203 											"        EmitVertex();\n"
1204 											"    }\n"
1205 											"\n"
1206 											"    EndPrimitive();\n"
1207 											"}\n";
1208 
1209 	static const glw::GLchar* tc_template =
1210 		"TEMPLATE_HEADER_DECLARATION\n"
1211 		"\n"
1212 		"TEMPLATE_LAYOUT_OUT\n"
1213 		"\n"
1214 		"out gl_PerVertex {\n"
1215 		"TEMPLATE_REDECLARE_CLIPDISTANCE\n"
1216 		"TEMPLATE_REDECLARE_CULLDISTANCE\n"
1217 		"vec4 gl_Position;\n"
1218 		"} gl_out[];\n"
1219 		"\n"
1220 		"#define ASSIGN_CLIP_DISTANCE(IDX) TEMPLATE_ASSIGN_CLIP_DISTANCE\n"
1221 		"#define ASSIGN_CULL_DISTANCE(IDX) TEMPLATE_ASSIGN_CULL_DISTANCE\n"
1222 		"\n"
1223 		"/* Tesselation control shader main function */\n"
1224 		"void main()\n"
1225 		"{\n"
1226 		"    gl_TessLevelInner[0] = 1.0;\n"
1227 		"    gl_TessLevelInner[1] = 1.0;\n"
1228 		"    gl_TessLevelOuter[0] = 1.0;\n"
1229 		"    gl_TessLevelOuter[1] = 1.0;\n"
1230 		"    gl_TessLevelOuter[2] = 1.0;\n"
1231 		"    gl_TessLevelOuter[3] = 1.0;\n"
1232 		"    /* Clipdistance and culldistance array setters */\n"
1233 		"    {\n"
1234 		"        TEMPLATE_ARRAY_SETTERS\n"
1235 		"    }\n"
1236 		"    gl_out[gl_InvocationID].gl_Position = gl_in[gl_InvocationID].gl_Position;\n"
1237 		"}\n";
1238 
1239 	static const glw::GLchar* te_template = "TEMPLATE_HEADER_DECLARATION\n"
1240 											"\n"
1241 											"TEMPLATE_LAYOUT_IN\n"
1242 											"\n"
1243 											"in gl_PerVertex {\n"
1244 											"TEMPLATE_REDECLARE_IN_CLIPDISTANCE\n"
1245 											"TEMPLATE_REDECLARE_IN_CULLDISTANCE\n"
1246 											"vec4 gl_Position;\n"
1247 											"} gl_in[];\n"
1248 											"\n"
1249 											"TEMPLATE_REDECLARE_CLIPDISTANCE\n"
1250 											"TEMPLATE_REDECLARE_CULLDISTANCE\n"
1251 											"\n"
1252 											"#define ASSIGN_CLIP_DISTANCE(IDX) TEMPLATE_ASSIGN_CLIP_DISTANCE\n"
1253 											"#define ASSIGN_CULL_DISTANCE(IDX) TEMPLATE_ASSIGN_CULL_DISTANCE\n"
1254 											"\n"
1255 											"/* Tesselation evaluation shader main function */\n"
1256 											"void main()\n"
1257 											"{\n"
1258 											"    /* Clipdistance and culldistance array setters */\n"
1259 											"    {\n"
1260 											"        TEMPLATE_ARRAY_SETTERS\n"
1261 											"    }\n"
1262 											"    gl_Position = TEMPLATE_OUT_FORMULA;\n"
1263 											"}\n";
1264 
1265 	static const glw::GLchar* vs_template =
1266 		"TEMPLATE_HEADER_DECLARATION\n"
1267 		"\n"
1268 		"in float clipdistance_data[TEMPLATE_CLIPDISTANCE_INPUT_SIZE];\n"
1269 		"in float culldistance_data[TEMPLATE_CULLDISTANCE_INPUT_SIZE];\n"
1270 		"in vec2  position;\n"
1271 		"\n"
1272 		"TEMPLATE_REDECLARE_CLIPDISTANCE\n"
1273 		"TEMPLATE_REDECLARE_CULLDISTANCE\n"
1274 		"\n"
1275 		"#define ASSIGN_CLIP_DISTANCE(IDX) TEMPLATE_ASSIGN_CLIP_DISTANCE\n"
1276 		"#define ASSIGN_CULL_DISTANCE(IDX) TEMPLATE_ASSIGN_CULL_DISTANCE\n"
1277 		"\n"
1278 		"/* Vertex shader main function */\n"
1279 		"void main()\n"
1280 		"{\n"
1281 		"    /* Clipdistance and culldistance array setters */\n"
1282 		"    {\n"
1283 		"        TEMPLATE_ARRAY_SETTERS\n"
1284 		"    }\n"
1285 		"    gl_Position = vec4(2.0 * position.x - 1.0, 2.0 * position.y - 1.0, 0.0, 1.0);\n"
1286 		"}\n";
1287 
1288 	std::string* shader_body_string_fs	 = DE_NULL;
1289 	std::string* shader_body_string_gs	 = DE_NULL;
1290 	std::string* shader_body_string_tc	 = DE_NULL;
1291 	std::string* shader_body_string_te	 = DE_NULL;
1292 	std::string* shader_body_string_vs	 = DE_NULL;
1293 	std::string  shader_header_declaration = use_core_functionality ? core_functionality : extention_functionality;
1294 
1295 	struct _shaders_configuration
1296 	{
1297 		glw::GLenum		   type;
1298 		const glw::GLchar* shader_template;
1299 		std::string		   body;
1300 		const bool		   use;
1301 	} shaders_configuration[] = { {
1302 									  GL_FRAGMENT_SHADER, fs_template, std::string(), true,
1303 								  },
1304 								  {
1305 									  GL_GEOMETRY_SHADER, gs_template, std::string(), use_gs,
1306 								  },
1307 								  {
1308 									  GL_TESS_CONTROL_SHADER, tc_template, std::string(), use_ts,
1309 								  },
1310 								  {
1311 									  GL_TESS_EVALUATION_SHADER, te_template, std::string(), use_ts,
1312 								  },
1313 								  {
1314 									  GL_VERTEX_SHADER, vs_template, std::string(), true,
1315 								  } };
1316 
1317 	const glw::GLuint n_shaders_configuration = sizeof(shaders_configuration) / sizeof(shaders_configuration[0]);
1318 
1319 	/* Construct shader bodies out of templates */
1320 	for (glw::GLuint n_shader_index = 0; n_shader_index < n_shaders_configuration; n_shader_index++)
1321 	{
1322 		if (shaders_configuration[n_shader_index].use)
1323 		{
1324 			std::string  array_setters;
1325 			std::string  clipdistance_array_declaration;
1326 			std::string  culldistance_array_declaration;
1327 			std::string  clipdistance_in_array_declaration;
1328 			std::string  culldistance_in_array_declaration;
1329 			std::string& shader_source = shaders_configuration[n_shader_index].body;
1330 
1331 			/* Copy template into shader body source */
1332 			shader_source = shaders_configuration[n_shader_index].shader_template;
1333 
1334 			CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_HEADER_DECLARATION"),
1335 												shader_header_declaration);
1336 
1337 			/* Shader-specific actions */
1338 			switch (shaders_configuration[n_shader_index].type)
1339 			{
1340 			case GL_FRAGMENT_SHADER:
1341 			{
1342 				shader_body_string_fs = &shaders_configuration[n_shader_index].body;
1343 
1344 				if (fetch_culldistance_from_fs)
1345 				{
1346 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_ASSIGN_RETURN_VALUE"),
1347 														std::string(fetch_function));
1348 
1349 					std::string fetch_sum_setters = "";
1350 					for (glw::GLuint i = 0; i < clipdistances_array_size; ++i)
1351 					{
1352 						fetch_sum_setters.append("    sum += abs(gl_ClipDistance[");
1353 						fetch_sum_setters.append(CullDistance::Utilities::intToString(i));
1354 						fetch_sum_setters.append("]) * ");
1355 						fetch_sum_setters.append(CullDistance::Utilities::intToString(i + 1));
1356 						fetch_sum_setters.append(".0;\n");
1357 					}
1358 
1359 					fetch_sum_setters.append("\n");
1360 
1361 					for (glw::GLuint i = 0; i < culldistances_array_size; ++i)
1362 					{
1363 						fetch_sum_setters.append("    sum += abs(gl_CullDistance[");
1364 						fetch_sum_setters.append(CullDistance::Utilities::intToString(i));
1365 						fetch_sum_setters.append("]) * ");
1366 						fetch_sum_setters.append(
1367 							CullDistance::Utilities::intToString(i + 1 + clipdistances_array_size));
1368 						fetch_sum_setters.append(".0;\n");
1369 					}
1370 
1371 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_SUM_SETTER"),
1372 														std::string(fetch_sum_setters));
1373 					CullDistance::Utilities::replaceAll(
1374 						shader_source, std::string("TEMPLATE_SUM_DIVIDER"),
1375 						std::string(CullDistance::Utilities::intToString(
1376 										(clipdistances_array_size + culldistances_array_size) *
1377 										((clipdistances_array_size + culldistances_array_size + 1))))
1378 							.append(".0"));
1379 				}
1380 				else
1381 				{
1382 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_ASSIGN_RETURN_VALUE"),
1383 														std::string("#define ASSIGN_RETURN_VALUE 1.0"));
1384 				}
1385 
1386 				break;
1387 			}
1388 
1389 			case GL_GEOMETRY_SHADER:
1390 			{
1391 				shader_body_string_gs = &shaders_configuration[n_shader_index].body;
1392 
1393 				CullDistance::Utilities::replaceAll(
1394 					shader_source, std::string("TEMPLATE_ASSIGN_CLIP_DISTANCE"),
1395 					std::string("gl_ClipDistance[IDX] = gl_in[n_vertex_index].gl_ClipDistance[IDX]"));
1396 				CullDistance::Utilities::replaceAll(
1397 					shader_source, std::string("TEMPLATE_ASSIGN_CULL_DISTANCE"),
1398 					std::string("gl_CullDistance[IDX] = gl_in[n_vertex_index].gl_CullDistance[IDX]"));
1399 
1400 				switch (primitive_mode)
1401 				{
1402 				case PRIMITIVE_MODE_LINES:
1403 				{
1404 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_IN"),
1405 														std::string("layout(lines)                        in;"));
1406 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_OUT"),
1407 														std::string("layout(line_strip, max_vertices = 2) out;"));
1408 
1409 					break;
1410 				}
1411 				case PRIMITIVE_MODE_POINTS:
1412 				{
1413 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_IN"),
1414 														std::string("layout(points)                   in;"));
1415 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_OUT"),
1416 														std::string("layout(points, max_vertices = 1) out;"));
1417 
1418 					break;
1419 				}
1420 				case PRIMITIVE_MODE_TRIANGLES:
1421 				{
1422 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_IN"),
1423 														std::string("layout(triangles)                        in;"));
1424 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_OUT"),
1425 														std::string("layout(triangle_strip, max_vertices = 3) out;"));
1426 
1427 					break;
1428 				}
1429 				default:
1430 					TCU_FAIL("Unknown primitive mode");
1431 				}
1432 
1433 				break;
1434 			}
1435 
1436 			case GL_TESS_CONTROL_SHADER:
1437 			{
1438 				shader_body_string_tc = &shaders_configuration[n_shader_index].body;
1439 
1440 				CullDistance::Utilities::replaceAll(
1441 					shader_source, std::string("TEMPLATE_ASSIGN_CLIP_DISTANCE"),
1442 					std::string(
1443 						"gl_out[gl_InvocationID].gl_ClipDistance[IDX] = gl_in[gl_InvocationID].gl_ClipDistance[IDX]"));
1444 				CullDistance::Utilities::replaceAll(
1445 					shader_source, std::string("TEMPLATE_ASSIGN_CULL_DISTANCE"),
1446 					std::string(
1447 						"gl_out[gl_InvocationID].gl_CullDistance[IDX] = gl_in[gl_InvocationID].gl_CullDistance[IDX]"));
1448 
1449 				switch (primitive_mode)
1450 				{
1451 				case PRIMITIVE_MODE_LINES:
1452 				{
1453 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_OUT"),
1454 														std::string("layout(vertices = 2) out;"));
1455 
1456 					break;
1457 				}
1458 				case PRIMITIVE_MODE_POINTS:
1459 				{
1460 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_OUT"),
1461 														std::string("layout(vertices = 1) out;"));
1462 
1463 					break;
1464 				}
1465 				case PRIMITIVE_MODE_TRIANGLES:
1466 				{
1467 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_OUT"),
1468 														std::string("layout(vertices = 3) out;"));
1469 
1470 					break;
1471 				}
1472 				default:
1473 					TCU_FAIL("Unknown primitive mode");
1474 				}
1475 
1476 				CullDistance::Utilities::replaceAll(
1477 						shader_source,
1478 						std::string("TEMPLATE_EXTENSIONS"),
1479 						std::string("#extension GL_ARB_tessellation_shader: require"));
1480 				break;
1481 			}
1482 
1483 			case GL_TESS_EVALUATION_SHADER:
1484 			{
1485 				shader_body_string_te = &shaders_configuration[n_shader_index].body;
1486 
1487 				switch (primitive_mode)
1488 				{
1489 				case PRIMITIVE_MODE_LINES:
1490 				{
1491 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_IN"),
1492 														std::string("layout(isolines) in;"));
1493 					CullDistance::Utilities::replaceAll(
1494 						shader_source, std::string("TEMPLATE_OUT_FORMULA"),
1495 						std::string("mix(gl_in[0].gl_Position, gl_in[1].gl_Position, gl_TessCoord.x)"));
1496 					CullDistance::Utilities::replaceAll(
1497 						shader_source, std::string("TEMPLATE_ASSIGN_CLIP_DISTANCE"),
1498 						std::string("gl_ClipDistance[IDX] = mix(gl_in[0].gl_ClipDistance[IDX], "
1499 									"gl_in[1].gl_ClipDistance[IDX], gl_TessCoord.x)"));
1500 					CullDistance::Utilities::replaceAll(
1501 						shader_source, std::string("TEMPLATE_ASSIGN_CULL_DISTANCE"),
1502 						std::string("gl_CullDistance[IDX] = mix(gl_in[0].gl_CullDistance[IDX], "
1503 									"gl_in[1].gl_CullDistance[IDX], gl_TessCoord.x)"));
1504 
1505 					break;
1506 				}
1507 				case PRIMITIVE_MODE_POINTS:
1508 				{
1509 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_IN"),
1510 														std::string("layout(isolines, point_mode) in;"));
1511 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_OUT_FORMULA"),
1512 														std::string("gl_in[0].gl_Position"));
1513 					CullDistance::Utilities::replaceAll(
1514 						shader_source, std::string("TEMPLATE_ASSIGN_CLIP_DISTANCE"),
1515 						std::string("gl_ClipDistance[IDX] = gl_in[0].gl_ClipDistance[IDX]"));
1516 					CullDistance::Utilities::replaceAll(
1517 						shader_source, std::string("TEMPLATE_ASSIGN_CULL_DISTANCE"),
1518 						std::string("gl_CullDistance[IDX] = gl_in[0].gl_CullDistance[IDX]"));
1519 
1520 					break;
1521 				}
1522 				case PRIMITIVE_MODE_TRIANGLES:
1523 				{
1524 					CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_LAYOUT_IN"),
1525 														std::string("layout(triangles) in;"));
1526 					CullDistance::Utilities::replaceAll(
1527 						shader_source, std::string("TEMPLATE_OUT_FORMULA"),
1528 						std::string("vec4(mat3(gl_in[0].gl_Position.xyz, gl_in[1].gl_Position.xyz, "
1529 									"gl_in[2].gl_Position.xyz) * gl_TessCoord, 1.0)"));
1530 					CullDistance::Utilities::replaceAll(
1531 						shader_source, std::string("TEMPLATE_ASSIGN_CLIP_DISTANCE"),
1532 						std::string("gl_ClipDistance[IDX] = dot(vec3(gl_in[0].gl_ClipDistance[IDX], "
1533 									"gl_in[1].gl_ClipDistance[IDX], gl_in[2].gl_ClipDistance[IDX]), gl_TessCoord)"));
1534 					CullDistance::Utilities::replaceAll(
1535 						shader_source, std::string("TEMPLATE_ASSIGN_CULL_DISTANCE"),
1536 						std::string("gl_CullDistance[IDX] = dot(vec3(gl_in[0].gl_CullDistance[IDX], "
1537 									"gl_in[1].gl_CullDistance[IDX], gl_in[2].gl_CullDistance[IDX]), gl_TessCoord)"));
1538 
1539 					break;
1540 				}
1541 				default:
1542 					TCU_FAIL("Unknown primitive mode");
1543 				}
1544 
1545 				CullDistance::Utilities::replaceAll(
1546 						shader_source,
1547 						std::string("TEMPLATE_EXTENSIONS"),
1548 						std::string("#extension GL_ARB_tessellation_shader: require"));
1549 				break;
1550 			}
1551 
1552 			case GL_VERTEX_SHADER:
1553 			{
1554 				shader_body_string_vs = &shaders_configuration[n_shader_index].body;
1555 
1556 				/* Specify input data size for clipdistances data */
1557 				CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_CLIPDISTANCE_INPUT_SIZE"),
1558 													CullDistance::Utilities::intToString(clipdistances_input_size));
1559 
1560 				/* Specify input data size for culldistances data */
1561 				CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_CULLDISTANCE_INPUT_SIZE"),
1562 													CullDistance::Utilities::intToString(culldistances_input_size));
1563 
1564 				CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_ASSIGN_CLIP_DISTANCE"),
1565 													std::string("gl_ClipDistance[IDX] = clipdistance_data[IDX]"));
1566 				CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_ASSIGN_CULL_DISTANCE"),
1567 													std::string("gl_CullDistance[IDX] = culldistance_data[IDX]"));
1568 
1569 				break;
1570 			}
1571 
1572 			default:
1573 				TCU_FAIL("Unknown shader type");
1574 			}
1575 
1576 			/* Clear out in case no specific exts were needed */
1577 			CullDistance::Utilities::replaceAll(
1578 					shader_source,
1579 					std::string("TEMPLATE_EXTENSIONS"),
1580 					std::string(""));
1581 
1582 			/* Adjust clipdistances declaration */
1583 			if (redeclare_clipdistances && clipdistances_array_size > 0)
1584 			{
1585 				if (shaders_configuration[n_shader_index].type == GL_FRAGMENT_SHADER)
1586 				{
1587 					if (fetch_culldistance_from_fs)
1588 					{
1589 						clipdistance_array_declaration =
1590 							std::string("in float gl_ClipDistance[") +
1591 							CullDistance::Utilities::intToString(clipdistances_array_size) + std::string("];");
1592 					}
1593 				}
1594 				else if (shaders_configuration[n_shader_index].type == GL_TESS_CONTROL_SHADER)
1595 				{
1596 					clipdistance_array_declaration = std::string("float gl_ClipDistance[") +
1597 													 CullDistance::Utilities::intToString(clipdistances_array_size) +
1598 													 std::string("];");
1599 				}
1600 				else
1601 				{
1602 					clipdistance_array_declaration = std::string("out float gl_ClipDistance[") +
1603 													 CullDistance::Utilities::intToString(clipdistances_array_size) +
1604 													 std::string("];");
1605 					clipdistance_in_array_declaration = std::string("in float gl_ClipDistance[") +
1606 														CullDistance::Utilities::intToString(clipdistances_array_size) +
1607 														std::string("];");
1608 				}
1609 			}
1610 			CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_REDECLARE_CLIPDISTANCE"),
1611 												clipdistance_array_declaration);
1612 			CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_REDECLARE_IN_CLIPDISTANCE"),
1613 												clipdistance_in_array_declaration);
1614 
1615 			/* Adjust culldistances declaration */
1616 			if (redeclare_culldistances && culldistances_array_size > 0)
1617 			{
1618 				if (shaders_configuration[n_shader_index].type == GL_FRAGMENT_SHADER)
1619 				{
1620 					if (fetch_culldistance_from_fs)
1621 					{
1622 						culldistance_array_declaration =
1623 							std::string("in float gl_CullDistance[") +
1624 							CullDistance::Utilities::intToString(culldistances_array_size) + std::string("];");
1625 					}
1626 				}
1627 				else if (shaders_configuration[n_shader_index].type == GL_TESS_CONTROL_SHADER)
1628 				{
1629 					culldistance_array_declaration = std::string("float gl_CullDistance[") +
1630 													 CullDistance::Utilities::intToString(culldistances_array_size) +
1631 													 std::string("];");
1632 				}
1633 				else
1634 				{
1635 					culldistance_array_declaration = std::string("out float gl_CullDistance[") +
1636 													 CullDistance::Utilities::intToString(culldistances_array_size) +
1637 													 std::string("];");
1638 					culldistance_in_array_declaration = std::string("in float gl_CullDistance[") +
1639 														CullDistance::Utilities::intToString(culldistances_array_size) +
1640 														std::string("];");
1641 				}
1642 			}
1643 			CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_REDECLARE_CULLDISTANCE"),
1644 												culldistance_array_declaration);
1645 			CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_REDECLARE_IN_CULLDISTANCE"),
1646 												culldistance_in_array_declaration);
1647 
1648 			/* Adjust clip/cull distances setters */
1649 			if (dynamic_index_writes)
1650 			{
1651 				array_setters = dynamic_array_setters;
1652 
1653 				CullDistance::Utilities::replaceAll(array_setters, std::string("TEMPLATE_N_GL_CLIPDISTANCE_ENTRIES"),
1654 													CullDistance::Utilities::intToString(clipdistances_array_size));
1655 				CullDistance::Utilities::replaceAll(array_setters, std::string("TEMPLATE_N_GL_CULLDISTANCE_ENTRIES"),
1656 													CullDistance::Utilities::intToString(culldistances_array_size));
1657 			}
1658 			else
1659 			{
1660 				std::stringstream static_array_setters_sstream;
1661 
1662 				static_array_setters_sstream << "\n";
1663 
1664 				for (glw::GLuint clipdistances_array_entry = 0; clipdistances_array_entry < clipdistances_array_size;
1665 					 ++clipdistances_array_entry)
1666 				{
1667 					static_array_setters_sstream << "        ASSIGN_CLIP_DISTANCE(" << clipdistances_array_entry
1668 												 << ");\n";
1669 				}
1670 
1671 				static_array_setters_sstream << "\n";
1672 
1673 				for (glw::GLuint culldistances_array_entry = 0; culldistances_array_entry < culldistances_array_size;
1674 					 ++culldistances_array_entry)
1675 				{
1676 					static_array_setters_sstream << "        ASSIGN_CULL_DISTANCE(" << culldistances_array_entry
1677 												 << ");\n";
1678 				}
1679 
1680 				array_setters = static_array_setters_sstream.str();
1681 			}
1682 
1683 			CullDistance::Utilities::replaceAll(shader_source, std::string("TEMPLATE_ARRAY_SETTERS"), array_setters);
1684 		}
1685 	}
1686 
1687 	/* Build the geometry shader */
1688 	CullDistance::Utilities::buildProgram(
1689 		m_context.getRenderContext().getFunctions(), m_testCtx, DE_NULL, /* Compute shader                    */
1690 		shader_body_string_fs != DE_NULL ? shader_body_string_fs->c_str() :
1691 										   DE_NULL, /* Fragment shader                   */
1692 		shader_body_string_gs != DE_NULL ? shader_body_string_gs->c_str() :
1693 										   DE_NULL, /* Geometry shader                   */
1694 		shader_body_string_tc != DE_NULL ? shader_body_string_tc->c_str() :
1695 										   DE_NULL, /* Tesselation control shader        */
1696 		shader_body_string_te != DE_NULL ? shader_body_string_te->c_str() :
1697 										   DE_NULL, /* Tesselation evaluation shader     */
1698 		shader_body_string_vs != DE_NULL ? shader_body_string_vs->c_str() :
1699 										   DE_NULL, /* Vertex shader                     */
1700 		0,											/* Transform feedback varyings count */
1701 		DE_NULL,									/* Transform feedback varyings       */
1702 		&m_po_id									/* Program object id                 */
1703 		);
1704 }
1705 
1706 /** Generates primitive data required to test a case with specified
1707  *  gl_ClipDistance and glCullDistance array sizes for specified
1708  *  primitive mode. Generated primitive data is stored in m_bo_data
1709  *  as well uploaded into buffer specified in m_bo_id buffer.
1710  *  Also the procedure binds vertex attribute locations to
1711  *  program object m_po_id.
1712  *
1713  *  @param clipdistances_array_size gl_ClipDistance array size. Can be 0.
1714  *  @param culldistances_array_size gl_CullDistance array size. Can be 0.
1715  *  @param _primitive_mode          Primitives to be generated. Can be:
1716  *                                  PRIMITIVE_MODE_POINTS,
1717  *                                  PRIMITIVE_MODE_LINES,
1718  *                                  PRIMITIVE_MODE_TRIANGLES.
1719  */
configureVAO(glw::GLuint clipdistances_array_size,glw::GLuint culldistances_array_size,_primitive_mode primitive_mode)1720 void CullDistance::FunctionalTest::configureVAO(glw::GLuint clipdistances_array_size,
1721 												glw::GLuint culldistances_array_size, _primitive_mode primitive_mode)
1722 {
1723 	/* Detailed test description.
1724 	 *
1725 	 * configureVAO() generates primitives layouted in grid. Primitve
1726 	 * consists of up to 3 vertices and each vertex is accompanied by:
1727 	 * - array of clipdistances (clipdistances_array_size floats);
1728 	 * - array of culldistances (culldistances_array_size floats);
1729 	 * - rendering position coordinates (x and y);
1730 	 * - check position coordinates (x and y).
1731 	 *
1732 	 * The grid has following layout:
1733 	 *
1734 	 *     Grid                       |         gl_CullDistance[x]         |
1735 	 *                                |  0 .. culldistances_array_size - 1 |
1736 	 *                                |  0th  |  1st  |  2nd  | .......... |
1737 	 *     ---------------------------+-------+-------+-------+------------+
1738 	 *     0th  gl_ClipDistance       |Subgrid|Subgrid|Subgrid| .......... |
1739 	 *     1st  gl_ClipDistance       |Subgrid|Subgrid|Subgrid| .......... |
1740 	 *     ...                        |  ...  |  ...  |  ...  | .......... |
1741 	 *     y-th gl_ClipDistance       |Subgrid|Subgrid|Subgrid| .......... |
1742 	 *     ...                        |  ...  |  ...  |  ...  | .......... |
1743 	 *     clipdistances_array_size-1 |Subgrid|Subgrid|Subgrid| .......... |
1744 	 *
1745 	 * Each grid cell contains subgrid of 3*3 items in size with following
1746 	 * structure:
1747 	 *
1748 	 *     Subgrid        |        x-th gl_CullDistance test           |
1749 	 *                    |                                            |
1750 	 *     y-th           | all vertices | 0th vertex   | all vertices |
1751 	 *     gl_ClipDistance| in primitive | in primitive | in primitive |
1752 	 *     tests          | dist[x] > 0  | dist[x] < 0  | dist[x] < 0  |
1753 	 *     ---------------+--------------+--------------+--------------+
1754 	 *        all vertices| primitive #0 | primitive #1 | primitive #2 |
1755 	 *        in primitive|              |              |              |
1756 	 *        dist[y] > 0 |   visible    |   visible    |    culled    |
1757 	 *     ---------------+--------------+--------------+--------------+
1758 	 *        0th vertex  | primitive #3 | primitive #4 | primitive #5 |
1759 	 *        in primitive|  0th vertex  |  0th vertex  |              |
1760 	 *        dist[y] < 0 |   clipped    |   clipped    |    culled    |
1761 	 *     ---------------+--------------+--------------+--------------+
1762 	 *        all vertices| primitive #6 | primitive #7 | primitive #8 |
1763 	 *        in primitive|              |              |              |
1764 	 *        dist[y] < 0 |   clipped    |   clipped    |    culled    |
1765 	 *     ---------------+--------------+--------------+--------------+
1766 	 *
1767 	 * Expected rendering result is specified in cell bottom.
1768 	 * It can be one of the following:
1769 	 * - "visible" means the primitive is not affected neither by gl_CullDistance
1770 	 *             nor by gl_ClipDistance and rendered as a whole;
1771 	 * - "clipped" for the vertex means the vertex is not rendered, while other
1772 	 *             primitive vertices and some filling fragments are rendered;
1773 	 * - "clipped" for primitive means none of primitive vertices and fragments
1774 	 *             are rendered and thus primitive is not rendered and is invisible;
1775 	 * - "culled"  means, that neither primitive vertices, nor primitive filling
1776 	 *             fragments are rendered (primitive is invisible).
1777 	 *
1778 	 * All subgrid items contain same primitive rendered. Depending on
1779 	 * test case running it would be either triangle, or line, or point:
1780 	 *
1781 	 *     triangle    line        point
1782 	 *     8x8 box     8x8 box     3x3 box
1783 	 *     ........    ........    ...
1784 	 *     .0----2.    .0......    .0.
1785 	 *     ..\@@@|.    ..\.....    ...
1786 	 *     ...\@@|.    ...\....
1787 	 *     ....\@|.    ....\...
1788 	 *     .....\|.    .....\..
1789 	 *     ......1.    ......1.
1790 	 *     ........    ........
1791 	 *
1792 	 *     where 0 - is a 0th vertex primitive
1793 	 *           1 - is a 1st vertex primitive
1794 	 *           2 - is a 2nd vertex primitive
1795 	 *
1796 	 * The culldistances_array_size can be 0. In that case, grid height
1797 	 * is assumed equal to 1, but 0 glCullDistances is specified.
1798 	 * Similar handled clipdistances_array_size.
1799 	 *
1800 	 * The data generated is used and checked in executeRenderTest().
1801 	 * After rendering each primitive vertex is tested:
1802 	 * - if it is rendered, if it have to be rendered (according distance);
1803 	 * - if it is not rendered, if it have to be not rendered (according distance).
1804 	 * Due to "top-left" rasterization rule check position is
1805 	 * different from rendering vertex position.
1806 	 *
1807 	 * Also one pixel width guarding box is checked to be clear.
1808 	 */
1809 
1810 	const glw::Functions& gl			   = m_context.getRenderContext().getFunctions();
1811 	const glw::GLuint	 n_sub_grid_cells = 3; /* Tested distance is positive for all vertices in the primitive;
1812 	 * Tested distance is negative for 0th vertex in the primitive;
1813 	 * Tested distance is negative for all vertices in the primitive;
1814 	 */
1815 	const glw::GLuint	 sub_grid_cell_size =
1816 		((primitive_mode == PRIMITIVE_MODE_LINES) ? 8 : (primitive_mode == PRIMITIVE_MODE_POINTS) ? 3 : 8);
1817 
1818 	const glw::GLuint grid_cell_size = n_sub_grid_cells * sub_grid_cell_size;
1819 	const glw::GLuint n_primitive_vertices =
1820 		((primitive_mode == PRIMITIVE_MODE_LINES) ? 2 : (primitive_mode == PRIMITIVE_MODE_POINTS) ? 1 : 3);
1821 
1822 	const glw::GLuint n_grid_cells_x			   = culldistances_array_size != 0 ? culldistances_array_size : 1;
1823 	const glw::GLuint n_grid_cells_y			   = clipdistances_array_size != 0 ? clipdistances_array_size : 1;
1824 	const glw::GLuint n_pervertex_float_attributes = clipdistances_array_size + culldistances_array_size +
1825 													 2 /* vertex' draw x, y */ + 2 /* vertex' checkpoint x, y */;
1826 	const glw::GLuint n_primitives_total	 = n_grid_cells_x * n_sub_grid_cells * n_grid_cells_y * n_sub_grid_cells;
1827 	const glw::GLuint n_vertices_total		 = n_primitives_total * n_primitive_vertices;
1828 	const glw::GLuint offsets_line_draw_x[2] = {
1829 		1, sub_grid_cell_size - 1
1830 	}; /* vertex x offsets to subgrid cell origin for line primitive     */
1831 	const glw::GLuint offsets_line_draw_y[2] = {
1832 		1, sub_grid_cell_size - 1
1833 	}; /* vertex y offsets to subgrid cell origin for line primitive     */
1834 	const glw::GLuint offsets_line_checkpoint_x[2] = {
1835 		1, sub_grid_cell_size - 2
1836 	}; /* pixel x offsets to subgrid cell origin for line primitive      */
1837 	const glw::GLuint offsets_line_checkpoint_y[2] = {
1838 		1, sub_grid_cell_size - 2
1839 	}; /* pixel y offsets to subgrid cell origin for line primitive      */
1840 	const glw::GLuint offsets_point_draw_x[1] = {
1841 		1
1842 	}; /* vertex x offsets to subgrid cell origin for point primitive    */
1843 	const glw::GLuint offsets_point_draw_y[1] = {
1844 		1
1845 	}; /* vertex y offsets to subgrid cell origin for point primitive    */
1846 	const glw::GLuint offsets_point_checkpoint_x[1] = {
1847 		1
1848 	}; /* pixel x offsets to subgrid cell origin for point primitive     */
1849 	const glw::GLuint offsets_point_checkpoint_y[1] = {
1850 		1
1851 	}; /* pixel y offsets to subgrid cell origin for point primitive     */
1852 	const glw::GLuint offsets_triangle_draw_x[3] = {
1853 		1, sub_grid_cell_size - 1, sub_grid_cell_size - 1
1854 	}; /* vertex x offsets to subgrid cell origin for triangle primitive */
1855 	const glw::GLuint offsets_triangle_draw_y[3] = {
1856 		1, sub_grid_cell_size - 1, 1
1857 	}; /* vertex y offsets to subgrid cell origin for triangle primitive */
1858 	const glw::GLuint offsets_triangle_checkpoint_x[3] = {
1859 		1, sub_grid_cell_size - 2, sub_grid_cell_size - 2
1860 	}; /* pixel x offsets to subgrid cell origin for triangle primitive  */
1861 	const glw::GLuint offsets_triangle_checkpoint_y[3] = {
1862 		1, sub_grid_cell_size - 2, 1
1863 	}; /* pixel y offsets to subgrid cell origin for triangle primitive  */
1864 	const glw::GLfloat offsets_pixel_center_x = (primitive_mode == PRIMITIVE_MODE_POINTS) ? 0.5f : 0;
1865 	const glw::GLfloat offsets_pixel_center_y = (primitive_mode == PRIMITIVE_MODE_POINTS) ? 0.5f : 0;
1866 	/* Clear data left from previous tests. */
1867 	m_bo_data.clear();
1868 
1869 	/* No data to render */
1870 	m_render_primitives = 0;
1871 	m_render_vertices   = 0;
1872 
1873 	/* Preallocate space for bo_points_count */
1874 	m_bo_data.reserve(n_vertices_total * n_pervertex_float_attributes);
1875 
1876 	/* Generate test data for cell_y-th clip distance */
1877 	for (glw::GLuint cell_y = 0; cell_y < n_grid_cells_y; cell_y++)
1878 	{
1879 		/* Generate test data for cell_x-th cull distance */
1880 		for (glw::GLuint cell_x = 0; cell_x < n_grid_cells_x; cell_x++)
1881 		{
1882 			/* Check clip distance sub cases:
1883 			 * 0. Tested distance is positive for all vertices in the primitive;
1884 			 * 1. Tested distance is negative for 0th vertex in the primitive;
1885 			 * 2. Tested distance is negative for all vertices in the primitive;
1886 			 */
1887 			for (glw::GLuint n_sub_cell_y = 0; n_sub_cell_y < n_sub_grid_cells; n_sub_cell_y++)
1888 			{
1889 				/* Check cull distance sub cases:
1890 				 * 0. Tested distance is positive for all vertices in the primitive;
1891 				 * 1. Tested distance is negative for 0th vertex in the primitive;
1892 				 * 2. Tested distance is negative for all vertices in the primitive;
1893 				 */
1894 				for (glw::GLuint n_sub_cell_x = 0; n_sub_cell_x < n_sub_grid_cells; n_sub_cell_x++)
1895 				{
1896 					/* Generate vertices in primitive */
1897 					for (glw::GLuint n_primitive_vertex = 0; n_primitive_vertex < n_primitive_vertices;
1898 						 n_primitive_vertex++)
1899 					{
1900 						/* Fill in clipdistance array for the n_primitive_vertex vertex in primitive */
1901 						for (glw::GLuint n_clipdistance_entry = 0; n_clipdistance_entry < clipdistances_array_size;
1902 							 n_clipdistance_entry++)
1903 						{
1904 							glw::GLfloat distance_value = 0.0f;
1905 							bool		 negative		= true;
1906 
1907 							/* Special approach to tested clipdistance entry. */
1908 							if (n_clipdistance_entry == cell_y)
1909 							{
1910 								/* The primitive vertex should be affected by the clip distance */
1911 								switch (n_sub_cell_y)
1912 								{
1913 								case 0:
1914 								{
1915 									/* subgrid row 0: all primitive vertices have tested distance value positive */
1916 									negative = false;
1917 
1918 									break;
1919 								}
1920 								case 1:
1921 								{
1922 									/* subgrid row 1: tested distance value for 0th primitive vertex is negative,
1923 									 all other primitive vertices have tested distance value positive */
1924 									negative = (n_primitive_vertex == 0) ? true : false;
1925 
1926 									break;
1927 								}
1928 								case 2:
1929 								{
1930 									/* subgrid row 2: tested distance value is negative for all primitive vertices */
1931 									negative = true;
1932 
1933 									break;
1934 								}
1935 								default:
1936 									TCU_FAIL("Invalid subgrid cell index");
1937 								}
1938 
1939 								distance_value = (negative ? -1.0f : 1.0f) * glw::GLfloat(n_clipdistance_entry + 1);
1940 							}
1941 							else
1942 							{
1943 								/* For clip distances other than tested: assign positive value to avoid its influence. */
1944 								distance_value = glw::GLfloat(clipdistances_array_size + n_clipdistance_entry + 1);
1945 							}
1946 
1947 							m_bo_data.push_back(distance_value / glw::GLfloat(clipdistances_array_size));
1948 						} /* for (all gl_ClipDistance[] array values) */
1949 
1950 						/* Fill in culldistance array for the n_primitive_vertex vertex in primitive */
1951 						for (glw::GLuint n_culldistance_entry = 0; n_culldistance_entry < culldistances_array_size;
1952 							 n_culldistance_entry++)
1953 						{
1954 							glw::GLfloat distance_value = 0.0f;
1955 							bool		 negative		= true;
1956 
1957 							/* Special approach to tested culldistance entry. */
1958 							if (n_culldistance_entry == cell_x)
1959 							{
1960 								/* The primitive vertex should be affected by the cull distance */
1961 								switch (n_sub_cell_x)
1962 								{
1963 								case 0:
1964 								{
1965 									/* subgrid column 0: all primitive vertices have tested distance value positive */
1966 									negative = false;
1967 
1968 									break;
1969 								}
1970 								case 1:
1971 								{
1972 									/* subgrid column 1: tested distance value for 0th primitive vertex is negative,
1973 									 all other primitive vertices have tested distance value positive */
1974 									negative = (n_primitive_vertex == 0) ? true : false;
1975 
1976 									break;
1977 								}
1978 								case 2:
1979 								{
1980 									/* subgrid column 2: tested distance value is negative for all primitive vertices */
1981 									negative = true;
1982 
1983 									break;
1984 								}
1985 								default:
1986 									TCU_FAIL("Invalid subgrid cell index");
1987 								}
1988 
1989 								distance_value = (negative ? -1.0f : 1.0f) * glw::GLfloat(n_culldistance_entry + 1);
1990 							}
1991 							else
1992 							{
1993 								/* For cull distances other than tested: assign 0th vertex negative value,
1994 								 to check absence of between-distances influence. */
1995 								if (n_primitive_vertices > 1 && n_primitive_vertex == 0)
1996 								{
1997 									distance_value = -glw::GLfloat(culldistances_array_size + n_culldistance_entry + 1);
1998 								}
1999 								else
2000 								{
2001 									/* This culldistance is out of interest: assign positive value. */
2002 									distance_value = glw::GLfloat(culldistances_array_size + n_culldistance_entry + 1);
2003 								}
2004 							}
2005 
2006 							m_bo_data.push_back(distance_value / glw::GLfloat(culldistances_array_size));
2007 						} /* for (all gl_CullDistance[] array values) */
2008 
2009 						/* Generate primitve vertex draw and checkpoint coordinates */
2010 						glw::GLint vertex_draw_pixel_offset_x		= 0;
2011 						glw::GLint vertex_draw_pixel_offset_y		= 0;
2012 						glw::GLint vertex_checkpoint_pixel_offset_x = 0;
2013 						glw::GLint vertex_checkpoint_pixel_offset_y = 0;
2014 
2015 						switch (primitive_mode)
2016 						{
2017 						case PRIMITIVE_MODE_LINES:
2018 						{
2019 							vertex_draw_pixel_offset_x		 = offsets_line_draw_x[n_primitive_vertex];
2020 							vertex_draw_pixel_offset_y		 = offsets_line_draw_y[n_primitive_vertex];
2021 							vertex_checkpoint_pixel_offset_x = offsets_line_checkpoint_x[n_primitive_vertex];
2022 							vertex_checkpoint_pixel_offset_y = offsets_line_checkpoint_y[n_primitive_vertex];
2023 
2024 							break;
2025 						}
2026 
2027 						case PRIMITIVE_MODE_POINTS:
2028 						{
2029 							vertex_draw_pixel_offset_x		 = offsets_point_draw_x[n_primitive_vertex];
2030 							vertex_draw_pixel_offset_y		 = offsets_point_draw_y[n_primitive_vertex];
2031 							vertex_checkpoint_pixel_offset_x = offsets_point_checkpoint_x[n_primitive_vertex];
2032 							vertex_checkpoint_pixel_offset_y = offsets_point_checkpoint_y[n_primitive_vertex];
2033 
2034 							break;
2035 						}
2036 
2037 						case PRIMITIVE_MODE_TRIANGLES:
2038 						{
2039 							vertex_draw_pixel_offset_x		 = offsets_triangle_draw_x[n_primitive_vertex];
2040 							vertex_draw_pixel_offset_y		 = offsets_triangle_draw_y[n_primitive_vertex];
2041 							vertex_checkpoint_pixel_offset_x = offsets_triangle_checkpoint_x[n_primitive_vertex];
2042 							vertex_checkpoint_pixel_offset_y = offsets_triangle_checkpoint_y[n_primitive_vertex];
2043 
2044 							break;
2045 						}
2046 
2047 						default:
2048 							TCU_FAIL("Unknown primitive mode");
2049 						}
2050 
2051 						/* Origin of sub_cell */
2052 						glw::GLint sub_cell_origin_x = cell_x * grid_cell_size + n_sub_cell_x * sub_grid_cell_size;
2053 						glw::GLint sub_cell_origin_y = cell_y * grid_cell_size + n_sub_cell_y * sub_grid_cell_size;
2054 						/* Normalized texture coordinates of vertex draw position. */
2055 						glw::GLfloat x =
2056 							(glw::GLfloat(sub_cell_origin_x + vertex_draw_pixel_offset_x) + offsets_pixel_center_x) /
2057 							glw::GLfloat(m_to_width);
2058 						glw::GLfloat y =
2059 							(glw::GLfloat(sub_cell_origin_y + vertex_draw_pixel_offset_y) + offsets_pixel_center_y) /
2060 							glw::GLfloat(m_to_height);
2061 						/* Normalized texture coordinates of vertex checkpoint position. */
2062 						glw::GLfloat checkpoint_x = glw::GLfloat(sub_cell_origin_x + vertex_checkpoint_pixel_offset_x) /
2063 													glw::GLfloat(m_to_width);
2064 						glw::GLfloat checkpoint_y = glw::GLfloat(sub_cell_origin_y + vertex_checkpoint_pixel_offset_y) /
2065 													glw::GLfloat(m_to_height);
2066 
2067 						/* Add vertex draw coordinates into buffer. */
2068 						m_bo_data.push_back(x);
2069 						m_bo_data.push_back(y);
2070 
2071 						/* Add vertex checkpoint coordinates into buffer. */
2072 						m_bo_data.push_back(checkpoint_x);
2073 						m_bo_data.push_back(checkpoint_y);
2074 					} /* for (all vertices in primitive) */
2075 				}	 /* for (all horizontal sub cells) */
2076 			}		  /* for (all vertical sub cells) */
2077 		}			  /* for (all horizontal cells) */
2078 	}				  /* for (all vertical cells) */
2079 
2080 	/* Sanity check: make sure we pushed required amount of data */
2081 	DE_ASSERT(m_bo_data.size() == n_vertices_total * n_pervertex_float_attributes);
2082 
2083 	/* Save number of primitives to render */
2084 	m_render_primitives  = n_primitives_total;
2085 	m_render_vertices	= n_vertices_total;
2086 	m_sub_grid_cell_size = sub_grid_cell_size;
2087 
2088 	/* Copy the data to the buffer object */
2089 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo_id);
2090 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer() call failed.");
2091 
2092 	gl.bufferData(GL_ARRAY_BUFFER, m_bo_data.size() * sizeof(glw::GLfloat), &m_bo_data[0], GL_STATIC_DRAW);
2093 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData() call failed.");
2094 
2095 	DE_ASSERT(m_po_id != 0);
2096 
2097 	/* Bind VAO data to program */
2098 	glw::GLint po_clipdistance_array_location = -1;
2099 	glw::GLint po_culldistance_array_location = -1;
2100 	glw::GLint po_position_location			  = -1;
2101 
2102 	/* Retrieve clipdistance and culldistance attribute locations */
2103 	gl.bindVertexArray(m_vao_id);
2104 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray() call failed.");
2105 
2106 	po_clipdistance_array_location = gl.getAttribLocation(m_po_id, "clipdistance_data[0]");
2107 	po_culldistance_array_location = gl.getAttribLocation(m_po_id, "culldistance_data[0]");
2108 	po_position_location		   = gl.getAttribLocation(m_po_id, "position");
2109 
2110 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGetAttribLocation() call(s) failed.");
2111 
2112 	if (clipdistances_array_size > 0)
2113 	{
2114 		DE_ASSERT(po_clipdistance_array_location != -1);
2115 	}
2116 
2117 	if (culldistances_array_size > 0)
2118 	{
2119 		DE_ASSERT(po_culldistance_array_location != -1);
2120 	}
2121 
2122 	DE_ASSERT(po_position_location != -1);
2123 
2124 	glw::GLintptr	current_offset = 0;
2125 	const glw::GLint stride			= static_cast<glw::GLint>(n_pervertex_float_attributes * sizeof(glw::GLfloat));
2126 
2127 	gl.bindVertexArray(m_vao_id);
2128 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray() call failed.");
2129 
2130 	for (glw::GLuint n_clipdistance_entry = 0; n_clipdistance_entry < clipdistances_array_size; ++n_clipdistance_entry)
2131 	{
2132 		gl.vertexAttribPointer(po_clipdistance_array_location + n_clipdistance_entry, 1, /* size */
2133 							   GL_FLOAT, GL_FALSE,										 /* normalized */
2134 							   stride, (const glw::GLvoid*)current_offset);
2135 		GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribPointer() call failed.");
2136 
2137 		gl.enableVertexAttribArray(po_clipdistance_array_location + n_clipdistance_entry);
2138 		GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray() call failed.");
2139 
2140 		current_offset += sizeof(glw::GLfloat);
2141 	} /* for (all clip distance array value attributes) */
2142 
2143 	for (glw::GLuint n_culldistance_entry = 0; n_culldistance_entry < culldistances_array_size; ++n_culldistance_entry)
2144 	{
2145 		gl.vertexAttribPointer(po_culldistance_array_location + n_culldistance_entry, 1, /* size */
2146 							   GL_FLOAT, GL_FALSE,										 /* normalized */
2147 							   stride, (const glw::GLvoid*)current_offset);
2148 		GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribPointer() call failed.");
2149 
2150 		gl.enableVertexAttribArray(po_culldistance_array_location + n_culldistance_entry);
2151 		GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray() call failed.");
2152 
2153 		current_offset += sizeof(glw::GLfloat);
2154 	} /* for (all cull distance array value attributes) */
2155 
2156 	gl.vertexAttribPointer(po_position_location, 2, /* size */
2157 						   GL_FLOAT, GL_FALSE,		/* normalized */
2158 						   stride, (const glw::GLvoid*)current_offset);
2159 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribPointer() call failed");
2160 
2161 	gl.enableVertexAttribArray(po_position_location);
2162 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray() call failed");
2163 }
2164 
2165 /** @brief Cull Distance Functional Test deinitialization */
deinit()2166 void CullDistance::FunctionalTest::deinit()
2167 {
2168 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2169 
2170 	if (m_fbo_id != 0)
2171 	{
2172 		gl.deleteFramebuffers(1, &m_fbo_id);
2173 
2174 		m_fbo_id = 0;
2175 	}
2176 
2177 	if (m_to_id != 0)
2178 	{
2179 		gl.deleteTextures(1, &m_to_id);
2180 
2181 		m_to_id = 0;
2182 	}
2183 
2184 	if (m_vao_id != 0)
2185 	{
2186 		gl.deleteVertexArrays(1, &m_vao_id);
2187 
2188 		m_vao_id = 0;
2189 	}
2190 
2191 	deinitPO();
2192 }
2193 
2194 /** @brief Cull Distance Functional Test deinitialization of OpenGL programs */
deinitPO()2195 void CullDistance::FunctionalTest::deinitPO()
2196 {
2197 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2198 
2199 	if (m_po_id != 0)
2200 	{
2201 		gl.deleteProgram(m_po_id);
2202 
2203 		m_po_id = 0;
2204 	}
2205 }
2206 
2207 /** @brief Executes single render test case
2208  *
2209  * @param [in]  clipdistances_array_size    Size of gl_ClipDistance[] array
2210  * @param [in]  culldistances_array_size    Size of gl_CullDistance[] array
2211  * @param [in]  primitive_mode              Type of primitives to be rendered (see enum _primitive_mode)
2212  * @param [in]  use_tesselation             Indicate whether to use tessellation shader
2213  * @param [in]  fetch_culldistance_from_fs  Indicate whether to fetch gl_CullDistance and gl_ClipDistance values from the fragment shader
2214  */
executeRenderTest(glw::GLuint clipdistances_array_size,glw::GLuint culldistances_array_size,_primitive_mode primitive_mode,bool use_tesselation,bool fetch_culldistance_from_fs)2215 void CullDistance::FunctionalTest::executeRenderTest(glw::GLuint	 clipdistances_array_size,
2216 													 glw::GLuint	 culldistances_array_size,
2217 													 _primitive_mode primitive_mode, bool use_tesselation,
2218 													 bool fetch_culldistance_from_fs)
2219 {
2220 	const glw::Functions& gl						  = m_context.getRenderContext().getFunctions();
2221 	glw::GLenum			  mode						  = GL_NONE;
2222 	glw::GLuint			  n_clipped_vertices_real	 = 0;
2223 	glw::GLuint			  n_culled_primitives_real	= 0;
2224 	glw::GLuint			  n_not_clipped_vertices_real = 0;
2225 	const glw::GLuint	 primitive_vertices_count =
2226 		((primitive_mode == PRIMITIVE_MODE_LINES) ? 2 : (primitive_mode == PRIMITIVE_MODE_POINTS) ? 1 : 3);
2227 	const glw::GLuint stride_in_floats =
2228 		clipdistances_array_size + culldistances_array_size + 2 /* position's x, y*/ + 2 /* checkpoint x,y */;
2229 
2230 	switch (primitive_mode)
2231 	{
2232 	case PRIMITIVE_MODE_LINES:
2233 	{
2234 		mode = GL_LINES;
2235 
2236 		break;
2237 	}
2238 	case PRIMITIVE_MODE_POINTS:
2239 	{
2240 		mode = GL_POINTS;
2241 
2242 		break;
2243 	}
2244 	case PRIMITIVE_MODE_TRIANGLES:
2245 	{
2246 		mode = GL_TRIANGLES;
2247 
2248 		break;
2249 	}
2250 	default:
2251 		TCU_FAIL("Unknown primitive mode");
2252 	}
2253 
2254 	if (use_tesselation)
2255 	{
2256 		mode = GL_PATCHES;
2257 
2258 		gl.patchParameteri(GL_PATCH_VERTICES, primitive_vertices_count);
2259 		GLU_EXPECT_NO_ERROR(gl.getError(), "glPatchParameteri() call failed.");
2260 	}
2261 
2262 	gl.clear(GL_COLOR_BUFFER_BIT);
2263 	GLU_EXPECT_NO_ERROR(gl.getError(), "glClear() call failed.");
2264 
2265 	gl.useProgram(m_po_id);
2266 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram() call failed.");
2267 
2268 	for (glw::GLuint n_clipdistance_entry = 0; n_clipdistance_entry < clipdistances_array_size; n_clipdistance_entry++)
2269 	{
2270 		gl.enable(GL_CLIP_DISTANCE0 + n_clipdistance_entry);
2271 		GLU_EXPECT_NO_ERROR(gl.getError(), "gl.enable(GL_CLIP_DISTANCE)() call failed.");
2272 	} /* for (all clip distance array value attributes) */
2273 
2274 	gl.drawArrays(mode, 0, m_render_vertices);
2275 	GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawArray() call(s) failed.");
2276 
2277 	for (glw::GLuint n_clipdistance_entry = 0; n_clipdistance_entry < clipdistances_array_size; n_clipdistance_entry++)
2278 	{
2279 		gl.disable(GL_CLIP_DISTANCE0 + n_clipdistance_entry);
2280 		GLU_EXPECT_NO_ERROR(gl.getError(), "gl.disable(GL_CLIP_DISTANCE)() call failed.");
2281 	} /* for (all clip distance array value attributes) */
2282 
2283 	gl.useProgram(0);
2284 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram() call failed.");
2285 
2286 	/* Read generated texture into m_to_pixel_data_cache */
2287 	readTexturePixels();
2288 
2289 	for (glw::GLint n_primitive_index = 0; n_primitive_index < m_render_primitives; n_primitive_index++)
2290 	{
2291 		glw::GLuint base_index_of_primitive		 = n_primitive_index * primitive_vertices_count * stride_in_floats;
2292 		bool		primitive_culled			 = false;
2293 		glw::GLint  primitive_culled_by_distance = -1;
2294 
2295 		/* Check the bounding box is clear */
2296 		glw::GLuint base_index_of_vertex	  = base_index_of_primitive;
2297 		glw::GLuint checkpoint_position_index = base_index_of_vertex + clipdistances_array_size +
2298 												culldistances_array_size + 2 /* ignore vertex coordinates */;
2299 		glw::GLint checkpoint_x = glw::GLint(glw::GLfloat(m_to_width) * m_bo_data[checkpoint_position_index]);
2300 		glw::GLint checkpoint_y = glw::GLint(glw::GLfloat(m_to_height) * m_bo_data[checkpoint_position_index + 1]);
2301 		glw::GLint origin_x		= checkpoint_x - 1;
2302 		glw::GLint origin_y		= checkpoint_y - 1;
2303 		for (glw::GLint pixel_offset = 0; pixel_offset < m_sub_grid_cell_size; pixel_offset++)
2304 		{
2305 			if (readRedPixelValue(origin_x + pixel_offset, origin_y) != 0)
2306 			{
2307 				TCU_FAIL("Top edge of bounding box is overwritten");
2308 			}
2309 
2310 			if (readRedPixelValue(origin_x + m_sub_grid_cell_size - 1, origin_y + pixel_offset) != 0)
2311 			{
2312 				TCU_FAIL("Right edge of bounding box is overwritten");
2313 			}
2314 
2315 			if (readRedPixelValue(origin_x + m_sub_grid_cell_size - 1 - pixel_offset,
2316 								  origin_y + m_sub_grid_cell_size - 1) != 0)
2317 			{
2318 				TCU_FAIL("Bottom edge of bounding box is overwritten");
2319 			}
2320 
2321 			if (readRedPixelValue(origin_x, origin_y + m_sub_grid_cell_size - 1 - pixel_offset) != 0)
2322 			{
2323 				TCU_FAIL("Left edge of bounding box is overwritten");
2324 			}
2325 		}
2326 
2327 		/* Determine if primitive has been culled */
2328 		for (glw::GLuint n_culldistance_entry = 0; n_culldistance_entry < culldistances_array_size;
2329 			 n_culldistance_entry++)
2330 		{
2331 			bool distance_negative_in_all_primitive_vertices = true;
2332 
2333 			for (glw::GLuint n_primitive_vertex = 0; n_primitive_vertex < primitive_vertices_count;
2334 				 n_primitive_vertex++)
2335 			{
2336 				glw::GLint base_index_of_vertex_internal =
2337 					base_index_of_primitive + n_primitive_vertex * stride_in_floats;
2338 				glw::GLint	culldistance_array_offset = base_index_of_vertex_internal + clipdistances_array_size;
2339 				glw::GLfloat* vertex_culldistance_array = &m_bo_data[culldistance_array_offset];
2340 
2341 				if (vertex_culldistance_array[n_culldistance_entry] >= 0)
2342 				{
2343 					/* Primitive is not culled, due to one of its distances is not negative */
2344 					distance_negative_in_all_primitive_vertices = false;
2345 
2346 					/* Skip left vertices for this distance */
2347 					break;
2348 				}
2349 			}
2350 
2351 			/* The distance is negative in all primitive vertices, so this distance culls the primitive */
2352 			if (distance_negative_in_all_primitive_vertices)
2353 			{
2354 				primitive_culled			 = true;
2355 				primitive_culled_by_distance = n_culldistance_entry;
2356 
2357 				n_culled_primitives_real++;
2358 
2359 				/* Skip left distances from check */
2360 				break;
2361 			}
2362 		}
2363 
2364 		/* Validate culling */
2365 		if (primitive_culled)
2366 		{
2367 			/* Check whether primitive was culled and all its vertices are invisible */
2368 			for (glw::GLuint n_primitive_vertex = 0; n_primitive_vertex < primitive_vertices_count;
2369 				 n_primitive_vertex++)
2370 			{
2371 				glw::GLint base_index_of_vertex_internal =
2372 					base_index_of_primitive + n_primitive_vertex * stride_in_floats;
2373 				glw::GLint checkpoint_position_index_internal = base_index_of_vertex_internal +
2374 																clipdistances_array_size + culldistances_array_size +
2375 																2 /* ignore vertex coordinates */;
2376 				glw::GLint checkpoint_x_internal =
2377 					glw::GLint(glw::GLfloat(m_to_width) * m_bo_data[checkpoint_position_index_internal]);
2378 				glw::GLint checkpoint_y_internal =
2379 					glw::GLint(glw::GLfloat(m_to_height) * m_bo_data[checkpoint_position_index_internal + 1]);
2380 				glw::GLint vertex_color_red_value = readRedPixelValue(checkpoint_x_internal, checkpoint_y_internal);
2381 
2382 				/* Make sure vertex is invisible */
2383 				if (vertex_color_red_value != 0)
2384 				{
2385 					m_testCtx.getLog() << tcu::TestLog::Message << "Primitive number [" << n_primitive_index << "] "
2386 									   << "should be culled by distance [" << primitive_culled_by_distance << "]"
2387 									   << "but primitive vertex at (" << checkpoint_x << "," << checkpoint_y
2388 									   << ") is visible." << tcu::TestLog::EndMessage;
2389 
2390 					TCU_FAIL("Primitive is expected to be culled, but one of its vertices is visible.");
2391 				}
2392 			}
2393 
2394 			/* Primitive is culled, no reason to check clipping */
2395 			continue;
2396 		}
2397 
2398 		bool all_vertices_are_clipped = true;
2399 
2400 		for (glw::GLuint n_primitive_vertex = 0; n_primitive_vertex < primitive_vertices_count; n_primitive_vertex++)
2401 		{
2402 			glw::GLuint base_index_of_vertex_internal = base_index_of_primitive + n_primitive_vertex * stride_in_floats;
2403 			glw::GLuint clipdistance_array_index	  = base_index_of_vertex_internal;
2404 			glw::GLuint checkpoint_position_index_internal = base_index_of_vertex_internal + clipdistances_array_size +
2405 															 culldistances_array_size +
2406 															 2 /* ignore vertex coordinates */;
2407 			glw::GLint checkpoint_x_internal =
2408 				glw::GLint(glw::GLfloat(m_to_width) * m_bo_data[checkpoint_position_index_internal]);
2409 			glw::GLint checkpoint_y_internal =
2410 				glw::GLint(glw::GLfloat(m_to_height) * m_bo_data[checkpoint_position_index_internal + 1]);
2411 			glw::GLfloat* vertex_clipdistance_array  = &m_bo_data[clipdistance_array_index];
2412 			bool		  vertex_clipped			 = false;
2413 			glw::GLint	vertex_clipped_by_distance = 0;
2414 			glw::GLint	vertex_color_red_value	 = readRedPixelValue(checkpoint_x_internal, checkpoint_y_internal);
2415 
2416 			/* Check whether pixel should be clipped */
2417 			for (glw::GLuint n_clipdistance_entry = 0; n_clipdistance_entry < clipdistances_array_size;
2418 				 n_clipdistance_entry++)
2419 			{
2420 				if (vertex_clipdistance_array[n_clipdistance_entry] < 0)
2421 				{
2422 					vertex_clipped			   = true;
2423 					vertex_clipped_by_distance = n_clipdistance_entry;
2424 
2425 					break;
2426 				}
2427 			}
2428 
2429 			all_vertices_are_clipped &= vertex_clipped;
2430 
2431 			/* Validate whether real data same as expected */
2432 			if (vertex_clipped)
2433 			{
2434 				if (vertex_color_red_value != 0)
2435 				{
2436 					m_testCtx.getLog() << tcu::TestLog::Message << "In primitive number [" << n_primitive_index << "] "
2437 									   << "vertex at (" << checkpoint_x << "," << checkpoint_y << ") "
2438 									   << "should be clipped by distance [" << vertex_clipped_by_distance << "] "
2439 									   << "(distance value [" << vertex_clipdistance_array[vertex_clipped_by_distance]
2440 									   << "])" << tcu::TestLog::EndMessage;
2441 
2442 					TCU_FAIL("Vertex is expected to be clipped and invisible, while it is visible.");
2443 				}
2444 				else
2445 				{
2446 					n_clipped_vertices_real++;
2447 				}
2448 			}
2449 			else
2450 			{
2451 				if (vertex_color_red_value == 0)
2452 				{
2453 					m_testCtx.getLog() << tcu::TestLog::Message << "In primitive number [" << n_primitive_index << "] "
2454 									   << "vertex at (" << checkpoint_x << "," << checkpoint_y << ") "
2455 									   << "should not be clipped." << tcu::TestLog::EndMessage;
2456 
2457 					TCU_FAIL("Vertex is unexpectedly clipped or invisible");
2458 				}
2459 				else
2460 				{
2461 					n_not_clipped_vertices_real++;
2462 				}
2463 			}
2464 		}
2465 
2466 		if (!all_vertices_are_clipped)
2467 		{
2468 			/* Check fetched values from the shader (Point 2 of Basic Outline : "Use program that...") */
2469 			if (fetch_culldistance_from_fs)
2470 			{
2471 				for (glw::GLuint n_primitive_vertex = 0; n_primitive_vertex < primitive_vertices_count;
2472 					 n_primitive_vertex++)
2473 				{
2474 					/* Get shader output value */
2475 					glw::GLuint base_index_of_vertex_internal =
2476 						base_index_of_primitive + n_primitive_vertex * stride_in_floats;
2477 					glw::GLuint checkpoint_position_index_internal =
2478 						base_index_of_vertex_internal + clipdistances_array_size + culldistances_array_size +
2479 						2 /* ignore vertex coordinates */;
2480 					glw::GLuint culldistances_index = base_index_of_vertex_internal + clipdistances_array_size;
2481 					glw::GLint  checkpoint_x_internal =
2482 						glw::GLint(glw::GLfloat(m_to_width) * m_bo_data[checkpoint_position_index_internal]);
2483 					glw::GLint checkpoint_y_internal =
2484 						glw::GLint(glw::GLfloat(m_to_height) * m_bo_data[checkpoint_position_index_internal + 1]);
2485 					glw::GLint vertex_color_red_value = readRedPixelValue(checkpoint_x_internal, checkpoint_y_internal);
2486 
2487 					/* Calculate culldistances check sum hash */
2488 					float sum = 0.f;
2489 
2490 					for (glw::GLuint n_clipdistance_entry = 0; n_clipdistance_entry < clipdistances_array_size;
2491 						 ++n_clipdistance_entry)
2492 					{
2493 						sum += de::abs(m_bo_data[base_index_of_vertex_internal + n_clipdistance_entry]) *
2494 							   float(n_clipdistance_entry + 1);
2495 					}
2496 
2497 					for (glw::GLuint n_culldistance_entry = 0; n_culldistance_entry < culldistances_array_size;
2498 						 ++n_culldistance_entry)
2499 					{
2500 						sum += de::abs(m_bo_data[culldistances_index + n_culldistance_entry]) *
2501 							   float(n_culldistance_entry + 1 + clipdistances_array_size);
2502 					}
2503 
2504 					/* limit sum and return */
2505 					glw::GLint sum_hash =
2506 						glw::GLint(sum / glw::GLfloat((clipdistances_array_size + culldistances_array_size) *
2507 													  (clipdistances_array_size + culldistances_array_size + 1)) *
2508 								   65535.f /* normalizing to short */);
2509 					sum_hash = (sum_hash < 65536) ? sum_hash : 65535; /* clamping to short */
2510 
2511 					/* Compare against setup value */
2512 					if (std::abs(vertex_color_red_value - sum_hash) > 4 /* precision 4/65536 */)
2513 					{
2514 						m_testCtx.getLog() << tcu::TestLog::Message << "Primitive number [" << n_primitive_index << "] "
2515 										   << "should have culldistance hash sum " << sum_hash
2516 										   << "but primitive vertex at (" << checkpoint_x << "," << checkpoint_y
2517 										   << ") has sum hash equal to " << vertex_color_red_value
2518 										   << tcu::TestLog::EndMessage;
2519 
2520 						TCU_FAIL("Culled distances returned from fragment shader dose not match expected values.");
2521 					}
2522 				}
2523 			}
2524 		}
2525 	}
2526 
2527 	/* sub_grid cell size is 3*3 */
2528 	DE_ASSERT(m_render_primitives % 9 == 0);
2529 
2530 	/* Sanity check */
2531 	switch (primitive_mode)
2532 	{
2533 	case PRIMITIVE_MODE_LINES:
2534 	case PRIMITIVE_MODE_TRIANGLES:
2535 	{
2536 		/* Validate culled primitives */
2537 		if (culldistances_array_size == 0)
2538 		{
2539 			DE_ASSERT(n_culled_primitives_real == 0);
2540 		}
2541 		else
2542 		{
2543 			/* Each 3rd line or triangle should be culled by test design */
2544 			DE_ASSERT(glw::GLsizei(n_culled_primitives_real) == m_render_primitives / 3);
2545 		}
2546 
2547 		/* Validate clipped vertices */
2548 		if (clipdistances_array_size == 0)
2549 		{
2550 			DE_ASSERT(n_clipped_vertices_real == 0);
2551 		}
2552 		else
2553 		{
2554 #if defined(DE_DEBUG) && !defined(DE_COVERAGE_BUILD)
2555 			glw::GLint one_third_of_rendered_primitives = (m_render_primitives - n_culled_primitives_real) / 3;
2556 			glw::GLint n_clipped_vertices_expected		= /* One third of primitives has 0th vertex clipped */
2557 				one_third_of_rendered_primitives +
2558 				/* One third of primitives clipped completely     */
2559 				one_third_of_rendered_primitives * primitive_vertices_count;
2560 
2561 			DE_ASSERT(glw::GLint(n_clipped_vertices_real) == n_clipped_vertices_expected);
2562 #endif
2563 		}
2564 		break;
2565 	}
2566 
2567 	case PRIMITIVE_MODE_POINTS:
2568 	{
2569 		/* Validate culled primitives */
2570 		if (culldistances_array_size == 0)
2571 		{
2572 			DE_ASSERT(n_culled_primitives_real == 0);
2573 		}
2574 		else
2575 		{
2576 			/* 2/3 points should be culled by test design */
2577 			DE_ASSERT(glw::GLsizei(n_culled_primitives_real) == m_render_primitives * 2 / 3);
2578 		}
2579 
2580 		/* Validate clipped vertices */
2581 		if (clipdistances_array_size == 0)
2582 		{
2583 			DE_ASSERT(n_clipped_vertices_real == 0);
2584 		}
2585 		else
2586 		{
2587 #if defined(DE_DEBUG) && !defined(DE_COVERAGE_BUILD)
2588 			glw::GLint one_third_of_rendered_primitives = (m_render_primitives - n_culled_primitives_real) / 3;
2589 
2590 			/* 2/3 of rendered points should be clipped by test design */
2591 			DE_ASSERT(glw::GLint(n_clipped_vertices_real) == 2 * one_third_of_rendered_primitives);
2592 #endif
2593 		}
2594 
2595 		break;
2596 	}
2597 	default:
2598 		TCU_FAIL("Unknown primitive mode");
2599 	}
2600 }
2601 
2602 /** Executes test iteration.
2603  *
2604  *  @return Returns STOP when test has finished executing, CONTINUE if more iterations are needed.
2605  */
iterate()2606 tcu::TestNode::IterateResult CullDistance::FunctionalTest::iterate()
2607 {
2608 	/* This test should only be executed if ARB_cull_distance is supported, or if
2609 	 * we're running a GL4.5 context
2610 	 */
2611 	if (!m_context.getContextInfo().isExtensionSupported("GL_ARB_cull_distance") &&
2612 		!glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)))
2613 	{
2614 		throw tcu::NotSupportedError("GL_ARB_cull_distance is not supported");
2615 	}
2616 
2617 	const glw::Functions& gl			= m_context.getRenderContext().getFunctions();
2618 	bool				  has_succeeded = true;
2619 	bool is_core = glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5));
2620 
2621 	/* Retrieve important GL constant values */
2622 	glw::GLint gl_max_clip_distances_value					 = 0;
2623 	glw::GLint gl_max_combined_clip_and_cull_distances_value = 0;
2624 	glw::GLint gl_max_cull_distances_value					 = 0;
2625 
2626 	gl.getIntegerv(GL_MAX_CLIP_DISTANCES, &gl_max_clip_distances_value);
2627 	gl.getIntegerv(GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES, &gl_max_combined_clip_and_cull_distances_value);
2628 	gl.getIntegerv(GL_MAX_CULL_DISTANCES, &gl_max_cull_distances_value);
2629 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv() call(s) failed.");
2630 
2631 	gl.genTextures(1, &m_to_id);
2632 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() call failed.");
2633 
2634 	gl.bindTexture(GL_TEXTURE_2D, m_to_id);
2635 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindTexture() call failed.");
2636 
2637 	gl.texStorage2D(GL_TEXTURE_2D, 1, /* levels */
2638 					GL_R32F, m_to_width, m_to_height);
2639 	GLU_EXPECT_NO_ERROR(gl.getError(), "glTexStorage2D() call failed.");
2640 
2641 	/* Set up the draw/read FBO */
2642 	gl.genFramebuffers(1, &m_fbo_id);
2643 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenFramebuffers() call failed.");
2644 
2645 	gl.bindFramebuffer(GL_FRAMEBUFFER, m_fbo_id);
2646 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindFramebuffer() call failed.");
2647 
2648 	gl.framebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, m_to_id, 0); /* level */
2649 	GLU_EXPECT_NO_ERROR(gl.getError(), "glFramebufferTexture2D() call failed.");
2650 
2651 	/* Prepare a buffer object */
2652 	gl.genBuffers(1, &m_bo_id);
2653 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers() call failed.");
2654 
2655 	/* Prepare a VAO. We will configure separately for each iteration. */
2656 	gl.genVertexArrays(1, &m_vao_id);
2657 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays() call failed.");
2658 
2659 	/* Iterate over all functional tests */
2660 	struct _test_item
2661 	{
2662 		bool redeclare_clipdistances_array;
2663 		bool redeclare_culldistances_array;
2664 		bool dynamic_index_writes;
2665 		bool use_passthrough_gs;
2666 		bool use_passthrough_ts;
2667 		bool use_core_functionality;
2668 		bool fetch_culldistances;
2669 	} test_items[] = { /* Use the basic outline to test the basic functionality of cull distances. */
2670 					   {
2671 						   true,	/* redeclare_clipdistances_array */
2672 						   true,	/* redeclare_culldistances_array */
2673 						   false,   /* dynamic_index_writes          */
2674 						   false,   /* use_passthrough_gs            */
2675 						   false,   /* use_passthrough_ts            */
2676 						   is_core, /* use_core_functionality        */
2677 						   false	/* fetch_culldistances           */
2678 					   },
2679 					   /* Use the basic outline but don't redeclare gl_ClipDistance with a size. */
2680 					   {
2681 						   false,   /* redeclare_clipdistances_array */
2682 						   true,	/* redeclare_culldistances_array */
2683 						   false,   /* dynamic_index_writes          */
2684 						   false,   /* use_passthrough_gs            */
2685 						   false,   /* use_passthrough_ts            */
2686 						   is_core, /* use_core_functionality        */
2687 						   false	/* fetch_culldistances           */
2688 					   },
2689 					   /* Use the basic outline but don't redeclare gl_CullDistance with a size. */
2690 					   {
2691 						   true,	/* redeclare_clipdistances_array  */
2692 						   false,   /* redeclare_culldistances_array  */
2693 						   false,   /* dynamic_index_writes           */
2694 						   false,   /* use_passthrough_gs             */
2695 						   false,   /* use_passthrough_ts             */
2696 						   is_core, /* use_core_functionality         */
2697 						   false	/* fetch_culldistances            */
2698 					   },
2699 					   /* Use the basic outline but don't redeclare either gl_ClipDistance or
2700 		 * gl_CullDistance with a size.
2701 		 */
2702 					   {
2703 						   false,   /* redeclare_clipdistances_array */
2704 						   false,   /* redeclare_culldistances_array */
2705 						   false,   /* dynamic_index_writes          */
2706 						   false,   /* use_passthrough_gs            */
2707 						   false,   /* use_passthrough_ts            */
2708 						   is_core, /* use_core_functionality        */
2709 						   false	/* fetch_culldistances           */
2710 					   },
2711 					   /* Use the basic outline but use dynamic indexing when writing the elements
2712 		 * of the gl_ClipDistance and gl_CullDistance arrays.
2713 		 */
2714 					   {
2715 						   true,	/* redeclare_clipdistances_array */
2716 						   true,	/* redeclare_culldistances_array */
2717 						   true,	/* dynamic_index_writes          */
2718 						   false,   /* use_passthrough_gs            */
2719 						   false,   /* use_passthrough_ts            */
2720 						   is_core, /* use_core_functionality        */
2721 						   false	/* fetch_culldistances           */
2722 					   },
2723 					   /* Use the basic outline but add a geometry shader to the program that
2724 		 * simply passes through all written clip and cull distances.
2725 		 */
2726 					   {
2727 						   true,	/* redeclare_clipdistances_array */
2728 						   true,	/* redeclare_culldistances_array */
2729 						   false,   /* dynamic_index_writes          */
2730 						   true,	/* use_passthrough_gs            */
2731 						   false,   /* use_passthrough_ts            */
2732 						   is_core, /* use_core_functionality        */
2733 						   false	/* fetch_culldistances           */
2734 					   },
2735 					   /* Use the basic outline but add a tessellation control and tessellation
2736 		 * evaluation shader to the program which simply pass through all written
2737 		 * clip and cull distances.
2738 		 */
2739 					   {
2740 						   true,	/* redeclare_clipdistances_array */
2741 						   true,	/* redeclare_culldistances_array */
2742 						   false,   /* dynamic_index_writes          */
2743 						   false,   /* use_passthrough_gs            */
2744 						   true,	/* use_passthrough_ts            */
2745 						   is_core, /* use_core_functionality        */
2746 						   false	/* fetch_culldistances           */
2747 					   },
2748 					   /* Test that using #extension with GL_ARB_cull_distance allows using the
2749 		 * feature even with an earlier version of GLSL. Also test that the
2750 		 * extension name is available as preprocessor #define.
2751 		 */
2752 					   {
2753 						   true,  /* redeclare_clipdistances_array */
2754 						   true,  /* redeclare_culldistances_array */
2755 						   false, /* dynamic_index_writes          */
2756 						   false, /* use_passthrough_gs            */
2757 						   false, /* use_passthrough_ts            */
2758 						   false, /* use_core_functionality        */
2759 						   false  /* fetch_culldistances           */
2760 					   },
2761 					   /* Use a program that has only a vertex shader and a fragment shader.
2762 		 * The vertex shader should redeclare gl_ClipDistance with a size that
2763 		 * fits all enabled cull distances. Also redeclare gl_CullDistance with a
2764 		 * size. The sum of the two sizes should not be more than MAX_COMBINED_-
2765 		 * CLIP_AND_CULL_DISTANCES. The fragment shader should output the cull
2766 		 * distances written by the vertex shader by reading them from the built-in
2767 		 * array gl_CullDistance.
2768 		 */
2769 					   {
2770 						   true,  /* redeclare_clipdistances_array */
2771 						   true,  /* redeclare_culldistances_array */
2772 						   false, /* dynamic_index_writes          */
2773 						   false, /* use_passthrough_gs            */
2774 						   false, /* use_passthrough_ts            */
2775 						   false, /* use_core_functionality        */
2776 						   true   /* fetch_culldistances           */
2777 					   }
2778 	};
2779 	const glw::GLuint n_test_items = sizeof(test_items) / sizeof(test_items[0]);
2780 
2781 	gl.viewport(0, 0, m_to_width, m_to_height);
2782 	GLU_EXPECT_NO_ERROR(gl.getError(), "glViewport() call failed.");
2783 
2784 	gl.clearColor(0.0f, 0.0f, 0.0f, 1.0f);
2785 	GLU_EXPECT_NO_ERROR(gl.getError(), "glClearColor() call failed.");
2786 
2787 	for (glw::GLuint n_test_item = 0; n_test_item < n_test_items; ++n_test_item)
2788 	{
2789 		/* Check for OpenGL feature support */
2790 		if (test_items[n_test_item].use_passthrough_ts)
2791 		{
2792 			if (!glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 0)) &&
2793 				!m_context.getContextInfo().isExtensionSupported("GL_ARB_tessellation_shader"))
2794 			{
2795 				continue; // no tessellation shader support
2796 			}
2797 		}
2798 
2799 		const _test_item&	 current_test_item						= test_items[n_test_item];
2800 		const _primitive_mode primitive_modes[PRIMITIVE_MODE_COUNT] = { PRIMITIVE_MODE_LINES, PRIMITIVE_MODE_POINTS,
2801 																		PRIMITIVE_MODE_TRIANGLES };
2802 
2803 		for (glw::GLuint primitive_mode_index = 0; primitive_mode_index < PRIMITIVE_MODE_COUNT; ++primitive_mode_index)
2804 		{
2805 			_primitive_mode primitive_mode = primitive_modes[primitive_mode_index];
2806 
2807 			/* Iterate over a set of gl_ClipDistances[] and gl_CullDistances[] array sizes */
2808 			for (glw::GLint n_iteration = 0; n_iteration <= gl_max_combined_clip_and_cull_distances_value;
2809 				 ++n_iteration)
2810 			{
2811 				glw::GLuint clipdistances_array_size = 0;
2812 				glw::GLuint culldistances_array_size = 0;
2813 
2814 				if (n_iteration != 0 && n_iteration <= gl_max_clip_distances_value)
2815 				{
2816 					clipdistances_array_size = n_iteration;
2817 				}
2818 
2819 				if ((gl_max_combined_clip_and_cull_distances_value - n_iteration) < gl_max_cull_distances_value)
2820 				{
2821 					culldistances_array_size = gl_max_combined_clip_and_cull_distances_value - n_iteration;
2822 				}
2823 				else
2824 				{
2825 					culldistances_array_size = gl_max_cull_distances_value;
2826 				}
2827 
2828 				if (clipdistances_array_size == 0 && culldistances_array_size == 0)
2829 				{
2830 					/* Skip the dummy iteration */
2831 					continue;
2832 				}
2833 
2834 				if (current_test_item.fetch_culldistances && (primitive_mode != PRIMITIVE_MODE_POINTS))
2835 				{
2836 					continue;
2837 				}
2838 
2839 				/* Create a program to run */
2840 				buildPO(clipdistances_array_size, culldistances_array_size, current_test_item.dynamic_index_writes,
2841 						primitive_mode, current_test_item.redeclare_clipdistances_array,
2842 						current_test_item.redeclare_culldistances_array, current_test_item.use_core_functionality,
2843 						current_test_item.use_passthrough_gs, current_test_item.use_passthrough_ts,
2844 						current_test_item.fetch_culldistances);
2845 
2846 				/* Initialize VAO data */
2847 				configureVAO(clipdistances_array_size, culldistances_array_size, primitive_mode);
2848 
2849 				/* Run GLSL program and check results */
2850 				executeRenderTest(clipdistances_array_size, culldistances_array_size, primitive_mode,
2851 								  current_test_item.use_passthrough_ts, current_test_item.fetch_culldistances);
2852 
2853 			} /* for (all iterations) */
2854 		}	 /* for (all test modes) */
2855 	}		  /* for (all test items) */
2856 
2857 	/* All done */
2858 	if (has_succeeded)
2859 	{
2860 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
2861 	}
2862 	else
2863 	{
2864 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
2865 	}
2866 
2867 	return STOP;
2868 }
2869 
2870 /** Returns pixel red component read from texture at position x, y.
2871  *
2872  *  @param x x-coordinate to read pixel color component from
2873  *  @param y y-coordinate to read pixel color component from
2874  **/
readRedPixelValue(glw::GLint x,glw::GLint y)2875 glw::GLint CullDistance::FunctionalTest::readRedPixelValue(glw::GLint x, glw::GLint y)
2876 {
2877 	glw::GLint result = -1;
2878 
2879 	DE_ASSERT(x >= 0 && (glw::GLuint)x < m_to_width);
2880 	DE_ASSERT(y >= 0 && (glw::GLuint)y < m_to_height);
2881 
2882 	result = m_to_pixel_data_cache[(m_to_width * y + x) * m_to_pixel_data_cache_color_components];
2883 
2884 	return result;
2885 }
2886 
2887 /** Reads texture into m_to_pixel_data_cache.
2888  *  Texture size determined by fields m_to_width, m_to_height
2889  **/
readTexturePixels()2890 void CullDistance::FunctionalTest::readTexturePixels()
2891 {
2892 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2893 
2894 	m_to_pixel_data_cache.clear();
2895 
2896 	m_to_pixel_data_cache.resize(m_to_width * m_to_height * m_to_pixel_data_cache_color_components);
2897 
2898 	/* Read vertex from texture */
2899 	gl.readPixels(0,		   /* x      */
2900 				  0,		   /* y      */
2901 				  m_to_width,  /* width  */
2902 				  m_to_height, /* height */
2903 				  GL_RGBA, GL_UNSIGNED_SHORT, &m_to_pixel_data_cache[0]);
2904 	GLU_EXPECT_NO_ERROR(gl.getError(), "glReadPixels() call failed.");
2905 }
2906 
2907 /** Constructor.
2908  *
2909  *  @param context Rendering context handle.
2910  **/
NegativeTest(deqp::Context & context)2911 CullDistance::NegativeTest::NegativeTest(deqp::Context& context)
2912 	: TestCase(context, "negative", "Cull Distance Negative Test")
2913 	, m_fs_id(0)
2914 	, m_po_id(0)
2915 	, m_temp_buffer(DE_NULL)
2916 	, m_vs_id(0)
2917 {
2918 	/* Left blank on purpose */
2919 }
2920 
2921 /** @brief Cull Distance Negative Test deinitialization */
deinit()2922 void CullDistance::NegativeTest::deinit()
2923 {
2924 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2925 
2926 	if (m_fs_id != 0)
2927 	{
2928 		gl.deleteShader(m_fs_id);
2929 
2930 		m_fs_id = 0;
2931 	}
2932 
2933 	if (m_po_id != 0)
2934 	{
2935 		gl.deleteProgram(m_po_id);
2936 
2937 		m_po_id = 0;
2938 	}
2939 
2940 	if (m_vs_id != 0)
2941 	{
2942 		gl.deleteShader(m_vs_id);
2943 
2944 		m_vs_id = 0;
2945 	}
2946 
2947 	if (m_temp_buffer != DE_NULL)
2948 	{
2949 		delete[] m_temp_buffer;
2950 
2951 		m_temp_buffer = DE_NULL;
2952 	}
2953 }
2954 
2955 /** @brief Get string description of test with given parameters
2956  *
2957  *  @param [in] n_test_iteration                    Test iteration number
2958  *  @param [in] should_redeclare_output_variables   Indicate whether test redeclared gl_ClipDistance and gl_CullDistance
2959  *  @param [in] use_dynamic_index_based_writes      Indicate whether test used dynamic index-based setters
2960  *
2961  *  @return String containing description.
2962  */
getTestDescription(int n_test_iteration,bool should_redeclare_output_variables,bool use_dynamic_index_based_writes)2963 std::string CullDistance::NegativeTest::getTestDescription(int n_test_iteration, bool should_redeclare_output_variables,
2964 														   bool use_dynamic_index_based_writes)
2965 {
2966 	std::stringstream stream;
2967 
2968 	stream << "Test iteration [" << n_test_iteration << "] which uses a vertex shader that:\n\n"
2969 		   << ((should_redeclare_output_variables) ?
2970 				   "* redeclares gl_ClipDistance and gl_CullDistance arrays\n" :
2971 				   "* does not redeclare gl_ClipDistance and gl_CullDistance arrays\n")
2972 		   << ((use_dynamic_index_based_writes) ? "* uses dynamic index-based writes\n" : "* uses static writes\n");
2973 
2974 	return stream.str();
2975 }
2976 
2977 /** Executes test iteration.
2978  *
2979  *  @return Returns STOP when test has finished executing, CONTINUE if more iterations are needed.
2980  */
iterate()2981 tcu::TestNode::IterateResult CullDistance::NegativeTest::iterate()
2982 {
2983 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2984 
2985 	/* Build the test shaders. */
2986 	const glw::GLchar* token_dynamic_index_based_writes = "DYNAMIC_INDEX_BASED_WRITES";
2987 	const glw::GLchar* token_insert_static_writes		= "INSERT_STATIC_WRITES";
2988 	const glw::GLchar* token_n_gl_clipdistance_entries  = "N_GL_CLIPDISTANCE_ENTRIES";
2989 	const glw::GLchar* token_n_gl_culldistance_entries  = "N_GL_CULLDISTANCE_ENTRIES";
2990 	const glw::GLchar* token_redeclare_output_variables = "REDECLARE_OUTPUT_VARIABLES";
2991 
2992 	const glw::GLchar* fs_body = "#version 130\n"
2993 								 "\n"
2994 								 "void main()\n"
2995 								 "{\n"
2996 								 "}\n";
2997 
2998 	const glw::GLchar* vs_body_preamble = "#version 130\n"
2999 										  "\n"
3000 										  "    #extension GL_ARB_cull_distance : require\n"
3001 										  "\n";
3002 
3003 	const glw::GLchar* vs_body_main = "#ifdef REDECLARE_OUTPUT_VARIABLES\n"
3004 									  "    out float gl_ClipDistance[N_GL_CLIPDISTANCE_ENTRIES];\n"
3005 									  "    out float gl_CullDistance[N_GL_CULLDISTANCE_ENTRIES];\n"
3006 									  "#endif\n"
3007 									  "\n"
3008 									  "void main()\n"
3009 									  "{\n"
3010 									  "#ifdef DYNAMIC_INDEX_BASED_WRITES\n"
3011 									  "    for (int n_clipdistance_entry = 0;\n"
3012 									  "             n_clipdistance_entry < N_GL_CLIPDISTANCE_ENTRIES;\n"
3013 									  "           ++n_clipdistance_entry)\n"
3014 									  "    {\n"
3015 									  "        gl_ClipDistance[n_clipdistance_entry] = float(n_clipdistance_entry) / "
3016 									  "float(N_GL_CLIPDISTANCE_ENTRIES);\n"
3017 									  "    }\n"
3018 									  "\n"
3019 									  "    for (int n_culldistance_entry = 0;\n"
3020 									  "             n_culldistance_entry < N_GL_CULLDISTANCE_ENTRIES;\n"
3021 									  "           ++n_culldistance_entry)\n"
3022 									  "    {\n"
3023 									  "        gl_CullDistance[n_culldistance_entry] = float(n_culldistance_entry) / "
3024 									  "float(N_GL_CULLDISTANCE_ENTRIES);\n"
3025 									  "    }\n"
3026 									  "#else\n"
3027 									  "    INSERT_STATIC_WRITES\n"
3028 									  "#endif\n"
3029 									  "}\n";
3030 
3031 	/* This test should only be executed if ARB_cull_distance is supported, or if
3032 	 * we're running a GL4.5 context
3033 	 */
3034 	if (!m_context.getContextInfo().isExtensionSupported("GL_ARB_cull_distance") &&
3035 		!glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)))
3036 	{
3037 		throw tcu::NotSupportedError("GL_ARB_cull_distance is not supported");
3038 	}
3039 
3040 	/* It only makes sense to run this test if GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES
3041 	 * is lower than a sum of GL_MAX_CLIP_DISTANCES and GL_MAX_CLIP_CULL_DISTANCES.
3042 	 */
3043 	glw::GLint  gl_max_clip_distances_value					  = 0;
3044 	glw::GLint  gl_max_combined_clip_and_cull_distances_value = 0;
3045 	glw::GLint  gl_max_cull_distances_value					  = 0;
3046 	glw::GLuint n_gl_clipdistance_array_items				  = 0;
3047 	std::string n_gl_clipdistance_array_items_string;
3048 	glw::GLuint n_gl_culldistance_array_items = 0;
3049 	std::string n_gl_culldistance_array_items_string;
3050 	std::string static_write_shader_body_part;
3051 
3052 	gl.getIntegerv(GL_MAX_CLIP_DISTANCES, &gl_max_clip_distances_value);
3053 	gl.getIntegerv(GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES, &gl_max_combined_clip_and_cull_distances_value);
3054 	gl.getIntegerv(GL_MAX_CULL_DISTANCES, &gl_max_cull_distances_value);
3055 
3056 	if (gl_max_clip_distances_value + gl_max_cull_distances_value < gl_max_combined_clip_and_cull_distances_value)
3057 	{
3058 		m_testCtx.getLog() << tcu::TestLog::Message
3059 						   << "GL_MAX_COMBINED_CLIP_AND_CULL_DISTANCES is larger than or equal to "
3060 							  "the sum of GL_MAX_CLIP_DISTANCES and GL_MAX_CULL_DISTANCES. Skipping."
3061 						   << tcu::TestLog::EndMessage;
3062 
3063 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
3064 
3065 		return STOP;
3066 	}
3067 
3068 	n_gl_clipdistance_array_items = gl_max_clip_distances_value;
3069 	n_gl_culldistance_array_items = gl_max_combined_clip_and_cull_distances_value - gl_max_clip_distances_value + 1;
3070 
3071 	/* Determine the number of items we will want the gl_ClipDistance and gl_CullDistance arrays
3072 	 * to hold for test iterations that will re-declare the built-in output variables.
3073 	 */
3074 	{
3075 		std::stringstream temp_sstream;
3076 
3077 		temp_sstream << n_gl_clipdistance_array_items;
3078 
3079 		n_gl_clipdistance_array_items_string = temp_sstream.str();
3080 	}
3081 
3082 	{
3083 		std::stringstream temp_sstream;
3084 
3085 		temp_sstream << n_gl_culldistance_array_items;
3086 
3087 		n_gl_culldistance_array_items_string = temp_sstream.str();
3088 	}
3089 
3090 	/* Form the "static write" shader body part. */
3091 	{
3092 		std::stringstream temp_sstream;
3093 
3094 		temp_sstream << "gl_ClipDistance[" << n_gl_clipdistance_array_items_string.c_str() << "] = 0.0f;\n"
3095 					 << "gl_CullDistance[" << n_gl_culldistance_array_items_string.c_str() << "] = 0.0f;\n";
3096 
3097 		static_write_shader_body_part = temp_sstream.str();
3098 	}
3099 
3100 	/* Prepare GL objects before we continue */
3101 	glw::GLint compile_status = GL_FALSE;
3102 
3103 	m_fs_id = gl.createShader(GL_FRAGMENT_SHADER);
3104 	m_po_id = gl.createProgram();
3105 	m_vs_id = gl.createShader(GL_VERTEX_SHADER);
3106 
3107 	GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram() / glCreateShader() calls failed.");
3108 
3109 	gl.attachShader(m_po_id, m_fs_id);
3110 	gl.attachShader(m_po_id, m_vs_id);
3111 
3112 	GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader() call(s) failed.");
3113 
3114 	gl.shaderSource(m_fs_id, 1,			/* count */
3115 					&fs_body, DE_NULL); /* length */
3116 	GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource() call failed.");
3117 
3118 	gl.compileShader(m_fs_id);
3119 	GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader() call failed.");
3120 
3121 	gl.getShaderiv(m_fs_id, GL_COMPILE_STATUS, &compile_status);
3122 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv() call failed.");
3123 
3124 	if (compile_status == GL_FALSE)
3125 	{
3126 		TCU_FAIL("Fragment shader failed to compile.");
3127 	}
3128 
3129 	/* Run three separate test iterations. */
3130 	struct _test_item
3131 	{
3132 		bool should_redeclare_output_variables;
3133 		bool use_dynamic_index_based_writes;
3134 	} test_items[] = { /* Negative Test 1 */
3135 					   { true, false },
3136 
3137 					   /* Negative Test 2 */
3138 					   { false, false },
3139 
3140 					   /* Negative Test 3 */
3141 					   { false, true }
3142 	};
3143 	const unsigned int n_test_items = sizeof(test_items) / sizeof(test_items[0]);
3144 
3145 	for (unsigned int n_test_item = 0; n_test_item < n_test_items; ++n_test_item)
3146 	{
3147 		const _test_item& current_test_item = test_items[n_test_item];
3148 
3149 		/* Prepare vertex shader body */
3150 		std::size_t		  token_position = std::string::npos;
3151 		std::stringstream vs_body_sstream;
3152 		std::string		  vs_body_string;
3153 
3154 		vs_body_sstream << vs_body_preamble << "\n";
3155 
3156 		if (current_test_item.should_redeclare_output_variables)
3157 		{
3158 			vs_body_sstream << "#define " << token_redeclare_output_variables << "\n";
3159 		}
3160 
3161 		if (current_test_item.use_dynamic_index_based_writes)
3162 		{
3163 			vs_body_sstream << "#define " << token_dynamic_index_based_writes << "\n";
3164 		}
3165 
3166 		vs_body_sstream << vs_body_main;
3167 
3168 		/* Replace tokens with meaningful values */
3169 		vs_body_string = vs_body_sstream.str();
3170 
3171 		while ((token_position = vs_body_string.find(token_n_gl_clipdistance_entries)) != std::string::npos)
3172 		{
3173 			vs_body_string = vs_body_string.replace(token_position, strlen(token_n_gl_clipdistance_entries),
3174 													n_gl_clipdistance_array_items_string);
3175 		}
3176 
3177 		while ((token_position = vs_body_string.find(token_n_gl_culldistance_entries)) != std::string::npos)
3178 		{
3179 			vs_body_string = vs_body_string.replace(token_position, strlen(token_n_gl_clipdistance_entries),
3180 													n_gl_culldistance_array_items_string);
3181 		}
3182 
3183 		while ((token_position = vs_body_string.find(token_insert_static_writes)) != std::string::npos)
3184 		{
3185 			vs_body_string = vs_body_string.replace(token_position, strlen(token_insert_static_writes),
3186 													static_write_shader_body_part);
3187 		}
3188 
3189 		/* Try to compile the vertex shader */
3190 		glw::GLint  compile_status_internal = GL_FALSE;
3191 		const char* vs_body_raw_ptr			= vs_body_string.c_str();
3192 
3193 		gl.shaderSource(m_vs_id, 1,					/* count */
3194 						&vs_body_raw_ptr, DE_NULL); /* length */
3195 		GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource() call failed.");
3196 
3197 		gl.compileShader(m_vs_id);
3198 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader() call failed.");
3199 
3200 		gl.getShaderiv(m_vs_id, GL_COMPILE_STATUS, &compile_status_internal);
3201 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv() call failed.");
3202 
3203 		if (compile_status_internal == GL_FALSE)
3204 		{
3205 			glw::GLint buffer_size = 0;
3206 
3207 			/* Log the compilation error */
3208 			m_testCtx.getLog() << tcu::TestLog::Message
3209 							   << getTestDescription(n_test_item, current_test_item.should_redeclare_output_variables,
3210 													 current_test_item.use_dynamic_index_based_writes)
3211 							   << "has failed (as expected) to compile with the following info log:\n\n"
3212 							   << tcu::TestLog::EndMessage;
3213 
3214 			gl.getShaderiv(m_vs_id, GL_INFO_LOG_LENGTH, &buffer_size);
3215 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv() call failed.");
3216 
3217 			m_temp_buffer = new glw::GLchar[buffer_size + 1];
3218 
3219 			memset(m_temp_buffer, 0, buffer_size + 1);
3220 
3221 			gl.getShaderInfoLog(m_vs_id, buffer_size, DE_NULL, /* length */
3222 								m_temp_buffer);
3223 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog() call failed.");
3224 
3225 			m_testCtx.getLog() << tcu::TestLog::Message << m_temp_buffer << tcu::TestLog::EndMessage;
3226 
3227 			delete[] m_temp_buffer;
3228 			m_temp_buffer = DE_NULL;
3229 
3230 			/* Move on to the next iteration */
3231 			continue;
3232 		}
3233 
3234 		/* Try to link the program object */
3235 		glw::GLint link_status = GL_FALSE;
3236 
3237 		gl.linkProgram(m_po_id);
3238 		GLU_EXPECT_NO_ERROR(gl.getError(), "glLinkProgram() call failed.");
3239 
3240 		gl.getProgramiv(m_po_id, GL_LINK_STATUS, &link_status);
3241 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv() call failed.");
3242 
3243 		if (link_status == GL_TRUE)
3244 		{
3245 			m_testCtx.getLog() << tcu::TestLog::Message
3246 							   << getTestDescription(n_test_item, current_test_item.should_redeclare_output_variables,
3247 													 current_test_item.use_dynamic_index_based_writes)
3248 							   << "has linked successfully which is invalid!" << tcu::TestLog::EndMessage;
3249 
3250 			TCU_FAIL("Program object has linked successfully, even though the process should have failed.");
3251 		}
3252 		else
3253 		{
3254 			glw::GLint buffer_size = 0;
3255 
3256 			m_testCtx.getLog() << tcu::TestLog::Message
3257 							   << getTestDescription(n_test_item, current_test_item.should_redeclare_output_variables,
3258 													 current_test_item.use_dynamic_index_based_writes)
3259 							   << "has failed (as expected) to link with the following info log:\n\n"
3260 							   << tcu::TestLog::EndMessage;
3261 
3262 			gl.getProgramiv(m_po_id, GL_INFO_LOG_LENGTH, &buffer_size);
3263 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv() call failed.");
3264 
3265 			m_temp_buffer = new glw::GLchar[buffer_size + 1];
3266 
3267 			memset(m_temp_buffer, 0, buffer_size + 1);
3268 
3269 			gl.getProgramInfoLog(m_po_id, buffer_size, DE_NULL, /* length */
3270 								 m_temp_buffer);
3271 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog() call failed.");
3272 
3273 			m_testCtx.getLog() << tcu::TestLog::Message << m_temp_buffer << tcu::TestLog::EndMessage;
3274 
3275 			delete[] m_temp_buffer;
3276 			m_temp_buffer = DE_NULL;
3277 		}
3278 	} /* for (all test items) */
3279 
3280 	/* All done */
3281 	m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
3282 
3283 	return STOP;
3284 }
3285 
3286 /** Constructor.
3287  *
3288  *  @param context Rendering context.
3289  */
Tests(deqp::Context & context)3290 CullDistance::Tests::Tests(deqp::Context& context) : TestCaseGroup(context, "cull_distance", "Cull Distance Test Suite")
3291 {
3292 }
3293 
3294 /** Initializes the test group contents. */
init()3295 void CullDistance::Tests::init()
3296 {
3297 	addChild(new CullDistance::APICoverageTest(m_context));
3298 	addChild(new CullDistance::FunctionalTest(m_context));
3299 	addChild(new CullDistance::NegativeTest(m_context));
3300 }
3301 } /* glcts namespace */
3302