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  gl4cDirectStateAccessVertexArraysTests.cpp
27  * \brief Conformance tests for the Direct State Access feature functionality (Vertex Array Objects access part).
28  */ /*-----------------------------------------------------------------------------------------------------------*/
29 
30 /* Includes. */
31 #include "gl4cDirectStateAccessTests.hpp"
32 
33 #include "deSharedPtr.hpp"
34 
35 #include "gluContextInfo.hpp"
36 #include "gluDefs.hpp"
37 #include "gluPixelTransfer.hpp"
38 #include "gluStrUtil.hpp"
39 
40 #include "tcuFuzzyImageCompare.hpp"
41 #include "tcuImageCompare.hpp"
42 #include "tcuRenderTarget.hpp"
43 #include "tcuSurface.hpp"
44 #include "tcuTestLog.hpp"
45 
46 #include "glw.h"
47 #include "glwFunctions.hpp"
48 
49 #include <algorithm>
50 #include <climits>
51 #include <cmath>
52 #include <set>
53 #include <sstream>
54 #include <stack>
55 
56 namespace gl4cts
57 {
58 namespace DirectStateAccess
59 {
60 namespace VertexArrays
61 {
62 /******************************** Creation Test Implementation   ********************************/
63 
64 /** @brief Creation Test constructor.
65  *
66  *  @param [in] context     OpenGL context.
67  */
CreationTest(deqp::Context & context)68 CreationTest::CreationTest(deqp::Context& context)
69 	: deqp::TestCase(context, "vertex_arrays_creation", "Vertex Array Objects Creation Test")
70 {
71 	/* Intentionally left blank. */
72 }
73 
74 /** @brief Iterate Creation Test cases.
75  *
76  *  @return Iteration result.
77  */
iterate()78 tcu::TestNode::IterateResult CreationTest::iterate()
79 {
80 	/* Shortcut for GL functionality. */
81 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
82 
83 	/* Get context setup. */
84 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
85 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
86 
87 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
88 	{
89 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
90 
91 		return STOP;
92 	}
93 
94 	/* Running tests. */
95 	bool is_ok	= true;
96 	bool is_error = false;
97 
98 	/* VertexArrays' objects */
99 	static const glw::GLuint vertex_arrays_count = 2;
100 
101 	glw::GLuint vertex_arrays_legacy[vertex_arrays_count] = {};
102 	glw::GLuint vertex_arrays_dsa[vertex_arrays_count]	= {};
103 
104 	try
105 	{
106 		/* Check legacy state creation. */
107 		gl.genVertexArrays(vertex_arrays_count, vertex_arrays_legacy);
108 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays have failed");
109 
110 		for (glw::GLuint i = 0; i < vertex_arrays_count; ++i)
111 		{
112 			if (gl.isVertexArray(vertex_arrays_legacy[i]))
113 			{
114 				is_ok = false;
115 
116 				/* Log. */
117 				m_context.getTestContext().getLog()
118 					<< tcu::TestLog::Message
119 					<< "GenVertexArrays has created default objects, but it should create only a names."
120 					<< tcu::TestLog::EndMessage;
121 			}
122 		}
123 
124 		/* Check direct state creation. */
125 		gl.createVertexArrays(vertex_arrays_count, vertex_arrays_dsa);
126 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays have failed");
127 
128 		for (glw::GLuint i = 0; i < vertex_arrays_count; ++i)
129 		{
130 			if (!gl.isVertexArray(vertex_arrays_dsa[i]))
131 			{
132 				is_ok = false;
133 
134 				/* Log. */
135 				m_context.getTestContext().getLog() << tcu::TestLog::Message
136 													<< "CreateVertexArrays has not created default objects."
137 													<< tcu::TestLog::EndMessage;
138 			}
139 		}
140 	}
141 	catch (...)
142 	{
143 		is_ok	= false;
144 		is_error = true;
145 	}
146 
147 	/* Cleanup. */
148 	for (glw::GLuint i = 0; i < vertex_arrays_count; ++i)
149 	{
150 		if (vertex_arrays_legacy[i])
151 		{
152 			gl.deleteVertexArrays(1, &vertex_arrays_legacy[i]);
153 
154 			vertex_arrays_legacy[i] = 0;
155 		}
156 
157 		if (vertex_arrays_dsa[i])
158 		{
159 			gl.deleteVertexArrays(1, &vertex_arrays_dsa[i]);
160 
161 			vertex_arrays_dsa[i] = 0;
162 		}
163 	}
164 
165 	/* Errors clean up. */
166 	while (gl.getError())
167 		;
168 
169 	/* Result's setup. */
170 	if (is_ok)
171 	{
172 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
173 	}
174 	else
175 	{
176 		if (is_error)
177 		{
178 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
179 		}
180 		else
181 		{
182 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
183 		}
184 	}
185 
186 	return STOP;
187 }
188 
189 /******************************** Vertex Array Object Enable Disable Attributes Test Implementation   ********************************/
190 
191 /** @brief Vertex Array Object Enable Disable Attributes Test constructor.
192  *
193  *  @param [in] context     OpenGL context.
194  */
EnableDisableAttributesTest(deqp::Context & context)195 EnableDisableAttributesTest::EnableDisableAttributesTest(deqp::Context& context)
196 	: deqp::TestCase(context, "vertex_arrays_enable_disable_attributes",
197 					 "Vertex Array Objects Enable Disable Attributes Test")
198 	, m_po_even(0)
199 	, m_po_odd(0)
200 	, m_vao(0)
201 	, m_bo(0)
202 	, m_bo_xfb(0)
203 	, m_max_attributes(16) /* Required Minimum: OpenGL 4.5 Table 23.57: Implementation Dependent Vertex Shader Limits */
204 {
205 	/* Intentionally left blank. */
206 }
207 
208 /** @brief Iterate Vertex Array Object Enable Disable Attributes Test cases.
209  *
210  *  @return Iteration result.
211  */
iterate()212 tcu::TestNode::IterateResult EnableDisableAttributesTest::iterate()
213 {
214 	/* Shortcut for GL functionality. */
215 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
216 
217 	/* Get context setup. */
218 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
219 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
220 
221 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
222 	{
223 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
224 
225 		return STOP;
226 	}
227 
228 	/* Running tests. */
229 	bool is_ok	= true;
230 	bool is_error = false;
231 
232 	try
233 	{
234 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIBS, &m_max_attributes);
235 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv(GL_MAX_VERTEX_ATTRIBS, ...) have failed");
236 
237 		m_po_even = PrepareProgram(false);
238 		m_po_odd  = PrepareProgram(true);
239 
240 		PrepareVAO();
241 		PrepareXFB();
242 
243 		is_ok &= TurnOnAttributes(true, false);
244 		is_ok &= DrawAndCheck(false);
245 
246 		is_ok &= TurnOnAttributes(false, true);
247 		is_ok &= DrawAndCheck(true);
248 	}
249 	catch (...)
250 	{
251 		is_ok	= false;
252 		is_error = true;
253 	}
254 
255 	/* Cleanup. */
256 	Clean();
257 
258 	/* Errors clean up. */
259 	while (gl.getError())
260 		;
261 
262 	/* Result's setup. */
263 	if (is_ok)
264 	{
265 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
266 	}
267 	else
268 	{
269 		if (is_error)
270 		{
271 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
272 		}
273 		else
274 		{
275 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
276 		}
277 	}
278 
279 	return STOP;
280 }
281 
PrepareProgram(const bool bind_even_or_odd)282 glw::GLuint EnableDisableAttributesTest::PrepareProgram(const bool bind_even_or_odd)
283 {
284 	/* Preprocess vertex shader sources. */
285 	std::string declarations = "";
286 	std::string copies		 = "    sum = 0;\n";
287 
288 	for (glw::GLint i = (glw::GLint)(bind_even_or_odd); i < m_max_attributes; i += 2)
289 	{
290 		declarations.append((std::string("in int a_").append(Utilities::itoa(i))).append(";\n"));
291 		copies.append((std::string("    sum += a_").append(Utilities::itoa(i))).append(";\n"));
292 	}
293 
294 	std::string vs_template(s_vertex_shader_template);
295 
296 	std::string vs_source = Utilities::replace(vs_template, "DECLARATION_TEMPLATE", declarations);
297 	vs_source			  = Utilities::replace(vs_source, "COPY_TEMPLATE", copies);
298 
299 	/* Build and compile. */
300 	return BuildProgram(vs_source.c_str(), bind_even_or_odd);
301 }
302 
303 /** @brief Build test's GLSL program.
304  *
305  *  @note The function may throw if unexpected error has occured.
306  */
BuildProgram(const char * vertex_shader,const bool bind_even_or_odd)307 glw::GLuint EnableDisableAttributesTest::BuildProgram(const char* vertex_shader, const bool bind_even_or_odd)
308 {
309 	/* Shortcut for GL functionality */
310 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
311 
312 	struct Shader
313 	{
314 		glw::GLchar const* const source;
315 		glw::GLenum const		 type;
316 		glw::GLuint				 id;
317 	} shader[] = { { vertex_shader, GL_VERTEX_SHADER, 0 }, { s_fragment_shader, GL_FRAGMENT_SHADER, 0 } };
318 
319 	glw::GLuint const shader_count = sizeof(shader) / sizeof(shader[0]);
320 
321 	glw::GLuint po = 0;
322 
323 	try
324 	{
325 		/* Create program. */
326 		po = gl.createProgram();
327 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram call failed.");
328 
329 		/* Shader compilation. */
330 
331 		for (glw::GLuint i = 0; i < shader_count; ++i)
332 		{
333 			if (DE_NULL != shader[i].source)
334 			{
335 				shader[i].id = gl.createShader(shader[i].type);
336 
337 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateShader call failed.");
338 
339 				gl.attachShader(po, shader[i].id);
340 
341 				GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader call failed.");
342 
343 				gl.shaderSource(shader[i].id, 1, &(shader[i].source), NULL);
344 
345 				GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource call failed.");
346 
347 				gl.compileShader(shader[i].id);
348 
349 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader call failed.");
350 
351 				glw::GLint status = GL_FALSE;
352 
353 				gl.getShaderiv(shader[i].id, GL_COMPILE_STATUS, &status);
354 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
355 
356 				if (GL_FALSE == status)
357 				{
358 					glw::GLint log_size = 0;
359 					gl.getShaderiv(shader[i].id, GL_INFO_LOG_LENGTH, &log_size);
360 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
361 
362 					glw::GLchar* log_text = new glw::GLchar[log_size];
363 
364 					gl.getShaderInfoLog(shader[i].id, log_size, NULL, &log_text[0]);
365 
366 					m_context.getTestContext().getLog() << tcu::TestLog::Message << "Shader compilation has failed.\n"
367 														<< "Shader type: " << glu::getShaderTypeStr(shader[i].type)
368 														<< "\n"
369 														<< "Shader compilation error log:\n"
370 														<< log_text << "\n"
371 														<< "Shader source code:\n"
372 														<< shader[i].source << "\n"
373 														<< tcu::TestLog::EndMessage;
374 
375 					delete[] log_text;
376 
377 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog call failed.");
378 
379 					throw 0;
380 				}
381 			}
382 		}
383 
384 		/* Transform Feedback setup. */
385 		static const glw::GLchar* xfb_varying = "sum";
386 
387 		gl.transformFeedbackVaryings(po, 1, &xfb_varying, GL_INTERLEAVED_ATTRIBS);
388 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
389 
390 		for (glw::GLint i = (glw::GLint)(bind_even_or_odd); i < m_max_attributes; i += 2)
391 		{
392 			std::string attribute = std::string("a_").append(Utilities::itoa(i));
393 
394 			gl.bindAttribLocation(po, i, attribute.c_str());
395 			GLU_EXPECT_NO_ERROR(gl.getError(), "glBindAttribLocation call failed.");
396 		}
397 
398 		/* Link. */
399 		gl.linkProgram(po);
400 
401 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
402 
403 		glw::GLint status = GL_FALSE;
404 
405 		gl.getProgramiv(po, GL_LINK_STATUS, &status);
406 
407 		if (GL_TRUE == status)
408 		{
409 			for (glw::GLuint i = 0; i < shader_count; ++i)
410 			{
411 				if (shader[i].id)
412 				{
413 					gl.detachShader(po, shader[i].id);
414 
415 					GLU_EXPECT_NO_ERROR(gl.getError(), "glDetachShader call failed.");
416 				}
417 			}
418 		}
419 		else
420 		{
421 			glw::GLint log_size = 0;
422 
423 			gl.getProgramiv(po, GL_INFO_LOG_LENGTH, &log_size);
424 
425 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv call failed.");
426 
427 			glw::GLchar* log_text = new glw::GLchar[log_size];
428 
429 			gl.getProgramInfoLog(po, log_size, NULL, &log_text[0]);
430 
431 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Program linkage has failed due to:\n"
432 												<< log_text << "\n"
433 												<< tcu::TestLog::EndMessage;
434 
435 			delete[] log_text;
436 
437 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog call failed.");
438 
439 			throw 0;
440 		}
441 	}
442 	catch (...)
443 	{
444 		if (po)
445 		{
446 			gl.deleteProgram(po);
447 
448 			po = 0;
449 		}
450 	}
451 
452 	for (glw::GLuint i = 0; i < shader_count; ++i)
453 	{
454 		if (0 != shader[i].id)
455 		{
456 			gl.deleteShader(shader[i].id);
457 
458 			shader[i].id = 0;
459 		}
460 	}
461 
462 	if (0 == po)
463 	{
464 		throw 0;
465 	}
466 
467 	return po;
468 }
469 
470 /** @brief Prepare vertex array object for the test.
471  */
PrepareVAO()472 void EnableDisableAttributesTest::PrepareVAO()
473 {
474 	/* Shortcut for GL functionality */
475 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
476 
477 	/* VAO creation. */
478 	gl.genVertexArrays(1, &m_vao);
479 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays call failed.");
480 
481 	gl.bindVertexArray(m_vao);
482 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
483 
484 	/* Buffer creation. */
485 	gl.genBuffers(1, &m_bo);
486 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
487 
488 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo);
489 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
490 
491 	glw::GLint* reference_data = new glw::GLint[m_max_attributes];
492 
493 	if (DE_NULL == reference_data)
494 	{
495 		throw 0;
496 	}
497 
498 	for (glw::GLint i = 0; i < m_max_attributes; ++i)
499 	{
500 		reference_data[i] = i;
501 	}
502 
503 	gl.bufferData(GL_ARRAY_BUFFER, sizeof(glw::GLint) * m_max_attributes, reference_data, GL_STATIC_DRAW);
504 
505 	delete[] reference_data;
506 
507 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
508 
509 	/* VAO setup. */
510 	for (glw::GLint i = 0; i < m_max_attributes; ++i)
511 	{
512 		gl.vertexAttribIPointer(i, 1, GL_INT, static_cast<glw::GLsizei>(sizeof(glw::GLint) * m_max_attributes),
513 								glu::BufferOffsetAsPointer(i * sizeof(glw::GLint)));
514 
515 		GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribIPointer call failed.");
516 	}
517 }
518 
519 /** @brief Prepare buffer object for test GLSL program transform feedback results.
520  */
PrepareXFB()521 void EnableDisableAttributesTest::PrepareXFB()
522 {
523 	/* Shortcut for GL functionality */
524 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
525 
526 	/* Buffer creation. */
527 	gl.genBuffers(1, &m_bo_xfb);
528 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
529 
530 	gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, m_bo_xfb);
531 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
532 
533 	/* Preparing storage. */
534 	gl.bufferStorage(GL_TRANSFORM_FEEDBACK_BUFFER, sizeof(glw::GLint), NULL, GL_MAP_READ_BIT);
535 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferStorage call failed.");
536 
537 	gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, m_bo_xfb);
538 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBufferBase call failed.");
539 }
540 
541 /** @brief Draw test program, fetch transform feedback results and compare them with expected values.
542  *
543  *  @param [in] bind_even_or_odd         Even or odd attribute are enabled.
544  *
545  *  @return True if expected results are equal to returned by XFB, false otherwise.
546  */
DrawAndCheck(bool bind_even_or_odd)547 bool EnableDisableAttributesTest::DrawAndCheck(bool bind_even_or_odd)
548 {
549 	/* Shortcut for GL functionality */
550 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
551 
552 	/* Setup state. */
553 	gl.useProgram(bind_even_or_odd ? m_po_odd : m_po_even);
554 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram call failed.");
555 
556 	gl.bindVertexArray(m_vao);
557 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
558 
559 	gl.beginTransformFeedback(GL_POINTS);
560 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBeginTransformFeedback call failed.");
561 
562 	/* Draw. */
563 	gl.drawArrays(GL_POINTS, 0, 1);
564 	GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawArrays call failed.");
565 
566 	/* State reset. */
567 	gl.endTransformFeedback();
568 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEndTransformFeedback call failed.");
569 
570 	/* Result query. */
571 	glw::GLint* result_ptr = (glw::GLint*)gl.mapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, GL_READ_ONLY);
572 	GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBuffer call failed.");
573 
574 	glw::GLint result = *result_ptr;
575 
576 	gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
577 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer call failed.");
578 
579 	/* Check result and return. */
580 	if (bind_even_or_odd)
581 	{
582 		glw::GLint reference_sum = 0;
583 
584 		for (glw::GLint i = 1; i < m_max_attributes; i += 2)
585 		{
586 			reference_sum += i;
587 		}
588 
589 		if (reference_sum == result)
590 		{
591 			return true;
592 		}
593 	}
594 	else
595 	{
596 		glw::GLint reference_sum = 0;
597 
598 		for (glw::GLint i = 0; i < m_max_attributes; i += 2)
599 		{
600 			reference_sum += i;
601 		}
602 
603 		if (reference_sum == result)
604 		{
605 			return true;
606 		}
607 	}
608 
609 	return false;
610 }
611 
612 /** @brief Turn on even or odd attributes (up to m_max_attributes) using EnableVertexArrayAttrib function.
613  *
614  *  @param [in] enable_even         Turn on even attribute indexes.
615  *  @param [in] enable_odd          Turn on odd  attribute indexes.
616  *
617  *  @return True if EnableVertexArrayAttrib does not generate any error, false otherwise.
618  */
TurnOnAttributes(bool enable_even,bool enable_odd)619 bool EnableDisableAttributesTest::TurnOnAttributes(bool enable_even, bool enable_odd)
620 {
621 	/* Shortcut for GL functionality */
622 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
623 
624 	gl.bindVertexArray(0);
625 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
626 
627 	for (glw::GLint i = 0; i < m_max_attributes; ++i)
628 	{
629 		bool disable = true;
630 
631 		if (i % 2) /* if odd */
632 		{
633 			if (enable_odd)
634 			{
635 				gl.enableVertexArrayAttrib(m_vao, i);
636 
637 				if (glw::GLenum error = gl.getError())
638 				{
639 					m_context.getTestContext().getLog()
640 						<< tcu::TestLog::Message << "glEnableVertexArrayAttrib generated error "
641 						<< glu::getErrorStr(error) << " when called with VAO " << m_vao << " and index " << i << "."
642 						<< tcu::TestLog::EndMessage;
643 
644 					return false;
645 				}
646 
647 				disable = false;
648 			}
649 		}
650 		else
651 		{
652 			if (enable_even)
653 			{
654 				gl.enableVertexArrayAttrib(m_vao, i);
655 
656 				if (glw::GLenum error = gl.getError())
657 				{
658 					m_context.getTestContext().getLog()
659 						<< tcu::TestLog::Message << "glEnableVertexArrayAttrib generated error "
660 						<< glu::getErrorStr(error) << " when called with VAO " << m_vao << " and index " << i << "."
661 						<< tcu::TestLog::EndMessage;
662 
663 					return false;
664 				}
665 
666 				disable = false;
667 			}
668 		}
669 
670 		if (disable)
671 		{
672 			gl.disableVertexArrayAttrib(m_vao, i);
673 
674 			if (glw::GLenum error = gl.getError())
675 			{
676 				m_context.getTestContext().getLog()
677 					<< tcu::TestLog::Message << "glDisableVertexArrayAttrib generated error " << glu::getErrorStr(error)
678 					<< " when called with VAO " << m_vao << " and index " << i << "." << tcu::TestLog::EndMessage;
679 
680 				return false;
681 			}
682 		}
683 	}
684 
685 	gl.bindVertexArray(m_vao);
686 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
687 
688 	return true;
689 }
690 
691 /** @brief Clean GL objects. */
Clean()692 void EnableDisableAttributesTest::Clean()
693 {
694 	/* Shortcut for GL functionality */
695 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
696 
697 	gl.useProgram(0);
698 
699 	if (m_po_even)
700 	{
701 		gl.deleteProgram(m_po_even);
702 
703 		m_po_even = 0;
704 	}
705 
706 	if (m_po_odd)
707 	{
708 		gl.deleteProgram(m_po_odd);
709 
710 		m_po_odd = 0;
711 	}
712 
713 	if (m_vao)
714 	{
715 		gl.deleteVertexArrays(1, &m_vao);
716 
717 		m_vao = 0;
718 	}
719 
720 	if (m_bo)
721 	{
722 		gl.deleteBuffers(1, &m_bo);
723 
724 		m_bo = 0;
725 	}
726 
727 	if (m_bo_xfb)
728 	{
729 		gl.deleteBuffers(1, &m_bo_xfb);
730 
731 		m_bo_xfb = 0;
732 	}
733 
734 	if (m_max_attributes)
735 	{
736 		m_max_attributes =
737 			16; /* OpenGL 4.5 Required Minimum Table 23.57: Implementation Dependent Vertex Shader Limits */
738 	}
739 
740 	while (gl.getError())
741 		;
742 }
743 
itoa(glw::GLuint i)744 std::string Utilities::itoa(glw::GLuint i)
745 {
746 	std::string s = "";
747 
748 	std::stringstream ss;
749 
750 	ss << i;
751 
752 	s = ss.str();
753 
754 	return s;
755 }
756 
replace(const std::string & src,const std::string & key,const std::string & value)757 std::string Utilities::replace(const std::string& src, const std::string& key, const std::string& value)
758 {
759 	size_t		pos = 0;
760 	std::string dst = src;
761 
762 	while (std::string::npos != (pos = dst.find(key, pos)))
763 	{
764 		dst.replace(pos, key.length(), value);
765 		pos += key.length();
766 	}
767 
768 	return dst;
769 }
770 
771 const glw::GLchar EnableDisableAttributesTest::s_vertex_shader_template[] = "#version 450\n"
772 																			"\n"
773 																			"DECLARATION_TEMPLATE"
774 																			"out int sum;\n"
775 																			"\n"
776 																			"void main()\n"
777 																			"{\n"
778 																			"COPY_TEMPLATE"
779 																			"}\n";
780 
781 const glw::GLchar EnableDisableAttributesTest::s_fragment_shader[] = "#version 450\n"
782 																	 "\n"
783 																	 "out vec4 color;\n"
784 																	 "\n"
785 																	 "void main()\n"
786 																	 "{\n"
787 																	 "    color = vec4(1.0);"
788 																	 "}\n";
789 
790 /******************************** Vertex Array Object Element Buffer Test Implementation   ********************************/
791 
792 /** @brief Vertex Array Object Element Buffer Test constructor.
793  *
794  *  @param [in] context     OpenGL context.
795  */
ElementBufferTest(deqp::Context & context)796 ElementBufferTest::ElementBufferTest(deqp::Context& context)
797 	: deqp::TestCase(context, "vertex_arrays_element_buffer", "Vertex Array Objects Element Buffer Test")
798 	, m_po(0)
799 	, m_vao(0)
800 	, m_bo_array(0)
801 	, m_bo_elements(0)
802 	, m_bo_xfb(0)
803 {
804 	/* Intentionally left blank. */
805 }
806 
807 /** @brief Iterate Vertex Array Object Enable Disable Attributes Test cases.
808  *
809  *  @return Iteration result.
810  */
iterate()811 tcu::TestNode::IterateResult ElementBufferTest::iterate()
812 {
813 	/* Shortcut for GL functionality. */
814 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
815 
816 	/* Get context setup. */
817 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
818 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
819 
820 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
821 	{
822 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
823 
824 		return STOP;
825 	}
826 
827 	/* Running tests. */
828 	bool is_ok	= true;
829 	bool is_error = false;
830 
831 	try
832 	{
833 		PrepareProgram();
834 		is_ok &= PrepareVAO();
835 		PrepareXFB();
836 		is_ok &= DrawAndCheck();
837 	}
838 	catch (...)
839 	{
840 		is_ok	= false;
841 		is_error = true;
842 	}
843 
844 	/* Cleanup. */
845 	Clean();
846 
847 	/* Errors clean up. */
848 	while (gl.getError())
849 		;
850 
851 	/* Result's setup. */
852 	if (is_ok)
853 	{
854 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
855 	}
856 	else
857 	{
858 		if (is_error)
859 		{
860 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
861 		}
862 		else
863 		{
864 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
865 		}
866 	}
867 
868 	return STOP;
869 }
870 
871 /** @brief Build test's GLSL program.
872  *
873  *  @note The function may throw if unexpected error has occured.
874  */
PrepareProgram()875 void ElementBufferTest::PrepareProgram()
876 {
877 	/* Shortcut for GL functionality */
878 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
879 
880 	struct Shader
881 	{
882 		glw::GLchar const* const source;
883 		glw::GLenum const		 type;
884 		glw::GLuint				 id;
885 	} shader[] = { { s_vertex_shader, GL_VERTEX_SHADER, 0 }, { s_fragment_shader, GL_FRAGMENT_SHADER, 0 } };
886 
887 	glw::GLuint const shader_count = sizeof(shader) / sizeof(shader[0]);
888 
889 	try
890 	{
891 		/* Create program. */
892 		m_po = gl.createProgram();
893 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram call failed.");
894 
895 		/* Shader compilation. */
896 
897 		for (glw::GLuint i = 0; i < shader_count; ++i)
898 		{
899 			if (DE_NULL != shader[i].source)
900 			{
901 				shader[i].id = gl.createShader(shader[i].type);
902 
903 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateShader call failed.");
904 
905 				gl.attachShader(m_po, shader[i].id);
906 
907 				GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader call failed.");
908 
909 				gl.shaderSource(shader[i].id, 1, &(shader[i].source), NULL);
910 
911 				GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource call failed.");
912 
913 				gl.compileShader(shader[i].id);
914 
915 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader call failed.");
916 
917 				glw::GLint status = GL_FALSE;
918 
919 				gl.getShaderiv(shader[i].id, GL_COMPILE_STATUS, &status);
920 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
921 
922 				if (GL_FALSE == status)
923 				{
924 					glw::GLint log_size = 0;
925 					gl.getShaderiv(shader[i].id, GL_INFO_LOG_LENGTH, &log_size);
926 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
927 
928 					glw::GLchar* log_text = new glw::GLchar[log_size];
929 
930 					gl.getShaderInfoLog(shader[i].id, log_size, NULL, &log_text[0]);
931 
932 					m_context.getTestContext().getLog() << tcu::TestLog::Message << "Shader compilation has failed.\n"
933 														<< "Shader type: " << glu::getShaderTypeStr(shader[i].type)
934 														<< "\n"
935 														<< "Shader compilation error log:\n"
936 														<< log_text << "\n"
937 														<< "Shader source code:\n"
938 														<< shader[i].source << "\n"
939 														<< tcu::TestLog::EndMessage;
940 
941 					delete[] log_text;
942 
943 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog call failed.");
944 
945 					throw 0;
946 				}
947 			}
948 		}
949 
950 		/* Transform Feedback setup. */
951 		static const glw::GLchar* xfb_varying = "result";
952 
953 		gl.transformFeedbackVaryings(m_po, 1, &xfb_varying, GL_INTERLEAVED_ATTRIBS);
954 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
955 
956 		/* Link. */
957 		gl.linkProgram(m_po);
958 
959 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
960 
961 		glw::GLint status = GL_FALSE;
962 
963 		gl.getProgramiv(m_po, GL_LINK_STATUS, &status);
964 
965 		if (GL_TRUE == status)
966 		{
967 			for (glw::GLuint i = 0; i < shader_count; ++i)
968 			{
969 				if (shader[i].id)
970 				{
971 					gl.detachShader(m_po, shader[i].id);
972 
973 					GLU_EXPECT_NO_ERROR(gl.getError(), "glDetachShader call failed.");
974 				}
975 			}
976 		}
977 		else
978 		{
979 			glw::GLint log_size = 0;
980 
981 			gl.getProgramiv(m_po, GL_INFO_LOG_LENGTH, &log_size);
982 
983 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv call failed.");
984 
985 			glw::GLchar* log_text = new glw::GLchar[log_size];
986 
987 			gl.getProgramInfoLog(m_po, log_size, NULL, &log_text[0]);
988 
989 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Program linkage has failed due to:\n"
990 												<< log_text << "\n"
991 												<< tcu::TestLog::EndMessage;
992 
993 			delete[] log_text;
994 
995 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog call failed.");
996 
997 			throw 0;
998 		}
999 	}
1000 	catch (...)
1001 	{
1002 		if (m_po)
1003 		{
1004 			gl.deleteProgram(m_po);
1005 
1006 			m_po = 0;
1007 		}
1008 	}
1009 
1010 	for (glw::GLuint i = 0; i < shader_count; ++i)
1011 	{
1012 		if (0 != shader[i].id)
1013 		{
1014 			gl.deleteShader(shader[i].id);
1015 
1016 			shader[i].id = 0;
1017 		}
1018 	}
1019 
1020 	if (0 == m_po)
1021 	{
1022 		throw 0;
1023 	}
1024 }
1025 
1026 /** @brief Prepare vertex array object for the test of VertexArrayElementBuffer function.
1027  *
1028  *  @return True if function VertexArrayElementBuffer* does not generate any error.
1029  */
PrepareVAO()1030 bool ElementBufferTest::PrepareVAO()
1031 {
1032 	/* Shortcut for GL functionality */
1033 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1034 
1035 	/* VAO creation. */
1036 	gl.genVertexArrays(1, &m_vao);
1037 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays call failed.");
1038 
1039 	gl.bindVertexArray(m_vao);
1040 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
1041 
1042 	/* Array buffer creation. */
1043 	glw::GLint array_data[3] = { 2, 1, 0 };
1044 
1045 	gl.genBuffers(1, &m_bo_array);
1046 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
1047 
1048 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo_array);
1049 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
1050 
1051 	gl.bufferData(GL_ARRAY_BUFFER, sizeof(array_data), array_data, GL_STATIC_DRAW);
1052 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
1053 
1054 	gl.vertexAttribIPointer(gl.getAttribLocation(m_po, "a"), 1, GL_INT, 0, NULL);
1055 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribIPointer call failed.");
1056 
1057 	gl.enableVertexAttribArray(0);
1058 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
1059 
1060 	gl.bindVertexArray(0);
1061 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
1062 
1063 	/* Element buffer creation. */
1064 	glw::GLuint elements_data[3] = { 2, 1, 0 };
1065 
1066 	gl.genBuffers(1, &m_bo_elements);
1067 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
1068 
1069 	gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_bo_elements);
1070 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
1071 
1072 	gl.bufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(elements_data), elements_data, GL_STATIC_DRAW);
1073 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
1074 
1075 	gl.vertexArrayElementBuffer(m_vao, m_bo_elements);
1076 
1077 	if (glw::GLenum error = gl.getError())
1078 	{
1079 		m_context.getTestContext().getLog() << tcu::TestLog::Message
1080 											<< "VertexArrayElementBuffer has unexpectedly generated "
1081 											<< glu::getErrorStr(error) << "error. Test fails.\n"
1082 											<< tcu::TestLog::EndMessage;
1083 
1084 		return false;
1085 	}
1086 
1087 	return true;
1088 }
1089 
1090 /** @brief Prepare buffer object for test GLSL program transform feedback results.
1091  */
PrepareXFB()1092 void ElementBufferTest::PrepareXFB()
1093 {
1094 	/* Shortcut for GL functionality */
1095 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1096 
1097 	/* Buffer creation. */
1098 	gl.genBuffers(1, &m_bo_xfb);
1099 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
1100 
1101 	gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, m_bo_xfb);
1102 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
1103 
1104 	/* Preparing storage. */
1105 	gl.bufferStorage(GL_TRANSFORM_FEEDBACK_BUFFER, 3 * sizeof(glw::GLint), NULL, GL_MAP_READ_BIT);
1106 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferStorage call failed.");
1107 
1108 	gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, m_bo_xfb);
1109 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBufferBase call failed.");
1110 }
1111 
1112 /** @brief Draw test program, fetch transform feedback results and compare them with expected values.
1113  *
1114  *  @return True if expected results are equal to returned by XFB, false otherwise.
1115  */
DrawAndCheck()1116 bool ElementBufferTest::DrawAndCheck()
1117 {
1118 	/* Shortcut for GL functionality */
1119 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1120 
1121 	/* Setup state. */
1122 	gl.useProgram(m_po);
1123 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram call failed.");
1124 
1125 	gl.bindVertexArray(m_vao);
1126 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
1127 
1128 	gl.beginTransformFeedback(GL_POINTS);
1129 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBeginTransformFeedback call failed.");
1130 
1131 	/* Draw. */
1132 	gl.drawElements(GL_POINTS, 3, GL_UNSIGNED_INT, NULL);
1133 	GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawArrays call failed.");
1134 
1135 	/* State reset. */
1136 	gl.endTransformFeedback();
1137 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEndTransformFeedback call failed.");
1138 
1139 	/* Result query. */
1140 	glw::GLint* result_ptr = (glw::GLint*)gl.mapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, GL_READ_ONLY);
1141 	GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBuffer call failed.");
1142 
1143 	glw::GLint result[3] = { result_ptr[0], result_ptr[1], result_ptr[2] };
1144 
1145 	gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
1146 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer call failed.");
1147 
1148 	/* Check result and return. */
1149 	for (glw::GLint i = 0; i < 3; ++i)
1150 	{
1151 		if (i != result[i])
1152 		{
1153 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Result vector is equal to [" << result[0]
1154 												<< ", " << result[1] << ", " << result[2]
1155 												<< "], but [0, 1, 2] was expected." << tcu::TestLog::EndMessage;
1156 
1157 			return false;
1158 		}
1159 	}
1160 
1161 	return true;
1162 }
1163 
1164 /** @brief Clean GL objects. */
Clean()1165 void ElementBufferTest::Clean()
1166 {
1167 	/* Shortcut for GL functionality */
1168 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1169 
1170 	gl.useProgram(0);
1171 
1172 	if (m_po)
1173 	{
1174 		gl.deleteProgram(m_po);
1175 
1176 		m_po = 0;
1177 	}
1178 
1179 	if (m_vao)
1180 	{
1181 		gl.deleteVertexArrays(1, &m_vao);
1182 
1183 		m_vao = 0;
1184 	}
1185 
1186 	if (m_bo_array)
1187 	{
1188 		gl.deleteBuffers(1, &m_bo_array);
1189 
1190 		m_bo_array = 0;
1191 	}
1192 
1193 	if (m_bo_elements)
1194 	{
1195 		gl.deleteBuffers(1, &m_bo_elements);
1196 
1197 		m_bo_elements = 0;
1198 	}
1199 
1200 	if (m_bo_xfb)
1201 	{
1202 		gl.deleteBuffers(1, &m_bo_xfb);
1203 
1204 		m_bo_xfb = 0;
1205 	}
1206 
1207 	while (gl.getError())
1208 		;
1209 }
1210 
1211 const glw::GLchar ElementBufferTest::s_vertex_shader[] = "#version 450\n"
1212 														 "\n"
1213 														 "in int a;"
1214 														 "out int result;\n"
1215 														 "\n"
1216 														 "void main()\n"
1217 														 "{\n"
1218 														 "    gl_Position = vec4(1.0);\n"
1219 														 "    result = a;"
1220 														 "}\n";
1221 
1222 const glw::GLchar ElementBufferTest::s_fragment_shader[] = "#version 450\n"
1223 														   "\n"
1224 														   "out vec4 color;\n"
1225 														   "\n"
1226 														   "void main()\n"
1227 														   "{\n"
1228 														   "    color = vec4(1.0);"
1229 														   "}\n";
1230 
1231 /******************************** Vertex Array Object Vertex Buffer and Buffers Test Implementation   ********************************/
1232 
1233 /** @brief Vertex Array Object Element Buffer Test constructor.
1234  *
1235  *  @param [in] context     OpenGL context.
1236  */
VertexBuffersTest(deqp::Context & context)1237 VertexBuffersTest::VertexBuffersTest(deqp::Context& context)
1238 	: deqp::TestCase(context, "vertex_arrays_vertex_buffers", "Vertex Array Object Vertex Buffer and Buffers Test")
1239 	, m_po(0)
1240 	, m_vao(0)
1241 	, m_bo_array_0(0)
1242 	, m_bo_array_1(0)
1243 	, m_bo_xfb(0)
1244 {
1245 	/* Intentionally left blank. */
1246 }
1247 
1248 /** @brief Iterate Vertex Array Object Enable Disable Attributes Test cases.
1249  *
1250  *  @return Iteration result.
1251  */
iterate()1252 tcu::TestNode::IterateResult VertexBuffersTest::iterate()
1253 {
1254 	/* Shortcut for GL functionality. */
1255 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1256 
1257 	/* Get context setup. */
1258 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
1259 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
1260 
1261 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
1262 	{
1263 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
1264 
1265 		return STOP;
1266 	}
1267 
1268 	/* Running tests. */
1269 	bool is_ok	= true;
1270 	bool is_error = false;
1271 
1272 	try
1273 	{
1274 		PrepareProgram();
1275 		is_ok &= PrepareVAO(false);
1276 		PrepareXFB();
1277 		is_ok &= DrawAndCheck();
1278 		Clean();
1279 
1280 		PrepareProgram();
1281 		is_ok &= PrepareVAO(true);
1282 		PrepareXFB();
1283 		is_ok &= DrawAndCheck();
1284 	}
1285 	catch (...)
1286 	{
1287 		is_ok	= false;
1288 		is_error = true;
1289 	}
1290 
1291 	/* Cleanup. */
1292 	Clean();
1293 
1294 	/* Errors clean up. */
1295 	while (gl.getError())
1296 		;
1297 
1298 	/* Result's setup. */
1299 	if (is_ok)
1300 	{
1301 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
1302 	}
1303 	else
1304 	{
1305 		if (is_error)
1306 		{
1307 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
1308 		}
1309 		else
1310 		{
1311 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
1312 		}
1313 	}
1314 
1315 	return STOP;
1316 }
1317 
1318 /** @brief Build test's GLSL program.
1319  *
1320  *  @note The function may throw if unexpected error has occured.
1321  */
PrepareProgram()1322 void VertexBuffersTest::PrepareProgram()
1323 {
1324 	/* Shortcut for GL functionality */
1325 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1326 
1327 	struct Shader
1328 	{
1329 		glw::GLchar const* const source;
1330 		glw::GLenum const		 type;
1331 		glw::GLuint				 id;
1332 	} shader[] = { { s_vertex_shader, GL_VERTEX_SHADER, 0 }, { s_fragment_shader, GL_FRAGMENT_SHADER, 0 } };
1333 
1334 	glw::GLuint const shader_count = sizeof(shader) / sizeof(shader[0]);
1335 
1336 	try
1337 	{
1338 		/* Create program. */
1339 		m_po = gl.createProgram();
1340 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram call failed.");
1341 
1342 		/* Shader compilation. */
1343 
1344 		for (glw::GLuint i = 0; i < shader_count; ++i)
1345 		{
1346 			if (DE_NULL != shader[i].source)
1347 			{
1348 				shader[i].id = gl.createShader(shader[i].type);
1349 
1350 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateShader call failed.");
1351 
1352 				gl.attachShader(m_po, shader[i].id);
1353 
1354 				GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader call failed.");
1355 
1356 				gl.shaderSource(shader[i].id, 1, &(shader[i].source), NULL);
1357 
1358 				GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource call failed.");
1359 
1360 				gl.compileShader(shader[i].id);
1361 
1362 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader call failed.");
1363 
1364 				glw::GLint status = GL_FALSE;
1365 
1366 				gl.getShaderiv(shader[i].id, GL_COMPILE_STATUS, &status);
1367 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
1368 
1369 				if (GL_FALSE == status)
1370 				{
1371 					glw::GLint log_size = 0;
1372 					gl.getShaderiv(shader[i].id, GL_INFO_LOG_LENGTH, &log_size);
1373 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
1374 
1375 					glw::GLchar* log_text = new glw::GLchar[log_size];
1376 
1377 					gl.getShaderInfoLog(shader[i].id, log_size, NULL, &log_text[0]);
1378 
1379 					m_context.getTestContext().getLog() << tcu::TestLog::Message << "Shader compilation has failed.\n"
1380 														<< "Shader type: " << glu::getShaderTypeStr(shader[i].type)
1381 														<< "\n"
1382 														<< "Shader compilation error log:\n"
1383 														<< log_text << "\n"
1384 														<< "Shader source code:\n"
1385 														<< shader[i].source << "\n"
1386 														<< tcu::TestLog::EndMessage;
1387 
1388 					delete[] log_text;
1389 
1390 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog call failed.");
1391 
1392 					throw 0;
1393 				}
1394 			}
1395 		}
1396 
1397 		/* Transform Feedback setup. */
1398 		static const glw::GLchar* xfb_varying = "result";
1399 
1400 		gl.transformFeedbackVaryings(m_po, 1, &xfb_varying, GL_INTERLEAVED_ATTRIBS);
1401 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
1402 
1403 		/* Link. */
1404 		gl.linkProgram(m_po);
1405 
1406 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
1407 
1408 		glw::GLint status = GL_FALSE;
1409 
1410 		gl.getProgramiv(m_po, GL_LINK_STATUS, &status);
1411 
1412 		if (GL_TRUE == status)
1413 		{
1414 			for (glw::GLuint i = 0; i < shader_count; ++i)
1415 			{
1416 				if (shader[i].id)
1417 				{
1418 					gl.detachShader(m_po, shader[i].id);
1419 
1420 					GLU_EXPECT_NO_ERROR(gl.getError(), "glDetachShader call failed.");
1421 				}
1422 			}
1423 		}
1424 		else
1425 		{
1426 			glw::GLint log_size = 0;
1427 
1428 			gl.getProgramiv(m_po, GL_INFO_LOG_LENGTH, &log_size);
1429 
1430 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv call failed.");
1431 
1432 			glw::GLchar* log_text = new glw::GLchar[log_size];
1433 
1434 			gl.getProgramInfoLog(m_po, log_size, NULL, &log_text[0]);
1435 
1436 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Program linkage has failed due to:\n"
1437 												<< log_text << "\n"
1438 												<< tcu::TestLog::EndMessage;
1439 
1440 			delete[] log_text;
1441 
1442 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog call failed.");
1443 
1444 			throw 0;
1445 		}
1446 	}
1447 	catch (...)
1448 	{
1449 		if (m_po)
1450 		{
1451 			gl.deleteProgram(m_po);
1452 
1453 			m_po = 0;
1454 		}
1455 	}
1456 
1457 	for (glw::GLuint i = 0; i < shader_count; ++i)
1458 	{
1459 		if (0 != shader[i].id)
1460 		{
1461 			gl.deleteShader(shader[i].id);
1462 
1463 			shader[i].id = 0;
1464 		}
1465 	}
1466 
1467 	if (0 == m_po)
1468 	{
1469 		throw 0;
1470 	}
1471 }
1472 
1473 /** @brief Prepare vertex array object for the test of gl.vertexArrayVertexBuffer* functions.
1474  *
1475  *  @param [in] use_multiple_buffers_function    Use gl.vertexArrayVertexBuffers instead of gl.vertexArrayVertexBuffer.
1476  *
1477  *  @return True if functions gl.vertexArrayVertexBuffer* do not generate any error.
1478  */
PrepareVAO(bool use_multiple_buffers_function)1479 bool VertexBuffersTest::PrepareVAO(bool use_multiple_buffers_function)
1480 {
1481 	/* Shortcut for GL functionality */
1482 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1483 
1484 	/* VAO creation. */
1485 	gl.genVertexArrays(1, &m_vao);
1486 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays call failed.");
1487 
1488 	gl.bindVertexArray(m_vao);
1489 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
1490 
1491 	/* Array buffer 0 creation. */
1492 	glw::GLint array_data_0[4] = { 0, 2, 1, 3 };
1493 
1494 	gl.genBuffers(1, &m_bo_array_0);
1495 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
1496 
1497 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo_array_0);
1498 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
1499 
1500 	gl.bufferData(GL_ARRAY_BUFFER, sizeof(array_data_0), array_data_0, GL_STATIC_DRAW);
1501 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
1502 
1503 	gl.vertexAttribBinding(gl.getAttribLocation(m_po, "a_0"), 0);
1504 
1505 	gl.vertexAttribBinding(gl.getAttribLocation(m_po, "a_1"), 1);
1506 
1507 	gl.vertexAttribIFormat(gl.getAttribLocation(m_po, "a_0"), 1, GL_INT, 0);
1508 
1509 	gl.vertexAttribIFormat(gl.getAttribLocation(m_po, "a_1"), 1, GL_INT, 0);
1510 
1511 	if (use_multiple_buffers_function)
1512 	{
1513 		const glw::GLuint		   buffers[2] = { m_bo_array_0, m_bo_array_0 };
1514 		static const glw::GLintptr offsets[2] = { 0, sizeof(glw::GLint) };
1515 		static const glw::GLsizei  strides[2] = { sizeof(glw::GLint) * 2, sizeof(glw::GLint) * 2 };
1516 
1517 		gl.vertexArrayVertexBuffers(m_vao, 0, 2, buffers, offsets, strides);
1518 
1519 		if (glw::GLenum error = gl.getError())
1520 		{
1521 			m_context.getTestContext().getLog() << tcu::TestLog::Message
1522 												<< "VertexArrayVertexBuffers has unexpectedly generated "
1523 												<< glu::getErrorStr(error) << "error. Test fails.\n"
1524 												<< tcu::TestLog::EndMessage;
1525 
1526 			return false;
1527 		}
1528 	}
1529 	else
1530 	{
1531 		gl.vertexArrayVertexBuffer(m_vao, 0, m_bo_array_0, (glw::GLintptr)NULL, sizeof(glw::GLint) * 2);
1532 
1533 		if (glw::GLenum error = gl.getError())
1534 		{
1535 			m_context.getTestContext().getLog() << tcu::TestLog::Message
1536 												<< "VertexArrayVertexBuffer has unexpectedly generated "
1537 												<< glu::getErrorStr(error) << "error. Test fails.\n"
1538 												<< tcu::TestLog::EndMessage;
1539 
1540 			return false;
1541 		}
1542 
1543 		gl.vertexArrayVertexBuffer(m_vao, 1, m_bo_array_0, sizeof(glw::GLint),
1544 								   sizeof(glw::GLint) * 2);
1545 
1546 		if (glw::GLenum error = gl.getError())
1547 		{
1548 			m_context.getTestContext().getLog() << tcu::TestLog::Message
1549 												<< "VertexArrayVertexBuffer has unexpectedly generated "
1550 												<< glu::getErrorStr(error) << "error. Test fails.\n"
1551 												<< tcu::TestLog::EndMessage;
1552 
1553 			return false;
1554 		}
1555 	}
1556 
1557 	gl.enableVertexAttribArray(0);
1558 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
1559 
1560 	gl.enableVertexAttribArray(1);
1561 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
1562 
1563 	/* Array buffer 1 creation. */
1564 	glw::GLint array_data_1[2] = { 4, 5 };
1565 
1566 	gl.genBuffers(1, &m_bo_array_1);
1567 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
1568 
1569 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo_array_1);
1570 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
1571 
1572 	gl.bufferData(GL_ARRAY_BUFFER, sizeof(array_data_1), array_data_1, GL_STATIC_DRAW);
1573 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
1574 
1575 	gl.vertexAttribBinding(gl.getAttribLocation(m_po, "a_2"), 2);
1576 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribBinding call failed.");
1577 
1578 	gl.vertexAttribIFormat(gl.getAttribLocation(m_po, "a_2"), 1, GL_INT, 0);
1579 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribIFormat call failed.");
1580 
1581 	if (use_multiple_buffers_function)
1582 	{
1583 		glw::GLintptr offset = (glw::GLintptr)NULL;
1584 		glw::GLsizei  stride = sizeof(glw::GLint);
1585 
1586 		gl.vertexArrayVertexBuffers(m_vao, 2, 1, &m_bo_array_1, &offset, &stride);
1587 
1588 		if (glw::GLenum error = gl.getError())
1589 		{
1590 			m_context.getTestContext().getLog() << tcu::TestLog::Message
1591 												<< "VertexArrayVertexBuffers has unexpectedly generated "
1592 												<< glu::getErrorStr(error) << "error. Test fails.\n"
1593 												<< tcu::TestLog::EndMessage;
1594 
1595 			return false;
1596 		}
1597 	}
1598 	else
1599 	{
1600 		gl.vertexArrayVertexBuffer(m_vao, 2, m_bo_array_1, (glw::GLintptr)NULL, sizeof(glw::GLint));
1601 
1602 		if (glw::GLenum error = gl.getError())
1603 		{
1604 			m_context.getTestContext().getLog() << tcu::TestLog::Message
1605 												<< "VertexArrayVertexBuffer has unexpectedly generated "
1606 												<< glu::getErrorStr(error) << "error. Test fails.\n"
1607 												<< tcu::TestLog::EndMessage;
1608 
1609 			return false;
1610 		}
1611 	}
1612 
1613 	gl.enableVertexAttribArray(2);
1614 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
1615 
1616 	gl.bindVertexArray(0);
1617 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
1618 
1619 	return true;
1620 }
1621 
1622 /** @brief Prepare buffer object for test GLSL program transform feedback results.
1623  */
PrepareXFB()1624 void VertexBuffersTest::PrepareXFB()
1625 {
1626 	/* Shortcut for GL functionality */
1627 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1628 
1629 	/* Buffer creation. */
1630 	gl.genBuffers(1, &m_bo_xfb);
1631 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
1632 
1633 	gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, m_bo_xfb);
1634 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
1635 
1636 	/* Preparing storage. */
1637 	gl.bufferStorage(GL_TRANSFORM_FEEDBACK_BUFFER, 2 * sizeof(glw::GLint), NULL, GL_MAP_READ_BIT);
1638 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferStorage call failed.");
1639 
1640 	gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, m_bo_xfb);
1641 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBufferBase call failed.");
1642 }
1643 
1644 /** @brief Draw test program, fetch transform feedback results and compare them with expected values.
1645  *
1646  *  @return True if expected results are equal to returned by XFB, false otherwise.
1647  */
DrawAndCheck()1648 bool VertexBuffersTest::DrawAndCheck()
1649 {
1650 	/* Shortcut for GL functionality */
1651 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1652 
1653 	/* Setup state. */
1654 	gl.useProgram(m_po);
1655 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram call failed.");
1656 
1657 	gl.bindVertexArray(m_vao);
1658 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
1659 
1660 	gl.beginTransformFeedback(GL_POINTS);
1661 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBeginTransformFeedback call failed.");
1662 
1663 	/* Draw. */
1664 	gl.drawArrays(GL_POINTS, 0, 2);
1665 	GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawArrays call failed.");
1666 
1667 	/* State reset. */
1668 	gl.endTransformFeedback();
1669 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEndTransformFeedback call failed.");
1670 
1671 	/* Result query. */
1672 	glw::GLint* result_ptr = (glw::GLint*)gl.mapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, GL_READ_ONLY);
1673 	GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBuffer call failed.");
1674 
1675 	glw::GLint result[2] = { result_ptr[0], result_ptr[1] };
1676 
1677 	gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
1678 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer call failed.");
1679 
1680 	static const glw::GLint reference[2] = { 0 + 2 + 4, 1 + 3 + 5 };
1681 
1682 	/* Check result and return. */
1683 	for (glw::GLint i = 0; i < 2; ++i)
1684 	{
1685 		if (reference[i] != result[i])
1686 		{
1687 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Result vector is equal to [" << result[0]
1688 												<< ", " << result[1] << "], but [" << reference[0] << ", "
1689 												<< reference[1] << "] was expected." << tcu::TestLog::EndMessage;
1690 
1691 			return false;
1692 		}
1693 	}
1694 
1695 	return true;
1696 }
1697 
1698 /** @brief Clean GL objects. */
Clean()1699 void VertexBuffersTest::Clean()
1700 {
1701 	/* Shortcut for GL functionality */
1702 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1703 
1704 	gl.useProgram(0);
1705 
1706 	if (m_po)
1707 	{
1708 		gl.deleteProgram(m_po);
1709 
1710 		m_po = 0;
1711 	}
1712 
1713 	if (m_vao)
1714 	{
1715 		gl.deleteVertexArrays(1, &m_vao);
1716 
1717 		m_vao = 0;
1718 	}
1719 
1720 	if (m_bo_array_0)
1721 	{
1722 		gl.deleteBuffers(1, &m_bo_array_0);
1723 
1724 		m_bo_array_0 = 0;
1725 	}
1726 
1727 	if (m_bo_array_1)
1728 	{
1729 		gl.deleteBuffers(1, &m_bo_array_1);
1730 
1731 		m_bo_array_1 = 0;
1732 	}
1733 
1734 	if (m_bo_xfb)
1735 	{
1736 		gl.deleteBuffers(1, &m_bo_xfb);
1737 
1738 		m_bo_xfb = 0;
1739 	}
1740 
1741 	while (gl.getError())
1742 		;
1743 }
1744 
1745 const glw::GLchar VertexBuffersTest::s_vertex_shader[] = "#version 450\n"
1746 														 "\n"
1747 														 "in  int a_0;"
1748 														 "in  int a_1;"
1749 														 "in  int a_2;"
1750 														 "\n"
1751 														 "out int result;\n"
1752 														 "\n"
1753 														 "void main()\n"
1754 														 "{\n"
1755 														 "    gl_Position = vec4(1.0);\n"
1756 														 "    result = a_0 + a_1 + a_2;"
1757 														 "}\n";
1758 
1759 const glw::GLchar VertexBuffersTest::s_fragment_shader[] = "#version 450\n"
1760 														   "\n"
1761 														   "out vec4 color;\n"
1762 														   "\n"
1763 														   "void main()\n"
1764 														   "{\n"
1765 														   "    color = vec4(1.0);"
1766 														   "}\n";
1767 
1768 /******************************** Vertex Array Object Attribute Format Test Implementation   ********************************/
1769 
1770 /** @brief Vertex Array Object Element Buffer Test constructor.
1771  *
1772  *  @param [in] context     OpenGL context.
1773  */
AttributeFormatTest(deqp::Context & context)1774 AttributeFormatTest::AttributeFormatTest(deqp::Context& context)
1775 	: deqp::TestCase(context, "vertex_arrays_attribute_format", "Vertex Array Object Attribute Format Test")
1776 	, m_po(0)
1777 	, m_vao(0)
1778 	, m_bo_array(0)
1779 	, m_bo_xfb(0)
1780 {
1781 	/* Intentionally left blank. */
1782 }
1783 
1784 /** @brief Iterate Vertex Array Object Enable Disable Attributes Test cases.
1785  *
1786  *  @return Iteration result.
1787  */
iterate()1788 tcu::TestNode::IterateResult AttributeFormatTest::iterate()
1789 {
1790 	/* Shortcut for GL functionality. */
1791 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1792 
1793 	/* Get context setup. */
1794 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
1795 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
1796 
1797 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
1798 	{
1799 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
1800 
1801 		return STOP;
1802 	}
1803 
1804 	/* Running tests. */
1805 	bool is_ok	= true;
1806 	bool is_error = false;
1807 
1808 	try
1809 	{
1810 		PrepareXFB();
1811 
1812 		/* Test floating function. */
1813 		for (glw::GLuint i = 1; i <= 4 /* max size */; ++i)
1814 		{
1815 			PrepareProgram(i, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1816 
1817 			is_ok &= PrepareVAO<glw::GLfloat>(i, GL_FLOAT, false, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1818 			is_ok &= DrawAndCheck<glw::GLfloat>(i, false);
1819 
1820 			CleanVAO();
1821 
1822 			is_ok &= PrepareVAO<glw::GLbyte>(i, GL_BYTE, false, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1823 			is_ok &= DrawAndCheck<glw::GLfloat>(i, false);
1824 
1825 			CleanVAO();
1826 
1827 			is_ok &= PrepareVAO<glw::GLbyte>(i, GL_BYTE, true, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1828 			is_ok &= DrawAndCheck<glw::GLfloat>(i, true);
1829 
1830 			CleanVAO();
1831 
1832 			is_ok &= PrepareVAO<glw::GLubyte>(i, GL_UNSIGNED_BYTE, false, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1833 			is_ok &= DrawAndCheck<glw::GLfloat>(i, false);
1834 
1835 			CleanVAO();
1836 
1837 			is_ok &= PrepareVAO<glw::GLshort>(i, GL_SHORT, false, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1838 			is_ok &= DrawAndCheck<glw::GLfloat>(i, false);
1839 
1840 			CleanVAO();
1841 
1842 			is_ok &= PrepareVAO<glw::GLushort>(i, GL_UNSIGNED_SHORT, false, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1843 			is_ok &= DrawAndCheck<glw::GLfloat>(i, false);
1844 
1845 			CleanVAO();
1846 
1847 			is_ok &= PrepareVAO<glw::GLint>(i, GL_INT, false, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1848 			is_ok &= DrawAndCheck<glw::GLfloat>(i, false);
1849 
1850 			CleanVAO();
1851 
1852 			is_ok &= PrepareVAO<glw::GLuint>(i, GL_UNSIGNED_INT, false, ATTRIBUTE_FORMAT_FUNCTION_FLOAT);
1853 			is_ok &= DrawAndCheck<glw::GLfloat>(i, false);
1854 
1855 			CleanVAO();
1856 
1857 			CleanProgram();
1858 		}
1859 
1860 		for (glw::GLuint i = 1; i <= 2 /* max size */; ++i)
1861 		{
1862 			PrepareProgram(i, ATTRIBUTE_FORMAT_FUNCTION_DOUBLE);
1863 
1864 			is_ok &= PrepareVAO<glw::GLdouble>(i, GL_DOUBLE, false, ATTRIBUTE_FORMAT_FUNCTION_DOUBLE);
1865 			is_ok &= DrawAndCheck<glw::GLdouble>(i, false);
1866 
1867 			CleanProgram();
1868 			CleanVAO();
1869 		}
1870 
1871 		for (glw::GLuint i = 1; i <= 4 /* max size */; ++i)
1872 		{
1873 			PrepareProgram(i, ATTRIBUTE_FORMAT_FUNCTION_INTEGER);
1874 
1875 			is_ok &= PrepareVAO<glw::GLbyte>(i, GL_BYTE, false, ATTRIBUTE_FORMAT_FUNCTION_INTEGER);
1876 			is_ok &= DrawAndCheck<glw::GLint>(i, false);
1877 
1878 			CleanVAO();
1879 
1880 			is_ok &= PrepareVAO<glw::GLubyte>(i, GL_UNSIGNED_BYTE, false, ATTRIBUTE_FORMAT_FUNCTION_INTEGER);
1881 			is_ok &= DrawAndCheck<glw::GLint>(i, false);
1882 
1883 			CleanVAO();
1884 
1885 			is_ok &= PrepareVAO<glw::GLshort>(i, GL_SHORT, false, ATTRIBUTE_FORMAT_FUNCTION_INTEGER);
1886 			is_ok &= DrawAndCheck<glw::GLint>(i, false);
1887 
1888 			CleanVAO();
1889 
1890 			is_ok &= PrepareVAO<glw::GLushort>(i, GL_UNSIGNED_SHORT, false, ATTRIBUTE_FORMAT_FUNCTION_INTEGER);
1891 			is_ok &= DrawAndCheck<glw::GLint>(i, false);
1892 
1893 			CleanVAO();
1894 
1895 			is_ok &= PrepareVAO<glw::GLint>(i, GL_INT, false, ATTRIBUTE_FORMAT_FUNCTION_INTEGER);
1896 			is_ok &= DrawAndCheck<glw::GLint>(i, false);
1897 
1898 			CleanVAO();
1899 
1900 			is_ok &= PrepareVAO<glw::GLuint>(i, GL_UNSIGNED_INT, false, ATTRIBUTE_FORMAT_FUNCTION_INTEGER);
1901 			is_ok &= DrawAndCheck<glw::GLint>(i, false);
1902 
1903 			CleanVAO();
1904 
1905 			CleanProgram();
1906 		}
1907 	}
1908 	catch (...)
1909 	{
1910 		is_ok	= false;
1911 		is_error = true;
1912 	}
1913 
1914 	/* Cleanup. */
1915 	CleanProgram();
1916 	CleanVAO();
1917 	CleanXFB();
1918 
1919 	/* Errors clean up. */
1920 	while (gl.getError())
1921 		;
1922 
1923 	/* Result's setup. */
1924 	if (is_ok)
1925 	{
1926 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
1927 	}
1928 	else
1929 	{
1930 		if (is_error)
1931 		{
1932 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
1933 		}
1934 		else
1935 		{
1936 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
1937 		}
1938 	}
1939 
1940 	return STOP;
1941 }
1942 
1943 /** @brief Build test's GLSL program.
1944  *
1945  *  @note The function may throw if unexpected error has occured.
1946  */
PrepareProgram(glw::GLint size,AtributeFormatFunctionType function_selector)1947 void AttributeFormatTest::PrepareProgram(glw::GLint size, AtributeFormatFunctionType function_selector)
1948 {
1949 	/* Shortcut for GL functionality */
1950 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1951 
1952 	struct Shader
1953 	{
1954 		glw::GLchar const* source[3];
1955 		glw::GLuint const  count;
1956 		glw::GLenum const  type;
1957 		glw::GLuint		   id;
1958 	} shader[] = { { { s_vertex_shader_head, s_vertex_shader_declaration[function_selector][size - 1],
1959 					   s_vertex_shader_body },
1960 					 3,
1961 					 GL_VERTEX_SHADER,
1962 					 0 },
1963 				   { { s_fragment_shader, DE_NULL, DE_NULL }, 1, GL_FRAGMENT_SHADER, 0 } };
1964 
1965 	glw::GLuint const shader_count = sizeof(shader) / sizeof(shader[0]);
1966 
1967 	try
1968 	{
1969 		/* Create program. */
1970 		m_po = gl.createProgram();
1971 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram call failed.");
1972 
1973 		/* Shader compilation. */
1974 
1975 		for (glw::GLuint i = 0; i < shader_count; ++i)
1976 		{
1977 			{
1978 				shader[i].id = gl.createShader(shader[i].type);
1979 
1980 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateShader call failed.");
1981 
1982 				gl.attachShader(m_po, shader[i].id);
1983 
1984 				GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader call failed.");
1985 
1986 				gl.shaderSource(shader[i].id, shader[i].count, shader[i].source, NULL);
1987 
1988 				GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource call failed.");
1989 
1990 				gl.compileShader(shader[i].id);
1991 
1992 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader call failed.");
1993 
1994 				glw::GLint status = GL_FALSE;
1995 
1996 				gl.getShaderiv(shader[i].id, GL_COMPILE_STATUS, &status);
1997 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
1998 
1999 				if (GL_FALSE == status)
2000 				{
2001 					glw::GLint log_size = 0;
2002 					gl.getShaderiv(shader[i].id, GL_INFO_LOG_LENGTH, &log_size);
2003 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
2004 
2005 					glw::GLchar* log_text = new glw::GLchar[log_size];
2006 
2007 					gl.getShaderInfoLog(shader[i].id, log_size, NULL, &log_text[0]);
2008 
2009 					m_context.getTestContext().getLog() << tcu::TestLog::Message << "Shader compilation has failed.\n"
2010 														<< "Shader type: " << glu::getShaderTypeStr(shader[i].type)
2011 														<< "\n"
2012 														<< "Shader compilation error log:\n"
2013 														<< log_text << "\n"
2014 														<< "Shader source code:\n"
2015 														<< shader[i].source << "\n"
2016 														<< tcu::TestLog::EndMessage;
2017 
2018 					delete[] log_text;
2019 
2020 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog call failed.");
2021 
2022 					throw 0;
2023 				}
2024 			}
2025 		}
2026 
2027 		/* Transform Feedback setup. */
2028 		static const glw::GLchar* xfb_varying = "result";
2029 
2030 		gl.transformFeedbackVaryings(m_po, 1, &xfb_varying, GL_INTERLEAVED_ATTRIBS);
2031 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
2032 
2033 		/* Link. */
2034 		gl.linkProgram(m_po);
2035 
2036 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
2037 
2038 		glw::GLint status = GL_FALSE;
2039 
2040 		gl.getProgramiv(m_po, GL_LINK_STATUS, &status);
2041 
2042 		if (GL_TRUE == status)
2043 		{
2044 			for (glw::GLuint i = 0; i < shader_count; ++i)
2045 			{
2046 				if (shader[i].id)
2047 				{
2048 					gl.detachShader(m_po, shader[i].id);
2049 
2050 					GLU_EXPECT_NO_ERROR(gl.getError(), "glDetachShader call failed.");
2051 				}
2052 			}
2053 		}
2054 		else
2055 		{
2056 			glw::GLint log_size = 0;
2057 
2058 			gl.getProgramiv(m_po, GL_INFO_LOG_LENGTH, &log_size);
2059 
2060 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv call failed.");
2061 
2062 			glw::GLchar* log_text = new glw::GLchar[log_size];
2063 
2064 			gl.getProgramInfoLog(m_po, log_size, NULL, &log_text[0]);
2065 
2066 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Program linkage has failed due to:\n"
2067 												<< log_text << "\n"
2068 												<< tcu::TestLog::EndMessage;
2069 
2070 			delete[] log_text;
2071 
2072 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog call failed.");
2073 
2074 			throw 0;
2075 		}
2076 	}
2077 	catch (...)
2078 	{
2079 		if (m_po)
2080 		{
2081 			gl.deleteProgram(m_po);
2082 
2083 			m_po = 0;
2084 		}
2085 	}
2086 
2087 	for (glw::GLuint i = 0; i < shader_count; ++i)
2088 	{
2089 		if (0 != shader[i].id)
2090 		{
2091 			gl.deleteShader(shader[i].id);
2092 
2093 			shader[i].id = 0;
2094 		}
2095 	}
2096 
2097 	if (0 == m_po)
2098 	{
2099 		throw 0;
2100 	}
2101 }
2102 
2103 template <>
NormalizationScaleFactor()2104 glw::GLdouble AttributeFormatTest::NormalizationScaleFactor<glw::GLuint>()
2105 {
2106 	return std::pow(2.0, (glw::GLdouble)(CHAR_BIT * sizeof(glw::GLuint) - 4 /* 1.0 / 16.0 */));
2107 }
2108 
2109 template <>
NormalizationScaleFactor()2110 glw::GLdouble AttributeFormatTest::NormalizationScaleFactor<glw::GLushort>()
2111 {
2112 	return std::pow(2.0, (glw::GLdouble)(CHAR_BIT * sizeof(glw::GLushort) - 4 /* 1.0 / 16.0 */));
2113 }
2114 
2115 template <>
NormalizationScaleFactor()2116 glw::GLdouble AttributeFormatTest::NormalizationScaleFactor<glw::GLubyte>()
2117 {
2118 	return std::pow(2.0, (glw::GLdouble)(CHAR_BIT * sizeof(glw::GLubyte) - 4 /* 1.0 / 16.0 */));
2119 }
2120 
2121 template <>
NormalizationScaleFactor()2122 glw::GLdouble AttributeFormatTest::NormalizationScaleFactor<glw::GLint>()
2123 {
2124 	return std::pow(2.0, (glw::GLdouble)(CHAR_BIT * sizeof(glw::GLint) - 4 /* 1.0 / 16.0 */ - 1 /* sign bit */));
2125 }
2126 
2127 template <>
NormalizationScaleFactor()2128 glw::GLdouble AttributeFormatTest::NormalizationScaleFactor<glw::GLshort>()
2129 {
2130 	return std::pow(2.0, (glw::GLdouble)(CHAR_BIT * sizeof(glw::GLshort) - 4 /* 1.0 / 16.0 */ - 1 /* sign bit */));
2131 }
2132 
2133 template <>
NormalizationScaleFactor()2134 glw::GLdouble AttributeFormatTest::NormalizationScaleFactor<glw::GLbyte>()
2135 {
2136 	return std::pow(2.0, (glw::GLdouble)(CHAR_BIT * sizeof(glw::GLbyte) - 4 /* 1.0 / 16.0 */ - 1 /* sign bit */));
2137 }
2138 
2139 template <typename T>
NormalizationScaleFactor()2140 glw::GLdouble AttributeFormatTest::NormalizationScaleFactor()
2141 {
2142 	return 1.0; /* Rest of the types cannot be normalized. */
2143 }
2144 
2145 /** @brief Prepare vertex array object for the test of VertexArrayAttrib*Format function.
2146  *
2147  *  @param [in] size                Size passed to VertexArrayAttrib*Format.
2148  *  @param [in] type_gl_name        Type passed to VertexArrayAttrib*Format.
2149  *  @param [in] function_selector   Selects one of VertexArrayAttrib*Format functions.
2150  *
2151  *  @return True if function VertexArrayAttrib*Format does not generate any error.
2152  */
2153 template <typename T>
PrepareVAO(glw::GLint size,glw::GLenum type_gl_name,bool normalized,AtributeFormatFunctionType function_selector)2154 bool AttributeFormatTest::PrepareVAO(glw::GLint size, glw::GLenum type_gl_name, bool normalized,
2155 									 AtributeFormatFunctionType function_selector)
2156 {
2157 	/* Shortcut for GL functionality */
2158 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2159 
2160 	/* VAO creation. */
2161 	gl.genVertexArrays(1, &m_vao);
2162 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays call failed.");
2163 
2164 	gl.bindVertexArray(m_vao);
2165 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
2166 
2167 	/* Array buffer 0 creation. */
2168 
2169 	const glw::GLdouble scale = normalized ? NormalizationScaleFactor<T>() : 1.0;
2170 
2171 	const T array_data[16] = { (T)(0.0 * scale),  (T)(1.0 * scale),  (T)(2.0 * scale),  (T)(3.0 * scale),
2172 							   (T)(4.0 * scale),  (T)(5.0 * scale),  (T)(6.0 * scale),  (T)(7.0 * scale),
2173 							   (T)(8.0 * scale),  (T)(9.0 * scale),  (T)(10.0 * scale), (T)(11.0 * scale),
2174 							   (T)(12.0 * scale), (T)(13.0 * scale), (T)(14.0 * scale), (T)(15.0 * scale) };
2175 
2176 	gl.genBuffers(1, &m_bo_array);
2177 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
2178 
2179 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo_array);
2180 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
2181 
2182 	gl.bufferData(GL_ARRAY_BUFFER, sizeof(array_data), array_data, GL_STATIC_DRAW);
2183 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
2184 
2185 	/* Attribute setup. */
2186 	gl.vertexAttribBinding(gl.getAttribLocation(m_po, "a_0"), 0);
2187 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribBinding call failed.");
2188 
2189 	gl.vertexAttribBinding(gl.getAttribLocation(m_po, "a_1"), 1);
2190 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribBinding call failed.");
2191 
2192 	/* Tested attribute format setup. */
2193 	switch (function_selector)
2194 	{
2195 	case ATTRIBUTE_FORMAT_FUNCTION_FLOAT:
2196 		gl.vertexArrayAttribFormat(m_vao, gl.getAttribLocation(m_po, "a_0"), size, type_gl_name, normalized, 0);
2197 		gl.vertexArrayAttribFormat(m_vao, gl.getAttribLocation(m_po, "a_1"), size, type_gl_name, normalized, 0);
2198 		break;
2199 
2200 	case ATTRIBUTE_FORMAT_FUNCTION_DOUBLE:
2201 		gl.vertexArrayAttribLFormat(m_vao, gl.getAttribLocation(m_po, "a_0"), size, type_gl_name, 0);
2202 		gl.vertexArrayAttribLFormat(m_vao, gl.getAttribLocation(m_po, "a_1"), size, type_gl_name, 0);
2203 		break;
2204 
2205 	case ATTRIBUTE_FORMAT_FUNCTION_INTEGER:
2206 		gl.vertexArrayAttribIFormat(m_vao, gl.getAttribLocation(m_po, "a_0"), size, type_gl_name, 0);
2207 		gl.vertexArrayAttribIFormat(m_vao, gl.getAttribLocation(m_po, "a_1"), size, type_gl_name, 0);
2208 		break;
2209 	default:
2210 		throw 0;
2211 	}
2212 
2213 	if (glw::GLenum error = gl.getError())
2214 	{
2215 		m_context.getTestContext().getLog()
2216 			<< tcu::TestLog::Message
2217 			<< ((ATTRIBUTE_FORMAT_FUNCTION_FLOAT == function_selector) ?
2218 					"VertexArrayAttribFormat" :
2219 					((ATTRIBUTE_FORMAT_FUNCTION_DOUBLE == function_selector) ?
2220 						 "VertexArrayAttribLFormat" :
2221 						 ((ATTRIBUTE_FORMAT_FUNCTION_INTEGER == function_selector) ? "VertexArrayAttribIFormat" :
2222 																					 "VertexArrayAttrib?Format")))
2223 			<< " has unexpectedly generated " << glu::getErrorStr(error) << "error for test with size = " << size
2224 			<< ", type = " << glu::getTypeStr(type_gl_name)
2225 			<< ((ATTRIBUTE_FORMAT_FUNCTION_FLOAT == function_selector) ?
2226 					(normalized ? ", which was normalized." : ", which was not normalized.") :
2227 					".")
2228 			<< " Test fails.\n"
2229 			<< tcu::TestLog::EndMessage;
2230 
2231 		return false;
2232 	}
2233 
2234 	gl.bindVertexBuffer(0, m_bo_array, 0, static_cast<glw::GLsizei>(sizeof(T) * size * 2));
2235 	gl.bindVertexBuffer(1, m_bo_array, size * sizeof(T),
2236 						static_cast<glw::GLsizei>(sizeof(T) * size * 2));
2237 
2238 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
2239 
2240 	gl.enableVertexAttribArray(0);
2241 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
2242 
2243 	gl.enableVertexAttribArray(1);
2244 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
2245 
2246 	gl.bindVertexArray(0);
2247 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
2248 
2249 	return true;
2250 }
2251 
2252 /** @brief Prepare buffer object for test GLSL program transform feedback results.
2253  */
PrepareXFB()2254 void AttributeFormatTest::PrepareXFB()
2255 {
2256 	/* Shortcut for GL functionality */
2257 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2258 
2259 	/* Buffer creation. */
2260 	gl.genBuffers(1, &m_bo_xfb);
2261 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
2262 
2263 	gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, m_bo_xfb);
2264 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
2265 
2266 	/* Calculating maximum size. */
2267 	glw::GLsizei size = static_cast<glw::GLsizei>(
2268 		de::max(sizeof(glw::GLubyte),
2269 				de::max(sizeof(glw::GLbyte),
2270 						de::max(sizeof(glw::GLushort),
2271 								de::max(sizeof(glw::GLshort),
2272 										de::max(sizeof(glw::GLhalf),
2273 												de::max(sizeof(glw::GLint),
2274 														de::max(sizeof(glw::GLuint),
2275 																de::max(sizeof(glw::GLfixed),
2276 																		de::max(sizeof(glw::GLfloat),
2277 																				sizeof(glw::GLdouble)))))))))) *
2278 		4 /* maximum number of components */);
2279 
2280 	/* Preparing storage. */
2281 	gl.bufferStorage(GL_TRANSFORM_FEEDBACK_BUFFER, size, NULL, GL_MAP_READ_BIT);
2282 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferStorage call failed.");
2283 
2284 	gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, m_bo_xfb);
2285 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBufferBase call failed.");
2286 }
2287 
2288 template <>
compare(glw::GLfloat a,glw::GLfloat b)2289 bool AttributeFormatTest::compare<glw::GLfloat>(glw::GLfloat a, glw::GLfloat b)
2290 {
2291 	if (de::abs(a - b) < 0.03125)
2292 	{
2293 		return true;
2294 	}
2295 
2296 	return false;
2297 }
2298 
2299 template <>
compare(glw::GLdouble a,glw::GLdouble b)2300 bool AttributeFormatTest::compare<glw::GLdouble>(glw::GLdouble a, glw::GLdouble b)
2301 {
2302 	if (de::abs(a - b) < 0.03125)
2303 	{
2304 		return true;
2305 	}
2306 
2307 	return false;
2308 }
2309 
2310 template <typename T>
compare(T a,T b)2311 bool AttributeFormatTest::compare(T a, T b)
2312 {
2313 	return (a == b);
2314 }
2315 
2316 /** @brief Draw test program, fetch transform feedback results and compare them with expected values.
2317  *
2318  *  @param [in] size         Count of elements of the XFB vector is expected.
2319  *  @param [in] normalized   Normalized values are expected.
2320  *
2321  *  @return True if expected results are equal to returned by XFB, false otherwise.
2322  */
2323 template <typename T>
DrawAndCheck(glw::GLint size,bool normalized)2324 bool AttributeFormatTest::DrawAndCheck(glw::GLint size, bool normalized)
2325 {
2326 	/* Shortcut for GL functionality */
2327 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2328 
2329 	/* Setup state. */
2330 	gl.useProgram(m_po);
2331 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram call failed.");
2332 
2333 	gl.bindVertexArray(m_vao);
2334 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
2335 
2336 	/* Draw. */
2337 	gl.beginTransformFeedback(GL_POINTS);
2338 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBeginTransformFeedback call failed.");
2339 
2340 	gl.drawArrays(GL_POINTS, 0, 2);
2341 	GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawArrays call failed.");
2342 
2343 	gl.endTransformFeedback();
2344 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEndTransformFeedback call failed.");
2345 
2346 	/* Result query. */
2347 	T* result_ptr = (T*)gl.mapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, GL_READ_ONLY);
2348 	GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBuffer call failed.");
2349 
2350 	T result[8] = { 0 };
2351 
2352 	for (glw::GLint i = 0; i < size * 2 /* two points */; ++i)
2353 	{
2354 		result[i] = result_ptr[i];
2355 	}
2356 
2357 	gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
2358 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer call failed.");
2359 
2360 	const glw::GLdouble scale = normalized ? (1.0 / 16.0) /* Floating point scalling factor. */ : 1.0;
2361 
2362 	const T array_data[16] = { (T)(0.0 * scale),  (T)(1.0 * scale),  (T)(2.0 * scale),  (T)(3.0 * scale),
2363 							   (T)(4.0 * scale),  (T)(5.0 * scale),  (T)(6.0 * scale),  (T)(7.0 * scale),
2364 							   (T)(8.0 * scale),  (T)(9.0 * scale),  (T)(10.0 * scale), (T)(11.0 * scale),
2365 							   (T)(12.0 * scale), (T)(13.0 * scale), (T)(14.0 * scale), (T)(15.0 * scale) };
2366 
2367 	T reference[8] = { 0 };
2368 
2369 	for (glw::GLint i = 0; i < 2 /* two points */; ++i)
2370 	{
2371 		for (glw::GLint j = 0; j < size /* size components */; ++j)
2372 		{
2373 			reference[i * size + j] = array_data[i * size * 2 + j] + array_data[i * size * 2 + j + size];
2374 		}
2375 	}
2376 
2377 	/* Check result and return. */
2378 	for (glw::GLint i = 0; i < size * 2 /* two points */; ++i)
2379 	{
2380 		if (!AttributeFormatTest::compare<T>(reference[i], result[i]))
2381 		{
2382 			std::string reference_str = "[ ";
2383 
2384 			for (glw::GLint j = 0; j < size * 2 /* two points */; ++j)
2385 			{
2386 				std::stringstream ss;
2387 
2388 				ss << reference[j];
2389 
2390 				reference_str.append(ss.str());
2391 
2392 				if (j < size * 2 - 1 /* if it is not the last value */)
2393 				{
2394 					reference_str.append(", ");
2395 				}
2396 				else
2397 				{
2398 					reference_str.append(" ]");
2399 				}
2400 			}
2401 
2402 			std::string result_str = "[ ";
2403 
2404 			for (glw::GLint j = 0; j < size * 2 /* two points */; ++j)
2405 			{
2406 				std::stringstream ss;
2407 
2408 				ss << result[j];
2409 
2410 				result_str.append(ss.str());
2411 
2412 				if (j < size * 2 - 1 /* if it is not the last value */)
2413 				{
2414 					result_str.append(", ");
2415 				}
2416 				else
2417 				{
2418 					result_str.append(" ]");
2419 				}
2420 			}
2421 
2422 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Result vector is equal to "
2423 												<< result_str.c_str() << ", but " << reference_str.c_str()
2424 												<< " was expected." << tcu::TestLog::EndMessage;
2425 
2426 			return false;
2427 		}
2428 	}
2429 
2430 	return true;
2431 }
2432 
2433 /** @brief Clean GLSL program object. */
CleanProgram()2434 void AttributeFormatTest::CleanProgram()
2435 {
2436 	/* Shortcut for GL functionality */
2437 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2438 
2439 	gl.useProgram(0);
2440 
2441 	if (m_po)
2442 	{
2443 		gl.deleteProgram(m_po);
2444 
2445 		m_po = 0;
2446 	}
2447 }
2448 
2449 /** @brief Clean Vertex Array Object and related buffer. */
CleanVAO()2450 void AttributeFormatTest::CleanVAO()
2451 {
2452 	/* Shortcut for GL functionality */
2453 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2454 
2455 	if (m_vao)
2456 	{
2457 		gl.deleteVertexArrays(1, &m_vao);
2458 
2459 		m_vao = 0;
2460 	}
2461 
2462 	if (m_bo_array)
2463 	{
2464 		gl.deleteBuffers(1, &m_bo_array);
2465 
2466 		m_bo_array = 0;
2467 	}
2468 }
2469 
2470 /** @brief Clean GL objects related to transform feedback. */
CleanXFB()2471 void AttributeFormatTest::CleanXFB()
2472 {
2473 	/* Shortcut for GL functionality */
2474 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2475 
2476 	if (m_bo_xfb)
2477 	{
2478 		gl.deleteBuffers(1, &m_bo_xfb);
2479 
2480 		m_bo_xfb = 0;
2481 	}
2482 
2483 	while (gl.getError())
2484 		;
2485 }
2486 
2487 const glw::GLchar* AttributeFormatTest::s_vertex_shader_head = "#version 450\n"
2488 															   "\n";
2489 
2490 const glw::GLchar* AttributeFormatTest::s_vertex_shader_body = "\n"
2491 															   "void main()\n"
2492 															   "{\n"
2493 															   "    gl_Position = vec4(1.0);\n"
2494 															   "    result = a_0 + a_1;"
2495 															   "}\n";
2496 
2497 const glw::GLchar* AttributeFormatTest::s_vertex_shader_declaration[ATTRIBUTE_FORMAT_FUNCTION_COUNT]
2498 																   [4 /* sizes count */] = { {
2499 																								 "in  float a_0;"
2500 																								 "in  float a_1;"
2501 																								 "out float result;\n",
2502 
2503 																								 "in  vec2 a_0;"
2504 																								 "in  vec2 a_1;"
2505 																								 "out vec2 result;\n",
2506 
2507 																								 "in  vec3 a_0;"
2508 																								 "in  vec3 a_1;"
2509 																								 "out vec3 result;\n",
2510 
2511 																								 "in  vec4 a_0;"
2512 																								 "in  vec4 a_1;"
2513 																								 "out vec4 result;\n",
2514 																							 },
2515 																							 {
2516 																								 "in  double a_0;"
2517 																								 "in  double a_1;"
2518 																								 "out double result;\n",
2519 
2520 																								 "in  dvec2 a_0;"
2521 																								 "in  dvec2 a_1;"
2522 																								 "out dvec2 result;\n",
2523 
2524 																								 "in  dvec3 a_0;"
2525 																								 "in  dvec3 a_1;"
2526 																								 "out dvec3 result;\n",
2527 
2528 																								 "in  dvec4 a_0;"
2529 																								 "in  dvec4 a_1;"
2530 																								 "out dvec4 result;\n",
2531 																							 },
2532 																							 {
2533 																								 "in  int a_0;"
2534 																								 "in  int a_1;"
2535 																								 "out int result;\n",
2536 
2537 																								 "in  ivec2 a_0;"
2538 																								 "in  ivec2 a_1;"
2539 																								 "out ivec2 result;\n",
2540 
2541 																								 "in  ivec3 a_0;"
2542 																								 "in  ivec3 a_1;"
2543 																								 "out ivec3 result;\n",
2544 
2545 																								 "in  ivec4 a_0;"
2546 																								 "in  ivec4 a_1;"
2547 																								 "out ivec4 result;\n",
2548 																							 } };
2549 
2550 const glw::GLchar* AttributeFormatTest::s_fragment_shader = "#version 450\n"
2551 															"\n"
2552 															"out vec4 color;\n"
2553 															"\n"
2554 															"void main()\n"
2555 															"{\n"
2556 															"    color = vec4(1.0);"
2557 															"}\n";
2558 
2559 /******************************** Vertex Array Object Attribute Binding Test Implementation   ********************************/
2560 
2561 /** @brief Attribute Binding Test constructor.
2562  *
2563  *  @param [in] context     OpenGL context.
2564  */
AttributeBindingTest(deqp::Context & context)2565 AttributeBindingTest::AttributeBindingTest(deqp::Context& context)
2566 	: deqp::TestCase(context, "vertex_arrays_attribute_binding", "Vertex Array Objects Attribute Binding Test")
2567 	, m_po(0)
2568 	, m_vao(0)
2569 	, m_bo_array(0)
2570 	, m_bo_xfb(0)
2571 {
2572 	/* Intentionally left blank. */
2573 }
2574 
2575 /** @brief Iterate Attribute Binding Test cases.
2576  *
2577  *  @return Iteration result.
2578  */
iterate()2579 tcu::TestNode::IterateResult AttributeBindingTest::iterate()
2580 {
2581 	/* Shortcut for GL functionality. */
2582 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2583 
2584 	/* Get context setup. */
2585 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
2586 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
2587 
2588 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
2589 	{
2590 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
2591 
2592 		return STOP;
2593 	}
2594 
2595 	/* Running tests. */
2596 	bool is_ok	= true;
2597 	bool is_error = false;
2598 
2599 	try
2600 	{
2601 		PrepareProgram();
2602 		is_ok &= PrepareVAO();
2603 		PrepareXFB();
2604 		is_ok &= DrawAndCheck();
2605 	}
2606 	catch (...)
2607 	{
2608 		is_ok	= false;
2609 		is_error = true;
2610 	}
2611 
2612 	/* Cleanup. */
2613 	Clean();
2614 
2615 	/* Errors clean up. */
2616 	while (gl.getError())
2617 		;
2618 
2619 	/* Result's setup. */
2620 	if (is_ok)
2621 	{
2622 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
2623 	}
2624 	else
2625 	{
2626 		if (is_error)
2627 		{
2628 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
2629 		}
2630 		else
2631 		{
2632 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
2633 		}
2634 	}
2635 
2636 	return STOP;
2637 }
2638 
2639 /** @brief Build test's GLSL program.
2640  *
2641  *  @note The function may throw if unexpected error has occured.
2642  */
PrepareProgram()2643 void AttributeBindingTest::PrepareProgram()
2644 {
2645 	/* Shortcut for GL functionality */
2646 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2647 
2648 	struct Shader
2649 	{
2650 		glw::GLchar const* const source;
2651 		glw::GLenum const		 type;
2652 		glw::GLuint				 id;
2653 	} shader[] = { { s_vertex_shader, GL_VERTEX_SHADER, 0 }, { s_fragment_shader, GL_FRAGMENT_SHADER, 0 } };
2654 
2655 	glw::GLuint const shader_count = sizeof(shader) / sizeof(shader[0]);
2656 
2657 	try
2658 	{
2659 		/* Create program. */
2660 		m_po = gl.createProgram();
2661 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram call failed.");
2662 
2663 		/* Shader compilation. */
2664 
2665 		for (glw::GLuint i = 0; i < shader_count; ++i)
2666 		{
2667 			if (DE_NULL != shader[i].source)
2668 			{
2669 				shader[i].id = gl.createShader(shader[i].type);
2670 
2671 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateShader call failed.");
2672 
2673 				gl.attachShader(m_po, shader[i].id);
2674 
2675 				GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader call failed.");
2676 
2677 				gl.shaderSource(shader[i].id, 1, &(shader[i].source), NULL);
2678 
2679 				GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource call failed.");
2680 
2681 				gl.compileShader(shader[i].id);
2682 
2683 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader call failed.");
2684 
2685 				glw::GLint status = GL_FALSE;
2686 
2687 				gl.getShaderiv(shader[i].id, GL_COMPILE_STATUS, &status);
2688 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
2689 
2690 				if (GL_FALSE == status)
2691 				{
2692 					glw::GLint log_size = 0;
2693 					gl.getShaderiv(shader[i].id, GL_INFO_LOG_LENGTH, &log_size);
2694 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
2695 
2696 					glw::GLchar* log_text = new glw::GLchar[log_size];
2697 
2698 					gl.getShaderInfoLog(shader[i].id, log_size, NULL, &log_text[0]);
2699 
2700 					m_context.getTestContext().getLog() << tcu::TestLog::Message << "Shader compilation has failed.\n"
2701 														<< "Shader type: " << glu::getShaderTypeStr(shader[i].type)
2702 														<< "\n"
2703 														<< "Shader compilation error log:\n"
2704 														<< log_text << "\n"
2705 														<< "Shader source code:\n"
2706 														<< shader[i].source << "\n"
2707 														<< tcu::TestLog::EndMessage;
2708 
2709 					delete[] log_text;
2710 
2711 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog call failed.");
2712 
2713 					throw 0;
2714 				}
2715 			}
2716 		}
2717 
2718 		/* Binding attributes. */
2719 		gl.bindAttribLocation(m_po, 0, "a_0");
2720 		gl.bindAttribLocation(m_po, 1, "a_1");
2721 		GLU_EXPECT_NO_ERROR(gl.getError(), "glBindAttribLocation call failed.");
2722 
2723 		/* Transform Feedback setup. */
2724 		static const glw::GLchar* xfb_varying = "result";
2725 		gl.transformFeedbackVaryings(m_po, 1, &xfb_varying, GL_INTERLEAVED_ATTRIBS);
2726 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
2727 
2728 		/* Link. */
2729 		gl.linkProgram(m_po);
2730 
2731 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
2732 
2733 		glw::GLint status = GL_FALSE;
2734 
2735 		gl.getProgramiv(m_po, GL_LINK_STATUS, &status);
2736 
2737 		if (GL_TRUE == status)
2738 		{
2739 			for (glw::GLuint i = 0; i < shader_count; ++i)
2740 			{
2741 				if (shader[i].id)
2742 				{
2743 					gl.detachShader(m_po, shader[i].id);
2744 
2745 					GLU_EXPECT_NO_ERROR(gl.getError(), "glDetachShader call failed.");
2746 				}
2747 			}
2748 		}
2749 		else
2750 		{
2751 			glw::GLint log_size = 0;
2752 
2753 			gl.getProgramiv(m_po, GL_INFO_LOG_LENGTH, &log_size);
2754 
2755 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv call failed.");
2756 
2757 			glw::GLchar* log_text = new glw::GLchar[log_size];
2758 
2759 			gl.getProgramInfoLog(m_po, log_size, NULL, &log_text[0]);
2760 
2761 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Program linkage has failed due to:\n"
2762 												<< log_text << "\n"
2763 												<< tcu::TestLog::EndMessage;
2764 
2765 			delete[] log_text;
2766 
2767 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog call failed.");
2768 
2769 			throw 0;
2770 		}
2771 	}
2772 	catch (...)
2773 	{
2774 		if (m_po)
2775 		{
2776 			gl.deleteProgram(m_po);
2777 
2778 			m_po = 0;
2779 		}
2780 	}
2781 
2782 	for (glw::GLuint i = 0; i < shader_count; ++i)
2783 	{
2784 		if (0 != shader[i].id)
2785 		{
2786 			gl.deleteShader(shader[i].id);
2787 
2788 			shader[i].id = 0;
2789 		}
2790 	}
2791 
2792 	if (0 == m_po)
2793 	{
2794 		throw 0;
2795 	}
2796 }
2797 
2798 /** @brief Prepare vertex array object for the test.
2799  *
2800  *  @return True if function VertexArrayAttribBinding does not generate any error.
2801  */
PrepareVAO()2802 bool AttributeBindingTest::PrepareVAO()
2803 {
2804 	/* Shortcut for GL functionality */
2805 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2806 
2807 	/* VAO creation. */
2808 	gl.genVertexArrays(1, &m_vao);
2809 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays call failed.");
2810 
2811 	gl.bindVertexArray(m_vao);
2812 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
2813 
2814 	/* Array buffer creation. */
2815 	glw::GLint array_data[2] = { 1, 0 };
2816 
2817 	gl.genBuffers(1, &m_bo_array);
2818 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
2819 
2820 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo_array);
2821 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
2822 
2823 	gl.bufferData(GL_ARRAY_BUFFER, sizeof(array_data), array_data, GL_STATIC_DRAW);
2824 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
2825 
2826 	gl.vertexAttribIPointer(0, 1, GL_INT, sizeof(glw::GLint) * 2, NULL);
2827 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribIPointer call failed.");
2828 
2829 	gl.vertexAttribIPointer(1, 1, GL_INT, sizeof(glw::GLint) * 2, glu::BufferOffsetAsPointer(1 * sizeof(glw::GLint)));
2830 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribIPointer call failed.");
2831 
2832 	gl.enableVertexAttribArray(0);
2833 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
2834 
2835 	gl.enableVertexAttribArray(1);
2836 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
2837 
2838 	gl.vertexArrayAttribBinding(m_vao, 0, 1);
2839 	gl.vertexArrayAttribBinding(m_vao, 1, 0);
2840 
2841 	if (glw::GLenum error = gl.getError())
2842 	{
2843 		m_context.getTestContext().getLog() << tcu::TestLog::Message
2844 											<< "VertexArrayAttribBinding has unexpectedly generated "
2845 											<< glu::getErrorStr(error) << "error. Test fails.\n"
2846 											<< tcu::TestLog::EndMessage;
2847 
2848 		return false;
2849 	}
2850 
2851 	return true;
2852 }
2853 
2854 /** @brief Prepare buffer object for test GLSL program transform feedback results.
2855  */
PrepareXFB()2856 void AttributeBindingTest::PrepareXFB()
2857 {
2858 	/* Shortcut for GL functionality */
2859 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2860 
2861 	/* Buffer creation. */
2862 	gl.genBuffers(1, &m_bo_xfb);
2863 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
2864 
2865 	gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, m_bo_xfb);
2866 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
2867 
2868 	/* Preparing storage. */
2869 	gl.bufferStorage(GL_TRANSFORM_FEEDBACK_BUFFER, 2 * sizeof(glw::GLint), NULL, GL_MAP_READ_BIT);
2870 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferStorage call failed.");
2871 
2872 	gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, m_bo_xfb);
2873 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBufferBase call failed.");
2874 }
2875 
2876 /** @brief Draw test program, fetch transform feedback results and compare them with expected values.
2877  *
2878  *  @return True if expected results are equal to returned by XFB, false otherwise.
2879  */
DrawAndCheck()2880 bool AttributeBindingTest::DrawAndCheck()
2881 {
2882 	/* Shortcut for GL functionality */
2883 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2884 
2885 	/* Setup state. */
2886 	gl.useProgram(m_po);
2887 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram call failed.");
2888 
2889 	gl.bindVertexArray(m_vao);
2890 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
2891 
2892 	gl.beginTransformFeedback(GL_POINTS);
2893 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBeginTransformFeedback call failed.");
2894 
2895 	/* Draw. */
2896 	gl.drawArrays(GL_POINTS, 0, 1);
2897 	GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawArrays call failed.");
2898 
2899 	/* State reset. */
2900 	gl.endTransformFeedback();
2901 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEndTransformFeedback call failed.");
2902 
2903 	/* Result query. */
2904 	glw::GLint* result_ptr = (glw::GLint*)gl.mapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, GL_READ_ONLY);
2905 	GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBuffer call failed.");
2906 
2907 	glw::GLint result[2] = { result_ptr[0], result_ptr[1] };
2908 
2909 	gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
2910 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer call failed.");
2911 
2912 	/* Check result and return. */
2913 	if ((0 == result[0]) || (1 == result[1]))
2914 	{
2915 		return true;
2916 	}
2917 
2918 	return false;
2919 }
2920 
2921 /** @brief Clean GL objects. */
Clean()2922 void AttributeBindingTest::Clean()
2923 {
2924 	/* Shortcut for GL functionality */
2925 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2926 
2927 	gl.useProgram(0);
2928 
2929 	if (m_po)
2930 	{
2931 		gl.deleteProgram(m_po);
2932 
2933 		m_po = 0;
2934 	}
2935 
2936 	if (m_vao)
2937 	{
2938 		gl.deleteVertexArrays(1, &m_vao);
2939 
2940 		m_vao = 0;
2941 	}
2942 
2943 	if (m_bo_array)
2944 	{
2945 		gl.deleteBuffers(1, &m_bo_array);
2946 
2947 		m_bo_array = 0;
2948 	}
2949 
2950 	if (m_bo_xfb)
2951 	{
2952 		gl.deleteBuffers(1, &m_bo_xfb);
2953 
2954 		m_bo_xfb = 0;
2955 	}
2956 
2957 	while (gl.getError())
2958 		;
2959 }
2960 
2961 const glw::GLchar AttributeBindingTest::s_vertex_shader[] = "#version 450\n"
2962 															"\n"
2963 															"in int a_0;\n"
2964 															"in int a_1;\n"
2965 															"out ivec2 result;\n"
2966 															"\n"
2967 															"void main()\n"
2968 															"{\n"
2969 															"    gl_Position = vec4(1.0);\n"
2970 															"    result[0] = a_0;\n"
2971 															"    result[1] = a_1;\n"
2972 															"}\n";
2973 
2974 const glw::GLchar AttributeBindingTest::s_fragment_shader[] = "#version 450\n"
2975 															  "\n"
2976 															  "out vec4 color;\n"
2977 															  "\n"
2978 															  "void main()\n"
2979 															  "{\n"
2980 															  "    color = vec4(1.0);"
2981 															  "}\n";
2982 
2983 /******************************** Vertex Array Attribute Binding Divisor Test Implementation   ********************************/
2984 
2985 /** @brief Vertex Array Attribute Binding Divisor Test constructor.
2986  *
2987  *  @param [in] context     OpenGL context.
2988  */
AttributeBindingDivisorTest(deqp::Context & context)2989 AttributeBindingDivisorTest::AttributeBindingDivisorTest(deqp::Context& context)
2990 	: deqp::TestCase(context, "vertex_arrays_attribute_binding_divisor", "Vertex Array Attribute Binding Divisor Test")
2991 	, m_po(0)
2992 	, m_vao(0)
2993 	, m_bo_array(0)
2994 	, m_bo_xfb(0)
2995 {
2996 	/* Intentionally left blank. */
2997 }
2998 
2999 /** @brief Iterate Vertex Array Attribute Binding Divisor Test cases.
3000  *
3001  *  @return Iteration result.
3002  */
iterate()3003 tcu::TestNode::IterateResult AttributeBindingDivisorTest::iterate()
3004 {
3005 	/* Shortcut for GL functionality. */
3006 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3007 
3008 	/* Get context setup. */
3009 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
3010 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
3011 
3012 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
3013 	{
3014 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
3015 
3016 		return STOP;
3017 	}
3018 
3019 	/* Running tests. */
3020 	bool is_ok	= true;
3021 	bool is_error = false;
3022 
3023 	try
3024 	{
3025 		PrepareProgram();
3026 		PrepareVAO();
3027 		PrepareXFB();
3028 
3029 		{
3030 			glw::GLint reference[] = { 0, 2 };
3031 			is_ok				   = SetDivisor(2);
3032 			Draw(1, 2);
3033 			is_ok = CheckXFB((sizeof(reference) / sizeof(reference[0])), reference,
3034 							 "Draw of 1 point with 2 instances with 2 divisor has failed.");
3035 		}
3036 
3037 		{
3038 			glw::GLint reference[] = { 0, 0, 1, 1 };
3039 			is_ok				   = SetDivisor(1);
3040 			Draw(2, 2);
3041 			is_ok = CheckXFB((sizeof(reference) / sizeof(reference[0])), reference,
3042 							 "Draw of 2 points with 2 instances with 1 divisor has failed.");
3043 		}
3044 
3045 		{
3046 			glw::GLint reference[] = { 0, 1, 2, 3 };
3047 			is_ok				   = SetDivisor(1);
3048 			Draw(1, 4);
3049 			is_ok = CheckXFB((sizeof(reference) / sizeof(reference[0])), reference,
3050 							 "Draw of 1 point with 4 instances with 1 divisor has failed.");
3051 		}
3052 
3053 		{
3054 			glw::GLint reference[] = { 0, 1, 0, 1 };
3055 			is_ok				   = SetDivisor(0);
3056 			Draw(2, 2);
3057 			is_ok = CheckXFB((sizeof(reference) / sizeof(reference[0])), reference,
3058 							 "Draw of 2 points with 2 instances with 0 divisor has failed.");
3059 		}
3060 
3061 		{
3062 			glw::GLint reference[] = { 0, 1, 2, 3 };
3063 			is_ok				   = SetDivisor(0);
3064 			Draw(4, 1);
3065 			is_ok = CheckXFB((sizeof(reference) / sizeof(reference[0])), reference,
3066 							 "Draw of 4 points with 1 instance with 0 divisor has failed.");
3067 		}
3068 	}
3069 	catch (...)
3070 	{
3071 		is_ok	= false;
3072 		is_error = true;
3073 	}
3074 
3075 	/* Cleanup. */
3076 	Clean();
3077 
3078 	/* Errors clean up. */
3079 	while (gl.getError())
3080 		;
3081 
3082 	/* Result's setup. */
3083 	if (is_ok)
3084 	{
3085 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
3086 	}
3087 	else
3088 	{
3089 		if (is_error)
3090 		{
3091 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
3092 		}
3093 		else
3094 		{
3095 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
3096 		}
3097 	}
3098 
3099 	return STOP;
3100 }
3101 
3102 /** @brief Build test's GLSL program.
3103  *
3104  *  @note The function may throw if unexpected error has occured.
3105  */
PrepareProgram()3106 void AttributeBindingDivisorTest::PrepareProgram()
3107 {
3108 	/* Shortcut for GL functionality */
3109 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3110 
3111 	struct Shader
3112 	{
3113 		glw::GLchar const* const source;
3114 		glw::GLenum const		 type;
3115 		glw::GLuint				 id;
3116 	} shader[] = { { s_vertex_shader, GL_VERTEX_SHADER, 0 }, { s_fragment_shader, GL_FRAGMENT_SHADER, 0 } };
3117 
3118 	glw::GLuint const shader_count = sizeof(shader) / sizeof(shader[0]);
3119 
3120 	try
3121 	{
3122 		/* Create program. */
3123 		m_po = gl.createProgram();
3124 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram call failed.");
3125 
3126 		/* Shader compilation. */
3127 
3128 		for (glw::GLuint i = 0; i < shader_count; ++i)
3129 		{
3130 			if (DE_NULL != shader[i].source)
3131 			{
3132 				shader[i].id = gl.createShader(shader[i].type);
3133 
3134 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateShader call failed.");
3135 
3136 				gl.attachShader(m_po, shader[i].id);
3137 
3138 				GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader call failed.");
3139 
3140 				gl.shaderSource(shader[i].id, 1, &(shader[i].source), NULL);
3141 
3142 				GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource call failed.");
3143 
3144 				gl.compileShader(shader[i].id);
3145 
3146 				GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader call failed.");
3147 
3148 				glw::GLint status = GL_FALSE;
3149 
3150 				gl.getShaderiv(shader[i].id, GL_COMPILE_STATUS, &status);
3151 				GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
3152 
3153 				if (GL_FALSE == status)
3154 				{
3155 					glw::GLint log_size = 0;
3156 					gl.getShaderiv(shader[i].id, GL_INFO_LOG_LENGTH, &log_size);
3157 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
3158 
3159 					glw::GLchar* log_text = new glw::GLchar[log_size];
3160 
3161 					gl.getShaderInfoLog(shader[i].id, log_size, NULL, &log_text[0]);
3162 
3163 					m_context.getTestContext().getLog() << tcu::TestLog::Message << "Shader compilation has failed.\n"
3164 														<< "Shader type: " << glu::getShaderTypeStr(shader[i].type)
3165 														<< "\n"
3166 														<< "Shader compilation error log:\n"
3167 														<< log_text << "\n"
3168 														<< "Shader source code:\n"
3169 														<< shader[i].source << "\n"
3170 														<< tcu::TestLog::EndMessage;
3171 
3172 					delete[] log_text;
3173 
3174 					GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog call failed.");
3175 
3176 					throw 0;
3177 				}
3178 			}
3179 		}
3180 
3181 		/* Transform Feedback setup. */
3182 		static const glw::GLchar* xfb_varying = "result";
3183 
3184 		gl.transformFeedbackVaryings(m_po, 1, &xfb_varying, GL_INTERLEAVED_ATTRIBS);
3185 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
3186 
3187 		/* Link. */
3188 		gl.linkProgram(m_po);
3189 
3190 		GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
3191 
3192 		glw::GLint status = GL_FALSE;
3193 
3194 		gl.getProgramiv(m_po, GL_LINK_STATUS, &status);
3195 
3196 		if (GL_TRUE == status)
3197 		{
3198 			for (glw::GLuint i = 0; i < shader_count; ++i)
3199 			{
3200 				if (shader[i].id)
3201 				{
3202 					gl.detachShader(m_po, shader[i].id);
3203 
3204 					GLU_EXPECT_NO_ERROR(gl.getError(), "glDetachShader call failed.");
3205 				}
3206 			}
3207 		}
3208 		else
3209 		{
3210 			glw::GLint log_size = 0;
3211 
3212 			gl.getProgramiv(m_po, GL_INFO_LOG_LENGTH, &log_size);
3213 
3214 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv call failed.");
3215 
3216 			glw::GLchar* log_text = new glw::GLchar[log_size];
3217 
3218 			gl.getProgramInfoLog(m_po, log_size, NULL, &log_text[0]);
3219 
3220 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Program linkage has failed due to:\n"
3221 												<< log_text << "\n"
3222 												<< tcu::TestLog::EndMessage;
3223 
3224 			delete[] log_text;
3225 
3226 			GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog call failed.");
3227 
3228 			throw 0;
3229 		}
3230 	}
3231 	catch (...)
3232 	{
3233 		if (m_po)
3234 		{
3235 			gl.deleteProgram(m_po);
3236 
3237 			m_po = 0;
3238 		}
3239 	}
3240 
3241 	for (glw::GLuint i = 0; i < shader_count; ++i)
3242 	{
3243 		if (0 != shader[i].id)
3244 		{
3245 			gl.deleteShader(shader[i].id);
3246 
3247 			shader[i].id = 0;
3248 		}
3249 	}
3250 
3251 	if (m_po)
3252 	{
3253 		gl.useProgram(m_po);
3254 		GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram call failed.");
3255 	}
3256 
3257 	if (0 == m_po)
3258 	{
3259 		throw 0;
3260 	}
3261 }
3262 
3263 /** @brief Prepare vertex array object for the test.
3264  */
PrepareVAO()3265 void AttributeBindingDivisorTest::PrepareVAO()
3266 {
3267 	/* Shortcut for GL functionality */
3268 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3269 
3270 	/* VAO creation. */
3271 	gl.genVertexArrays(1, &m_vao);
3272 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays call failed.");
3273 
3274 	gl.bindVertexArray(m_vao);
3275 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
3276 
3277 	/* Array buffer 0 creation. */
3278 	glw::GLint array_data[4] = { 0, 1, 2, 3 };
3279 
3280 	gl.genBuffers(1, &m_bo_array);
3281 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
3282 
3283 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo_array);
3284 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
3285 
3286 	gl.bufferData(GL_ARRAY_BUFFER, sizeof(array_data), array_data, GL_STATIC_DRAW);
3287 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData call failed.");
3288 
3289 	gl.vertexAttribBinding(gl.getAttribLocation(m_po, "a"), 0);
3290 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribBinding call failed.");
3291 
3292 	gl.vertexAttribIFormat(gl.getAttribLocation(m_po, "a"), 1, GL_INT, 0);
3293 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribIFormat call failed.");
3294 
3295 	gl.bindVertexBuffer(0, m_bo_array, 0, sizeof(glw::GLint));
3296 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexBuffer call failed.");
3297 
3298 	gl.enableVertexAttribArray(gl.getAttribLocation(m_po, "a"));
3299 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
3300 }
3301 
3302 /** @brief Prepare buffer object for test GLSL program transform feedback results.
3303  */
PrepareXFB()3304 void AttributeBindingDivisorTest::PrepareXFB()
3305 {
3306 	/* Shortcut for GL functionality */
3307 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3308 
3309 	/* Buffer creation. */
3310 	gl.genBuffers(1, &m_bo_xfb);
3311 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers call failed.");
3312 
3313 	gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, m_bo_xfb);
3314 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
3315 
3316 	/* Preparing storage. */
3317 	gl.bufferStorage(GL_TRANSFORM_FEEDBACK_BUFFER, 4 * sizeof(glw::GLint), NULL, GL_MAP_READ_BIT);
3318 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferStorage call failed.");
3319 
3320 	gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, m_bo_xfb);
3321 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBufferBase call failed.");
3322 }
3323 
3324 /** @brief Draw number of points and number of instances with XFB environment.
3325  *
3326  *  @param [in] number_of_points        Number of points to be drawn.
3327  *  @param [in] number_of_instances     Number of instances to be drawn.
3328  */
Draw(glw::GLuint number_of_points,glw::GLuint number_of_instances)3329 void AttributeBindingDivisorTest::Draw(glw::GLuint number_of_points, glw::GLuint number_of_instances)
3330 {
3331 	/* Shortcut for GL functionality */
3332 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3333 
3334 	/* Setup state. */
3335 	gl.beginTransformFeedback(GL_POINTS);
3336 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBeginTransformFeedback call failed.");
3337 
3338 	/* Draw. */
3339 	gl.drawArraysInstanced(GL_POINTS, 0, number_of_points, number_of_instances);
3340 	GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawArraysInstanced call failed.");
3341 
3342 	/* State reset. */
3343 	gl.endTransformFeedback();
3344 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEndTransformFeedback call failed.");
3345 }
3346 
3347 /** @brief Call VertexArrayBindingDivisor on m_vao object and check errors.
3348  *
3349  *  @param [in] divisor        Divisor to be passed.
3350  *
3351  *  @return True if VertexArrayBindingDivisor doe not generate any error, false otherwise.
3352  */
SetDivisor(glw::GLuint divisor)3353 bool AttributeBindingDivisorTest::SetDivisor(glw::GLuint divisor)
3354 {
3355 	/* Shortcut for GL functionality */
3356 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3357 
3358 	/* Setup. */
3359 	gl.vertexArrayBindingDivisor(m_vao, 0, divisor);
3360 
3361 	/* Checking for errors (this is tested function so it fail the test if there is error). */
3362 	if (glw::GLenum error = gl.getError())
3363 	{
3364 		m_context.getTestContext().getLog()
3365 			<< tcu::TestLog::Message << "VertexArrayBindingDivisor unexpectedl generated " << glu::getErrorStr(error)
3366 			<< " error when called with divisor" << divisor << ". " << tcu::TestLog::EndMessage;
3367 
3368 		return false;
3369 	}
3370 
3371 	return true;
3372 }
3373 
3374 /** @brief Check transform feedback results and log.
3375  *
3376  *  @param [in] count           Number of results to be checked.
3377  *  @param [in] expected        Expected results.
3378  *  @param [in] log_message     Message to be logged if expected values are not equal to queried.
3379  *
3380  *  @return True if expected values are equal to queried, false otherwise.
3381  */
CheckXFB(const glw::GLuint count,const glw::GLint expected[],const glw::GLchar * log_message)3382 bool AttributeBindingDivisorTest::CheckXFB(const glw::GLuint count, const glw::GLint expected[],
3383 										   const glw::GLchar* log_message)
3384 {
3385 	/* Shortcut for GL functionality */
3386 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3387 
3388 	/* Result setup */
3389 	bool is_ok = true;
3390 
3391 	/* Result query. */
3392 	glw::GLint* result = (glw::GLint*)gl.mapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, GL_READ_ONLY);
3393 	GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBuffer call failed.");
3394 
3395 	/* Check result and return. */
3396 	for (glw::GLuint i = 0; i < count; ++i)
3397 	{
3398 		if (expected[i] != result[i])
3399 		{
3400 			std::string expected_str = "[";
3401 			std::string result_str   = "[";
3402 
3403 			for (glw::GLuint j = 0; j < count; ++j)
3404 			{
3405 				expected_str.append(Utilities::itoa((glw::GLuint)expected[j]));
3406 				result_str.append(Utilities::itoa((glw::GLuint)result[j]));
3407 
3408 				if (j < count - 1)
3409 				{
3410 					expected_str.append(", ");
3411 					result_str.append(", ");
3412 				}
3413 				else
3414 				{
3415 					expected_str.append("]");
3416 					result_str.append("]");
3417 				}
3418 			}
3419 
3420 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "Result is " << result_str << ", but "
3421 												<< expected_str << " was expected. " << log_message
3422 												<< tcu::TestLog::EndMessage;
3423 
3424 			is_ok = false;
3425 			break;
3426 		}
3427 	}
3428 
3429 	/* Unmaping GL buffer. */
3430 	gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
3431 	GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer call failed.");
3432 
3433 	return is_ok;
3434 }
3435 
3436 /** @brief Clean GL objects. */
Clean()3437 void AttributeBindingDivisorTest::Clean()
3438 {
3439 	/* Shortcut for GL functionality */
3440 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3441 
3442 	gl.useProgram(0);
3443 
3444 	if (m_po)
3445 	{
3446 		gl.deleteProgram(m_po);
3447 
3448 		m_po = 0;
3449 	}
3450 
3451 	if (m_vao)
3452 	{
3453 		gl.deleteVertexArrays(1, &m_vao);
3454 
3455 		m_vao = 0;
3456 	}
3457 
3458 	if (m_bo_array)
3459 	{
3460 		gl.deleteBuffers(1, &m_bo_array);
3461 
3462 		m_bo_array = 0;
3463 	}
3464 
3465 	if (m_bo_xfb)
3466 	{
3467 		gl.deleteBuffers(1, &m_bo_xfb);
3468 
3469 		m_bo_xfb = 0;
3470 	}
3471 
3472 	while (gl.getError())
3473 		;
3474 }
3475 
3476 const glw::GLchar AttributeBindingDivisorTest::s_vertex_shader[] = "#version 450\n"
3477 																   "\n"
3478 																   "in  int a;\n"
3479 																   "out int result;\n"
3480 																   "\n"
3481 																   "void main()\n"
3482 																   "{\n"
3483 																   "    gl_Position = vec4(1.0);\n"
3484 																   "    result = a;"
3485 																   "}\n";
3486 
3487 const glw::GLchar AttributeBindingDivisorTest::s_fragment_shader[] = "#version 450\n"
3488 																	 "\n"
3489 																	 "out vec4 color;\n"
3490 																	 "\n"
3491 																	 "void main()\n"
3492 																	 "{\n"
3493 																	 "    color = vec4(1.0);"
3494 																	 "}\n";
3495 
3496 /******************************** Get Vertex Array Test Implementation   ********************************/
3497 
3498 /** @brief Get Vertex Array Test constructor.
3499  *
3500  *  @param [in] context     OpenGL context.
3501  */
GetVertexArrayTest(deqp::Context & context)3502 GetVertexArrayTest::GetVertexArrayTest(deqp::Context& context)
3503 	: deqp::TestCase(context, "vertex_arrays_get_vertex_array", "Get Vertex Array Test")
3504 {
3505 	/* Intentionally left blank. */
3506 }
3507 
3508 /** @brief Iterate Vertex Array Attribute Binding Divisor Test cases.
3509  *
3510  *  @return Iteration result.
3511  */
iterate()3512 tcu::TestNode::IterateResult GetVertexArrayTest::iterate()
3513 {
3514 	/* Shortcut for GL functionality. */
3515 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3516 
3517 	/* Get context setup. */
3518 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
3519 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
3520 
3521 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
3522 	{
3523 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
3524 
3525 		return STOP;
3526 	}
3527 
3528 	/* Running tests. */
3529 	bool is_ok	= true;
3530 	bool is_error = false;
3531 
3532 	/* Test objects. */
3533 	glw::GLuint vao = 0;
3534 	glw::GLuint bo  = 0;
3535 
3536 	try
3537 	{
3538 		gl.genVertexArrays(1, &vao);
3539 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
3540 
3541 		gl.bindVertexArray(vao);
3542 		GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
3543 
3544 		gl.genBuffers(1, &bo);
3545 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateBuffers call failed.");
3546 
3547 		gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, bo);
3548 		GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
3549 
3550 		glw::GLint result = 0;
3551 		gl.getVertexArrayiv(vao, GL_ELEMENT_ARRAY_BUFFER_BINDING, &result);
3552 
3553 		if (glw::GLenum error = gl.getError())
3554 		{
3555 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "GetVertexArrayiv unexpectedly generated "
3556 												<< glu::getErrorStr(error) << "error. Test fails."
3557 												<< tcu::TestLog::EndMessage;
3558 
3559 			is_ok = false;
3560 		}
3561 
3562 		if ((glw::GLuint)result != bo)
3563 		{
3564 			m_context.getTestContext().getLog() << tcu::TestLog::Message << "GetVertexArrayiv was expected to return "
3565 												<< bo << ", but " << result << " was observed. Test fails."
3566 												<< tcu::TestLog::EndMessage;
3567 
3568 			is_ok = false;
3569 		}
3570 	}
3571 	catch (...)
3572 	{
3573 		is_ok	= false;
3574 		is_error = true;
3575 	}
3576 
3577 	/* Cleanup. */
3578 	if (vao)
3579 	{
3580 		gl.deleteVertexArrays(1, &vao);
3581 	}
3582 
3583 	if (bo)
3584 	{
3585 		gl.deleteBuffers(1, &bo);
3586 	}
3587 
3588 	/* Errors clean up. */
3589 	while (gl.getError())
3590 		;
3591 
3592 	/* Result's setup. */
3593 	if (is_ok)
3594 	{
3595 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
3596 	}
3597 	else
3598 	{
3599 		if (is_error)
3600 		{
3601 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
3602 		}
3603 		else
3604 		{
3605 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
3606 		}
3607 	}
3608 
3609 	return STOP;
3610 }
3611 
3612 /******************************** Get Vertex Array Test Indexed Implementation   ********************************/
3613 
3614 /** @brief Get Vertex Array Indexed Test constructor.
3615  *
3616  *  @param [in] context     OpenGL context.
3617  */
GetVertexArrayIndexedTest(deqp::Context & context)3618 GetVertexArrayIndexedTest::GetVertexArrayIndexedTest(deqp::Context& context)
3619 	: deqp::TestCase(context, "vertex_arrays_get_vertex_array_indexed", "Get Vertex Array Indexed Test"), m_vao(0)
3620 {
3621 	m_bo[0] = 0;
3622 	m_bo[1] = 0;
3623 	m_bo[2] = 0;
3624 	m_bo[3] = 0;
3625 }
3626 
3627 /** @brief Iterate Vertex Array Attribute Binding Divisor Test cases.
3628  *
3629  *  @return Iteration result.
3630  */
iterate()3631 tcu::TestNode::IterateResult GetVertexArrayIndexedTest::iterate()
3632 {
3633 	/* Shortcut for GL functionality. */
3634 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3635 
3636 	/* Get context setup. */
3637 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
3638 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
3639 
3640 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
3641 	{
3642 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
3643 
3644 		return STOP;
3645 	}
3646 
3647 	/* Running tests. */
3648 	bool is_ok	= true;
3649 	bool is_error = false;
3650 
3651 	try
3652 	{
3653 		PrepareVAO();
3654 
3655 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_ENABLED, 0, GL_TRUE);
3656 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_ENABLED, 1, GL_TRUE);
3657 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_ENABLED, 2, GL_TRUE);
3658 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_ENABLED, 3, GL_TRUE);
3659 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_ENABLED, 5, GL_FALSE);
3660 
3661 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_STRIDE, 0, 0);
3662 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_STRIDE, 1, 2);
3663 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_STRIDE, 2, 0);
3664 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_STRIDE, 3, 8);
3665 
3666 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_TYPE, 0, GL_BYTE);
3667 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_TYPE, 1, GL_SHORT);
3668 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_TYPE, 2, GL_FLOAT);
3669 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_TYPE, 3, GL_UNSIGNED_INT_2_10_10_10_REV);
3670 
3671 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_NORMALIZED, 0, GL_TRUE);
3672 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_NORMALIZED, 1, GL_FALSE);
3673 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_NORMALIZED, 2, GL_FALSE);
3674 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_NORMALIZED, 3, GL_FALSE);
3675 
3676 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_INTEGER, 0, GL_FALSE);
3677 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_INTEGER, 1, GL_TRUE);
3678 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_INTEGER, 2, GL_FALSE);
3679 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_INTEGER, 3, GL_FALSE);
3680 
3681 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_DIVISOR, 0, 3);
3682 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_DIVISOR, 1, 2);
3683 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_DIVISOR, 2, 1);
3684 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_DIVISOR, 3, 0);
3685 
3686 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_LONG, 0, GL_FALSE);
3687 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_LONG, 1, GL_FALSE);
3688 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_LONG, 2, GL_FALSE);
3689 		is_ok &= Check(GL_VERTEX_ATTRIB_ARRAY_LONG, 3, GL_FALSE);
3690 
3691 		is_ok &= Check(GL_VERTEX_ATTRIB_RELATIVE_OFFSET, 0, 0);
3692 		is_ok &= Check(GL_VERTEX_ATTRIB_RELATIVE_OFFSET, 1, 0);
3693 		is_ok &= Check(GL_VERTEX_ATTRIB_RELATIVE_OFFSET, 2, 4);
3694 		is_ok &= Check(GL_VERTEX_ATTRIB_RELATIVE_OFFSET, 3, 0);
3695 
3696 		is_ok &= Check64(GL_VERTEX_BINDING_OFFSET, 0, 0);
3697 		is_ok &= Check64(GL_VERTEX_BINDING_OFFSET, 1, 2);
3698 		is_ok &= Check64(GL_VERTEX_BINDING_OFFSET, 2, 8);
3699 		is_ok &= Check64(GL_VERTEX_BINDING_OFFSET, 3, 4);
3700 	}
3701 	catch (...)
3702 	{
3703 		is_ok	= false;
3704 		is_error = true;
3705 	}
3706 
3707 	/* Cleanup. */
3708 	if (m_vao)
3709 	{
3710 		gl.deleteVertexArrays(1, &m_vao);
3711 
3712 		m_vao = 0;
3713 	}
3714 
3715 	if (m_bo[0] || m_bo[1] || m_bo[2] || m_bo[3])
3716 	{
3717 		gl.deleteBuffers(4, m_bo);
3718 
3719 		m_bo[0] = 0;
3720 		m_bo[1] = 0;
3721 		m_bo[2] = 0;
3722 		m_bo[3] = 0;
3723 	}
3724 
3725 	/* Errors clean up. */
3726 	while (gl.getError())
3727 		;
3728 
3729 	/* Result's setup. */
3730 	if (is_ok)
3731 	{
3732 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
3733 	}
3734 	else
3735 	{
3736 		if (is_error)
3737 		{
3738 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
3739 		}
3740 		else
3741 		{
3742 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
3743 		}
3744 	}
3745 
3746 	return STOP;
3747 }
3748 
3749 /** @brief Prepare vertex array object for the test.
3750  */
PrepareVAO()3751 void GetVertexArrayIndexedTest::PrepareVAO()
3752 {
3753 	/* Shortcut for GL functionality. */
3754 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3755 
3756 	gl.genVertexArrays(1, &m_vao);
3757 	GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
3758 
3759 	gl.bindVertexArray(m_vao);
3760 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray call failed.");
3761 
3762 	gl.genBuffers(4, m_bo);
3763 	GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateBuffers call failed.");
3764 
3765 	/* Attribute 0. */
3766 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo[0]);
3767 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
3768 
3769 	gl.vertexAttribPointer(0, 1, GL_BYTE, GL_TRUE, 0, NULL);
3770 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribPointer call failed.");
3771 
3772 	gl.enableVertexAttribArray(0);
3773 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
3774 
3775 	gl.vertexAttribDivisor(0, 3);
3776 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribDivisor call failed.");
3777 
3778 	/* Attribute 1. */
3779 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo[1]);
3780 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
3781 
3782 	gl.vertexAttribIPointer(1, 2, GL_SHORT, 2, glu::BufferOffsetAsPointer(2 * sizeof(glw::GLchar)));
3783 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribPointer call failed.");
3784 
3785 	gl.enableVertexAttribArray(1);
3786 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
3787 
3788 	gl.vertexAttribDivisor(1, 2);
3789 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribDivisor call failed.");
3790 
3791 	/* Attribute 2. */
3792 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo[2]);
3793 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
3794 
3795 	gl.vertexAttribBinding(2, 2);
3796 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribBinding call failed.");
3797 
3798 	gl.vertexAttribFormat(2, 3, GL_FLOAT, GL_FALSE, 4);
3799 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribIFormat call failed.");
3800 
3801 	gl.bindVertexBuffer(2, m_bo[2], 8, 0);
3802 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexBuffer call failed.");
3803 
3804 	gl.enableVertexAttribArray(2);
3805 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
3806 
3807 	gl.vertexAttribDivisor(2, 1);
3808 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribDivisor call failed.");
3809 
3810 	/* Attribute 3. */
3811 	gl.bindBuffer(GL_ARRAY_BUFFER, m_bo[3]);
3812 	GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer call failed.");
3813 
3814 	gl.vertexAttribPointer(3, 4, GL_UNSIGNED_INT_2_10_10_10_REV, GL_FALSE, 8, glu::BufferOffsetAsPointer(4 * sizeof(glw::GLchar)));
3815 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribPointer call failed.");
3816 
3817 	gl.enableVertexAttribArray(3);
3818 	GLU_EXPECT_NO_ERROR(gl.getError(), "glEnableVertexAttribArray call failed.");
3819 
3820 	gl.vertexAttribDivisor(3, 0);
3821 	GLU_EXPECT_NO_ERROR(gl.getError(), "glVertexAttribDivisor call failed.");
3822 }
3823 
3824 /** @brief Compare value queried using GetVertexArrayIndexediv with expected value and log.
3825  *
3826  *  @param [in] pname        Parameter to be queried.
3827  *  @param [in] index        Index to be queried.
3828  *  @param [in] expected     Expected error.
3829  *
3830  *  @return True if value is equal to expected, false otherwise.
3831  */
Check(const glw::GLenum pname,const glw::GLuint index,const glw::GLint expected)3832 bool GetVertexArrayIndexedTest::Check(const glw::GLenum pname, const glw::GLuint index, const glw::GLint expected)
3833 {
3834 	/* Shortcut for GL functionality. */
3835 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3836 
3837 	glw::GLint result = 0;
3838 
3839 	gl.getVertexArrayIndexediv(m_vao, index, pname, &result);
3840 
3841 	if (glw::GLenum error = gl.getError())
3842 	{
3843 		m_context.getTestContext().getLog() << tcu::TestLog::Message << "GetVertexArrayIndexediv called with index "
3844 											<< index << ", with pname" << glu::getVertexAttribParameterNameStr(pname)
3845 											<< " unexpectedly generated " << glu::getErrorStr(error)
3846 											<< "error. Test fails." << tcu::TestLog::EndMessage;
3847 
3848 		return false;
3849 	}
3850 
3851 	if (result != expected)
3852 	{
3853 		m_context.getTestContext().getLog() << tcu::TestLog::Message << "GetVertexArrayIndexediv called with index "
3854 											<< index << " and with pname" << glu::getVertexAttribParameterNameStr(pname)
3855 											<< " returned " << result << ", but " << expected
3856 											<< " was expected. Test fails." << tcu::TestLog::EndMessage;
3857 
3858 		return false;
3859 	}
3860 
3861 	return true;
3862 }
3863 
3864 /** @brief Compare value queried using GetVertexArrayIndexed64iv with expected value and log.
3865  *
3866  *  @param [in] pname        Parameter to be queried.
3867  *  @param [in] index        Index to be queried.
3868  *  @param [in] expected     Expected error.
3869  *
3870  *  @return True if value is equal to expected, false otherwise.
3871  */
Check64(const glw::GLenum pname,const glw::GLuint index,const glw::GLint64 expected)3872 bool GetVertexArrayIndexedTest::Check64(const glw::GLenum pname, const glw::GLuint index, const glw::GLint64 expected)
3873 {
3874 	/* Shortcut for GL functionality. */
3875 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3876 
3877 	glw::GLint64 result = 0;
3878 
3879 	gl.getVertexArrayIndexed64iv(m_vao, index, pname, &result);
3880 
3881 	if (glw::GLenum error = gl.getError())
3882 	{
3883 		m_context.getTestContext().getLog() << tcu::TestLog::Message << "GetVertexArrayIndexed64iv called with index "
3884 											<< index << ", with pname" << glu::getVertexAttribParameterNameStr(pname)
3885 											<< " unexpectedly generated " << glu::getErrorStr(error)
3886 											<< "error. Test fails." << tcu::TestLog::EndMessage;
3887 
3888 		return false;
3889 	}
3890 
3891 	if (result != expected)
3892 	{
3893 		m_context.getTestContext().getLog() << tcu::TestLog::Message << "GetVertexArrayIndexed64iv called with index "
3894 											<< index << " and with pname" << glu::getVertexAttribParameterNameStr(pname)
3895 											<< " returned " << result << ", but " << expected
3896 											<< " was expected. Test fails." << tcu::TestLog::EndMessage;
3897 
3898 		return false;
3899 	}
3900 
3901 	return true;
3902 }
3903 
3904 /******************************** Defaults Test Implementation   ********************************/
3905 
3906 /** @brief Defaults Test constructor.
3907  *
3908  *  @param [in] context     OpenGL context.
3909  */
DefaultsTest(deqp::Context & context)3910 DefaultsTest::DefaultsTest(deqp::Context& context)
3911 	: deqp::TestCase(context, "vertex_arrays_defaults", "Defaults Test"), m_vao(0)
3912 {
3913 }
3914 
3915 /** @brief Iterate Defaults Test cases.
3916  *
3917  *  @return Iteration result.
3918  */
iterate()3919 tcu::TestNode::IterateResult DefaultsTest::iterate()
3920 {
3921 	/* Shortcut for GL functionality. */
3922 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
3923 
3924 	/* Get context setup. */
3925 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
3926 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
3927 
3928 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
3929 	{
3930 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
3931 
3932 		return STOP;
3933 	}
3934 
3935 	/* Running tests. */
3936 	bool is_ok	= true;
3937 	bool is_error = false;
3938 
3939 	/* Test objects. */
3940 	glw::GLint max_attributes = 8;
3941 
3942 	try
3943 	{
3944 		/* Query limits. */
3945 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIBS, &max_attributes);
3946 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv call failed.");
3947 
3948 		/* Prepare default Vertex Array Object. */
3949 		PrepareVAO();
3950 
3951 		/* Check default values per attribute index. */
3952 		for (glw::GLint i = 0; i < max_attributes; ++i)
3953 		{
3954 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_ARRAY_ENABLED, i, GL_FALSE);
3955 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_ARRAY_SIZE, i, 4);
3956 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_ARRAY_STRIDE, i, 0);
3957 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_ARRAY_TYPE, i, GL_FLOAT);
3958 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_ARRAY_NORMALIZED, i, GL_FALSE);
3959 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_ARRAY_INTEGER, i, GL_FALSE);
3960 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_ARRAY_DIVISOR, i, 0);
3961 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_ARRAY_LONG, i, GL_FALSE);
3962 			is_ok &= CheckIndexed(GL_VERTEX_ATTRIB_RELATIVE_OFFSET, i, 0);
3963 			is_ok &= CheckIndexed64(GL_VERTEX_BINDING_OFFSET, i, 0);
3964 		}
3965 
3966 		/* Check default values per vertex array object. */
3967 		is_ok &= Check(GL_ELEMENT_ARRAY_BUFFER_BINDING, 0);
3968 	}
3969 	catch (...)
3970 	{
3971 		is_ok	= false;
3972 		is_error = true;
3973 	}
3974 
3975 	/* Cleanup. */
3976 	if (m_vao)
3977 	{
3978 		gl.deleteVertexArrays(1, &m_vao);
3979 
3980 		m_vao = 0;
3981 	}
3982 
3983 	/* Errors clean up. */
3984 	while (gl.getError())
3985 		;
3986 
3987 	/* Result's setup. */
3988 	if (is_ok)
3989 	{
3990 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
3991 	}
3992 	else
3993 	{
3994 		if (is_error)
3995 		{
3996 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
3997 		}
3998 		else
3999 		{
4000 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
4001 		}
4002 	}
4003 
4004 	return STOP;
4005 }
4006 
4007 /** @brief Prepare vertex array object for the test.
4008  */
PrepareVAO()4009 void DefaultsTest::PrepareVAO()
4010 {
4011 	/* Shortcut for GL functionality. */
4012 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4013 
4014 	gl.createVertexArrays(1, &m_vao);
4015 	GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
4016 }
4017 
4018 /** @brief Compare value queried using GetVertexArrayiv with expected value and log.
4019  *
4020  *  @param [in] pname        Parameter to be queried.
4021  *  @param [in] expected     Expected error.
4022  *
4023  *  @return True if value is equal to expected, false otherwise.
4024  */
Check(const glw::GLenum pname,const glw::GLint expected)4025 bool DefaultsTest::Check(const glw::GLenum pname, const glw::GLint expected)
4026 {
4027 	/* Shortcut for GL functionality. */
4028 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4029 
4030 	glw::GLint result = 0;
4031 
4032 	gl.getVertexArrayiv(m_vao, pname, &result);
4033 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGetVertexArrayiv call failed.");
4034 
4035 	if (result != expected)
4036 	{
4037 		m_context.getTestContext().getLog() << tcu::TestLog::Message << "Default Vertex Array Object has parameter "
4038 											<< glu::getVertexAttribParameterNameStr(pname) << " equal to " << result
4039 											<< ", but " << expected << " was expected. Test fails."
4040 											<< tcu::TestLog::EndMessage;
4041 
4042 		return false;
4043 	}
4044 
4045 	return true;
4046 }
4047 
4048 /** @brief Compare value queried using GetVertexArrayIndexediv with expected value and log.
4049  *
4050  *  @param [in] pname        Parameter to be queried.
4051  *  @param [in] index        Index to be queried.
4052  *  @param [in] expected     Expected error.
4053  *
4054  *  @return True if value is equal to expected, false otherwise.
4055  */
CheckIndexed(const glw::GLenum pname,const glw::GLuint index,const glw::GLint expected)4056 bool DefaultsTest::CheckIndexed(const glw::GLenum pname, const glw::GLuint index, const glw::GLint expected)
4057 {
4058 	/* Shortcut for GL functionality. */
4059 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4060 
4061 	glw::GLint result = 0;
4062 
4063 	gl.getVertexArrayIndexediv(m_vao, index, pname, &result);
4064 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGetVertexArrayIndexediv call failed.");
4065 
4066 	if (result != expected)
4067 	{
4068 		m_context.getTestContext().getLog() << tcu::TestLog::Message << "Default Vertex Array Object at index " << index
4069 											<< " has parameter " << glu::getVertexAttribParameterNameStr(pname)
4070 											<< " equal to " << result << ", but " << expected
4071 											<< " was expected. Test fails." << tcu::TestLog::EndMessage;
4072 
4073 		return false;
4074 	}
4075 
4076 	return true;
4077 }
4078 
4079 /** @brief Compare value queried using GetVertexArrayIndexed64iv with expected value and log.
4080  *
4081  *  @param [in] pname        Parameter to be queried.
4082  *  @param [in] index        Index to be queried.
4083  *  @param [in] expected     Expected error.
4084  *
4085  *  @return True if value is equal to expected, false otherwise.
4086  */
CheckIndexed64(const glw::GLenum pname,const glw::GLuint index,const glw::GLint64 expected)4087 bool DefaultsTest::CheckIndexed64(const glw::GLenum pname, const glw::GLuint index, const glw::GLint64 expected)
4088 {
4089 	/* Shortcut for GL functionality. */
4090 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4091 
4092 	glw::GLint64 result = 0;
4093 
4094 	gl.getVertexArrayIndexed64iv(m_vao, index, pname, &result);
4095 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGetVertexArrayIndexed64iv call failed.");
4096 
4097 	if (result != expected)
4098 	{
4099 		m_context.getTestContext().getLog() << tcu::TestLog::Message << "Default Vertex Array Object at index " << index
4100 											<< " has parameter " << glu::getVertexAttribParameterNameStr(pname)
4101 											<< " equal to " << result << ", but " << expected
4102 											<< " was expected. Test fails." << tcu::TestLog::EndMessage;
4103 
4104 		return false;
4105 	}
4106 
4107 	return true;
4108 }
4109 
4110 /******************************** Creation Error Test Implementation   ********************************/
4111 
4112 /** @brief Creation Error Test constructor.
4113  *
4114  *  @param [in] context     OpenGL context.
4115  */
CreationErrorTest(deqp::Context & context)4116 CreationErrorTest::CreationErrorTest(deqp::Context& context)
4117 	: deqp::TestCase(context, "vertex_arrays_creation_error", "Creation Error Test")
4118 {
4119 }
4120 
4121 /** @brief Iterate Creation Error Test cases.
4122  *
4123  *  @return Iteration result.
4124  */
iterate()4125 tcu::TestNode::IterateResult CreationErrorTest::iterate()
4126 {
4127 	/* Shortcut for GL functionality. */
4128 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4129 
4130 	/* Get context setup. */
4131 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
4132 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
4133 
4134 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
4135 	{
4136 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
4137 
4138 		return STOP;
4139 	}
4140 
4141 	/* Running tests. */
4142 	bool is_ok	= true;
4143 	bool is_error = false;
4144 
4145 	try
4146 	{
4147 		glw::GLuint negative_vao = 0;
4148 
4149 		gl.createVertexArrays(-1, &negative_vao);
4150 
4151 		is_ok = CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated if n is negative.");
4152 	}
4153 	catch (...)
4154 	{
4155 		is_ok	= false;
4156 		is_error = true;
4157 	}
4158 
4159 	/* Errors clean up. */
4160 	while (gl.getError())
4161 		;
4162 
4163 	/* Result's setup. */
4164 	if (is_ok)
4165 	{
4166 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
4167 	}
4168 	else
4169 	{
4170 		if (is_error)
4171 		{
4172 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
4173 		}
4174 		else
4175 		{
4176 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
4177 		}
4178 	}
4179 
4180 	return STOP;
4181 }
4182 
4183 /** @brief Compare error returned by GL with expected value and log.
4184  *
4185  *  @param [in] expected        Expected error.
4186  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
4187  *
4188  *  @return True if GL error is equal to expected, false otherwise.
4189  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)4190 bool CreationErrorTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
4191 {
4192 	/* Shortcut for GL functionality. */
4193 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4194 
4195 	glw::GLenum error = 0;
4196 
4197 	if (expected != (error = gl.getError()))
4198 	{
4199 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
4200 											<< "was observed instead." << tcu::TestLog::EndMessage;
4201 
4202 		return false;
4203 	}
4204 
4205 	return true;
4206 }
4207 
4208 /******************************** Enable Disable Attribute Errors Test Implementation   ********************************/
4209 
4210 /** @brief Enable Disable Attribute Errors Test constructor.
4211  *
4212  *  @param [in] context     OpenGL context.
4213  */
EnableDisableAttributeErrorsTest(deqp::Context & context)4214 EnableDisableAttributeErrorsTest::EnableDisableAttributeErrorsTest(deqp::Context& context)
4215 	: deqp::TestCase(context, "vertex_arrays_enable_disable_attribute_errors", "Enable Disable Attribute Errors Test")
4216 {
4217 }
4218 
4219 /** @brief Enable Disable Attribute Errors Test cases.
4220  *
4221  *  @return Iteration result.
4222  */
iterate()4223 tcu::TestNode::IterateResult EnableDisableAttributeErrorsTest::iterate()
4224 {
4225 	/* Shortcut for GL functionality. */
4226 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4227 
4228 	/* Get context setup. */
4229 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
4230 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
4231 
4232 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
4233 	{
4234 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
4235 
4236 		return STOP;
4237 	}
4238 
4239 	/* Running tests. */
4240 	bool is_ok	= true;
4241 	bool is_error = false;
4242 
4243 	/* Test objects. */
4244 	glw::GLint max_attributes = 8;
4245 
4246 	/* Tested VAOs. */
4247 	glw::GLuint vao		  = 0;
4248 	glw::GLuint not_a_vao = 0;
4249 	try
4250 	{
4251 		/* Query limits. */
4252 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIBS, &max_attributes);
4253 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv call failed.");
4254 
4255 		/* Prepare valid VAO. */
4256 		gl.createVertexArrays(1, &vao);
4257 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
4258 
4259 		/* Prepare invalid VAO. */
4260 		while (gl.isVertexArray(++not_a_vao))
4261 			;
4262 
4263 		/* Test not a VAO. */
4264 		gl.enableVertexArrayAttrib(0, not_a_vao);
4265 
4266 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by EnableVertexArrayAttrib if "
4267 												  "vaobj is not the name of an existing vertex array object.");
4268 
4269 		gl.disableVertexArrayAttrib(0, not_a_vao);
4270 
4271 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by DisableVertexArrayAttrib if "
4272 												  "vaobj is not the name of an existing vertex array object.");
4273 
4274 		/* Test to big attribute index. */
4275 		gl.enableVertexArrayAttrib(max_attributes, vao);
4276 
4277 		is_ok &= CheckError(
4278 			GL_INVALID_OPERATION,
4279 			"INVALID_VALUE was not generated by EnableVertexArrayAttrib if index is equal to MAX_VERTEX_ATTRIBS.");
4280 
4281 		gl.disableVertexArrayAttrib(max_attributes, vao);
4282 
4283 		is_ok &= CheckError(
4284 			GL_INVALID_OPERATION,
4285 			"INVALID_VALUE was not generated by DisableVertexArrayAttrib if index is equal to MAX_VERTEX_ATTRIBS.");
4286 
4287 		gl.enableVertexArrayAttrib(max_attributes + 1, vao);
4288 
4289 		is_ok &= CheckError(
4290 			GL_INVALID_OPERATION,
4291 			"INVALID_VALUE was not generated by EnableVertexArrayAttrib if index is greater than MAX_VERTEX_ATTRIBS.");
4292 
4293 		gl.disableVertexArrayAttrib(max_attributes + 1, vao);
4294 
4295 		is_ok &= CheckError(
4296 			GL_INVALID_OPERATION,
4297 			"INVALID_VALUE was not generated by DisableVertexArrayAttrib if index is greater than MAX_VERTEX_ATTRIBS.");
4298 	}
4299 	catch (...)
4300 	{
4301 		is_ok	= false;
4302 		is_error = true;
4303 	}
4304 
4305 	/* Clean up. */
4306 	if (vao)
4307 	{
4308 		gl.deleteVertexArrays(1, &vao);
4309 	}
4310 
4311 	/* Errors clean up. */
4312 	while (gl.getError())
4313 		;
4314 
4315 	/* Result's setup. */
4316 	if (is_ok)
4317 	{
4318 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
4319 	}
4320 	else
4321 	{
4322 		if (is_error)
4323 		{
4324 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
4325 		}
4326 		else
4327 		{
4328 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
4329 		}
4330 	}
4331 
4332 	return STOP;
4333 }
4334 
4335 /** @brief Compare error returned by GL with expected value and log.
4336  *
4337  *  @param [in] expected        Expected error.
4338  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
4339  *
4340  *  @return True if GL error is equal to expected, false otherwise.
4341  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)4342 bool EnableDisableAttributeErrorsTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
4343 {
4344 	/* Shortcut for GL functionality. */
4345 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4346 
4347 	glw::GLenum error = 0;
4348 
4349 	if (expected != (error = gl.getError()))
4350 	{
4351 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
4352 											<< "was observed instead." << tcu::TestLog::EndMessage;
4353 
4354 		return false;
4355 	}
4356 
4357 	return true;
4358 }
4359 
4360 /******************************** Element Buffer Errors Test Implementation   ********************************/
4361 
4362 /** @brief Element Buffer Errors Test constructor.
4363  *
4364  *  @param [in] context     OpenGL context.
4365  */
ElementBufferErrorsTest(deqp::Context & context)4366 ElementBufferErrorsTest::ElementBufferErrorsTest(deqp::Context& context)
4367 	: deqp::TestCase(context, "vertex_arrays_element_buffer_errors", "Element Buffer Errors Test")
4368 {
4369 }
4370 
4371 /** @brief Element Buffer Errors Test cases.
4372  *
4373  *  @return Iteration result.
4374  */
iterate()4375 tcu::TestNode::IterateResult ElementBufferErrorsTest::iterate()
4376 {
4377 	/* Shortcut for GL functionality. */
4378 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4379 
4380 	/* Get context setup. */
4381 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
4382 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
4383 
4384 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
4385 	{
4386 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
4387 
4388 		return STOP;
4389 	}
4390 
4391 	/* Running tests. */
4392 	bool is_ok	= true;
4393 	bool is_error = false;
4394 
4395 	/* Tested Objects. */
4396 	glw::GLuint vao		  = 0;
4397 	glw::GLuint not_a_vao = 0;
4398 	glw::GLuint bo		  = 0;
4399 	glw::GLuint not_a_bo  = 0;
4400 
4401 	try
4402 	{
4403 		/* Prepare valid Objects. */
4404 		gl.createVertexArrays(1, &vao);
4405 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
4406 
4407 		gl.createBuffers(1, &bo);
4408 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateBuffers call failed.");
4409 
4410 		/* Prepare invalid VAO. */
4411 		while (gl.isVertexArray(++not_a_vao))
4412 			;
4413 		while (gl.isBuffer(++not_a_bo))
4414 			;
4415 
4416 		/* Test not a VAO. */
4417 		gl.vertexArrayElementBuffer(not_a_vao, bo);
4418 
4419 		is_ok &=
4420 			CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION error was not generated by VertexArrayElementBuffer if "
4421 											 "vaobj is not the name of an existing vertex array object.");
4422 
4423 		/* Test not a BO. */
4424 		gl.vertexArrayElementBuffer(vao, not_a_bo);
4425 
4426 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION error is generated by VertexArrayElementBuffer if "
4427 												  "buffer is not zero or the name of an existing buffer object.");
4428 	}
4429 	catch (...)
4430 	{
4431 		is_ok	= false;
4432 		is_error = true;
4433 	}
4434 
4435 	/* Clean up. */
4436 	if (vao)
4437 	{
4438 		gl.deleteVertexArrays(1, &vao);
4439 	}
4440 
4441 	if (bo)
4442 	{
4443 		gl.deleteBuffers(1, &bo);
4444 	}
4445 
4446 	/* Errors clean up. */
4447 	while (gl.getError())
4448 		;
4449 
4450 	/* Result's setup. */
4451 	if (is_ok)
4452 	{
4453 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
4454 	}
4455 	else
4456 	{
4457 		if (is_error)
4458 		{
4459 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
4460 		}
4461 		else
4462 		{
4463 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
4464 		}
4465 	}
4466 
4467 	return STOP;
4468 }
4469 
4470 /** @brief Compare error returned by GL with expected value and log.
4471  *
4472  *  @param [in] expected        Expected error.
4473  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
4474  *
4475  *  @return True if GL error is equal to expected, false otherwise.
4476  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)4477 bool ElementBufferErrorsTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
4478 {
4479 	/* Shortcut for GL functionality. */
4480 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4481 
4482 	glw::GLenum error = 0;
4483 
4484 	if (expected != (error = gl.getError()))
4485 	{
4486 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
4487 											<< " was observed instead." << tcu::TestLog::EndMessage;
4488 
4489 		return false;
4490 	}
4491 
4492 	return true;
4493 }
4494 
4495 /******************************** Vertex Buffers Errors Test Implementation   ********************************/
4496 
4497 /** @brief Vertex Buffers Errors Test constructor.
4498  *
4499  *  @param [in] context     OpenGL context.
4500  */
VertexBuffersErrorsTest(deqp::Context & context)4501 VertexBuffersErrorsTest::VertexBuffersErrorsTest(deqp::Context& context)
4502 	: deqp::TestCase(context, "vertex_arrays_vertex_buffers_errors", "Vertex Buffers Errors Test")
4503 {
4504 }
4505 
4506 /** @brief Vertex Buffers Errors Test cases.
4507  *
4508  *  @return Iteration result.
4509  */
iterate()4510 tcu::TestNode::IterateResult VertexBuffersErrorsTest::iterate()
4511 {
4512 	/* Shortcut for GL functionality. */
4513 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4514 
4515 	/* Get context setup. */
4516 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
4517 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
4518 
4519 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
4520 	{
4521 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
4522 
4523 		return STOP;
4524 	}
4525 
4526 	/* Running tests. */
4527 	bool is_ok	= true;
4528 	bool is_error = false;
4529 
4530 	/* Tested Objects. */
4531 	glw::GLuint vao		  = 0;
4532 	glw::GLuint not_a_vao = 0;
4533 	glw::GLuint bo		  = 0;
4534 	glw::GLuint not_a_bo  = 0;
4535 
4536 	/* Valid setup. */
4537 	glw::GLintptr valid_offset = 0;
4538 	glw::GLsizei  valid_stride = 1;
4539 
4540 	/* Limits. (Minimum values - OpenGL 4.5 Core Specification, Table 23.55) */
4541 	glw::GLint max_vertex_attrib_bindings = 16;
4542 	glw::GLint max_vertex_attrib_stride   = 2048;
4543 
4544 	try
4545 	{
4546 		/* Prepare valid Objects. */
4547 		gl.createVertexArrays(1, &vao);
4548 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
4549 
4550 		gl.createBuffers(1, &bo);
4551 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateBuffers call failed.");
4552 
4553 		/* Prepare invalid VAO. */
4554 		while (gl.isVertexArray(++not_a_vao))
4555 			;
4556 		while (gl.isBuffer(++not_a_bo))
4557 			;
4558 
4559 		/* Prepare limits. */
4560 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIB_BINDINGS, &max_vertex_attrib_bindings);
4561 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIB_STRIDE, &max_vertex_attrib_stride);
4562 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv call failed.");
4563 
4564 		/* Invalid setup. */
4565 		glw::GLintptr invalid_offset   = -1;
4566 		glw::GLsizei  invalid_stride_0 = -1;
4567 		glw::GLsizei  invalid_stride_1 = max_vertex_attrib_stride + 1;
4568 
4569 		/* Test not a VAO. */
4570 		gl.vertexArrayVertexBuffer(not_a_vao, 0, bo, valid_offset, valid_stride);
4571 
4572 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayVertexBuffer if "
4573 												  "vaobj is not the name of an existing vertex array object.");
4574 
4575 		gl.vertexArrayVertexBuffers(not_a_vao, 0, 1, &bo, &valid_offset, &valid_stride);
4576 
4577 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayVertexBuffers if "
4578 												  "vaobj is not the name of an existing vertex array object.");
4579 
4580 		/* Test not a BO. */
4581 		gl.vertexArrayVertexBuffer(vao, 0, not_a_bo, valid_offset, valid_stride);
4582 
4583 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayVertexBuffer if "
4584 												  "vaobj is not the name of an existing vertex array object.");
4585 
4586 		gl.vertexArrayVertexBuffers(vao, 0, 1, &not_a_bo, &valid_offset, &valid_stride);
4587 
4588 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayVertexBuffers if "
4589 												  "vaobj is not the name of an existing vertex array object.");
4590 
4591 		/* Test too big binding index. */
4592 		gl.vertexArrayVertexBuffer(vao, max_vertex_attrib_bindings, bo, valid_offset, valid_stride);
4593 
4594 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayVertexBuffer if "
4595 											  "bindingindex is equal to the value of MAX_VERTEX_ATTRIB_BINDINGS.");
4596 
4597 		gl.vertexArrayVertexBuffer(vao, max_vertex_attrib_bindings + 1, bo, valid_offset, valid_stride);
4598 
4599 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayVertexBuffer if "
4600 											  "bindingindex is greater than the value of MAX_VERTEX_ATTRIB_BINDINGS.");
4601 
4602 		gl.vertexArrayVertexBuffers(vao, max_vertex_attrib_bindings, 1, &bo, &valid_offset, &valid_stride);
4603 
4604 		is_ok &=
4605 			CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayVertexBuffers if "
4606 											 "first+count is greater than the value of MAX_VERTEX_ATTRIB_BINDINGS.");
4607 
4608 		/* Test too big stride. */
4609 		gl.vertexArrayVertexBuffer(vao, 0, bo, -1, valid_stride);
4610 
4611 		is_ok &= CheckError(GL_INVALID_VALUE,
4612 							"INVALID_VALUE is generated by VertexArrayVertexBuffer if offset less than zero.");
4613 
4614 		gl.vertexArrayVertexBuffer(vao, 0, bo, valid_offset, -1);
4615 
4616 		is_ok &= CheckError(GL_INVALID_VALUE,
4617 							"INVALID_VALUE is generated by VertexArrayVertexBuffer if stride is less than zero.");
4618 
4619 		gl.vertexArrayVertexBuffer(vao, 0, bo, valid_offset, max_vertex_attrib_stride + 1);
4620 
4621 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE is generated by VertexArrayVertexBuffer if stride is "
4622 											  "greater than the value of MAX_VERTEX_ATTRIB_STRIDE.");
4623 
4624 		gl.vertexArrayVertexBuffers(vao, 0, 1, &bo, &invalid_offset, &valid_stride);
4625 
4626 		is_ok &=
4627 			CheckError(GL_INVALID_VALUE,
4628 					   "INVALID_VALUE is generated by VertexArrayVertexBuffers if any value in offsets is negative.");
4629 
4630 		gl.vertexArrayVertexBuffers(vao, 0, 1, &bo, &valid_offset, &invalid_stride_0);
4631 
4632 		is_ok &=
4633 			CheckError(GL_INVALID_VALUE,
4634 					   "INVALID_VALUE is generated by VertexArrayVertexBuffers if any value in strides is negative.");
4635 
4636 		gl.vertexArrayVertexBuffers(vao, 0, 1, &bo, &valid_offset, &invalid_stride_1);
4637 
4638 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE is generated by VertexArrayVertexBuffers if a value in "
4639 											  "strides is greater than the value of MAX_VERTEX_ATTRIB_STRIDE.");
4640 	}
4641 	catch (...)
4642 	{
4643 		is_ok	= false;
4644 		is_error = true;
4645 	}
4646 
4647 	/* Clean up. */
4648 	if (vao)
4649 	{
4650 		gl.deleteVertexArrays(1, &vao);
4651 	}
4652 
4653 	if (bo)
4654 	{
4655 		gl.deleteBuffers(1, &bo);
4656 	}
4657 
4658 	/* Errors clean up. */
4659 	while (gl.getError())
4660 		;
4661 
4662 	/* Result's setup. */
4663 	if (is_ok)
4664 	{
4665 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
4666 	}
4667 	else
4668 	{
4669 		if (is_error)
4670 		{
4671 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
4672 		}
4673 		else
4674 		{
4675 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
4676 		}
4677 	}
4678 
4679 	return STOP;
4680 }
4681 
4682 /** @brief Compare error returned by GL with expected value and log.
4683  *
4684  *  @param [in] expected        Expected error.
4685  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
4686  *
4687  *  @return True if GL error is equal to expected, false otherwise.
4688  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)4689 bool VertexBuffersErrorsTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
4690 {
4691 	/* Shortcut for GL functionality. */
4692 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4693 
4694 	glw::GLenum error = 0;
4695 
4696 	if (expected != (error = gl.getError()))
4697 	{
4698 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
4699 											<< " was observed instead." << tcu::TestLog::EndMessage;
4700 
4701 		return false;
4702 	}
4703 
4704 	return true;
4705 }
4706 
4707 /******************************** Attribute Format Errors Test Implementation   ********************************/
4708 
4709 /** @brief Attribute Format Errors Test constructor.
4710  *
4711  *  @param [in] context     OpenGL context.
4712  */
AttributeFormatErrorsTest(deqp::Context & context)4713 AttributeFormatErrorsTest::AttributeFormatErrorsTest(deqp::Context& context)
4714 	: deqp::TestCase(context, "vertex_arrays_attribute_format_errors", "Attribute Format Errors Test")
4715 {
4716 }
4717 
4718 /** @brief Attribute Format Errors Test cases.
4719  *
4720  *  @return Iteration result.
4721  */
iterate()4722 tcu::TestNode::IterateResult AttributeFormatErrorsTest::iterate()
4723 {
4724 	/* Shortcut for GL functionality. */
4725 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
4726 
4727 	/* Get context setup. */
4728 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
4729 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
4730 
4731 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
4732 	{
4733 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
4734 
4735 		return STOP;
4736 	}
4737 
4738 	/* Running tests. */
4739 	bool is_ok	= true;
4740 	bool is_error = false;
4741 
4742 	/* Tested Objects. */
4743 	glw::GLuint vao		  = 0;
4744 	glw::GLuint not_a_vao = 0;
4745 
4746 	/* Limits. (Minimum values - OpenGL 4.5 Core Specification, Table 23.55) */
4747 	glw::GLint max_vertex_attribs				 = 16;
4748 	glw::GLint max_vertex_attrib_relative_offset = 2047;
4749 
4750 	/* Invalid values. */
4751 	glw::GLenum bad_type = 0;
4752 
4753 	static const glw::GLenum accepted_types[] = { GL_BYTE,
4754 												  GL_SHORT,
4755 												  GL_INT,
4756 												  GL_FIXED,
4757 												  GL_FLOAT,
4758 												  GL_HALF_FLOAT,
4759 												  GL_DOUBLE,
4760 												  GL_UNSIGNED_BYTE,
4761 												  GL_UNSIGNED_SHORT,
4762 												  GL_UNSIGNED_INT,
4763 												  GL_INT_2_10_10_10_REV,
4764 												  GL_UNSIGNED_INT_2_10_10_10_REV,
4765 												  GL_UNSIGNED_INT_10F_11F_11F_REV };
4766 
4767 	{
4768 		bool is_accepted_type = true;
4769 		while (is_accepted_type)
4770 		{
4771 			bad_type++;
4772 			is_accepted_type = false;
4773 			for (glw::GLuint i = 0; i < sizeof(accepted_types) / sizeof(accepted_types); ++i)
4774 			{
4775 				if (accepted_types[i] == bad_type)
4776 				{
4777 					is_accepted_type = true;
4778 					break;
4779 				}
4780 			}
4781 		}
4782 	}
4783 
4784 	try
4785 	{
4786 		/* Prepare valid Objects. */
4787 		gl.createVertexArrays(1, &vao);
4788 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
4789 
4790 		/* Prepare invalid VAO. */
4791 		while (gl.isVertexArray(++not_a_vao))
4792 			;
4793 
4794 		/* Prepare limits. */
4795 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIBS, &max_vertex_attribs);
4796 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIB_RELATIVE_OFFSET, &max_vertex_attrib_relative_offset);
4797 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv call failed.");
4798 
4799 		/* TESTS OF VERTEXARRAYATTRIBFORMAT */
4800 
4801 		/* MAX_VERTEX_ATTRIBS < */
4802 		gl.vertexArrayAttribFormat(vao, max_vertex_attribs, 1, GL_BYTE, GL_FALSE, 0);
4803 
4804 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribFormat if "
4805 											  "attribindex is equal to the value of MAX_VERTEX_ATTRIBS.");
4806 
4807 		gl.vertexArrayAttribFormat(vao, max_vertex_attribs + 1, 1, GL_BYTE, GL_FALSE, 0);
4808 
4809 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribFormat if "
4810 											  "attribindex is greater than the value of MAX_VERTEX_ATTRIBS.");
4811 
4812 		gl.vertexArrayAttribIFormat(vao, max_vertex_attribs, 1, GL_BYTE, 0);
4813 
4814 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribIFormat if "
4815 											  "attribindex is equal to the value of MAX_VERTEX_ATTRIBS.");
4816 
4817 		gl.vertexArrayAttribIFormat(vao, max_vertex_attribs + 1, 1, GL_BYTE, 0);
4818 
4819 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribIFormat if "
4820 											  "attribindex is greater than the value of MAX_VERTEX_ATTRIBS.");
4821 
4822 		gl.vertexArrayAttribLFormat(vao, max_vertex_attribs, 1, GL_DOUBLE, 0);
4823 
4824 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribLFormat if "
4825 											  "attribindex is equal to the value of MAX_VERTEX_ATTRIBS.");
4826 
4827 		gl.vertexArrayAttribLFormat(vao, max_vertex_attribs + 1, 1, GL_DOUBLE, 0);
4828 
4829 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribLFormat if "
4830 											  "attribindex is greater than the value of MAX_VERTEX_ATTRIBS.");
4831 
4832 		/* size */
4833 		gl.vertexArrayAttribFormat(vao, 0, 0, GL_BYTE, GL_FALSE, 0);
4834 
4835 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttrib*Format if size is "
4836 											  "not one of the accepted values (0).");
4837 
4838 		gl.vertexArrayAttribFormat(vao, 0, 5, GL_BYTE, GL_FALSE, 0);
4839 
4840 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttrib*Format if size is "
4841 											  "not one of the accepted values (5).");
4842 
4843 		gl.vertexArrayAttribIFormat(vao, 0, 0, GL_BYTE, 0);
4844 
4845 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribIFormat if size is "
4846 											  "not one of the accepted values (0).");
4847 
4848 		gl.vertexArrayAttribIFormat(vao, 0, 5, GL_BYTE, 0);
4849 
4850 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribIFormat if size is "
4851 											  "not one of the accepted values (5).");
4852 
4853 		gl.vertexArrayAttribLFormat(vao, 0, 0, GL_DOUBLE, 0);
4854 
4855 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribLFormat if size is "
4856 											  "not one of the accepted values (0).");
4857 
4858 		gl.vertexArrayAttribLFormat(vao, 0, 5, GL_DOUBLE, 0);
4859 
4860 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribLFormat if size is "
4861 											  "not one of the accepted values (5).");
4862 
4863 		/* relative offset */
4864 		gl.vertexArrayAttribFormat(vao, 0, 1, GL_BYTE, GL_FALSE, max_vertex_attrib_relative_offset + 1);
4865 
4866 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttrib*Format if "
4867 											  "relativeoffset is greater than the value of "
4868 											  "MAX_VERTEX_ATTRIB_RELATIVE_OFFSET.");
4869 
4870 		gl.vertexArrayAttribIFormat(vao, 0, 1, GL_BYTE, max_vertex_attrib_relative_offset + 1);
4871 
4872 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribIFormat if "
4873 											  "relativeoffset is greater than the value of "
4874 											  "MAX_VERTEX_ATTRIB_RELATIVE_OFFSET.");
4875 
4876 		gl.vertexArrayAttribLFormat(vao, 0, 1, GL_DOUBLE, max_vertex_attrib_relative_offset + 1);
4877 
4878 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribLFormat if "
4879 											  "relativeoffset is greater than the value of "
4880 											  "MAX_VERTEX_ATTRIB_RELATIVE_OFFSET.");
4881 
4882 		/* type */
4883 		gl.vertexArrayAttribFormat(vao, 0, 1, bad_type, GL_FALSE, 0);
4884 
4885 		is_ok &= CheckError(
4886 			GL_INVALID_ENUM,
4887 			"INVALID_ENUM was not generated by VertexArrayAttribFormat if type is not one of the accepted tokens.");
4888 
4889 		gl.vertexArrayAttribIFormat(vao, 0, 1, bad_type, 0);
4890 
4891 		is_ok &= CheckError(
4892 			GL_INVALID_ENUM,
4893 			"INVALID_ENUM was not generated by VertexArrayAttribIFormat if type is not one of the accepted tokens.");
4894 
4895 		gl.vertexArrayAttribLFormat(vao, 0, 1, bad_type, 0);
4896 
4897 		is_ok &= CheckError(
4898 			GL_INVALID_ENUM,
4899 			"INVALID_ENUM was not generated by VertexArrayAttribLFormat if type is not one of the accepted tokens.");
4900 
4901 		/* type UNSIGNED_INT_10F_11F_11F_REV case */
4902 		gl.vertexArrayAttribIFormat(vao, 0, 1, GL_UNSIGNED_INT_10F_11F_11F_REV, 0);
4903 
4904 		is_ok &= CheckError(
4905 			GL_INVALID_ENUM,
4906 			"INVALID_ENUM was not generated by VertexArrayAttribIFormat if type is UNSIGNED_INT_10F_11F_11F_REV.");
4907 
4908 		gl.vertexArrayAttribLFormat(vao, 0, 1, GL_UNSIGNED_INT_10F_11F_11F_REV, 0);
4909 
4910 		is_ok &= CheckError(
4911 			GL_INVALID_ENUM,
4912 			"INVALID_ENUM was not generated by VertexArrayAttribLFormat if type is UNSIGNED_INT_10F_11F_11F_REV.");
4913 
4914 		/* Test not a VAO. */
4915 		gl.vertexArrayAttribFormat(not_a_vao, 0, 1, GL_BYTE, GL_FALSE, 0);
4916 
4917 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayAttribFormat if "
4918 												  "vaobj is not the name of an existing vertex array object.");
4919 
4920 		gl.vertexArrayAttribIFormat(not_a_vao, 0, 1, GL_BYTE, 0);
4921 
4922 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayAttribIFormat if "
4923 												  "vaobj is not the name of an existing vertex array object.");
4924 
4925 		gl.vertexArrayAttribLFormat(not_a_vao, 0, 1, GL_DOUBLE, 0);
4926 
4927 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayAttribLFormat if "
4928 												  "vaobj is not the name of an existing vertex array object.");
4929 
4930 		/* BGRA */
4931 		gl.vertexArrayAttribFormat(vao, 0, GL_BGRA, GL_BYTE, GL_TRUE, 0);
4932 
4933 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayAttribFormat if "
4934 												  "size is BGRA and type is not UNSIGNED_BYTE, INT_2_10_10_10_REV or "
4935 												  "UNSIGNED_INT_2_10_10_10_REV.");
4936 
4937 		gl.vertexArrayAttribFormat(vao, 0, 1, GL_INT_2_10_10_10_REV, GL_TRUE, 0);
4938 
4939 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayAttribFormat if "
4940 												  "type is INT_2_10_10_10_REV and size is neither 4 nor BGRA.");
4941 
4942 		gl.vertexArrayAttribFormat(vao, 0, 1, GL_UNSIGNED_INT_2_10_10_10_REV, GL_TRUE, 0);
4943 
4944 		is_ok &=
4945 			CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayAttribFormat if type "
4946 											 "is UNSIGNED_INT_2_10_10_10_REV and size is neither 4 nor BGRA.");
4947 
4948 		gl.vertexArrayAttribFormat(vao, 0, 1, GL_UNSIGNED_INT_10F_11F_11F_REV, GL_TRUE, 0);
4949 
4950 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayAttribFormat if "
4951 												  "type is UNSIGNED_INT_10F_11F_11F_REV and size is not 3.");
4952 
4953 		gl.vertexArrayAttribFormat(vao, 0, GL_BGRA, GL_UNSIGNED_BYTE, GL_FALSE, 0);
4954 
4955 		is_ok &= CheckError(
4956 			GL_INVALID_OPERATION,
4957 			"INVALID_OPERATION was not generated by VertexArrayAttribFormat if size is BGRA and normalized is FALSE.");
4958 	}
4959 	catch (...)
4960 	{
4961 		is_ok	= false;
4962 		is_error = true;
4963 	}
4964 
4965 	/* Clean up. */
4966 	if (vao)
4967 	{
4968 		gl.deleteVertexArrays(1, &vao);
4969 	}
4970 
4971 	/* Errors clean up. */
4972 	while (gl.getError())
4973 		;
4974 
4975 	/* Result's setup. */
4976 	if (is_ok)
4977 	{
4978 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
4979 	}
4980 	else
4981 	{
4982 		if (is_error)
4983 		{
4984 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
4985 		}
4986 		else
4987 		{
4988 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
4989 		}
4990 	}
4991 
4992 	return STOP;
4993 }
4994 
4995 /** @brief Compare error returned by GL with expected value and log.
4996  *
4997  *  @param [in] expected        Expected error.
4998  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
4999  *
5000  *  @return True if GL error is equal to expected, false otherwise.
5001  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)5002 bool AttributeFormatErrorsTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
5003 {
5004 	/* Shortcut for GL functionality. */
5005 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5006 
5007 	glw::GLenum error = 0;
5008 
5009 	if (expected != (error = gl.getError()))
5010 	{
5011 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
5012 											<< " was observed instead." << tcu::TestLog::EndMessage;
5013 
5014 		return false;
5015 	}
5016 
5017 	return true;
5018 }
5019 
5020 /******************************** Attribute Binding Errors Test Implementation   ********************************/
5021 
5022 /** @brief Attribute Binding Errors Test constructor.
5023  *
5024  *  @param [in] context     OpenGL context.
5025  */
AttributeBindingErrorsTest(deqp::Context & context)5026 AttributeBindingErrorsTest::AttributeBindingErrorsTest(deqp::Context& context)
5027 	: deqp::TestCase(context, "vertex_arrays_attribute_binding_errors", "Attribute Binding Errors Test")
5028 {
5029 }
5030 
5031 /** @brief Attribute Binding Errors Test cases.
5032  *
5033  *  @return Iteration result.
5034  */
iterate()5035 tcu::TestNode::IterateResult AttributeBindingErrorsTest::iterate()
5036 {
5037 	/* Shortcut for GL functionality. */
5038 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5039 
5040 	/* Get context setup. */
5041 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
5042 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
5043 
5044 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
5045 	{
5046 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
5047 
5048 		return STOP;
5049 	}
5050 
5051 	/* Running tests. */
5052 	bool is_ok	= true;
5053 	bool is_error = false;
5054 
5055 	/* Tested Objects. */
5056 	glw::GLuint vao		  = 0;
5057 	glw::GLuint not_a_vao = 0;
5058 
5059 	/* Limits. (Minimum values - OpenGL 4.5 Core Specification, Table 23.55) */
5060 	glw::GLint max_vertex_attribs		  = 16;
5061 	glw::GLint max_vertex_attrib_bindings = 16;
5062 
5063 	try
5064 	{
5065 		/* Prepare valid Objects. */
5066 		gl.createVertexArrays(1, &vao);
5067 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
5068 
5069 		/* Prepare invalid VAO. */
5070 		while (gl.isVertexArray(++not_a_vao))
5071 			;
5072 
5073 		/* Prepare limits. */
5074 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIBS, &max_vertex_attribs);
5075 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIB_BINDINGS, &max_vertex_attrib_bindings);
5076 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv call failed.");
5077 
5078 		/* Not a VAO. */
5079 		gl.vertexArrayAttribBinding(not_a_vao, 0, 0);
5080 
5081 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayAttribBinding if "
5082 												  "vaobj is not the name of an existing vertex array object.");
5083 
5084 		/* Too big attribute index. */
5085 		gl.vertexArrayAttribBinding(vao, max_vertex_attribs, 0);
5086 
5087 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribBinding if "
5088 											  "attribindex is equal to the value of MAX_VERTEX_ATTRIBS.");
5089 
5090 		gl.vertexArrayAttribBinding(vao, max_vertex_attribs + 1, 0);
5091 
5092 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribBinding if "
5093 											  "attribindex is greater than the value of MAX_VERTEX_ATTRIBS.");
5094 
5095 		/* Too big binding index. */
5096 		gl.vertexArrayAttribBinding(vao, 0, max_vertex_attrib_bindings);
5097 
5098 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribBinding if "
5099 											  "bindingindex is equal to the value of MAX_VERTEX_ATTRIB_BINDINGS.");
5100 
5101 		gl.vertexArrayAttribBinding(vao, 0, max_vertex_attrib_bindings + 1);
5102 
5103 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayAttribBinding if "
5104 											  "bindingindex is greater than the value of MAX_VERTEX_ATTRIB_BINDINGS.");
5105 	}
5106 	catch (...)
5107 	{
5108 		is_ok	= false;
5109 		is_error = true;
5110 	}
5111 
5112 	/* Clean up. */
5113 	if (vao)
5114 	{
5115 		gl.deleteVertexArrays(1, &vao);
5116 	}
5117 
5118 	/* Errors clean up. */
5119 	while (gl.getError())
5120 		;
5121 
5122 	/* Result's setup. */
5123 	if (is_ok)
5124 	{
5125 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
5126 	}
5127 	else
5128 	{
5129 		if (is_error)
5130 		{
5131 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
5132 		}
5133 		else
5134 		{
5135 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
5136 		}
5137 	}
5138 
5139 	return STOP;
5140 }
5141 
5142 /** @brief Compare error returned by GL with expected value and log.
5143  *
5144  *  @param [in] expected        Expected error.
5145  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
5146  *
5147  *  @return True if GL error is equal to expected, false otherwise.
5148  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)5149 bool AttributeBindingErrorsTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
5150 {
5151 	/* Shortcut for GL functionality. */
5152 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5153 
5154 	glw::GLenum error = 0;
5155 
5156 	if (expected != (error = gl.getError()))
5157 	{
5158 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
5159 											<< " was observed instead." << tcu::TestLog::EndMessage;
5160 
5161 		return false;
5162 	}
5163 
5164 	return true;
5165 }
5166 
5167 /******************************** Attribute Binding Divisor Errors Test Implementation   ********************************/
5168 
5169 /** @brief Attribute Binding Divisor Errors Test constructor.
5170  *
5171  *  @param [in] context     OpenGL context.
5172  */
AttributeBindingDivisorErrorsTest(deqp::Context & context)5173 AttributeBindingDivisorErrorsTest::AttributeBindingDivisorErrorsTest(deqp::Context& context)
5174 	: deqp::TestCase(context, "vertex_arrays_attribute_binding_divisor_errors", "Attribute Binding Divisor Errors Test")
5175 {
5176 }
5177 
5178 /** @brief Attribute Binding Divisor Errors Test cases.
5179  *
5180  *  @return Iteration result.
5181  */
iterate()5182 tcu::TestNode::IterateResult AttributeBindingDivisorErrorsTest::iterate()
5183 {
5184 	/* Shortcut for GL functionality. */
5185 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5186 
5187 	/* Get context setup. */
5188 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
5189 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
5190 
5191 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
5192 	{
5193 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
5194 
5195 		return STOP;
5196 	}
5197 
5198 	/* Running tests. */
5199 	bool is_ok	= true;
5200 	bool is_error = false;
5201 
5202 	/* Tested Objects. */
5203 	glw::GLuint vao		  = 0;
5204 	glw::GLuint not_a_vao = 0;
5205 
5206 	/* Limits. (Minimum values - OpenGL 4.5 Core Specification, Table 23.55) */
5207 	glw::GLint max_vertex_attrib_bindings = 16;
5208 
5209 	try
5210 	{
5211 		/* Prepare valid Objects. */
5212 		gl.createVertexArrays(1, &vao);
5213 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
5214 
5215 		/* Prepare invalid VAO. */
5216 		while (gl.isVertexArray(++not_a_vao))
5217 			;
5218 
5219 		/* Prepare limits. */
5220 		gl.getIntegerv(GL_MAX_VERTEX_ATTRIB_BINDINGS, &max_vertex_attrib_bindings);
5221 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv call failed.");
5222 
5223 		/* Not a VAO. */
5224 		gl.vertexArrayBindingDivisor(not_a_vao, 0, 0);
5225 
5226 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION was not generated by VertexArrayBindingDivisor if "
5227 												  "vaobj is not the name of an existing vertex array object.");
5228 
5229 		/* Too big binding index. */
5230 		gl.vertexArrayBindingDivisor(vao, max_vertex_attrib_bindings, 0);
5231 
5232 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayBindingDivisor if "
5233 											  "bindingindex is equal to the value of MAX_VERTEX_ATTRIB_BINDINGS.");
5234 
5235 		gl.vertexArrayBindingDivisor(vao, max_vertex_attrib_bindings + 1, 0);
5236 
5237 		is_ok &= CheckError(GL_INVALID_VALUE, "INVALID_VALUE was not generated by VertexArrayBindingDivisor if "
5238 											  "bindingindex is greater than the value of MAX_VERTEX_ATTRIB_BINDINGS.");
5239 	}
5240 	catch (...)
5241 	{
5242 		is_ok	= false;
5243 		is_error = true;
5244 	}
5245 
5246 	/* Clean up. */
5247 	if (vao)
5248 	{
5249 		gl.deleteVertexArrays(1, &vao);
5250 	}
5251 
5252 	/* Errors clean up. */
5253 	while (gl.getError())
5254 		;
5255 
5256 	/* Result's setup. */
5257 	if (is_ok)
5258 	{
5259 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
5260 	}
5261 	else
5262 	{
5263 		if (is_error)
5264 		{
5265 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
5266 		}
5267 		else
5268 		{
5269 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
5270 		}
5271 	}
5272 
5273 	return STOP;
5274 }
5275 
5276 /** @brief Compare error returned by GL with expected value and log.
5277  *
5278  *  @param [in] expected        Expected error.
5279  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
5280  *
5281  *  @return True if GL error is equal to expected, false otherwise.
5282  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)5283 bool AttributeBindingDivisorErrorsTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
5284 {
5285 	/* Shortcut for GL functionality. */
5286 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5287 
5288 	glw::GLenum error = 0;
5289 
5290 	if (expected != (error = gl.getError()))
5291 	{
5292 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
5293 											<< " was observed instead." << tcu::TestLog::EndMessage;
5294 
5295 		return false;
5296 	}
5297 
5298 	return true;
5299 }
5300 
5301 /******************************** Get Vertex Array Errors Test Implementation   ********************************/
5302 
5303 /** @brief Get Vertex Array Errors Test constructor.
5304  *
5305  *  @param [in] context     OpenGL context.
5306  */
GetVertexArrayErrorsTest(deqp::Context & context)5307 GetVertexArrayErrorsTest::GetVertexArrayErrorsTest(deqp::Context& context)
5308 	: deqp::TestCase(context, "vertex_arrays_get_vertex_array_errors", "Get Vertex Array Errors Test")
5309 {
5310 }
5311 
5312 /** @brief Iterate over Get Vertex Array Errors Test cases.
5313  *
5314  *  @return Iteration result.
5315  */
iterate()5316 tcu::TestNode::IterateResult GetVertexArrayErrorsTest::iterate()
5317 {
5318 	/* Shortcut for GL functionality. */
5319 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5320 
5321 	/* Get context setup. */
5322 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
5323 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
5324 
5325 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
5326 	{
5327 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
5328 
5329 		return STOP;
5330 	}
5331 
5332 	/* Running tests. */
5333 	bool is_ok	= true;
5334 	bool is_error = false;
5335 
5336 	/* Tested Objects. */
5337 	glw::GLuint vao		  = 0;
5338 	glw::GLuint not_a_vao = 0;
5339 
5340 	glw::GLint storage = 0;
5341 
5342 	try
5343 	{
5344 		/* Prepare valid Objects. */
5345 		gl.createVertexArrays(1, &vao);
5346 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
5347 
5348 		/* Prepare invalid VAO. */
5349 		while (gl.isVertexArray(++not_a_vao))
5350 			;
5351 
5352 		/* Not a VAO. */
5353 		gl.getVertexArrayiv(not_a_vao, GL_ELEMENT_ARRAY_BUFFER_BINDING, &storage);
5354 
5355 		is_ok &= CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION error was not generated by GetVertexArrayiv if "
5356 												  "vaobj is not the name of an existing vertex array object.");
5357 
5358 		/* Bad parameter. */
5359 		gl.getVertexArrayiv(vao, GL_ELEMENT_ARRAY_BUFFER_BINDING + 1, &storage);
5360 
5361 		is_ok &= CheckError(
5362 			GL_INVALID_ENUM,
5363 			"INVALID_ENUM error was not generated by GetVertexArrayiv if pname is not ELEMENT_ARRAY_BUFFER_BINDING.");
5364 	}
5365 	catch (...)
5366 	{
5367 		is_ok	= false;
5368 		is_error = true;
5369 	}
5370 
5371 	/* Clean up. */
5372 	if (vao)
5373 	{
5374 		gl.deleteVertexArrays(1, &vao);
5375 	}
5376 
5377 	/* Errors clean up. */
5378 	while (gl.getError())
5379 		;
5380 
5381 	/* Result's setup. */
5382 	if (is_ok)
5383 	{
5384 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
5385 	}
5386 	else
5387 	{
5388 		if (is_error)
5389 		{
5390 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
5391 		}
5392 		else
5393 		{
5394 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
5395 		}
5396 	}
5397 
5398 	return STOP;
5399 }
5400 
5401 /** @brief Compare error returned by GL with expected value and log.
5402  *
5403  *  @param [in] expected        Expected error.
5404  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
5405  *
5406  *  @return True if GL error is equal to expected, false otherwise.
5407  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)5408 bool GetVertexArrayErrorsTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
5409 {
5410 	/* Shortcut for GL functionality. */
5411 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5412 
5413 	glw::GLenum error = 0;
5414 
5415 	if (expected != (error = gl.getError()))
5416 	{
5417 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
5418 											<< " was observed instead." << tcu::TestLog::EndMessage;
5419 
5420 		return false;
5421 	}
5422 
5423 	return true;
5424 }
5425 
5426 /******************************** Get Vertex Array Indexed Errors Test Implementation   ********************************/
5427 
5428 /** @brief Get Vertex Array Indexed Errors Test constructor.
5429  *
5430  *  @param [in] context     OpenGL context.
5431  */
GetVertexArrayIndexedErrorsTest(deqp::Context & context)5432 GetVertexArrayIndexedErrorsTest::GetVertexArrayIndexedErrorsTest(deqp::Context& context)
5433 	: deqp::TestCase(context, "vertex_arrays_get_vertex_array_indexed_errors", "Get Vertex Array Indexed Errors Test")
5434 {
5435 }
5436 
5437 /** @brief Iterate over Get Vertex Array Indexed Errors Test cases.
5438  *
5439  *  @return Iteration result.
5440  */
iterate()5441 tcu::TestNode::IterateResult GetVertexArrayIndexedErrorsTest::iterate()
5442 {
5443 	/* Shortcut for GL functionality. */
5444 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5445 
5446 	/* Get context setup. */
5447 	bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
5448 	bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
5449 
5450 	if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
5451 	{
5452 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
5453 
5454 		return STOP;
5455 	}
5456 
5457 	/* Running tests. */
5458 	bool is_ok	= true;
5459 	bool is_error = false;
5460 
5461 	/* Tested Objects. */
5462 	glw::GLuint vao		  = 0;
5463 	glw::GLuint not_a_vao = 0;
5464 
5465 	/* Dummy storage. */
5466 	glw::GLint   storage   = 0;
5467 	glw::GLint64 storage64 = 0;
5468 
5469 	/* Bad parameter setup. */
5470 	glw::GLenum bad_pname = 0;
5471 
5472 	static const glw::GLenum accepted_pnames[] = { GL_VERTEX_ATTRIB_ARRAY_ENABLED,	GL_VERTEX_ATTRIB_ARRAY_SIZE,
5473 												   GL_VERTEX_ATTRIB_ARRAY_STRIDE,	 GL_VERTEX_ATTRIB_ARRAY_TYPE,
5474 												   GL_VERTEX_ATTRIB_ARRAY_NORMALIZED, GL_VERTEX_ATTRIB_ARRAY_INTEGER,
5475 												   GL_VERTEX_ATTRIB_ARRAY_LONG,		  GL_VERTEX_ATTRIB_ARRAY_DIVISOR,
5476 												   GL_VERTEX_ATTRIB_RELATIVE_OFFSET };
5477 
5478 	{
5479 		bool is_accepted_pname = true;
5480 		while (is_accepted_pname)
5481 		{
5482 			bad_pname++;
5483 			is_accepted_pname = false;
5484 			for (glw::GLuint i = 0; i < sizeof(accepted_pnames) / sizeof(accepted_pnames); ++i)
5485 			{
5486 				if (accepted_pnames[i] == bad_pname)
5487 				{
5488 					is_accepted_pname = true;
5489 					break;
5490 				}
5491 			}
5492 		}
5493 	}
5494 
5495 	try
5496 	{
5497 		/* Prepare valid Objects. */
5498 		gl.createVertexArrays(1, &vao);
5499 		GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateVertexArrays call failed.");
5500 
5501 		/* Prepare invalid VAO. */
5502 		while (gl.isVertexArray(++not_a_vao))
5503 			;
5504 
5505 		/* Not a VAO. */
5506 		gl.getVertexArrayIndexediv(not_a_vao, 0, GL_VERTEX_ATTRIB_ARRAY_ENABLED, &storage);
5507 
5508 		is_ok &=
5509 			CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION error was not generated by GetVertexArrayIndexediv if "
5510 											 "vaobj is not the name of an existing vertex array object.");
5511 
5512 		gl.getVertexArrayIndexed64iv(not_a_vao, 0, GL_VERTEX_BINDING_OFFSET, &storage64);
5513 
5514 		is_ok &=
5515 			CheckError(GL_INVALID_OPERATION, "INVALID_OPERATION error was not generated by GetVertexArrayIndexed64iv "
5516 											 "if vaobj is not the name of an existing vertex array object.");
5517 
5518 		/* Bad parameter. */
5519 		gl.getVertexArrayIndexediv(vao, 0, bad_pname, &storage);
5520 
5521 		is_ok &= CheckError(
5522 			GL_INVALID_ENUM,
5523 			"INVALID_ENUM error was not generated by GetVertexArrayIndexediv if pname is not one of the valid values.");
5524 
5525 		/* Bad parameter 64. */
5526 		gl.getVertexArrayIndexed64iv(vao, 0, GL_VERTEX_BINDING_OFFSET + 1, &storage64);
5527 
5528 		is_ok &= CheckError(
5529 			GL_INVALID_ENUM,
5530 			"INVALID_ENUM error was not generated by GetVertexArrayIndexed64iv if pname is not VERTEX_BINDING_OFFSET.");
5531 	}
5532 	catch (...)
5533 	{
5534 		is_ok	= false;
5535 		is_error = true;
5536 	}
5537 
5538 	/* Clean up. */
5539 	if (vao)
5540 	{
5541 		gl.deleteVertexArrays(1, &vao);
5542 	}
5543 
5544 	/* Errors clean up. */
5545 	while (gl.getError())
5546 		;
5547 
5548 	/* Result's setup. */
5549 	if (is_ok)
5550 	{
5551 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
5552 	}
5553 	else
5554 	{
5555 		if (is_error)
5556 		{
5557 			m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
5558 		}
5559 		else
5560 		{
5561 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
5562 		}
5563 	}
5564 
5565 	return STOP;
5566 }
5567 
5568 /** @brief Compare error returned by GL with expected value and log.
5569  *
5570  *  @param [in] expected        Expected error.
5571  *  @param [in] log_message   Message to be logged if expected error is not the equal to the reported one.
5572  *
5573  *  @return True if GL error is equal to expected, false otherwise.
5574  */
CheckError(const glw::GLenum expected,const glw::GLchar * log_message)5575 bool GetVertexArrayIndexedErrorsTest::CheckError(const glw::GLenum expected, const glw::GLchar* log_message)
5576 {
5577 	/* Shortcut for GL functionality. */
5578 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
5579 
5580 	glw::GLenum error = 0;
5581 
5582 	if (expected != (error = gl.getError()))
5583 	{
5584 		m_context.getTestContext().getLog() << tcu::TestLog::Message << log_message << " " << glu::getErrorStr(error)
5585 											<< " was observed instead." << tcu::TestLog::EndMessage;
5586 
5587 		return false;
5588 	}
5589 
5590 	return true;
5591 }
5592 } /* VertexArrays namespace. */
5593 } /* DirectStateAccess namespace. */
5594 } /* gl4cts namespace. */
5595