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 gl4cDirectStateAccessQueriesTests.cpp
27 * \brief Conformance tests for the Direct State Access feature functionality (Queries 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 /* Define OpenGL enumerations not available in framework. */
50 #ifndef GL_QUERY_TARGET
51 #define GL_QUERY_TARGET 0x82EA
52 #endif
53
54 namespace gl4cts
55 {
56 namespace DirectStateAccess
57 {
58 namespace Queries
59 {
60 /******************************** Creation Test Implementation ********************************/
61
62 /** @brief Creation Test constructor.
63 *
64 * @param [in] context OpenGL context.
65 */
CreationTest(deqp::Context & context)66 CreationTest::CreationTest(deqp::Context& context)
67 : deqp::TestCase(context, "queries_creation", "Query Objects Creation Test")
68 {
69 /* Intentionally left blank. */
70 }
71
72 /** @brief Iterate Creation Test cases.
73 *
74 * @return Iteration result.
75 */
iterate()76 tcu::TestNode::IterateResult CreationTest::iterate()
77 {
78 /* Shortcut for GL functionality. */
79 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
80
81 /* Get context setup. */
82 bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
83 bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
84
85 if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
86 {
87 m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
88
89 return STOP;
90 }
91
92 /* Running tests. */
93 bool is_ok = true;
94 bool is_error = false;
95
96 /* Query targets */
97 static const glw::GLenum targets[] = {
98 GL_SAMPLES_PASSED, GL_ANY_SAMPLES_PASSED, GL_ANY_SAMPLES_PASSED_CONSERVATIVE, GL_TIME_ELAPSED,
99 GL_TIMESTAMP, GL_PRIMITIVES_GENERATED, GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN
100 };
101 static const glw::GLuint targets_count = sizeof(targets) / sizeof(targets[0]);
102
103 /* Queries objects */
104 static const glw::GLuint queries_count = 2;
105
106 glw::GLuint queries_legacy[queries_count] = {};
107 glw::GLuint queries_dsa[targets_count][queries_count] = {};
108
109 try
110 {
111 /* Check legacy state creation. */
112 gl.genQueries(queries_count, queries_legacy);
113 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenQueries have failed");
114
115 for (glw::GLuint i = 0; i < queries_count; ++i)
116 {
117 if (gl.isQuery(queries_legacy[i]))
118 {
119 is_ok = false;
120
121 /* Log. */
122 m_context.getTestContext().getLog()
123 << tcu::TestLog::Message
124 << "GenQueries has created default objects, but it should create only a names."
125 << tcu::TestLog::EndMessage;
126 }
127 }
128
129 /* Check direct state creation. */
130 for (glw::GLuint i = 0; i < targets_count; ++i)
131 {
132 gl.createQueries(targets[i], queries_count, queries_dsa[i]);
133 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateQueries have failed");
134
135 for (glw::GLuint j = 0; j < queries_count; ++j)
136 {
137 if (!gl.isQuery(queries_dsa[i][j]))
138 {
139 is_ok = false;
140
141 /* Log. */
142 m_context.getTestContext().getLog() << tcu::TestLog::Message
143 << "CreateQueries has not created default objects."
144 << tcu::TestLog::EndMessage;
145 }
146 }
147 }
148 }
149 catch (...)
150 {
151 is_ok = false;
152 is_error = true;
153 }
154
155 /* Cleanup. */
156 for (glw::GLuint j = 0; j < queries_count; ++j)
157 {
158 if (queries_legacy[j])
159 {
160 gl.deleteQueries(1, &queries_legacy[j]);
161
162 queries_legacy[j] = 0;
163 }
164
165 for (glw::GLuint i = 0; i < targets_count; ++i)
166 {
167 if (queries_dsa[i][j])
168 {
169 gl.deleteQueries(1, &queries_dsa[i][j]);
170
171 queries_dsa[i][j] = 0;
172 }
173 }
174 }
175
176 /* Result's setup. */
177 if (is_ok)
178 {
179 m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
180 }
181 else
182 {
183 if (is_error)
184 {
185 m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
186 }
187 else
188 {
189 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
190 }
191 }
192
193 return STOP;
194 }
195
196 /******************************** Defaults Test Implementation ********************************/
197
198 /** @brief Defaults Test constructor.
199 *
200 * @param [in] context OpenGL context.
201 */
DefaultsTest(deqp::Context & context)202 DefaultsTest::DefaultsTest(deqp::Context& context)
203 : deqp::TestCase(context, "queries_defaults", "Queries Defaults Test"), m_query_dsa(0)
204 {
205 /* Intentionally left blank. */
206 }
207
208 /** @brief Iterate Defaults Test cases.
209 *
210 * @return Iteration result.
211 */
iterate()212 tcu::TestNode::IterateResult DefaultsTest::iterate()
213 {
214 /* Get context setup. */
215 bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
216 bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
217
218 if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
219 {
220 m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
221
222 return STOP;
223 }
224
225 /* Running tests. */
226 bool is_ok = true;
227 bool is_error = false;
228
229 /* Query targets. */
230 static const glw::GLenum targets[] = {
231 GL_SAMPLES_PASSED, GL_ANY_SAMPLES_PASSED, GL_ANY_SAMPLES_PASSED_CONSERVATIVE, GL_TIME_ELAPSED,
232 GL_TIMESTAMP, GL_PRIMITIVES_GENERATED, GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN
233 };
234
235 static const glw::GLchar* target_names[] = {
236 "GL_SAMPLES_PASSED", "GL_ANY_SAMPLES_PASSED", "GL_ANY_SAMPLES_PASSED_CONSERVATIVE", "GL_TIME_ELAPSED",
237 "GL_TIMESTAMP", "GL_PRIMITIVES_GENERATED", "GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN"
238 };
239
240 static const glw::GLuint targets_count = sizeof(targets) / sizeof(targets[0]);
241
242 try
243 {
244 /* Check direct state creation. */
245 for (glw::GLuint i = 0; i < targets_count; ++i)
246 {
247 prepare(targets[i]);
248
249 is_ok &= testQueryParameter(GL_QUERY_RESULT, GL_FALSE, target_names[i]);
250 is_ok &= testQueryParameter(GL_QUERY_RESULT_AVAILABLE, GL_TRUE, target_names[i]);
251
252 clean();
253 }
254 }
255 catch (...)
256 {
257 is_ok = false;
258 is_error = true;
259
260 clean();
261 }
262
263 /* Result's setup. */
264 if (is_ok)
265 {
266 m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
267 }
268 else
269 {
270 if (is_error)
271 {
272 m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
273 }
274 else
275 {
276 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
277 }
278 }
279
280 return STOP;
281 }
282
283 /** @brief Create Query Objects.
284 *
285 * @note The function may throw if unexpected error has occured.
286 *
287 * @return True if test succeeded, false otherwise.
288 */
prepare(const glw::GLenum target)289 void DefaultsTest::prepare(const glw::GLenum target)
290 {
291 /* Shortcut for GL functionality. */
292 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
293
294 /* Query object creation */
295 gl.createQueries(target, 1, &m_query_dsa);
296 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateTransformFeedbacks have failed");
297 }
298
299 /** @brief Test Query Integer Parameter.
300 *
301 * @note The function may throw if unexpected error has occured.
302 *
303 * @param [in] pname Parameter name to be tested.
304 * @param [in] expected_value Expected value for comparison.
305 * @param [in] target_name Target name of the tested query object - for logging purposes.
306 *
307 * @return True if test succeeded, false otherwise.
308 */
testQueryParameter(const glw::GLenum pname,const glw::GLuint expected_value,const glw::GLchar * target_name)309 bool DefaultsTest::testQueryParameter(const glw::GLenum pname, const glw::GLuint expected_value,
310 const glw::GLchar* target_name)
311 {
312 /* Shortcut for GL functionality. */
313 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
314
315 /* Get data. */
316 glw::GLuint value = 0;
317
318 gl.getQueryObjectuiv(m_query_dsa, pname, &value);
319 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetQueryObjectuiv have failed");
320
321 if (expected_value != value)
322 {
323 m_context.getTestContext().getLog()
324 << tcu::TestLog::Message << "glGetQueryObjectuiv of query object with target " << target_name
325 << " with parameter " << pname << " has returned " << value << ", however " << expected_value
326 << " was expected." << tcu::TestLog::EndMessage;
327
328 return false;
329 }
330
331 return true;
332 }
333
334 /** @brief Release GL objects.
335 */
clean()336 void DefaultsTest::clean()
337 {
338 /* Shortcut for GL functionality. */
339 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
340
341 if (m_query_dsa)
342 {
343 gl.deleteQueries(1, &m_query_dsa);
344
345 m_query_dsa = 0;
346 }
347 }
348
349 /******************************** Errors Test Implementation ********************************/
350
351 /** @brief Errors Test constructor.
352 *
353 * @param [in] context OpenGL context.
354 */
ErrorsTest(deqp::Context & context)355 ErrorsTest::ErrorsTest(deqp::Context& context)
356 : deqp::TestCase(context, "queries_errors", "Queries Errors Test")
357 , m_pGetQueryBufferObjectiv(DE_NULL)
358 , m_pGetQueryBufferObjectuiv(DE_NULL)
359 , m_pGetQueryBufferObjecti64v(DE_NULL)
360 , m_pGetQueryBufferObjectui64v(DE_NULL)
361 {
362 /* Intentionally left blank. */
363 }
364
365 /** @brief Iterate Errors Test cases.
366 *
367 * @return Iteration result.
368 */
iterate()369 tcu::TestNode::IterateResult ErrorsTest::iterate()
370 {
371 /* Shortcut for GL functionality. */
372 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
373
374 /* Get context setup. */
375 bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
376 bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
377
378 if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
379 {
380 m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
381
382 return STOP;
383 }
384
385 /* Getting function pointers. */
386 m_pGetQueryBufferObjectiv = (PFNGLGETQUERYBUFFEROBJECT)gl.getQueryBufferObjectiv;
387 m_pGetQueryBufferObjectuiv = (PFNGLGETQUERYBUFFEROBJECT)gl.getQueryBufferObjectuiv;
388 m_pGetQueryBufferObjecti64v = (PFNGLGETQUERYBUFFEROBJECT)gl.getQueryBufferObjecti64v;
389 m_pGetQueryBufferObjectui64v = (PFNGLGETQUERYBUFFEROBJECT)gl.getQueryBufferObjectui64v;
390
391 /* Running tests. */
392 bool is_ok = true;
393 bool is_error = false;
394
395 try
396 {
397 if ((DE_NULL == m_pGetQueryBufferObjectiv) || (DE_NULL == m_pGetQueryBufferObjectuiv) ||
398 (DE_NULL == m_pGetQueryBufferObjecti64v) || (DE_NULL == m_pGetQueryBufferObjectui64v))
399 {
400 m_context.getTestContext().getLog()
401 << tcu::TestLog::Message << "Test could not get the pointers for glGetQueryBufferObject* functions."
402 << tcu::TestLog::EndMessage;
403
404 throw 0;
405 }
406
407 is_ok &= testNegativeNumberOfObjects();
408 is_ok &= testInvalidTarget();
409 is_ok &= testInvalidQueryName();
410 is_ok &= testInvalidBufferName();
411 is_ok &= testInvalidParameterName();
412 is_ok &= testBufferOverflow();
413 is_ok &= testBufferNegativeOffset();
414 }
415 catch (...)
416 {
417 is_ok = false;
418 is_error = true;
419 }
420
421 /* Result's setup. */
422 if (is_ok)
423 {
424 m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
425 }
426 else
427 {
428 if (is_error)
429 {
430 m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
431 }
432 else
433 {
434 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
435 }
436 }
437
438 return STOP;
439 }
440
441 /** @brief Check that CreateQueries generates INVALID_VALUE
442 * error if number of query objects to create is
443 * negative.
444 *
445 * @return True if test succeded, false otherwise.
446 */
testNegativeNumberOfObjects()447 bool ErrorsTest::testNegativeNumberOfObjects()
448 {
449 /* Shortcut for GL functionality. */
450 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
451
452 /* Test for each target. */
453 for (glw::GLuint i = 0; i < s_targets_count; ++i)
454 {
455 glw::GLuint query = 0;
456
457 gl.createQueries(s_targets[i], -1, &query); /* Create negative number of queries. */
458
459 glw::GLenum error = gl.getError();
460
461 if (GL_INVALID_VALUE != error)
462 {
463 m_context.getTestContext().getLog()
464 << tcu::TestLog::Message << "glCreateQueries called with target " << s_target_names[i]
465 << " with negative number of objects to be created (-1) has generated error " << glu::getErrorStr(error)
466 << ", however GL_INVALID_VALUE was expected." << tcu::TestLog::EndMessage;
467
468 if (query)
469 {
470 gl.deleteQueries(1, &query);
471
472 while (error == gl.getError())
473 ;
474
475 m_context.getTestContext().getLog()
476 << tcu::TestLog::Message << "glCreateQueries called with target " << s_target_names[i]
477 << " with negative number of objects to be created (-1) has created at least one object."
478 << tcu::TestLog::EndMessage;
479 }
480
481 return false;
482 }
483 }
484
485 return true;
486 }
487
488 /** @brief Check that CreateQueries generates INVALID_ENUM error if target is not
489 * one of accepted values:
490 * - SAMPLES_PASSED,
491 * - ANY_SAMPLES_PASSED,
492 * - ANY_SAMPLES_PASSED_CONSERVATIVE,
493 * - TIME_ELAPSED,
494 * - TIMESTAMP,
495 * - PRIMITIVES_GENERATED or
496 * - TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN.
497 *
498 * @return True if test succeded, false otherwise.
499 */
testInvalidTarget()500 bool ErrorsTest::testInvalidTarget()
501 {
502 /* Shortcut for GL functionality. */
503 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
504
505 /* Creating invalid target. */
506 glw::GLenum invalid_target = 0;
507
508 while (isTarget(++invalid_target))
509 ;
510
511 /* Test. */
512 glw::GLuint query = 0;
513
514 gl.createQueries(invalid_target, 1, &query); /* Create negative number of queries. */
515
516 glw::GLenum error = gl.getError();
517
518 if (GL_INVALID_ENUM != error)
519 {
520 m_context.getTestContext().getLog() << tcu::TestLog::Message << "glCreateQueries called with invalid target ("
521 << invalid_target << ") has generated error " << glu::getErrorStr(error)
522 << ", however GL_INVALID_ENUM was expected." << tcu::TestLog::EndMessage;
523
524 if (query)
525 {
526 gl.deleteQueries(1, &query);
527
528 while (error == gl.getError())
529 ;
530
531 m_context.getTestContext().getLog() << tcu::TestLog::Message
532 << "glCreateQueries called with invalid target (" << invalid_target
533 << ") has created an object." << tcu::TestLog::EndMessage;
534 }
535
536 return false;
537 }
538
539 return true;
540 }
541
542 /** @brief Check that GetQueryBufferObjectiv, GetQueryBufferObjectuiv,
543 * GetQueryBufferObjecti64v and GetQueryBufferObjectui64v generate
544 * INVALID_OPERATION error if <id> is not the name of a query object, or
545 * if the query object named by <id> is currently active.
546 *
547 * @return True if test succeded, false otherwise.
548 */
testInvalidQueryName()549 bool ErrorsTest::testInvalidQueryName()
550 {
551 /* Shortcut for GL functionality. */
552 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
553
554 /* Creating invalid query name. */
555 glw::GLuint invalid_query = 0;
556
557 /* Default result. */
558 bool is_ok = true;
559 bool is_error = false;
560
561 while (gl.isQuery(++invalid_query))
562 ;
563
564 /* Test's objects. */
565 glw::GLuint buffer = 0;
566 glw::GLuint query = 0;
567
568 try
569 {
570 /* Creating buffer for the test. */
571 gl.genBuffers(1, &buffer);
572 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers have failed");
573
574 gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, buffer);
575 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffers have failed");
576
577 gl.bufferData(GL_TRANSFORM_FEEDBACK_BUFFER, sizeof(glw::GLint64), DE_NULL, GL_DYNAMIC_COPY);
578 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffers have failed");
579
580 /* Test invalid query object name (integer version). */
581 m_pGetQueryBufferObjectiv(invalid_query, buffer, GL_QUERY_RESULT, 0);
582
583 glw::GLenum error = gl.getError();
584
585 if (GL_INVALID_OPERATION != error)
586 {
587 m_context.getTestContext().getLog()
588 << tcu::TestLog::Message
589 << "glGetQueryBufferObjectiv called with invalid query name has generated error "
590 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
591 << tcu::TestLog::EndMessage;
592
593 is_ok = false;
594 }
595
596 /* Test invalid query object name (unsigned integer version). */
597 m_pGetQueryBufferObjectuiv(invalid_query, buffer, GL_QUERY_RESULT, 0);
598
599 error = gl.getError();
600
601 if (GL_INVALID_OPERATION != error)
602 {
603 m_context.getTestContext().getLog()
604 << tcu::TestLog::Message
605 << "glGetQueryBufferObjectuiv called with invalid query name has generated error "
606 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
607 << tcu::TestLog::EndMessage;
608
609 is_ok = false;
610 }
611
612 /* Test invalid query object name (64-bit integer version). */
613 m_pGetQueryBufferObjecti64v(invalid_query, buffer, GL_QUERY_RESULT, 0);
614
615 error = gl.getError();
616
617 if (GL_INVALID_OPERATION != error)
618 {
619 m_context.getTestContext().getLog()
620 << tcu::TestLog::Message
621 << "glGetQueryBufferObjecti64v called with invalid query name has generated error "
622 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
623 << tcu::TestLog::EndMessage;
624
625 is_ok = false;
626 }
627
628 /* Test invalid query object name (64-bit unsigned integer version). */
629 m_pGetQueryBufferObjectui64v(invalid_query, buffer, GL_QUERY_RESULT, 0);
630
631 error = gl.getError();
632
633 if (GL_INVALID_OPERATION != error)
634 {
635 m_context.getTestContext().getLog()
636 << tcu::TestLog::Message
637 << "glGetQueryBufferObjectui64v called with invalid query name has generated error "
638 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
639 << tcu::TestLog::EndMessage;
640
641 is_ok = false;
642 }
643
644 /* Create query object for the test. */
645 gl.createQueries(s_targets[0], 1, &query);
646 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateQueries have failed");
647
648 gl.beginQuery(s_targets[0], query);
649 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateQueries have failed");
650
651 /* Test query of active query object name (integer version). */
652 m_pGetQueryBufferObjectiv(query, buffer, GL_QUERY_RESULT, 0);
653
654 error = gl.getError();
655
656 if (GL_INVALID_OPERATION != error)
657 {
658 m_context.getTestContext().getLog()
659 << tcu::TestLog::Message
660 << "glGetQueryBufferObjectiv called with active query object has generated error "
661 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
662 << tcu::TestLog::EndMessage;
663
664 is_ok = false;
665 }
666
667 /* Test query of active query object name (unsigned integer version). */
668 m_pGetQueryBufferObjectuiv(query, buffer, GL_QUERY_RESULT, 0);
669
670 error = gl.getError();
671
672 if (GL_INVALID_OPERATION != error)
673 {
674 m_context.getTestContext().getLog()
675 << tcu::TestLog::Message
676 << "glGetQueryBufferObjectuiv called with active query object has generated error "
677 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
678 << tcu::TestLog::EndMessage;
679
680 is_ok = false;
681 }
682
683 /* Test query of active query object name (64-bit integer version). */
684 m_pGetQueryBufferObjecti64v(query, buffer, GL_QUERY_RESULT, 0);
685
686 error = gl.getError();
687
688 if (GL_INVALID_OPERATION != error)
689 {
690 m_context.getTestContext().getLog()
691 << tcu::TestLog::Message
692 << "glGetQueryBufferObjecti64v called with active query object has generated error "
693 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
694 << tcu::TestLog::EndMessage;
695
696 is_ok = false;
697 }
698
699 /* Test query of active query object name (64-bit unsigned integer version). */
700 m_pGetQueryBufferObjectui64v(query, buffer, GL_QUERY_RESULT, 0);
701
702 error = gl.getError();
703
704 if (GL_INVALID_OPERATION != error)
705 {
706 m_context.getTestContext().getLog()
707 << tcu::TestLog::Message
708 << "glGetQueryBufferObjectui64v called with active query object has generated error "
709 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
710 << tcu::TestLog::EndMessage;
711
712 is_ok = false;
713 }
714 }
715 catch (...)
716 {
717 is_error = true;
718 }
719
720 /* Releasing objects. */
721 if (query)
722 {
723 gl.endQuery(s_targets[0]);
724
725 gl.deleteQueries(1, &query);
726 }
727
728 if (buffer)
729 {
730 gl.deleteBuffers(1, &buffer);
731 }
732
733 /* Error cleanup. */
734 while (gl.getError())
735 ;
736
737 if (is_error)
738 {
739 throw 0;
740 }
741
742 return is_ok;
743 }
744
745 /** @brief Check that GetQueryBufferObjectiv, GetQueryBufferObjectuiv,
746 * GetQueryBufferObjecti64v and GetQueryBufferObjectui64v generate
747 * INVALID_OPERATION error if <buffer> is not the name of an existing
748 * buffer object.
749 *
750 * @return True if test succeded, false otherwise.
751 */
testInvalidBufferName()752 bool ErrorsTest::testInvalidBufferName()
753 {
754 /* Shortcut for GL functionality. */
755 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
756
757 /* Default result. */
758 bool is_ok = true;
759 bool is_error = false;
760
761 /* Creating invalid buffer name. */
762 glw::GLuint invalid_buffer = 0;
763
764 while (gl.isBuffer(++invalid_buffer))
765 ;
766
767 /* Test's objects. */
768 glw::GLuint query = 0;
769
770 try
771 {
772 /* Create query object for the test. */
773 gl.createQueries(s_targets[0], 1, &query);
774 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateQueries have failed");
775
776 /* Test query of invalid buffer name (integer version). */
777 m_pGetQueryBufferObjectiv(query, invalid_buffer, GL_QUERY_RESULT_AVAILABLE, 0);
778
779 glw::GLenum error = gl.getError();
780
781 if (GL_INVALID_OPERATION != error)
782 {
783 m_context.getTestContext().getLog()
784 << tcu::TestLog::Message
785 << "glGetQueryBufferObjectiv which could generate buffers overflow generated error "
786 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
787 << tcu::TestLog::EndMessage;
788
789 is_ok = false;
790 }
791
792 /* Test query of invalid buffer name (unsigned integer version). */
793 m_pGetQueryBufferObjectuiv(query, invalid_buffer, GL_QUERY_RESULT_AVAILABLE, 0);
794
795 error = gl.getError();
796
797 if (GL_INVALID_OPERATION != error)
798 {
799 m_context.getTestContext().getLog()
800 << tcu::TestLog::Message
801 << "glGetQueryBufferObjectuiv which could generate buffers overflow generated error "
802 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
803 << tcu::TestLog::EndMessage;
804
805 is_ok = false;
806 }
807
808 /* Test query of invalid buffer name (64-bit integer version). */
809 m_pGetQueryBufferObjecti64v(query, invalid_buffer, GL_QUERY_RESULT_AVAILABLE, 0);
810
811 error = gl.getError();
812
813 if (GL_INVALID_OPERATION != error)
814 {
815 m_context.getTestContext().getLog()
816 << tcu::TestLog::Message
817 << "glGetQueryBufferObjecti64v which could generate buffers overflow generated error "
818 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
819 << tcu::TestLog::EndMessage;
820
821 is_ok = false;
822 }
823
824 /* Test query of invalid buffer name (64-bit unsigned integer version). */
825 m_pGetQueryBufferObjectui64v(query, invalid_buffer, GL_QUERY_RESULT_AVAILABLE, 0);
826
827 error = gl.getError();
828
829 if (GL_INVALID_OPERATION != error)
830 {
831 m_context.getTestContext().getLog()
832 << tcu::TestLog::Message
833 << "glGetQueryBufferObjectui64v which could generate buffers overflow generated error "
834 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
835 << tcu::TestLog::EndMessage;
836
837 is_ok = false;
838 }
839 }
840 catch (...)
841 {
842 is_error = true;
843 }
844
845 /* Releasing objects. */
846 if (query)
847 {
848 gl.deleteQueries(1, &query);
849 }
850
851 /* Error cleanup. */
852 while (gl.getError())
853 ;
854
855 if (is_error)
856 {
857 throw 0;
858 }
859
860 return is_ok;
861 }
862
863 /** @brief Check that GetQueryBufferObjectiv, GetQueryBufferObjectuiv,
864 * GetQueryBufferObjecti64v and GetQueryBufferObjectui64v generate
865 * INVALID_ENUM error if <pname> is not QUERY_RESULT,
866 * QUERY_RESULT_AVAILABLE, QUERY_RESULT_NO_WAIT or QUERY_TARGET.
867 *
868 * @return True if test succeded, false otherwise.
869 */
testInvalidParameterName()870 bool ErrorsTest::testInvalidParameterName()
871 {
872 /* Shortcut for GL functionality. */
873 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
874
875 /* Creating invalid parameter name. */
876 glw::GLuint invalid_pname = 0;
877
878 while (isParameterName(++invalid_pname))
879 ;
880
881 /* Default result. */
882 bool is_ok = true;
883 bool is_error = false;
884
885 /* Test's objects. */
886 glw::GLuint buffer = 0;
887 glw::GLuint query = 0;
888
889 try
890 {
891 /* Creating buffer for the test. */
892 gl.genBuffers(1, &buffer);
893 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers have failed");
894
895 gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, buffer);
896 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffers have failed");
897
898 gl.bufferData(GL_TRANSFORM_FEEDBACK_BUFFER, sizeof(glw::GLint64), DE_NULL, GL_DYNAMIC_COPY);
899 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffers have failed");
900
901 /* Create query object for the test. */
902 gl.createQueries(s_targets[0], 1, &query);
903 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateQueries have failed");
904
905 /* Test query of invalid parameter name (integer version). */
906 m_pGetQueryBufferObjectiv(query, buffer, invalid_pname, 0);
907
908 glw::GLenum error = gl.getError();
909
910 if (GL_INVALID_ENUM != error)
911 {
912 m_context.getTestContext().getLog()
913 << tcu::TestLog::Message
914 << "glGetQueryBufferObjectiv called with invalid parameter name has generated error "
915 << glu::getErrorStr(error) << ", however GL_INVALID_ENUM was expected." << tcu::TestLog::EndMessage;
916
917 is_ok = false;
918 }
919
920 /* Test query of invalid parameter name (unsigned integer version). */
921 m_pGetQueryBufferObjectuiv(query, buffer, invalid_pname, 0);
922
923 error = gl.getError();
924
925 if (GL_INVALID_ENUM != error)
926 {
927 m_context.getTestContext().getLog()
928 << tcu::TestLog::Message
929 << "glGetQueryBufferObjectuiv called with invalid parameter name has generated error "
930 << glu::getErrorStr(error) << ", however GL_INVALID_ENUM was expected." << tcu::TestLog::EndMessage;
931
932 is_ok = false;
933 }
934
935 /* Test query of invalid parameter name (64-bit integer version). */
936 m_pGetQueryBufferObjecti64v(query, buffer, invalid_pname, 0);
937
938 error = gl.getError();
939
940 if (GL_INVALID_ENUM != error)
941 {
942 m_context.getTestContext().getLog()
943 << tcu::TestLog::Message
944 << "glGetQueryBufferObjecti64v called with invalid parameter name has generated error "
945 << glu::getErrorStr(error) << ", however GL_INVALID_ENUM was expected." << tcu::TestLog::EndMessage;
946
947 is_ok = false;
948 }
949
950 /* Test query of invalid parameter name (64-bit unsigned integer version). */
951 m_pGetQueryBufferObjectui64v(query, buffer, invalid_pname, 0);
952
953 error = gl.getError();
954
955 if (GL_INVALID_ENUM != error)
956 {
957 m_context.getTestContext().getLog()
958 << tcu::TestLog::Message
959 << "glGetQueryBufferObjectui64v called with invalid parameter name has generated error "
960 << glu::getErrorStr(error) << ", however GL_INVALID_ENUM was expected." << tcu::TestLog::EndMessage;
961
962 is_ok = false;
963 }
964 }
965 catch (...)
966 {
967 is_error = true;
968 }
969
970 /* Releasing objects. */
971 if (query)
972 {
973 gl.deleteQueries(1, &query);
974 }
975
976 if (buffer)
977 {
978 gl.deleteBuffers(1, &buffer);
979 }
980
981 /* Error cleanup. */
982 while (gl.getError())
983 ;
984
985 if (is_error)
986 {
987 throw 0;
988 }
989
990 return is_ok;
991 }
992
993 /** @brief Check that GetQueryBufferObjectiv, GetQueryBufferObjectuiv,
994 * GetQueryBufferObjecti64v and GetQueryBufferObjectui64v generate
995 * INVALID_OPERATION error if the query writes to a buffer object, and the
996 * specified buffer offset would cause data to be written beyond the bounds
997 * of that buffer object.
998 *
999 * @return True if test succeded, false otherwise.
1000 */
testBufferOverflow()1001 bool ErrorsTest::testBufferOverflow()
1002 {
1003 /* Shortcut for GL functionality. */
1004 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1005
1006 /* Default result. */
1007 bool is_ok = true;
1008 bool is_error = false;
1009
1010 /* Test's objects. */
1011 glw::GLuint buffer = 0;
1012 glw::GLuint query = 0;
1013
1014 try
1015 {
1016 /* Creating buffer for the test. */
1017 gl.genBuffers(1, &buffer);
1018 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers have failed");
1019
1020 gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, buffer);
1021 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffers have failed");
1022
1023 gl.bufferData(GL_TRANSFORM_FEEDBACK_BUFFER, sizeof(glw::GLint64), DE_NULL, GL_DYNAMIC_COPY);
1024 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffers have failed");
1025
1026 /* Create query object for the test. */
1027 gl.createQueries(s_targets[0], 1, &query);
1028 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateQueries have failed");
1029
1030 /* Test query of buffer overflow (integer version). */
1031 m_pGetQueryBufferObjectiv(query, buffer, GL_QUERY_RESULT_AVAILABLE, sizeof(glw::GLint64));
1032
1033 glw::GLenum error = gl.getError();
1034
1035 if (GL_INVALID_OPERATION != error)
1036 {
1037 m_context.getTestContext().getLog()
1038 << tcu::TestLog::Message
1039 << "glGetQueryBufferObjectiv which could generate buffers overflow generated error "
1040 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
1041 << tcu::TestLog::EndMessage;
1042
1043 is_ok = false;
1044 }
1045
1046 /* Test query of buffer overflow (unsigned integer version). */
1047 m_pGetQueryBufferObjectuiv(query, buffer, GL_QUERY_RESULT_AVAILABLE, sizeof(glw::GLint64));
1048
1049 error = gl.getError();
1050
1051 if (GL_INVALID_OPERATION != error)
1052 {
1053 m_context.getTestContext().getLog()
1054 << tcu::TestLog::Message
1055 << "glGetQueryBufferObjectuiv which could generate buffers overflow generated error "
1056 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
1057 << tcu::TestLog::EndMessage;
1058
1059 is_ok = false;
1060 }
1061
1062 /* Test query of buffer overflow (64-bit integer version). */
1063 m_pGetQueryBufferObjecti64v(query, buffer, GL_QUERY_RESULT_AVAILABLE, sizeof(glw::GLint64));
1064
1065 error = gl.getError();
1066
1067 if (GL_INVALID_OPERATION != error)
1068 {
1069 m_context.getTestContext().getLog()
1070 << tcu::TestLog::Message
1071 << "glGetQueryBufferObjecti64v which could generate buffers overflow generated error "
1072 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
1073 << tcu::TestLog::EndMessage;
1074
1075 is_ok = false;
1076 }
1077
1078 /* Test query of buffer overflow (64-bit unsigned integer version). */
1079 m_pGetQueryBufferObjectui64v(query, buffer, GL_QUERY_RESULT_AVAILABLE, sizeof(glw::GLint64));
1080
1081 error = gl.getError();
1082
1083 if (GL_INVALID_OPERATION != error)
1084 {
1085 m_context.getTestContext().getLog()
1086 << tcu::TestLog::Message
1087 << "glGetQueryBufferObjectui64v which could generate buffers overflow generated error "
1088 << glu::getErrorStr(error) << ", however GL_INVALID_OPERATION was expected."
1089 << tcu::TestLog::EndMessage;
1090
1091 is_ok = false;
1092 }
1093 }
1094 catch (...)
1095 {
1096 is_error = true;
1097 }
1098
1099 /* Releasing objects. */
1100 if (query)
1101 {
1102 gl.deleteQueries(1, &query);
1103 }
1104
1105 if (buffer)
1106 {
1107 gl.deleteBuffers(1, &buffer);
1108 }
1109
1110 /* Error cleanup. */
1111 while (gl.getError())
1112 ;
1113
1114 if (is_error)
1115 {
1116 throw 0;
1117 }
1118
1119 return is_ok;
1120 }
1121
1122 /** @brief Check that GetQueryBufferObjectiv, GetQueryBufferObjectuiv,
1123 * GetQueryBufferObjecti64v and GetQueryBufferObjectui64v generate
1124 * INVALID_VALUE error if <offset> is negative.
1125 *
1126 * @return True if test succeded, false otherwise.
1127 */
testBufferNegativeOffset()1128 bool ErrorsTest::testBufferNegativeOffset()
1129 {
1130 /* Shortcut for GL functionality. */
1131 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1132
1133 /* Default result. */
1134 bool is_ok = true;
1135 bool is_error = false;
1136
1137 /* Test's objects. */
1138 glw::GLuint buffer = 0;
1139 glw::GLuint query = 0;
1140
1141 try
1142 {
1143 /* Creating buffer for the test. */
1144 gl.genBuffers(1, &buffer);
1145 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers have failed");
1146
1147 gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, buffer);
1148 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffers have failed");
1149
1150 gl.bufferData(GL_TRANSFORM_FEEDBACK_BUFFER, sizeof(glw::GLint64), DE_NULL, GL_DYNAMIC_COPY);
1151 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffers have failed");
1152
1153 /* Create query object for the test. */
1154 gl.createQueries(s_targets[0], 1, &query);
1155 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateQueries have failed");
1156
1157 /* Test query with negative offset (integer version). */
1158 m_pGetQueryBufferObjectiv(query, buffer, GL_QUERY_RESULT_AVAILABLE, -1);
1159
1160 glw::GLenum error = gl.getError();
1161
1162 if (GL_INVALID_VALUE != error)
1163 {
1164 m_context.getTestContext().getLog()
1165 << tcu::TestLog::Message << "glGetQueryBufferObjectiv called with negative offset generated error "
1166 << glu::getErrorStr(error) << ", however GL_INVALID_VALUE was expected." << tcu::TestLog::EndMessage;
1167
1168 is_ok = false;
1169 }
1170
1171 /* Test query with negative offset (unsigned integer version). */
1172 m_pGetQueryBufferObjectuiv(query, buffer, GL_QUERY_RESULT_AVAILABLE, -1);
1173
1174 error = gl.getError();
1175
1176 if (GL_INVALID_VALUE != error)
1177 {
1178 m_context.getTestContext().getLog()
1179 << tcu::TestLog::Message << "glGetQueryBufferObjectuiv called with negative offset generated error "
1180 << glu::getErrorStr(error) << ", however GL_INVALID_VALUE was expected." << tcu::TestLog::EndMessage;
1181
1182 is_ok = false;
1183 }
1184
1185 /* Test query with negative offset (64-bit integer version). */
1186 m_pGetQueryBufferObjecti64v(query, buffer, GL_QUERY_RESULT_AVAILABLE, -1);
1187
1188 error = gl.getError();
1189
1190 if (GL_INVALID_VALUE != error)
1191 {
1192 m_context.getTestContext().getLog()
1193 << tcu::TestLog::Message << "glGetQueryBufferObjecti64v called with negative offset generated error "
1194 << glu::getErrorStr(error) << ", however GL_INVALID_VALUE was expected." << tcu::TestLog::EndMessage;
1195
1196 is_ok = false;
1197 }
1198
1199 /* Test query with negative offset (64-bit unsigned integer version). */
1200 m_pGetQueryBufferObjectui64v(query, buffer, GL_QUERY_RESULT_AVAILABLE, -1);
1201
1202 error = gl.getError();
1203
1204 if (GL_INVALID_VALUE != error)
1205 {
1206 m_context.getTestContext().getLog()
1207 << tcu::TestLog::Message << "glGetQueryBufferObjectui64v called with negative offset generated error "
1208 << glu::getErrorStr(error) << ", however GL_INVALID_VALUE was expected." << tcu::TestLog::EndMessage;
1209
1210 is_ok = false;
1211 }
1212 }
1213 catch (...)
1214 {
1215 is_error = true;
1216 }
1217
1218 /* Releasing objects. */
1219 if (query)
1220 {
1221 gl.deleteQueries(1, &query);
1222 }
1223
1224 if (buffer)
1225 {
1226 gl.deleteBuffers(1, &buffer);
1227 }
1228
1229 /* Error cleanup. */
1230 while (gl.getError())
1231 ;
1232
1233 if (is_error)
1234 {
1235 throw 0;
1236 }
1237
1238 return is_ok;
1239 }
1240
1241 /** @brief Check if argument is one of the target names:
1242 * - SAMPLES_PASSED,
1243 * - TIME_ELAPSED,
1244 * - PRIMITIVES_GENERATED,
1245 * - TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN.
1246 *
1247 * @param [in] maybe_target Target to be checked.
1248 *
1249 * @return True if argument is one of the listed targets, false otherwise.
1250 */
isTarget(glw::GLenum maybe_target)1251 bool ErrorsTest::isTarget(glw::GLenum maybe_target)
1252 {
1253 for (glw::GLuint i = 0; i < s_targets_count; ++i)
1254 {
1255 if (maybe_target == s_targets[i])
1256 {
1257 return true;
1258 }
1259 }
1260
1261 return false;
1262 }
1263
1264 /** @brief Check if argument is one of the parameter names:
1265 * - QUERY_RESULT,
1266 * - QUERY_RESULT_AVAILABLE,
1267 * - QUERY_RESULT_NO_WAIT,
1268 * - QUERY_TARGET.
1269 *
1270 * @param [in] maybe_pname Parameter name to be checked.
1271 *
1272 * @return True if argument is one of the listed parameters, false otherwise.
1273 */
isParameterName(glw::GLenum maybe_pname)1274 bool ErrorsTest::isParameterName(glw::GLenum maybe_pname)
1275 {
1276 glw::GLenum pnames[] = { GL_QUERY_RESULT, GL_QUERY_RESULT_AVAILABLE, GL_QUERY_RESULT_NO_WAIT, GL_QUERY_TARGET };
1277 glw::GLuint pnames_count = sizeof(pnames) / sizeof(pnames[0]);
1278
1279 for (glw::GLuint i = 0; i < pnames_count; ++i)
1280 {
1281 if (maybe_pname == pnames[i])
1282 {
1283 return true;
1284 }
1285 }
1286
1287 return false;
1288 }
1289
1290 /** Targets to be tested. */
1291 const glw::GLenum ErrorsTest::s_targets[] = {
1292 GL_SAMPLES_PASSED, GL_ANY_SAMPLES_PASSED, GL_ANY_SAMPLES_PASSED_CONSERVATIVE, GL_TIME_ELAPSED,
1293 GL_TIMESTAMP, GL_PRIMITIVES_GENERATED, GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN
1294 };
1295
1296 /** Names of targets to be tested. */
1297 const glw::GLchar* ErrorsTest::s_target_names[] = {
1298 "GL_SAMPLES_PASSED", "GL_ANY_SAMPLES_PASSED", "GL_ANY_SAMPLES_PASSED_CONSERVATIVE", "GL_TIME_ELAPSED",
1299 "GL_TIMESTAMP", "GL_PRIMITIVES_GENERATED", "GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN"
1300 };
1301
1302 /** Number of targets. */
1303 const glw::GLuint ErrorsTest::s_targets_count = sizeof(s_targets) / sizeof(s_targets[0]);
1304
1305 /******************************** Functional Test Implementation ********************************/
1306
1307 /** @brief Functional Test constructor.
1308 *
1309 * @param [in] context OpenGL context.
1310 */
FunctionalTest(deqp::Context & context)1311 FunctionalTest::FunctionalTest(deqp::Context& context)
1312 : deqp::TestCase(context, "queries_functional", "Queries Functional Test")
1313 , m_pGetQueryBufferObjectiv(DE_NULL)
1314 , m_pGetQueryBufferObjectuiv(DE_NULL)
1315 , m_pGetQueryBufferObjecti64v(DE_NULL)
1316 , m_pGetQueryBufferObjectui64v(DE_NULL)
1317 , m_fbo(0)
1318 , m_rbo(0)
1319 , m_vao(0)
1320 , m_bo_query(0)
1321 , m_bo_xfb(0)
1322 , m_qo(DE_NULL)
1323 , m_po(0)
1324 {
1325 /* Intentionally left blank. */
1326 }
1327
1328 /** @brief Do comparison (a == b).
1329 *
1330 * @tparam Type of the values to be compared.
1331 *
1332 * @param [in] a First value to be compared.
1333 * @param [in] b Second value to be compared.
1334 *
1335 * @return Result of the comparison.
1336 */
1337 template <typename T>
equal(T a,T b)1338 bool FunctionalTest::equal(T a, T b)
1339 {
1340 return (a == b);
1341 }
1342
1343 /** @brief Do comparison (a < b).
1344 *
1345 * @tparam Type of the values to be compared.
1346 *
1347 * @param [in] a First value to be compared.
1348 * @param [in] b Second value to be compared.
1349 *
1350 * @return Result of the comparison.
1351 */
1352 template <typename T>
less(T a,T b)1353 bool FunctionalTest::less(T a, T b)
1354 {
1355 return (a < b);
1356 }
1357
1358 /** @brief Template specialization of glGetQueryBufferObject* function for GLint.
1359 * This is pass through function to glGetQueryBufferObjectiv
1360 *
1361 * @param [in] id Query object identifier.
1362 * @param [in] buffer Buffer object identifier.
1363 * @param [in] pname Parameter name to be queried.
1364 * @param [in] offset Offset of the buffer to be saved at.
1365 */
1366 template <>
GetQueryBufferObject(glw::GLuint id,glw::GLuint buffer,glw::GLenum pname,glw::GLintptr offset)1367 void FunctionalTest::GetQueryBufferObject<glw::GLint>(glw::GLuint id, glw::GLuint buffer, glw::GLenum pname,
1368 glw::GLintptr offset)
1369 {
1370 m_pGetQueryBufferObjectiv(id, buffer, pname, offset);
1371 }
1372
1373 /** @brief Template specialization of glGetQueryBufferObject* function for GLuint.
1374 * This is pass through function to glGetQueryBufferObjectuiv
1375 *
1376 * @param [in] id Query object identifier.
1377 * @param [in] buffer Buffer object identifier.
1378 * @param [in] pname Parameter name to be queried.
1379 * @param [in] offset Offset of the buffer to be saved at.
1380 */
1381 template <>
GetQueryBufferObject(glw::GLuint id,glw::GLuint buffer,glw::GLenum pname,glw::GLintptr offset)1382 void FunctionalTest::GetQueryBufferObject<glw::GLuint>(glw::GLuint id, glw::GLuint buffer, glw::GLenum pname,
1383 glw::GLintptr offset)
1384 {
1385 m_pGetQueryBufferObjectuiv(id, buffer, pname, offset);
1386 }
1387
1388 /** @brief Template specialization of glGetQueryBufferObject* function for GLint64.
1389 * This is pass through function to glGetQueryBufferObjecti64v
1390 *
1391 * @param [in] id Query object identifier.
1392 * @param [in] buffer Buffer object identifier.
1393 * @param [in] pname Parameter name to be queried.
1394 * @param [in] offset Offset of the buffer to be saved at.
1395 */
1396 template <>
GetQueryBufferObject(glw::GLuint id,glw::GLuint buffer,glw::GLenum pname,glw::GLintptr offset)1397 void FunctionalTest::GetQueryBufferObject<glw::GLint64>(glw::GLuint id, glw::GLuint buffer, glw::GLenum pname,
1398 glw::GLintptr offset)
1399 {
1400 m_pGetQueryBufferObjecti64v(id, buffer, pname, offset);
1401 }
1402
1403 /** @brief Template specialization of glGetQueryBufferObject* function for GLuint64.
1404 * This is pass through function to glGetQueryBufferObjectui64v
1405 *
1406 * @param [in] id Query object identifier.
1407 * @param [in] buffer Buffer object identifier.
1408 * @param [in] pname Parameter name to be queried.
1409 * @param [in] offset Offset of the buffer to be saved at.
1410 */
1411 template <>
GetQueryBufferObject(glw::GLuint id,glw::GLuint buffer,glw::GLenum pname,glw::GLintptr offset)1412 void FunctionalTest::GetQueryBufferObject<glw::GLuint64>(glw::GLuint id, glw::GLuint buffer, glw::GLenum pname,
1413 glw::GLintptr offset)
1414 {
1415 m_pGetQueryBufferObjectui64v(id, buffer, pname, offset);
1416 }
1417
1418 /** @brief Function template fetches query result to buffer object using
1419 * glGetQueryBufferObject* function (selected on the basis of template parameter).
1420 * Then buffer is mapped and the result is compared with expected_value using comparison function.
1421 *
1422 * @tparam Templated type of fetched data.
1423 * It shall be one of glw::GL[u]int[64].
1424 *
1425 * @param [in] query Query object to be queried.
1426 * @param [in] pname Parameter name to be queried.
1427 * @param [in] expected_value Reference value to be compared.
1428 * @param [in] comparison Comparison function pointer.
1429 * Comparsion function shall NOT throw.
1430 *
1431 * @note The function may throw if unexpected error has occured.
1432 *
1433 * @return True if comparison succeeded, false otherwise.
1434 */
1435 template <typename T>
checkQueryBufferObject(glw::GLuint query,glw::GLenum pname,T expected_value,bool (* comparison)(T,T))1436 bool FunctionalTest::checkQueryBufferObject(glw::GLuint query, glw::GLenum pname, T expected_value,
1437 bool (*comparison)(T, T))
1438 {
1439 /* Shortcut for GL functionality. */
1440 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1441
1442 /* Default result. */
1443 bool is_ok = true;
1444
1445 /* Saving results to buffer. */
1446 GetQueryBufferObject<T>(query, m_bo_query, pname, 0);
1447 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetQueryBufferObject* have failed");
1448
1449 /* Mapping buffer to user space. */
1450 T* value = (T*)gl.mapBuffer(GL_QUERY_BUFFER, GL_READ_ONLY);
1451 GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBuffer have failed");
1452
1453 /* Doing test. */
1454 if (!comparison(expected_value, *value))
1455 {
1456 is_ok = false;
1457 }
1458
1459 /* Cleanup. */
1460 gl.unmapBuffer(GL_QUERY_BUFFER);
1461 GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer have failed");
1462
1463 /* Return test result. */
1464 return is_ok;
1465 }
1466
1467 /** @brief Iterate Functional Test cases.
1468 *
1469 * @return Iteration result.
1470 */
iterate()1471 tcu::TestNode::IterateResult FunctionalTest::iterate()
1472 {
1473 /* Shortcut for GL functionality. */
1474 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1475
1476 /* Get context setup. */
1477 bool is_at_least_gl_45 = (glu::contextSupports(m_context.getRenderContext().getType(), glu::ApiType::core(4, 5)));
1478 bool is_arb_direct_state_access = m_context.getContextInfo().isExtensionSupported("GL_ARB_direct_state_access");
1479
1480 if ((!is_at_least_gl_45) && (!is_arb_direct_state_access))
1481 {
1482 m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not Supported");
1483
1484 return STOP;
1485 }
1486
1487 /* Fetching access point to GL functions. */
1488 m_pGetQueryBufferObjectiv = (PFNGLGETQUERYBUFFEROBJECT)gl.getQueryBufferObjectiv;
1489 m_pGetQueryBufferObjectuiv = (PFNGLGETQUERYBUFFEROBJECT)gl.getQueryBufferObjectuiv;
1490 m_pGetQueryBufferObjecti64v = (PFNGLGETQUERYBUFFEROBJECT)gl.getQueryBufferObjecti64v;
1491 m_pGetQueryBufferObjectui64v = (PFNGLGETQUERYBUFFEROBJECT)gl.getQueryBufferObjectui64v;
1492
1493 /* Running tests. */
1494 bool is_ok = true;
1495 bool is_error = false;
1496
1497 try
1498 {
1499 if ((DE_NULL == m_pGetQueryBufferObjectiv) || (DE_NULL == m_pGetQueryBufferObjectuiv) ||
1500 (DE_NULL == m_pGetQueryBufferObjecti64v) || (DE_NULL == m_pGetQueryBufferObjectui64v))
1501 {
1502 m_context.getTestContext().getLog()
1503 << tcu::TestLog::Message << "Test could not get the pointers for glGetQueryBufferObject* functions."
1504 << tcu::TestLog::EndMessage;
1505
1506 throw 0;
1507 }
1508
1509 prepareView();
1510 prepareVertexArray();
1511 prepareBuffers();
1512 prepareQueries();
1513 prepareProgram();
1514
1515 draw();
1516
1517 /* Make sure that framebuffer and transform feedback buffer are filled with expectd data. */
1518 is_ok &= checkView();
1519 is_ok &= checkXFB();
1520
1521 /* Make comparisons for each query object. */
1522 for (glw::GLuint i = 0; i < s_targets_count; ++i)
1523 {
1524 /* Checking targets. */
1525 is_ok &= checkQueryBufferObject<glw::GLint>(m_qo[i], GL_QUERY_TARGET, s_targets[i],
1526 &FunctionalTest::equal<glw::GLint>);
1527 is_ok &= checkQueryBufferObject<glw::GLuint>(m_qo[i], GL_QUERY_TARGET, s_targets[i],
1528 &FunctionalTest::equal<glw::GLuint>);
1529 is_ok &= checkQueryBufferObject<glw::GLint64>(m_qo[i], GL_QUERY_TARGET, s_targets[i],
1530 &FunctionalTest::equal<glw::GLint64>);
1531 is_ok &= checkQueryBufferObject<glw::GLuint64>(m_qo[i], GL_QUERY_TARGET, s_targets[i],
1532 &FunctionalTest::equal<glw::GLuint64>);
1533
1534 /* Checking result availability. */
1535 is_ok &= checkQueryBufferObject<glw::GLint>(m_qo[i], GL_QUERY_RESULT_AVAILABLE, GL_TRUE,
1536 &FunctionalTest::equal<glw::GLint>);
1537 is_ok &= checkQueryBufferObject<glw::GLuint>(m_qo[i], GL_QUERY_RESULT_AVAILABLE, GL_TRUE,
1538 &FunctionalTest::equal<glw::GLuint>);
1539 is_ok &= checkQueryBufferObject<glw::GLint64>(m_qo[i], GL_QUERY_RESULT_AVAILABLE, GL_TRUE,
1540 &FunctionalTest::equal<glw::GLint64>);
1541 is_ok &= checkQueryBufferObject<glw::GLuint64>(m_qo[i], GL_QUERY_RESULT_AVAILABLE, GL_TRUE,
1542 &FunctionalTest::equal<glw::GLuint64>);
1543
1544 if (GL_TIME_ELAPSED == s_targets[i])
1545 {
1546 /* Checking result. */
1547 is_ok &= checkQueryBufferObject<glw::GLint>(m_qo[i], GL_QUERY_RESULT, (glw::GLint)s_results[i],
1548 &FunctionalTest::less<glw::GLint>);
1549 is_ok &= checkQueryBufferObject<glw::GLuint>(m_qo[i], GL_QUERY_RESULT, (glw::GLuint)s_results[i],
1550 &FunctionalTest::less<glw::GLuint>);
1551 is_ok &= checkQueryBufferObject<glw::GLint64>(m_qo[i], GL_QUERY_RESULT, (glw::GLint64)s_results[i],
1552 &FunctionalTest::less<glw::GLint64>);
1553 is_ok &= checkQueryBufferObject<glw::GLuint64>(m_qo[i], GL_QUERY_RESULT, (glw::GLuint64)s_results[i],
1554 &FunctionalTest::less<glw::GLuint64>);
1555
1556 /* Checking result (no-wait). */
1557 is_ok &= checkQueryBufferObject<glw::GLint>(m_qo[i], GL_QUERY_RESULT_NO_WAIT, (glw::GLint)s_results[i],
1558 &FunctionalTest::less<glw::GLint>);
1559 is_ok &= checkQueryBufferObject<glw::GLuint>(
1560 m_qo[i], GL_QUERY_RESULT_NO_WAIT, (glw::GLuint)s_results[i], &FunctionalTest::less<glw::GLuint>);
1561 is_ok &= checkQueryBufferObject<glw::GLint64>(
1562 m_qo[i], GL_QUERY_RESULT_NO_WAIT, (glw::GLint64)s_results[i], &FunctionalTest::less<glw::GLint64>);
1563 is_ok &=
1564 checkQueryBufferObject<glw::GLuint64>(m_qo[i], GL_QUERY_RESULT_NO_WAIT, (glw::GLuint64)s_results[i],
1565 &FunctionalTest::less<glw::GLuint64>);
1566 }
1567 else
1568 {
1569 /* Checking result. */
1570 is_ok &= checkQueryBufferObject<glw::GLint>(m_qo[i], GL_QUERY_RESULT, (glw::GLint)s_results[i],
1571 &FunctionalTest::equal<glw::GLint>);
1572 is_ok &= checkQueryBufferObject<glw::GLuint>(m_qo[i], GL_QUERY_RESULT, (glw::GLuint)s_results[i],
1573 &FunctionalTest::equal<glw::GLuint>);
1574 is_ok &= checkQueryBufferObject<glw::GLint64>(m_qo[i], GL_QUERY_RESULT, (glw::GLint64)s_results[i],
1575 &FunctionalTest::equal<glw::GLint64>);
1576 is_ok &= checkQueryBufferObject<glw::GLuint64>(m_qo[i], GL_QUERY_RESULT, (glw::GLuint64)s_results[i],
1577 &FunctionalTest::equal<glw::GLuint64>);
1578
1579 /* Checking result (no-wait). */
1580 is_ok &= checkQueryBufferObject<glw::GLint>(m_qo[i], GL_QUERY_RESULT_NO_WAIT, (glw::GLint)s_results[i],
1581 &FunctionalTest::equal<glw::GLint>);
1582 is_ok &= checkQueryBufferObject<glw::GLuint>(
1583 m_qo[i], GL_QUERY_RESULT_NO_WAIT, (glw::GLuint)s_results[i], &FunctionalTest::equal<glw::GLuint>);
1584 is_ok &= checkQueryBufferObject<glw::GLint64>(
1585 m_qo[i], GL_QUERY_RESULT_NO_WAIT, (glw::GLint64)s_results[i], &FunctionalTest::equal<glw::GLint64>);
1586 is_ok &=
1587 checkQueryBufferObject<glw::GLuint64>(m_qo[i], GL_QUERY_RESULT_NO_WAIT, (glw::GLuint64)s_results[i],
1588 &FunctionalTest::equal<glw::GLuint64>);
1589 }
1590 }
1591 }
1592 catch (...)
1593 {
1594 is_ok = false;
1595 is_error = true;
1596 }
1597
1598 /* Clean up. */
1599 clean();
1600
1601 /* Result's setup. */
1602 if (is_ok)
1603 {
1604 m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
1605 }
1606 else
1607 {
1608 if (is_error)
1609 {
1610 m_testCtx.setTestResult(QP_TEST_RESULT_INTERNAL_ERROR, "Error");
1611 }
1612 else
1613 {
1614 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
1615 }
1616 }
1617
1618 return STOP;
1619 }
1620
1621 /** @brief Function prepares framebuffer with RGBA8 color attachment.
1622 * Viewport is set up. Content of the framebuffer is cleared.
1623 *
1624 * @note The function may throw if unexpected error has occured.
1625 */
prepareView()1626 void FunctionalTest::prepareView()
1627 {
1628 /* Shortcut for GL functionality. */
1629 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1630
1631 /* Prepare framebuffer. */
1632 gl.genFramebuffers(1, &m_fbo);
1633 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenFramebuffers call failed.");
1634
1635 gl.genRenderbuffers(1, &m_rbo);
1636 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenRenderbuffers call failed.");
1637
1638 gl.bindFramebuffer(GL_FRAMEBUFFER, m_fbo);
1639 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindFramebuffer call failed.");
1640
1641 gl.bindRenderbuffer(GL_RENDERBUFFER, m_rbo);
1642 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindRenderbuffer call failed.");
1643
1644 gl.renderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 1 /* x size */, 1 /* y size */);
1645 GLU_EXPECT_NO_ERROR(gl.getError(), "glRenderbufferStorage call failed.");
1646
1647 gl.framebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, m_rbo);
1648 GLU_EXPECT_NO_ERROR(gl.getError(), "glFramebufferRenderbuffer call failed.");
1649
1650 if (gl.checkFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
1651 {
1652 throw 0;
1653 }
1654
1655 gl.viewport(0, 0, 1, 1);
1656 GLU_EXPECT_NO_ERROR(gl.getError(), "glViewport call failed.");
1657
1658 /* Clear framebuffer's content. */
1659 gl.clearColor(0.0f, 0.0f, 0.0f, 0.0f);
1660 GLU_EXPECT_NO_ERROR(gl.getError(), "glClearColor call failed.");
1661
1662 gl.clear(GL_COLOR_BUFFER_BIT);
1663 GLU_EXPECT_NO_ERROR(gl.getError(), "glClear call failed.");
1664 }
1665
1666 /** @brief Function creates and binds empty vertex array.
1667 *
1668 * @note The function may throw if unexpected error has occured.
1669 */
prepareVertexArray()1670 void FunctionalTest::prepareVertexArray()
1671 {
1672 /* Shortcut for GL functionality. */
1673 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1674
1675 /* Creating and binding VAO. */
1676 gl.genVertexArrays(1, &m_vao);
1677 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenVertexArrays have failed");
1678
1679 gl.bindVertexArray(m_vao);
1680 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindVertexArray have failed");
1681 }
1682
1683 /** @brief Function creates buffers for query and transform feedback data storage.
1684 * The storage is allocated and buffers are bound to QUERY_BUFFER and
1685 * TRANSFORM_FEEDBACK_BUFFER binding points respectively.
1686 *
1687 * @note The function may throw if unexpected error has occured.
1688 */
prepareBuffers()1689 void FunctionalTest::prepareBuffers()
1690 {
1691 /* Shortcut for GL functionality. */
1692 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1693
1694 /* Buffer for storing query's result. */
1695 gl.genBuffers(1, &m_bo_query);
1696 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers have failed");
1697
1698 gl.bindBuffer(GL_QUERY_BUFFER, m_bo_query);
1699 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer have failed");
1700
1701 gl.bufferData(GL_QUERY_BUFFER, sizeof(glw::GLint64), DE_NULL, GL_DYNAMIC_COPY);
1702 GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData have failed");
1703
1704 /* Buffer for storing transform feedback results. */
1705 gl.genBuffers(1, &m_bo_xfb);
1706 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers have failed");
1707
1708 gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, m_bo_xfb);
1709 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBuffer have failed");
1710
1711 gl.bufferData(GL_TRANSFORM_FEEDBACK_BUFFER,
1712 3 /* number of vertices per triangle */ * 2 /* number of triangles */ * sizeof(glw::GLint), DE_NULL,
1713 GL_DYNAMIC_COPY);
1714 GLU_EXPECT_NO_ERROR(gl.getError(), "glBufferData have failed");
1715
1716 gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, m_bo_xfb);
1717 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindBufferBase have failed");
1718 }
1719
1720 /** @brief Function creates array of query objects using DSA-style method.
1721 *
1722 * @note The function may throw if unexpected error has occured.
1723 */
prepareQueries()1724 void FunctionalTest::prepareQueries()
1725 {
1726 /* Shortcut for GL functionality. */
1727 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1728
1729 /* Allocating memory for queries array. */
1730 m_qo = new glw::GLuint[s_targets_count];
1731
1732 if (DE_NULL == m_qo)
1733 {
1734 throw 0;
1735 }
1736
1737 /* Creating query object for each target. */
1738 for (glw::GLuint i = 0; i < s_targets_count; ++i)
1739 {
1740 gl.createQueries(s_targets[i], 1, &m_qo[i]);
1741
1742 /* Error checking. */
1743 if (GL_NO_ERROR != gl.getError())
1744 {
1745 /* Remove previous. */
1746 if (i > 0)
1747 {
1748 gl.deleteQueries(i, m_qo);
1749 }
1750
1751 /* Deallocate storage. */
1752 delete[] m_qo;
1753
1754 m_qo = DE_NULL;
1755
1756 /* Signalise test failure. */
1757 throw 0;
1758 }
1759 }
1760 }
1761
1762 /** @brief Function builds test's GLSL program.
1763 * If succeded, the program will be set to be used.
1764 *
1765 * @note The function may throw if unexpected error has occured.
1766 */
prepareProgram()1767 void FunctionalTest::prepareProgram()
1768 {
1769 /* Shortcut for GL functionality */
1770 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1771
1772 struct Shader
1773 {
1774 glw::GLchar const* const source;
1775 glw::GLenum const type;
1776 glw::GLuint id;
1777 } shader[] = { { s_vertex_shader, GL_VERTEX_SHADER, 0 }, { s_fragment_shader, GL_FRAGMENT_SHADER, 0 } };
1778
1779 glw::GLuint const shader_count = sizeof(shader) / sizeof(shader[0]);
1780
1781 try
1782 {
1783 /* Create program. */
1784 m_po = gl.createProgram();
1785 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateProgram call failed.");
1786
1787 /* Shader compilation. */
1788
1789 for (glw::GLuint i = 0; i < shader_count; ++i)
1790 {
1791 if (DE_NULL != shader[i].source)
1792 {
1793 shader[i].id = gl.createShader(shader[i].type);
1794
1795 GLU_EXPECT_NO_ERROR(gl.getError(), "glCreateShader call failed.");
1796
1797 gl.attachShader(m_po, shader[i].id);
1798
1799 GLU_EXPECT_NO_ERROR(gl.getError(), "glAttachShader call failed.");
1800
1801 gl.shaderSource(shader[i].id, 1, &(shader[i].source), NULL);
1802
1803 GLU_EXPECT_NO_ERROR(gl.getError(), "glShaderSource call failed.");
1804
1805 gl.compileShader(shader[i].id);
1806
1807 GLU_EXPECT_NO_ERROR(gl.getError(), "glCompileShader call failed.");
1808
1809 glw::GLint status = GL_FALSE;
1810
1811 gl.getShaderiv(shader[i].id, GL_COMPILE_STATUS, &status);
1812 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
1813
1814 if (GL_FALSE == status)
1815 {
1816 glw::GLint log_size = 0;
1817 gl.getShaderiv(shader[i].id, GL_INFO_LOG_LENGTH, &log_size);
1818 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderiv call failed.");
1819
1820 glw::GLchar* log_text = new glw::GLchar[log_size];
1821
1822 gl.getShaderInfoLog(shader[i].id, log_size, NULL, &log_text[0]);
1823
1824 m_context.getTestContext().getLog() << tcu::TestLog::Message << "Shader compilation has failed.\n"
1825 << "Shader type: " << glu::getShaderTypeStr(shader[i].type)
1826 << "\n"
1827 << "Shader compilation error log:\n"
1828 << log_text << "\n"
1829 << "Shader source code:\n"
1830 << shader[i].source << "\n"
1831 << tcu::TestLog::EndMessage;
1832
1833 delete[] log_text;
1834
1835 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetShaderInfoLog call failed.");
1836
1837 throw 0;
1838 }
1839 }
1840 }
1841
1842 /* Transform Feedback setup. */
1843 gl.transformFeedbackVaryings(m_po, 1, &s_xfb_varying_name, GL_INTERLEAVED_ATTRIBS);
1844 GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
1845
1846 /* Link. */
1847 gl.linkProgram(m_po);
1848
1849 GLU_EXPECT_NO_ERROR(gl.getError(), "glTransformFeedbackVaryings call failed.");
1850
1851 glw::GLint status = GL_FALSE;
1852
1853 gl.getProgramiv(m_po, GL_LINK_STATUS, &status);
1854
1855 if (GL_TRUE == status)
1856 {
1857 for (glw::GLuint i = 0; i < shader_count; ++i)
1858 {
1859 if (shader[i].id)
1860 {
1861 gl.detachShader(m_po, shader[i].id);
1862
1863 GLU_EXPECT_NO_ERROR(gl.getError(), "glDetachShader call failed.");
1864 }
1865 }
1866 }
1867 else
1868 {
1869 glw::GLint log_size = 0;
1870
1871 gl.getProgramiv(m_po, GL_INFO_LOG_LENGTH, &log_size);
1872
1873 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramiv call failed.");
1874
1875 glw::GLchar* log_text = new glw::GLchar[log_size];
1876
1877 gl.getProgramInfoLog(m_po, log_size, NULL, &log_text[0]);
1878
1879 m_context.getTestContext().getLog() << tcu::TestLog::Message << "Program linkage has failed due to:\n"
1880 << log_text << "\n"
1881 << tcu::TestLog::EndMessage;
1882
1883 delete[] log_text;
1884
1885 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetProgramInfoLog call failed.");
1886
1887 throw 0;
1888 }
1889 }
1890 catch (...)
1891 {
1892 if (m_po)
1893 {
1894 gl.deleteProgram(m_po);
1895
1896 m_po = 0;
1897 }
1898 }
1899
1900 for (glw::GLuint i = 0; i < shader_count; ++i)
1901 {
1902 if (0 != shader[i].id)
1903 {
1904 gl.deleteShader(shader[i].id);
1905
1906 shader[i].id = 0;
1907 }
1908 }
1909
1910 if (m_po)
1911 {
1912 gl.useProgram(m_po);
1913 GLU_EXPECT_NO_ERROR(gl.getError(), "glUseProgram call failed.");
1914 }
1915
1916 if (0 == m_po)
1917 {
1918 throw 0;
1919 }
1920 }
1921
1922 /** @brief Function draws full screen quad. Queries are measured during the process.
1923 * Also, transform feedback data is captured.
1924 *
1925 * @note The function may throw if unexpected error has occured.
1926 */
draw()1927 void FunctionalTest::draw()
1928 {
1929 /* Shortcut for GL functionality. */
1930 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1931
1932 /* Start queries. */
1933 for (glw::GLuint i = 0; i < s_targets_count; ++i)
1934 {
1935 gl.beginQuery(s_targets[i], m_qo[i]);
1936 }
1937
1938 /* Start XFB. */
1939 gl.beginTransformFeedback(GL_TRIANGLES);
1940
1941 /* Draw full screen quad. */
1942 gl.drawArrays(GL_TRIANGLE_STRIP, 0, 4);
1943
1944 /* Finish XFB. */
1945 gl.endTransformFeedback();
1946
1947 /* Finish queries. */
1948 for (glw::GLuint i = 0; i < s_targets_count; ++i)
1949 {
1950 gl.endQuery(s_targets[i]);
1951 }
1952
1953 /* Make sure OpenGL finished drawing. */
1954 gl.finish();
1955
1956 /* Error checking. */
1957 GLU_EXPECT_NO_ERROR(gl.getError(), "Drawing function have failed.");
1958 }
1959
1960 /** @brief Check that framebuffer is filled with red color.
1961 */
checkView()1962 bool FunctionalTest::checkView()
1963 {
1964 /* Shortcut for GL functionality. */
1965 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1966
1967 /* Fetch framebuffer data. */
1968 glw::GLubyte pixel[4] = { 0 };
1969
1970 gl.readPixels(0, 0, 1, 1, GL_RGBA, GL_UNSIGNED_BYTE, pixel);
1971 GLU_EXPECT_NO_ERROR(gl.getError(), "glReadPixels have failed");
1972
1973 /* Comparison with expected values. */
1974 if ((255 != pixel[0]) || (0 != pixel[1]) || (0 != pixel[2]) || (255 != pixel[3]))
1975 {
1976 m_context.getTestContext().getLog()
1977 << tcu::TestLog::Message << "Frameuffer content (" << (unsigned int)pixel[0] << ", "
1978 << (unsigned int)pixel[1] << ", " << (unsigned int)pixel[2] << ", " << (unsigned int)pixel[3]
1979 << ") is different than expected (255, 0, 0, 255)." << tcu::TestLog::EndMessage;
1980
1981 return false;
1982 }
1983
1984 return true;
1985 }
1986
1987 /** @brief Check that transform feedback buffer
1988 * contains values representing quad.
1989 */
checkXFB()1990 bool FunctionalTest::checkXFB()
1991 {
1992 /* Shortcut for GL functionality */
1993 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1994
1995 /* Default result. */
1996 bool is_ok = true;
1997
1998 /* Mapping buffer object to the user-space. */
1999 glw::GLint* buffer = (glw::GLint*)gl.mapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, GL_READ_ONLY);
2000 GLU_EXPECT_NO_ERROR(gl.getError(), "glMapBuffer call failed.");
2001
2002 if ((0 != buffer[0]) || (1 != buffer[1]) || (2 != buffer[2]) ||
2003
2004 (2 != buffer[3]) || (1 != buffer[4]) || (3 != buffer[5]))
2005 {
2006 is_ok = false;
2007 }
2008
2009 gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
2010 GLU_EXPECT_NO_ERROR(gl.getError(), "glUnmapBuffer call failed.");
2011
2012 return is_ok;
2013 }
2014
2015 /** @brief Release all created objects.
2016 */
clean()2017 void FunctionalTest::clean()
2018 {
2019 /* Shortcut for GL functionality. */
2020 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2021
2022 /* Releasing queries. */
2023 if (DE_NULL != m_qo)
2024 {
2025 gl.deleteQueries(s_targets_count, m_qo);
2026
2027 delete[] m_qo;
2028
2029 m_qo = DE_NULL;
2030 }
2031
2032 /* Release framebuffer. */
2033 if (m_fbo)
2034 {
2035 gl.deleteFramebuffers(1, &m_fbo);
2036
2037 m_fbo = 0;
2038 }
2039
2040 /* Release renderbuffer. */
2041 if (m_rbo)
2042 {
2043 gl.deleteRenderbuffers(1, &m_rbo);
2044
2045 m_rbo = 0;
2046 }
2047
2048 /* Release vertex array object. */
2049 if (m_vao)
2050 {
2051 gl.deleteVertexArrays(1, &m_vao);
2052
2053 m_vao = 0;
2054 }
2055
2056 /* Release buffer object for storing queries' results. */
2057 if (m_bo_query)
2058 {
2059 gl.deleteBuffers(1, &m_bo_query);
2060
2061 m_bo_query = 0;
2062 }
2063
2064 /* Release transform feedback buffer. */
2065 if (m_bo_xfb)
2066 {
2067 gl.deleteBuffers(1, &m_bo_xfb);
2068
2069 m_bo_xfb = 0;
2070 }
2071
2072 /* Release GLSL program. */
2073 if (m_po)
2074 {
2075 gl.useProgram(0);
2076
2077 gl.deleteProgram(m_po);
2078
2079 m_po = 0;
2080 }
2081 }
2082
2083 /** Targets to be tested. */
2084 const glw::GLenum FunctionalTest::s_targets[] = { GL_SAMPLES_PASSED, GL_TIME_ELAPSED, GL_PRIMITIVES_GENERATED,
2085 GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN };
2086
2087 /** Expected result for each target. */
2088 const glw::GLint FunctionalTest::s_results[] = { 1, 0, 2, 2 };
2089
2090 /** Number of targets. */
2091 const glw::GLuint FunctionalTest::s_targets_count = sizeof(s_targets) / sizeof(s_targets[0]);
2092
2093 /** Vertex shader source code. */
2094 const glw::GLchar FunctionalTest::s_vertex_shader[] = "#version 450\n"
2095 "\n"
2096 "out int xfb_result;\n"
2097 "\n"
2098 "void main()\n"
2099 "{\n"
2100 " switch(gl_VertexID)\n"
2101 " {\n"
2102 " case 0:\n"
2103 " gl_Position = vec4(-1.0, 1.0, 0.0, 1.0);\n"
2104 " break;\n"
2105 " case 1:\n"
2106 " gl_Position = vec4( 1.0, 1.0, 0.0, 1.0);\n"
2107 " break;\n"
2108 " case 2:\n"
2109 " gl_Position = vec4(-1.0,-1.0, 0.0, 1.0);\n"
2110 " break;\n"
2111 " case 3:\n"
2112 " gl_Position = vec4( 1.0,-1.0, 0.0, 1.0);\n"
2113 " break;\n"
2114 " }\n"
2115 "\n"
2116 " xfb_result = gl_VertexID;\n"
2117 "}\n";
2118
2119 /** Fragment shader source program. */
2120 const glw::GLchar FunctionalTest::s_fragment_shader[] = "#version 450\n"
2121 "\n"
2122 "out vec4 color;\n"
2123 "\n"
2124 "void main()\n"
2125 "{\n"
2126 " color = vec4(1.0, 0.0, 0.0, 1.0);\n"
2127 "}\n";
2128
2129 /** Name of transform feedback varying in vertex shader. */
2130 const glw::GLchar* FunctionalTest::s_xfb_varying_name = "xfb_result";
2131
2132 } /* Queries namespace. */
2133 } /* DirectStateAccess namespace. */
2134 } /* gl4cts namespace. */
2135