1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program OpenGL ES 3.1 Module
3  * -------------------------------------------------
4  *
5  * Copyright 2014 The Android Open Source Project
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 Debug output (KHR_debug) tests
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es31fDebugTests.hpp"
25 
26 #include "es31fNegativeTestShared.hpp"
27 #include "es31fNegativeBufferApiTests.hpp"
28 #include "es31fNegativeTextureApiTests.hpp"
29 #include "es31fNegativeShaderApiTests.hpp"
30 #include "es31fNegativeFragmentApiTests.hpp"
31 #include "es31fNegativeVertexArrayApiTests.hpp"
32 #include "es31fNegativeStateApiTests.hpp"
33 #include "es31fNegativeAtomicCounterTests.hpp"
34 #include "es31fNegativeShaderImageLoadStoreTests.hpp"
35 #include "es31fNegativeShaderFunctionTests.hpp"
36 #include "es31fNegativeShaderDirectiveTests.hpp"
37 #include "es31fNegativeSSBOBlockTests.hpp"
38 #include "es31fNegativePreciseTests.hpp"
39 #include "es31fNegativeAdvancedBlendEquationTests.hpp"
40 #include "es31fNegativeShaderStorageTests.hpp"
41 #include "es31fNegativeTessellationTests.hpp"
42 #include "es31fNegativeComputeTests.hpp"
43 #include "es31fNegativeSampleVariablesTests.hpp"
44 #include "es31fNegativeShaderFramebufferFetchTests.hpp"
45 
46 #include "deUniquePtr.hpp"
47 #include "deRandom.hpp"
48 #include "deStringUtil.hpp"
49 #include "deSTLUtil.hpp"
50 #include "deMutex.hpp"
51 #include "deThread.h"
52 
53 #include "gluRenderContext.hpp"
54 #include "gluContextInfo.hpp"
55 #include "gluCallLogWrapper.hpp"
56 #include "gluStrUtil.hpp"
57 
58 #include "glwDefs.hpp"
59 #include "glwEnums.hpp"
60 #include "glwFunctions.hpp"
61 
62 #include "tes31Context.hpp"
63 #include "tcuTestContext.hpp"
64 #include "tcuCommandLine.hpp"
65 #include "tcuResultCollector.hpp"
66 
67 #include "glsStateQueryUtil.hpp"
68 
69 namespace deqp
70 {
71 namespace gles31
72 {
73 namespace Functional
74 {
75 namespace
76 {
77 using namespace glw;
78 
79 using std::string;
80 using std::vector;
81 using std::set;
82 using std::map;
83 using de::MovePtr;
84 
85 using tcu::ResultCollector;
86 using tcu::TestLog;
87 using glu::CallLogWrapper;
88 
89 using NegativeTestShared::NegativeTestContext;
90 
91 static const GLenum s_debugTypes[] =
92 {
93 	GL_DEBUG_TYPE_ERROR,
94 	GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR,
95 	GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR,
96 	GL_DEBUG_TYPE_PORTABILITY,
97 	GL_DEBUG_TYPE_PERFORMANCE,
98 	GL_DEBUG_TYPE_OTHER,
99 	GL_DEBUG_TYPE_MARKER,
100 	GL_DEBUG_TYPE_PUSH_GROUP,
101 	GL_DEBUG_TYPE_POP_GROUP,
102 };
103 
104 static const GLenum s_debugSeverities[] =
105 {
106 	GL_DEBUG_SEVERITY_HIGH,
107     GL_DEBUG_SEVERITY_MEDIUM,
108     GL_DEBUG_SEVERITY_LOW,
109     GL_DEBUG_SEVERITY_NOTIFICATION,
110 };
111 
isKHRDebugSupported(Context & ctx)112 static bool isKHRDebugSupported (Context& ctx)
113 {
114 	const bool supportsES32 = glu::contextSupports(ctx.getRenderContext().getType(), glu::ApiType::es(3, 2));
115 	return supportsES32 || ctx.getContextInfo().isExtensionSupported("GL_KHR_debug");
116 }
117 
118 class BaseCase;
119 
120 class DebugMessageTestContext : public NegativeTestContext
121 {
122 public:
123 				DebugMessageTestContext		(BaseCase&					host,
124 											 glu::RenderContext&		renderCtx,
125 											 const glu::ContextInfo&	ctxInfo,
126 											 tcu::TestLog&				log,
127 											 tcu::ResultCollector&		results,
128 											 bool						enableLog);
129 				~DebugMessageTestContext	(void);
130 
131 	void		expectMessage				(GLenum source, GLenum type);
132 
133 private:
134 	BaseCase&	m_debugHost;
135 };
136 
137 class TestFunctionWrapper
138 {
139 public:
140 	typedef void (*CoreTestFunc)(NegativeTestContext& ctx);
141 	typedef void (*DebugTestFunc)(DebugMessageTestContext& ctx);
142 
143 				TestFunctionWrapper	(void);
144 	explicit	TestFunctionWrapper	(CoreTestFunc func);
145 	explicit	TestFunctionWrapper	(DebugTestFunc func);
146 
147 	void		call				(DebugMessageTestContext& ctx) const;
148 
149 private:
150 	enum FuncType
151 	{
152 		TYPE_NULL = 0,
153 		TYPE_CORE,
154 		TYPE_DEBUG,
155 	};
156 	FuncType m_type;
157 
158 	union
159 	{
160 		CoreTestFunc	coreFn;
161 		DebugTestFunc	debugFn;
162 	} m_func;
163 };
164 
TestFunctionWrapper(void)165 TestFunctionWrapper::TestFunctionWrapper (void)
166 	: m_type(TYPE_NULL)
167 {
168 }
169 
TestFunctionWrapper(CoreTestFunc func)170 TestFunctionWrapper::TestFunctionWrapper (CoreTestFunc func)
171 	: m_type(TYPE_CORE)
172 {
173 	m_func.coreFn = func;
174 }
175 
TestFunctionWrapper(DebugTestFunc func)176 TestFunctionWrapper::TestFunctionWrapper (DebugTestFunc func)
177 	: m_type(TYPE_DEBUG)
178 {
179 	m_func.debugFn = func;
180 }
181 
call(DebugMessageTestContext & ctx) const182 void TestFunctionWrapper::call (DebugMessageTestContext& ctx) const
183 {
184 	if (m_type == TYPE_CORE)
185 		m_func.coreFn(static_cast<NegativeTestContext&>(ctx));
186 	else if (m_type == TYPE_DEBUG)
187 		m_func.debugFn(ctx);
188 	else
189 		DE_ASSERT(false);
190 }
191 
emitMessages(DebugMessageTestContext & ctx,GLenum source)192 void emitMessages (DebugMessageTestContext& ctx, GLenum source)
193 {
194 	for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(s_debugTypes); typeNdx++)
195 	{
196 		for (int severityNdx = 0; severityNdx < DE_LENGTH_OF_ARRAY(s_debugSeverities); severityNdx++)
197 		{
198 			const GLenum type		= s_debugTypes[typeNdx];
199 			const GLenum severity	= s_debugSeverities[severityNdx];
200 			const string msg		= string("Application generated message with type ") + glu::getDebugMessageTypeName(type)
201 									  + " and severity " + glu::getDebugMessageSeverityName(severity);
202 
203 			// Use severity as ID, guaranteed unique
204 			ctx.glDebugMessageInsert(source, type, severity, severity, -1, msg.c_str());
205 			ctx.expectMessage(source, type);
206 		}
207 	}
208 }
209 
application_messages(DebugMessageTestContext & ctx)210 void application_messages (DebugMessageTestContext& ctx)
211 {
212 	ctx.beginSection("Messages with source of GL_DEBUG_SOURCE_APPLICATION");
213 	emitMessages(ctx, GL_DEBUG_SOURCE_APPLICATION);
214 	ctx.endSection();
215 }
216 
thirdparty_messages(DebugMessageTestContext & ctx)217 void thirdparty_messages (DebugMessageTestContext& ctx)
218 {
219 	ctx.beginSection("Messages with source of GL_DEBUG_SOURCE_THIRD_PARTY");
220 	emitMessages(ctx, GL_DEBUG_SOURCE_THIRD_PARTY);
221 	ctx.endSection();
222 }
223 
push_pop_messages(DebugMessageTestContext & ctx)224 void push_pop_messages (DebugMessageTestContext& ctx)
225 {
226 	ctx.beginSection("Push/Pop Debug Group");
227 
228 	ctx.glPushDebugGroup(GL_DEBUG_SOURCE_APPLICATION, 1, -1, "Application group 1");
229 	ctx.expectMessage(GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PUSH_GROUP);
230 	ctx.glPushDebugGroup(GL_DEBUG_SOURCE_APPLICATION, 2, -1, "Application group 1-1");
231 	ctx.expectMessage(GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PUSH_GROUP);
232 	ctx.glPushDebugGroup(GL_DEBUG_SOURCE_APPLICATION, 3, -1, "Application group 1-1-1");
233 	ctx.expectMessage(GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PUSH_GROUP);
234 	ctx.glPopDebugGroup();
235 	ctx.expectMessage(GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_POP_GROUP);
236 	ctx.glPopDebugGroup();
237 	ctx.expectMessage(GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_POP_GROUP);
238 
239 	ctx.glPushDebugGroup(GL_DEBUG_SOURCE_APPLICATION, 4, -1, "Application group 1-2");
240 	ctx.expectMessage(GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PUSH_GROUP);
241 	ctx.glPopDebugGroup();
242 	ctx.expectMessage(GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_POP_GROUP);
243 
244 	ctx.glPushDebugGroup(GL_DEBUG_SOURCE_THIRD_PARTY, 4, -1, "3rd Party group 1-3");
245 	ctx.expectMessage(GL_DEBUG_SOURCE_THIRD_PARTY, GL_DEBUG_TYPE_PUSH_GROUP);
246 	ctx.glPopDebugGroup();
247 	ctx.expectMessage(GL_DEBUG_SOURCE_THIRD_PARTY, GL_DEBUG_TYPE_POP_GROUP);
248 	ctx.glPopDebugGroup();
249 	ctx.expectMessage(GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_POP_GROUP);
250 
251 	ctx.glPushDebugGroup(GL_DEBUG_SOURCE_THIRD_PARTY, 4, -1, "3rd Party group 2");
252 	ctx.expectMessage(GL_DEBUG_SOURCE_THIRD_PARTY, GL_DEBUG_TYPE_PUSH_GROUP);
253 	ctx.glPopDebugGroup();
254 	ctx.expectMessage(GL_DEBUG_SOURCE_THIRD_PARTY, GL_DEBUG_TYPE_POP_GROUP);
255 
256 	ctx.endSection();
257 }
258 
259 struct FunctionContainer
260 {
261 	TestFunctionWrapper	function;
262 	const char*			name;
263 	const char*			desc;
264 };
265 
getUserMessageFuncs(void)266 vector<FunctionContainer> getUserMessageFuncs (void)
267 {
268 	FunctionContainer funcs[] =
269 	{
270 		{ TestFunctionWrapper(application_messages),	"application_messages", "Externally generated messages from the application"	},
271 		{ TestFunctionWrapper(thirdparty_messages),		"third_party_messages",	"Externally generated messages from a third party"		},
272 		{ TestFunctionWrapper(push_pop_messages),		"push_pop_stack",		"Messages from pushing/popping debug groups"			},
273 	};
274 
275 	return std::vector<FunctionContainer>(DE_ARRAY_BEGIN(funcs), DE_ARRAY_END(funcs));
276 }
277 
278 // Data required to uniquely identify a debug message
279 struct MessageID
280 {
281 	GLenum source;
282 	GLenum type;
283 	GLuint id;
284 
MessageIDdeqp::gles31::Functional::__anon0b3d5a1f0111::MessageID285 	MessageID (void) : source(GL_NONE), type(GL_NONE), id(0) {}
MessageIDdeqp::gles31::Functional::__anon0b3d5a1f0111::MessageID286 	MessageID (GLenum source_, GLenum type_, GLuint id_) : source(source_), type(type_), id(id_) {}
287 
operator ==deqp::gles31::Functional::__anon0b3d5a1f0111::MessageID288 	bool operator== (const MessageID& rhs) const { return source == rhs.source && type == rhs.type && id == rhs.id;}
operator !=deqp::gles31::Functional::__anon0b3d5a1f0111::MessageID289 	bool operator!= (const MessageID& rhs) const { return source != rhs.source || type != rhs.type || id != rhs.id;}
operator <deqp::gles31::Functional::__anon0b3d5a1f0111::MessageID290 	bool operator<  (const MessageID& rhs) const
291 	{
292 		return source < rhs.source || (source == rhs.source && (type < rhs.type || (type == rhs.type && id < rhs.id)));
293 	}
294 };
295 
operator <<(std::ostream & str,const MessageID & id)296 std::ostream& operator<< (std::ostream& str, const MessageID &id)
297 {
298 	return str << glu::getDebugMessageSourceStr(id.source) << ", "	<< glu::getDebugMessageTypeStr(id.type) << ", " << id.id;
299 }
300 
301 // All info from a single debug message
302 struct MessageData
303 {
304 	MessageID	id;
305 	GLenum		severity;
306 	string		message;
307 
MessageDatadeqp::gles31::Functional::__anon0b3d5a1f0111::MessageData308 	MessageData (void) : id(MessageID()), severity(GL_NONE) {}
MessageDatadeqp::gles31::Functional::__anon0b3d5a1f0111::MessageData309 	MessageData (const MessageID& id_, GLenum severity_, const string& message_) : id(id_) , severity(severity_) , message(message_) {}
310 };
311 
312 extern "C" typedef void GLW_APIENTRY DebugCallbackFunc(GLenum, GLenum, GLuint, GLenum, GLsizei, const char*, const void*);
313 
314 // Base class
315 class BaseCase : public NegativeTestShared::ErrorCase
316 {
317 public:
318 								BaseCase			(Context&					ctx,
319 													 const char*				name,
320 													 const char*				desc);
~BaseCase(void)321 	virtual						~BaseCase			(void) {}
322 
323 	virtual IterateResult		iterate				(void) = 0;
324 
325 	virtual void				expectMessage		(GLenum source, GLenum type);
326 	virtual void				expectError			(GLenum error0, GLenum error1);
327 
328 protected:
329 	struct VerificationResult
330 	{
331 		const qpTestResult	result;
332 		const string		resultMessage;
333 		const string		logMessage;
334 
VerificationResultdeqp::gles31::Functional::__anon0b3d5a1f0111::BaseCase::VerificationResult335 		VerificationResult (qpTestResult result_, const string& resultMessage_, const string& logMessage_)
336 			: result(result_), resultMessage(resultMessage_), logMessage(logMessage_) {}
337 	};
338 
339 	static DebugCallbackFunc	callbackHandle;
340 	virtual void				callback			(GLenum source, GLenum type, GLuint id, GLenum severity, const std::string& message);
341 
342 
343 	VerificationResult			verifyMessageCount	(const MessageID& id, GLenum severity, int refCount, int resCount, bool messageEnabled) const;
344 
345 	// Verify a single message instance against expected attributes
346 	void						verifyMessage		(const MessageData& message, GLenum source, GLenum type, GLuint id, GLenum severity);
347 	void						verifyMessage		(const MessageData& message, GLenum source, GLenum type);
348 
349 	bool						verifyMessageExists	(const MessageData& message, GLenum source, GLenum type);
350 	void						verifyMessageGroup	(const MessageData& message, GLenum source, GLenum type);
351 	void						verifyMessageString	(const MessageData& message);
352 
353 	bool						isDebugContext		(void) const;
354 
355 	tcu::ResultCollector		m_results;
356 };
357 
callbackHandle(GLenum source,GLenum type,GLuint id,GLenum severity,GLsizei length,const char * message,const void * userParam)358 void BaseCase::callbackHandle (GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const char* message, const void* userParam)
359 {
360 	static_cast<BaseCase*>(const_cast<void*>(userParam))->callback(source, type, id, severity, string(message, &message[length]));
361 }
362 
BaseCase(Context & ctx,const char * name,const char * desc)363 BaseCase::BaseCase (Context& ctx, const char* name, const char* desc)
364 	: ErrorCase(ctx, name, desc)
365 {
366 }
367 
expectMessage(GLenum source,GLenum type)368 void BaseCase::expectMessage (GLenum source, GLenum type)
369 {
370 	DE_UNREF(source);
371 	DE_UNREF(type);
372 }
373 
expectError(GLenum error0,GLenum error1)374 void BaseCase::expectError (GLenum error0, GLenum error1)
375 {
376 	if (error0 != GL_NO_ERROR || error1 != GL_NO_ERROR)
377 		expectMessage(GL_DEBUG_SOURCE_API, GL_DEBUG_TYPE_ERROR);
378 	else
379 		expectMessage(GL_DONT_CARE, GL_DONT_CARE);
380 }
381 
callback(GLenum source,GLenum type,GLuint id,GLenum severity,const string & message)382 void BaseCase::callback (GLenum source, GLenum type, GLuint id, GLenum severity, const string& message)
383 {
384 	DE_UNREF(source);
385 	DE_UNREF(type);
386 	DE_UNREF(id);
387 	DE_UNREF(severity);
388 	DE_UNREF(message);
389 }
390 
verifyMessageCount(const MessageID & id,GLenum severity,int refCount,int resCount,bool messageEnabled) const391 BaseCase::VerificationResult BaseCase::verifyMessageCount (const MessageID& id, GLenum severity, int refCount, int resCount, bool messageEnabled) const
392 {
393 	std::stringstream log;
394 
395 	// This message should not be filtered out
396 	if (messageEnabled)
397 	{
398 		if (resCount != refCount)
399 		{
400 			/*
401 			 * Technically nothing requires the implementation to be consistent in terms
402 			 * of the messages it produces in most situations, allowing the set of messages
403 			 * produced to vary between executions. This function splits messages
404 			 * into deterministic and non-deterministic to facilitate handling of such messages.
405 			 *
406 			 * Non-deterministic messages that are present in differing quantities in filtered and
407 			 * unfiltered runs will not fail the test case unless in direct violation of a filter:
408 			 * the implementation may produce an arbitrary number of such messages when they are
409 			 * not filtered out and none when they are filtered.
410 			 *
411 			 * A list of error source/type combinations with their assumed behaviour and
412 			 * the rationale for expecting such behaviour follows
413 			 *
414 			 * For API/shader messages we assume that the following types are deterministic:
415 			 *   DEBUG_TYPE_ERROR                 Errors specified by spec and should always be produced
416 			 *
417 			 * For API messages the following types are assumed to be non-deterministic
418 			 * and treated as quality warnings since the underlying reported issue does not change between calls:
419 			 *   DEBUG_TYPE_DEPRECATED_BEHAVIOR   Reasonable to only report first instance
420              *   DEBUG_TYPE_UNDEFINED_BEHAVIOR    Reasonable to only report first instance
421              *   DEBUG_TYPE_PORTABILITY           Reasonable to only report first instance
422 			 *
423 			 * For API messages the following types are assumed to be non-deterministic
424 			 * and do not affect test results.
425 			 *   DEBUG_TYPE_PERFORMANCE           May be tied to arbitrary factors, reasonable to report only first instance
426              *   DEBUG_TYPE_OTHER                 Definition allows arbitrary contents
427 			 *
428 			 * For 3rd party and application messages the following types are deterministic:
429              *   DEBUG_TYPE_MARKER                Only generated by test
430 			 *   DEBUG_TYPE_PUSH_GROUP            Only generated by test
431 			 *   DEBUG_TYPE_POP_GROUP             Only generated by test
432 			 *   All others                       Only generated by test
433 			 *
434 			 * All messages with category of window system or other are treated as non-deterministic
435 			 * and do not effect test results since they can be assumed to be outside control of
436 			 * both the implementation and test case
437 			 *
438 			 */
439 
440 			const bool isDeterministic	= id.source == GL_DEBUG_SOURCE_APPLICATION ||
441 										  id.source == GL_DEBUG_SOURCE_THIRD_PARTY ||
442 										  ((id.source == GL_DEBUG_SOURCE_API || id.source == GL_DEBUG_SOURCE_SHADER_COMPILER) && id.type == GL_DEBUG_TYPE_ERROR);
443 
444 			const bool canIgnore		= id.source == GL_DEBUG_SOURCE_WINDOW_SYSTEM || id.source == GL_DEBUG_SOURCE_OTHER;
445 
446 			if (isDeterministic)
447 			{
448 				if (resCount > refCount)
449 				{
450 					log << "Extra instances of message were found: (" << id << ") with "
451 						<< glu::getDebugMessageSeverityStr(severity)
452 						<< " (got " << resCount << ", expected " << refCount << ")";
453 					return VerificationResult(QP_TEST_RESULT_FAIL, "Extra instances of a deterministic message were present", log.str());
454 				}
455 				else
456 				{
457 					log << "Instances of message were missing: (" << id << ") with "
458 						<< glu::getDebugMessageSeverityStr(severity)
459 						<< " (got " << resCount << ", expected " << refCount << ")";
460 					return VerificationResult(QP_TEST_RESULT_FAIL, "Message missing", log.str());
461 				}
462 			}
463 			else if(!canIgnore)
464 			{
465 				if (resCount > refCount)
466 				{
467 					log << "Extra instances of message were found but the message is non-deterministic(warning): (" << id << ") with "
468 						<< glu::getDebugMessageSeverityStr(severity)
469 						<< " (got " << resCount << ", expected " << refCount << ")";
470 					return VerificationResult(QP_TEST_RESULT_QUALITY_WARNING, "Extra instances of a message were present", log.str());
471 				}
472 				else
473 				{
474 					log << "Instances of message were missing but the message is non-deterministic(warning): (" << id << ") with "
475 						<< glu::getDebugMessageSeverityStr(severity)
476 						<< " (got " << resCount << ", expected " << refCount << ")";
477 					return VerificationResult(QP_TEST_RESULT_QUALITY_WARNING, "Message missing", log.str());
478 				}
479 			}
480 			else
481 			{
482 				if (resCount > refCount)
483 				{
484 					log << "Extra instances of message were found but the message is non-deterministic(ignored): (" << id << ") with "
485 						<< glu::getDebugMessageSeverityStr(severity)
486 						<< " (got " << resCount << ", expected " << refCount << ")";
487 					return VerificationResult(QP_TEST_RESULT_PASS, "", log.str());
488 				}
489 				else
490 				{
491 					log << "Instances of message were missing but the message is non-deterministic(ignored): (" << id << ") with "
492 						<< glu::getDebugMessageSeverityStr(severity)
493 						<< " (got " << resCount << ", expected " << refCount << ")";
494 					return VerificationResult(QP_TEST_RESULT_PASS, "", log.str());
495 				}
496 			}
497 		}
498 		else // Passed as appropriate
499 		{
500 			log << "Message was found when expected: ("<< id << ") with "
501 				<< glu::getDebugMessageSeverityStr(severity);
502 			return VerificationResult(QP_TEST_RESULT_PASS, "", log.str());
503 		}
504 	}
505 	// Message should be filtered out
506 	else
507 	{
508 		// Filtered out
509 		if (resCount == 0)
510 		{
511 			log << "Message was excluded correctly:  (" << id << ") with "
512 				<< glu::getDebugMessageSeverityStr(severity);
513 			return VerificationResult(QP_TEST_RESULT_PASS, "", log.str());
514 		}
515 		// Only present in filtered run (ERROR)
516 		else if (resCount > 0 && refCount == 0)
517 		{
518 			log << "A message was not excluded as it should have been: (" << id << ") with "
519 				<< glu::getDebugMessageSeverityStr(severity)
520 				<< ". This message was not present in the reference run";
521 			return VerificationResult(QP_TEST_RESULT_FAIL, "A message was not filtered out", log.str());
522 		}
523 		// Present in both runs (ERROR)
524 		else
525 		{
526 			log << "A message was not excluded as it should have been: (" << id << ") with "
527 				<< glu::getDebugMessageSeverityStr(severity);
528 			return VerificationResult(QP_TEST_RESULT_FAIL, "A message was not filtered out", log.str());
529 		}
530 	}
531 }
532 
533 // Return true if message needs further verification
verifyMessageExists(const MessageData & message,GLenum source,GLenum type)534 bool BaseCase::verifyMessageExists (const MessageData& message, GLenum source, GLenum type)
535 {
536 	TestLog& log = m_testCtx.getLog();
537 
538 	if (source == GL_DONT_CARE || type == GL_DONT_CARE)
539 		return false;
540 	else if (message.id.source == GL_NONE || message.id.type == GL_NONE)
541 	{
542 		if (isDebugContext())
543 		{
544 			m_results.addResult(QP_TEST_RESULT_FAIL, "Message was not reported as expected");
545 			log << TestLog::Message << "A message was expected but none was reported" << TestLog::EndMessage;
546 		}
547 		else
548 		{
549 			m_results.addResult(QP_TEST_RESULT_QUALITY_WARNING, "Verification accuracy is lacking without a debug context");
550 			log << TestLog::Message << "A message was expected but none was reported. Running without a debug context" << TestLog::EndMessage;
551 		}
552 		return false;
553 	}
554 	else
555 		return true;
556 }
557 
verifyMessageGroup(const MessageData & message,GLenum source,GLenum type)558 void BaseCase::verifyMessageGroup (const MessageData& message, GLenum source, GLenum type)
559 {
560 	TestLog& log = m_testCtx.getLog();
561 
562 	if (message.id.source != source)
563 	{
564 		m_results.addResult(QP_TEST_RESULT_FAIL, "Incorrect message source");
565 		log << TestLog::Message << "Message source was " << glu::getDebugMessageSourceStr(message.id.source)
566 			<< " when it should have been "  << glu::getDebugMessageSourceStr(source) << TestLog::EndMessage;
567 	}
568 
569 	if (message.id.type != type)
570 	{
571 		m_results.addResult(QP_TEST_RESULT_FAIL, "Incorrect message type");
572 		log << TestLog::Message << "Message type was " << glu::getDebugMessageTypeStr(message.id.type)
573 			<< " when it should have been " << glu::getDebugMessageTypeStr(type) << TestLog::EndMessage;
574 	}
575 }
576 
verifyMessageString(const MessageData & message)577 void BaseCase::verifyMessageString (const MessageData& message)
578 {
579 	TestLog& log = m_testCtx.getLog();
580 
581 	log << TestLog::Message << "Driver says: \"" << message.message << "\"" << TestLog::EndMessage;
582 
583 	if (message.message.empty())
584 	{
585 		m_results.addResult(QP_TEST_RESULT_QUALITY_WARNING, "Empty message");
586 		log << TestLog::Message << "Message message was empty" << TestLog::EndMessage;
587 	}
588 }
589 
verifyMessage(const MessageData & message,GLenum source,GLenum type)590 void BaseCase::verifyMessage (const MessageData& message, GLenum source, GLenum type)
591 {
592 	if (verifyMessageExists(message, source, type))
593 	{
594 		verifyMessageString(message);
595 		verifyMessageGroup(message, source, type);
596 	}
597 }
598 
verifyMessage(const MessageData & message,GLenum source,GLenum type,GLuint id,GLenum severity)599 void BaseCase::verifyMessage (const MessageData& message, GLenum source, GLenum type, GLuint id, GLenum severity)
600 {
601 	TestLog& log = m_testCtx.getLog();
602 
603 	if (verifyMessageExists(message, source, type))
604 	{
605 		verifyMessageString(message);
606 		verifyMessageGroup(message, source, type);
607 
608 		if (message.id.id != id)
609 		{
610 			m_results.addResult(QP_TEST_RESULT_FAIL, "Incorrect message id");
611 			log << TestLog::Message << "Message id was " << message.id.id
612 				<< " when it should have been " << id << TestLog::EndMessage;
613 		}
614 
615 		if (message.severity != severity)
616 		{
617 			m_results.addResult(QP_TEST_RESULT_FAIL, "Incorrect message severity");
618 			log << TestLog::Message << "Message severity was " << glu::getDebugMessageSeverityStr(message.severity)
619 				<< " when it should have been " << glu::getDebugMessageSeverityStr(severity) << TestLog::EndMessage;
620 		}
621 	}
622 }
623 
isDebugContext(void) const624 bool BaseCase::isDebugContext (void) const
625 {
626 	return (m_context.getRenderContext().getType().getFlags() & glu::CONTEXT_DEBUG) != 0;
627 }
628 
629 // Generate errors, verify that each error results in a callback call
630 class CallbackErrorCase : public BaseCase
631 {
632 public:
633 								CallbackErrorCase	(Context&				ctx,
634 													 const char*			name,
635 													 const char*			desc,
636 													 TestFunctionWrapper	errorFunc);
~CallbackErrorCase(void)637 	virtual						~CallbackErrorCase	(void) {}
638 
639 	virtual IterateResult		iterate				(void);
640 
641 	virtual void				expectMessage		(GLenum source, GLenum type);
642 
643 private:
644 	virtual void				callback			(GLenum source, GLenum type, GLuint id, GLenum severity, const string& message);
645 
646 	const TestFunctionWrapper	m_errorFunc;
647 	MessageData					m_lastMessage;
648 };
649 
CallbackErrorCase(Context & ctx,const char * name,const char * desc,TestFunctionWrapper errorFunc)650 CallbackErrorCase::CallbackErrorCase (Context&				ctx,
651 									  const char*			name,
652 									  const char*			desc,
653 									  TestFunctionWrapper	errorFunc)
654 	: BaseCase		(ctx, name, desc)
655 	, m_errorFunc	(errorFunc)
656 {
657 }
658 
iterate(void)659 CallbackErrorCase::IterateResult CallbackErrorCase::iterate (void)
660 {
661 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
662 
663 	const glw::Functions&	gl		= m_context.getRenderContext().getFunctions();
664 	tcu::TestLog&			log		= m_testCtx.getLog();
665 	DebugMessageTestContext	context	= DebugMessageTestContext(*this, m_context.getRenderContext(), m_context.getContextInfo(), log, m_results, true);
666 
667 	gl.enable(GL_DEBUG_OUTPUT);
668 	gl.enable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
669 	gl.debugMessageControl(GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, false); // disable all
670 	gl.debugMessageControl(GL_DEBUG_SOURCE_API, GL_DEBUG_TYPE_ERROR, GL_DONT_CARE, 0, DE_NULL, true); // enable API errors
671 	gl.debugMessageControl(GL_DEBUG_SOURCE_APPLICATION, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, true); // enable application messages
672 	gl.debugMessageControl(GL_DEBUG_SOURCE_THIRD_PARTY, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, true); // enable third party messages
673 	gl.debugMessageCallback(callbackHandle, this);
674 
675 	m_errorFunc.call(context);
676 
677 	gl.debugMessageCallback(DE_NULL, DE_NULL);
678 	gl.disable(GL_DEBUG_OUTPUT);
679 
680 	m_results.setTestContextResult(m_testCtx);
681 
682 	return STOP;
683 }
684 
expectMessage(GLenum source,GLenum type)685 void CallbackErrorCase::expectMessage (GLenum source, GLenum type)
686 {
687 	verifyMessage(m_lastMessage, source, type);
688 	m_lastMessage = MessageData();
689 
690 	// Reset error so that code afterwards (such as glu::ShaderProgram) doesn't break because of
691 	// lingering error state.
692 	m_context.getRenderContext().getFunctions().getError();
693 }
694 
callback(GLenum source,GLenum type,GLuint id,GLenum severity,const string & message)695 void CallbackErrorCase::callback (GLenum source, GLenum type, GLuint id, GLenum severity, const string& message)
696 {
697 	m_lastMessage = MessageData(MessageID(source, type, id), severity, message);
698 }
699 
700 // Generate errors, verify that each error results in a log entry
701 class LogErrorCase : public BaseCase
702 {
703 public:
704 								LogErrorCase	(Context&				context,
705 												 const char*			name,
706 												 const char*			desc,
707 												 TestFunctionWrapper	errorFunc);
~LogErrorCase(void)708 	virtual						~LogErrorCase	(void) {}
709 
710 	virtual IterateResult		iterate			(void);
711 
712 	virtual void				expectMessage	(GLenum source, GLenum type);
713 
714 private:
715 	const TestFunctionWrapper	m_errorFunc;
716 	MessageData					m_lastMessage;
717 };
718 
LogErrorCase(Context & ctx,const char * name,const char * desc,TestFunctionWrapper errorFunc)719 LogErrorCase::LogErrorCase (Context&			ctx,
720 							const char*			name,
721 							const char*			desc,
722 							TestFunctionWrapper	errorFunc)
723 	: BaseCase		(ctx, name, desc)
724 	, m_errorFunc	(errorFunc)
725 {
726 }
727 
iterate(void)728 LogErrorCase::IterateResult LogErrorCase::iterate (void)
729 {
730 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
731 
732 	const glw::Functions&	gl		= m_context.getRenderContext().getFunctions();
733 	tcu::TestLog&			log		= m_testCtx.getLog();
734 	DebugMessageTestContext	context	= DebugMessageTestContext(*this, m_context.getRenderContext(), m_context.getContextInfo(), log, m_results, true);
735 	GLint					numMsg	= 0;
736 
737 	gl.enable(GL_DEBUG_OUTPUT);
738 	gl.enable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
739 	gl.debugMessageControl(GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, false); // disable all
740 	gl.debugMessageControl(GL_DEBUG_SOURCE_API, GL_DEBUG_TYPE_ERROR, GL_DONT_CARE, 0, DE_NULL, true); // enable API errors
741 	gl.debugMessageCallback(DE_NULL, DE_NULL); // enable logging
742 	gl.getIntegerv(GL_DEBUG_LOGGED_MESSAGES, &numMsg);
743 	gl.getDebugMessageLog(numMsg, 0, DE_NULL, DE_NULL, DE_NULL, DE_NULL, DE_NULL, DE_NULL); // clear log
744 
745 	m_errorFunc.call(context);
746 
747 	gl.disable(GL_DEBUG_OUTPUT);
748 	m_results.setTestContextResult(m_testCtx);
749 
750 	return STOP;
751 }
752 
expectMessage(GLenum source,GLenum type)753 void LogErrorCase::expectMessage (GLenum source, GLenum type)
754 {
755 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
756 	int						numMsg		= 0;
757 	TestLog&				log			= m_testCtx.getLog();
758 	MessageData				lastMsg;
759 
760 	if (source == GL_DONT_CARE || type == GL_DONT_CARE)
761 		return;
762 
763 	gl.getIntegerv(GL_DEBUG_LOGGED_MESSAGES, &numMsg);
764 
765 	if (numMsg == 0)
766 	{
767 		if (isDebugContext())
768 		{
769 			m_results.addResult(QP_TEST_RESULT_FAIL, "Error was not reported as expected");
770 			log << TestLog::Message << "A message was expected but none was reported (empty message log)" << TestLog::EndMessage;
771 		}
772 		else
773 		{
774 			m_results.addResult(QP_TEST_RESULT_QUALITY_WARNING, "Verification accuracy is lacking without a debug context");
775 			log << TestLog::Message << "A message was expected but none was reported (empty message log). Running without a debug context" << TestLog::EndMessage;
776 		}
777 		return;
778 	}
779 
780 	// There may be messages other than the error we are looking for in the log.
781 	// Strictly nothing prevents the implementation from producing more than the
782 	// required error from an API call with a defined error. however we assume that
783 	// since calls that produce an error should not change GL state the implementation
784 	// should have nothing else to report.
785 	if (numMsg > 1)
786 		gl.getDebugMessageLog(numMsg-1, 0, DE_NULL, DE_NULL, DE_NULL, DE_NULL, DE_NULL, DE_NULL); // Clear all but last
787 
788 	{
789 		int  msgLen = 0;
790 		gl.getIntegerv(GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH, &msgLen);
791 
792 		TCU_CHECK_MSG(msgLen >= 0, "Negative message length");
793 		TCU_CHECK_MSG(msgLen < 100000, "Excessively long message");
794 
795 		lastMsg.message.resize(msgLen);
796 		gl.getDebugMessageLog(1, msgLen, &lastMsg.id.source, &lastMsg.id.type, &lastMsg.id.id, &lastMsg.severity, &msgLen, &lastMsg.message[0]);
797 	}
798 
799 	log << TestLog::Message << "Driver says: \"" << lastMsg.message << "\"" << TestLog::EndMessage;
800 
801 	verifyMessage(lastMsg, source, type);
802 
803 	// Reset error so that code afterwards (such as glu::ShaderProgram) doesn't break because of
804 	// lingering error state.
805 	m_context.getRenderContext().getFunctions().getError();
806 }
807 
808 // Generate errors, verify that calling glGetError afterwards produces desired result
809 class GetErrorCase : public BaseCase
810 {
811 public:
812 								GetErrorCase	(Context&				ctx,
813 												 const char*			name,
814 												 const char*			desc,
815 												 TestFunctionWrapper	errorFunc);
~GetErrorCase(void)816 	virtual						~GetErrorCase	(void) {}
817 
818 	virtual IterateResult		iterate			(void);
819 
820 	virtual void				expectMessage	(GLenum source, GLenum type);
821 	virtual void				expectError		(glw::GLenum error0, glw::GLenum error1);
822 
823 private:
824 	const TestFunctionWrapper	m_errorFunc;
825 };
826 
GetErrorCase(Context & ctx,const char * name,const char * desc,TestFunctionWrapper errorFunc)827 GetErrorCase::GetErrorCase (Context&			ctx,
828 							const char*			name,
829 							const char*			desc,
830 							TestFunctionWrapper	errorFunc)
831 	: BaseCase		(ctx, name, desc)
832 	, m_errorFunc	(errorFunc)
833 {
834 }
835 
iterate(void)836 GetErrorCase::IterateResult GetErrorCase::iterate (void)
837 {
838 	tcu::TestLog&			log		= m_testCtx.getLog();
839 	DebugMessageTestContext	context	= DebugMessageTestContext(*this, m_context.getRenderContext(), m_context.getContextInfo(), log, m_results, true);
840 
841 	m_errorFunc.call(context);
842 
843 	m_results.setTestContextResult(m_testCtx);
844 
845 	return STOP;
846 }
847 
expectMessage(GLenum source,GLenum type)848 void GetErrorCase::expectMessage (GLenum source, GLenum type)
849 {
850 	DE_UNREF(source);
851 	DE_UNREF(type);
852 	DE_FATAL("GetErrorCase cannot handle anything other than error codes");
853 }
854 
expectError(glw::GLenum error0,glw::GLenum error1)855 void GetErrorCase::expectError (glw::GLenum error0, glw::GLenum error1)
856 {
857 	const glw::Functions&	gl		= m_context.getRenderContext().getFunctions();
858 	TestLog&				log		= m_testCtx.getLog();
859 
860 	const GLenum			result	= gl.getError();
861 
862 	if (result != error0 && result != error1)
863 	{
864 		m_results.addResult(QP_TEST_RESULT_FAIL, "Incorrect error was reported");
865 		if (error0 == error1)
866 			log << TestLog::Message
867 				<< glu::getErrorStr(error0) << " was expected but got "
868 				<< glu::getErrorStr(result)
869 				<< TestLog::EndMessage;
870 		else
871 			log << TestLog::Message
872 				<< glu::getErrorStr(error0) << " or "
873 				<< glu::getErrorStr(error1) << " was expected but got "
874 				<< glu::getErrorStr(result)
875 				<< TestLog::EndMessage;
876 		return;
877 	}
878 }
879 
880 // Generate errors, log the types, disable some, regenerate errors, verify correct errors (not)reported
881 class FilterCase : public BaseCase
882 {
883 public:
884 										FilterCase		(Context&							ctx,
885 														 const char*						name,
886 														 const char*						desc,
887 														 const vector<TestFunctionWrapper>&	errorFuncs);
~FilterCase(void)888 	virtual								~FilterCase		(void) {}
889 
890 	virtual IterateResult				iterate			(void);
891 
892 	virtual void						expectMessage	(GLenum source, GLenum type);
893 
894 protected:
895 	struct MessageFilter
896 	{
MessageFilterdeqp::gles31::Functional::__anon0b3d5a1f0111::FilterCase::MessageFilter897 		MessageFilter() : source(GL_DONT_CARE), type(GL_DONT_CARE), severity(GL_DONT_CARE), enabled(true) {} // Default to enable all
MessageFilterdeqp::gles31::Functional::__anon0b3d5a1f0111::FilterCase::MessageFilter898 		MessageFilter(GLenum source_, GLenum type_, GLenum severity_, const vector<GLuint>& ids_, bool enabled_) : source(source_), type(type_), severity(severity_), ids(ids_), enabled(enabled_) {}
899 
900 		GLenum			source;
901 		GLenum			type;
902 		GLenum			severity;
903 		vector<GLuint>	ids;
904 		bool			enabled;
905 	};
906 
907 	virtual void						callback			(GLenum source, GLenum type, GLuint id, GLenum severity, const string& message);
908 
909 	vector<MessageData>					genMessages			(bool uselog, const string& desc);
910 
911 	vector<MessageFilter>				genFilters			(const vector<MessageData>& messages, const vector<MessageFilter>& initial, deUint32 seed, int iterations) const;
912 	void								applyFilters		(const vector<MessageFilter>& filters) const;
913 	bool								isEnabled			(const vector<MessageFilter>& filters, const MessageData& message) const;
914 
915 	void								verify				(const vector<MessageData>&		refMessages,
916 															 const vector<MessageData>&		filteredMessages,
917 															 const vector<MessageFilter>&	filters);
918 
919 	const vector<TestFunctionWrapper>	m_errorFuncs;
920 
921 	vector<MessageData>*				m_currentErrors;
922 };
923 
FilterCase(Context & ctx,const char * name,const char * desc,const vector<TestFunctionWrapper> & errorFuncs)924 FilterCase::FilterCase (Context&							ctx,
925 						const char*							name,
926 						const char*							desc,
927 						const vector<TestFunctionWrapper>&	errorFuncs)
928 	: BaseCase			(ctx, name, desc)
929 	, m_errorFuncs		(errorFuncs)
930 	, m_currentErrors	(DE_NULL)
931 {
932 }
933 
iterate(void)934 FilterCase::IterateResult FilterCase::iterate (void)
935 {
936 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
937 
938 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
939 
940 	gl.enable(GL_DEBUG_OUTPUT);
941 	gl.enable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
942 	gl.debugMessageCallback(callbackHandle, this);
943 
944 	try
945 	{
946 		gl.debugMessageControl(GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, true);
947 
948 		{
949 			const vector<MessageData>	refMessages		= genMessages(true, "Reference run");
950 			const MessageFilter			baseFilter		(GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, vector<GLuint>(), true);
951 			const deUint32				baseSeed		= deStringHash(getName()) ^ m_testCtx.getCommandLine().getBaseSeed();
952 			const vector<MessageFilter>	filters			= genFilters(refMessages, vector<MessageFilter>(1, baseFilter), baseSeed, 4);
953 			vector<MessageData>			filteredMessages;
954 
955 			applyFilters(filters);
956 
957 			// Generate errors
958 			filteredMessages = genMessages(false, "Filtered run");
959 
960 			// Verify
961 			verify(refMessages, filteredMessages, filters);
962 
963 			if (!isDebugContext() && refMessages.empty())
964 				m_results.addResult(QP_TEST_RESULT_QUALITY_WARNING, "Verification accuracy is lacking without a debug context");
965 		}
966 	}
967 	catch (...)
968 	{
969 		gl.disable(GL_DEBUG_OUTPUT);
970 		gl.debugMessageCallback(DE_NULL, DE_NULL);
971 		throw;
972 	}
973 
974 	gl.disable(GL_DEBUG_OUTPUT);
975 	gl.debugMessageCallback(DE_NULL, DE_NULL);
976 	m_results.setTestContextResult(m_testCtx);
977 
978 	return STOP;
979 }
980 
expectMessage(GLenum source,GLenum type)981 void FilterCase::expectMessage (GLenum source, GLenum type)
982 {
983 	DE_UNREF(source);
984 	DE_UNREF(type);
985 }
986 
callback(GLenum source,GLenum type,GLuint id,GLenum severity,const string & message)987 void FilterCase::callback (GLenum source, GLenum type, GLuint id, GLenum severity, const string& message)
988 {
989 	if (m_currentErrors)
990 		m_currentErrors->push_back(MessageData(MessageID(source, type, id), severity, message));
991 }
992 
genMessages(bool uselog,const string & desc)993 vector<MessageData> FilterCase::genMessages (bool uselog, const string& desc)
994 {
995 	tcu::TestLog&			log			= m_testCtx.getLog();
996 	DebugMessageTestContext	context		= DebugMessageTestContext(*this, m_context.getRenderContext(), m_context.getContextInfo(), log, m_results, uselog);
997 	tcu::ScopedLogSection	section		(log, "message gen", desc);
998 	vector<MessageData>		messages;
999 
1000 	m_currentErrors = &messages;
1001 
1002 	for (int ndx = 0; ndx < int(m_errorFuncs.size()); ndx++)
1003 		m_errorFuncs[ndx].call(context);
1004 
1005 	m_currentErrors = DE_NULL;
1006 
1007 	return messages;
1008 }
1009 
genFilters(const vector<MessageData> & messages,const vector<MessageFilter> & initial,deUint32 seed,int iterations) const1010 vector<FilterCase::MessageFilter> FilterCase::genFilters (const vector<MessageData>& messages, const vector<MessageFilter>& initial, deUint32 seed, int iterations) const
1011 {
1012 	de::Random				rng				(seed ^ deInt32Hash(deStringHash(getName())));
1013 
1014 	set<MessageID>			tempMessageIds;
1015 	set<GLenum>				tempSources;
1016 	set<GLenum>				tempTypes;
1017 	set<GLenum>				tempSeverities;
1018 
1019 	if (messages.empty())
1020 		return initial;
1021 
1022 	for (int ndx = 0; ndx < int(messages.size()); ndx++)
1023 	{
1024 		const MessageData& msg = messages[ndx];
1025 
1026 		tempMessageIds.insert(msg.id);
1027 		tempSources.insert(msg.id.source);
1028 		tempTypes.insert(msg.id.type);
1029 		tempSeverities.insert(msg.severity);
1030 	}
1031 
1032 	{
1033 		// Fetchable by index
1034 		const vector<MessageID> messageIds	(tempMessageIds.begin(), tempMessageIds.end());
1035 		const vector<GLenum>	sources		(tempSources.begin(), tempSources.end());
1036 		const vector<GLenum>	types		(tempTypes.begin(), tempTypes.end());
1037 		const vector<GLenum>	severities	(tempSeverities.begin(), tempSeverities.end());
1038 
1039 		vector<MessageFilter>	filters		= initial;
1040 
1041 		for (int iteration = 0; iteration < iterations; iteration++)
1042 		{
1043 			switch(rng.getInt(0, 8)) // Distribute so that per-message randomization (the default branch) is prevalent
1044 			{
1045 				case 0:
1046 				{
1047 					const GLenum	source	= sources[rng.getInt(0, int(sources.size()-1))];
1048 					const bool		enabled	= rng.getBool();
1049 
1050 					filters.push_back(MessageFilter(source, GL_DONT_CARE, GL_DONT_CARE, vector<GLuint>(), enabled));
1051 					break;
1052 				}
1053 
1054 				case 1:
1055 				{
1056 					const GLenum	type	= types[rng.getUint32()%types.size()];
1057 					const bool		enabled	= rng.getBool();
1058 
1059 					filters.push_back(MessageFilter(GL_DONT_CARE, type, GL_DONT_CARE, vector<GLuint>(), enabled));
1060 					break;
1061 				}
1062 
1063 				case 2:
1064 				{
1065 					const GLenum	severity	= severities[rng.getUint32()%severities.size()];
1066 					const bool		enabled		= rng.getBool();
1067 
1068 					filters.push_back(MessageFilter(GL_DONT_CARE, GL_DONT_CARE, severity, vector<GLuint>(), enabled));
1069 					break;
1070 				}
1071 
1072 				default:
1073 				{
1074 					const int start = rng.getInt(0, int(messageIds.size()));
1075 
1076 					for (int itr = 0; itr < 4; itr++)
1077 					{
1078 						const MessageID&	id		= messageIds[(start+itr)%messageIds.size()];
1079 						const bool			enabled = rng.getBool();
1080 
1081 						filters.push_back(MessageFilter(id.source, id.type, GL_DONT_CARE, vector<GLuint>(1, id.id), enabled));
1082 					}
1083 				}
1084 			}
1085 		}
1086 
1087 		return filters;
1088 	}
1089 }
1090 
applyFilters(const vector<MessageFilter> & filters) const1091 void FilterCase::applyFilters (const vector<MessageFilter>& filters) const
1092 {
1093 	TestLog&					log		= m_testCtx.getLog();
1094 	const tcu::ScopedLogSection	section	(log, "", "Setting message filters");
1095 	const glw::Functions&		gl		= m_context.getRenderContext().getFunctions();
1096 
1097 	for (size_t filterNdx = 0; filterNdx < filters.size(); filterNdx++)
1098 	{
1099 		const MessageFilter& filter = filters[filterNdx];
1100 
1101 		if (filter.ids.empty())
1102 			log << TestLog::Message << "Setting messages with"
1103 				<< " source " << glu::getDebugMessageSourceStr(filter.source)
1104 				<< ", type " << glu::getDebugMessageTypeStr(filter.type)
1105 				<< " and severity " << glu::getDebugMessageSeverityStr(filter.severity)
1106 				<< (filter.enabled ? " to enabled" : " to disabled")
1107 				<< TestLog::EndMessage;
1108 		else
1109 		{
1110 			for (size_t ndx = 0; ndx < filter.ids.size(); ndx++)
1111 				log << TestLog::Message << "Setting message (" << MessageID(filter.source, filter.type, filter.ids[ndx]) << ") to " << (filter.enabled ? "enabled" : "disabled") << TestLog::EndMessage;
1112 		}
1113 
1114 		gl.debugMessageControl(filter.source, filter.type, filter.severity, GLsizei(filter.ids.size()), filter.ids.empty() ? DE_NULL : &filter.ids[0], filter.enabled);
1115 	}
1116 }
1117 
isEnabled(const vector<MessageFilter> & filters,const MessageData & message) const1118 bool FilterCase::isEnabled (const vector<MessageFilter>& filters, const MessageData& message) const
1119 {
1120 	bool retval = true;
1121 
1122 	for (size_t filterNdx = 0; filterNdx < filters.size(); filterNdx++)
1123 	{
1124 		const MessageFilter&	filter	= filters[filterNdx];
1125 
1126 		if (filter.ids.empty())
1127 		{
1128 			if (filter.source != GL_DONT_CARE && filter.source != message.id.source)
1129 				continue;
1130 
1131 			if (filter.type != GL_DONT_CARE && filter.type != message.id.type)
1132 				continue;
1133 
1134 			if (filter.severity != GL_DONT_CARE && filter.severity != message.severity)
1135 				continue;
1136 		}
1137 		else
1138 		{
1139 			DE_ASSERT(filter.source != GL_DONT_CARE);
1140 			DE_ASSERT(filter.type != GL_DONT_CARE);
1141 			DE_ASSERT(filter.severity == GL_DONT_CARE);
1142 
1143 			if (filter.source != message.id.source || filter.type != message.id.type)
1144 				continue;
1145 
1146 			if (!de::contains(filter.ids.begin(), filter.ids.end(), message.id.id))
1147 				continue;
1148 		}
1149 
1150 		retval = filter.enabled;
1151 	}
1152 
1153 	return retval;
1154 }
1155 
1156 struct MessageMeta
1157 {
1158 	int		refCount;
1159 	int		resCount;
1160 	GLenum	severity;
1161 
MessageMetadeqp::gles31::Functional::__anon0b3d5a1f0111::MessageMeta1162 	MessageMeta (void) : refCount(0), resCount(0), severity(GL_NONE) {}
1163 };
1164 
verify(const vector<MessageData> & refMessages,const vector<MessageData> & resMessages,const vector<MessageFilter> & filters)1165 void FilterCase::verify (const vector<MessageData>& refMessages, const vector<MessageData>& resMessages, const vector<MessageFilter>& filters)
1166 {
1167 	TestLog&						log		= m_testCtx.getLog();
1168 	map<MessageID, MessageMeta>		counts;
1169 
1170 	log << TestLog::Section("verification", "Verifying");
1171 
1172 	// Gather message counts & severities, report severity mismatches if found
1173 	for (size_t refNdx = 0; refNdx < refMessages.size(); refNdx++)
1174 	{
1175 		const MessageData&	msg  = refMessages[refNdx];
1176 		MessageMeta&		meta = counts[msg.id];
1177 
1178 		if (meta.severity != GL_NONE && meta.severity != msg.severity)
1179 		{
1180 			log << TestLog::Message << "A message has variable severity between instances: (" << msg.id << ") with severity "
1181 				<< glu::getDebugMessageSeverityStr(meta.severity) << " and " << glu::getDebugMessageSeverityStr(msg.severity) << TestLog::EndMessage;
1182 			m_results.addResult(QP_TEST_RESULT_FAIL, "Message severity changed between instances of the same message");
1183 		}
1184 
1185 		meta.refCount++;
1186 		meta.severity = msg.severity;
1187 	}
1188 
1189 	for (size_t resNdx = 0; resNdx < resMessages.size(); resNdx++)
1190 	{
1191 		const MessageData&	msg  = resMessages[resNdx];
1192 		MessageMeta&		meta = counts[msg.id];
1193 
1194 		if (meta.severity != GL_NONE && meta.severity != msg.severity)
1195 		{
1196 			log << TestLog::Message << "A message has variable severity between instances: (" << msg.id << ") with severity "
1197 				<< glu::getDebugMessageSeverityStr(meta.severity) << " and " << glu::getDebugMessageSeverityStr(msg.severity) << TestLog::EndMessage;
1198 			m_results.addResult(QP_TEST_RESULT_FAIL, "Message severity changed between instances of the same message");
1199 		}
1200 
1201 		meta.resCount++;
1202 		meta.severity = msg.severity;
1203 	}
1204 
1205 	for (map<MessageID, MessageMeta>::const_iterator itr = counts.begin(); itr != counts.end(); itr++)
1206 	{
1207 		const MessageID&	id			= itr->first;
1208 		const GLenum		severity	= itr->second.severity;
1209 
1210 		const int			refCount	= itr->second.refCount;
1211 		const int			resCount	= itr->second.resCount;
1212 		const bool			enabled		= isEnabled(filters, MessageData(id, severity, ""));
1213 
1214 		VerificationResult	result		= verifyMessageCount(id, severity, refCount, resCount, enabled);
1215 
1216 		log << TestLog::Message << result.logMessage << TestLog::EndMessage;
1217 
1218 		if (result.result != QP_TEST_RESULT_PASS)
1219 			m_results.addResult(result.result, result.resultMessage);
1220 	}
1221 
1222 	log << TestLog::EndSection;
1223 }
1224 
1225 // Filter case that uses debug groups
1226 class GroupFilterCase : public FilterCase
1227 {
1228 public:
1229 							GroupFilterCase		(Context&							ctx,
1230 												 const char*						name,
1231 												 const char*						desc,
1232 												 const vector<TestFunctionWrapper>&	errorFuncs);
~GroupFilterCase(void)1233 	virtual					~GroupFilterCase	(void) {}
1234 
1235 	virtual IterateResult	iterate				(void);
1236 };
1237 
GroupFilterCase(Context & ctx,const char * name,const char * desc,const vector<TestFunctionWrapper> & errorFuncs)1238 GroupFilterCase::GroupFilterCase (Context&								ctx,
1239 								  const char*							name,
1240 								  const char*							desc,
1241 								  const vector<TestFunctionWrapper>&	errorFuncs)
1242 	: FilterCase(ctx, name, desc, errorFuncs)
1243 {
1244 }
1245 
1246 template<typename T>
join(const vector<T> & a,const vector<T> & b)1247 vector<T> join(const vector<T>& a, const vector<T>&b)
1248 {
1249 	vector<T> retval;
1250 
1251 	retval.reserve(a.size()+b.size());
1252 	retval.insert(retval.end(), a.begin(), a.end());
1253 	retval.insert(retval.end(), b.begin(), b.end());
1254 	return retval;
1255 }
1256 
iterate(void)1257 GroupFilterCase::IterateResult GroupFilterCase::iterate (void)
1258 {
1259 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
1260 
1261 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
1262 	tcu::TestLog&			log			= m_testCtx.getLog();
1263 
1264 	gl.enable(GL_DEBUG_OUTPUT);
1265 	gl.enable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1266 	gl.debugMessageCallback(callbackHandle, this);
1267 
1268 	try
1269 	{
1270 		gl.debugMessageControl(GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, true);
1271 
1272 		{
1273 
1274 			// Generate reference (all errors)
1275 			const vector<MessageData>	refMessages		= genMessages(true, "Reference run");
1276 			const deUint32				baseSeed		= deStringHash(getName()) ^ m_testCtx.getCommandLine().getBaseSeed();
1277 			const MessageFilter			baseFilter		 (GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, vector<GLuint>(), true);
1278 			const vector<MessageFilter>	filter0			= genFilters(refMessages, vector<MessageFilter>(1, baseFilter), baseSeed, 4);
1279 			vector<MessageData>			resMessages0;
1280 
1281 			applyFilters(filter0);
1282 
1283 			resMessages0 = genMessages(false, "Filtered run, default debug group");
1284 
1285 			// Initial verification
1286 			verify(refMessages, resMessages0, filter0);
1287 
1288 			{
1289 				// Generate reference (filters inherited from parent)
1290 				const vector<MessageFilter> filter1base		= genFilters(refMessages, vector<MessageFilter>(), baseSeed ^ 0xDEADBEEF, 4);
1291 				const vector<MessageFilter>	filter1full		= join(filter0, filter1base);
1292 				tcu::ScopedLogSection		section1		(log, "", "Pushing Debug Group");
1293 				vector<MessageData>			resMessages1;
1294 
1295 				gl.pushDebugGroup(GL_DEBUG_SOURCE_APPLICATION, 1, -1, "Test Group");
1296 				applyFilters(filter1base);
1297 
1298 				// First nested verification
1299 				resMessages1 = genMessages(false, "Filtered run, pushed one debug group");
1300 				verify(refMessages, resMessages1, filter1full);
1301 
1302 				{
1303 					// Generate reference (filters iherited again)
1304 					const vector<MessageFilter>	filter2base		= genFilters(refMessages, vector<MessageFilter>(), baseSeed ^ 0x43211234, 4);
1305 					const vector<MessageFilter>	filter2full		= join(filter1full, filter2base);
1306 					tcu::ScopedLogSection		section2		(log, "", "Pushing Debug Group");
1307 					vector<MessageData>			resMessages2;
1308 
1309 					gl.pushDebugGroup(GL_DEBUG_SOURCE_APPLICATION, 1, -1, "Nested Test Group");
1310 					applyFilters(filter2base);
1311 
1312 					// Second nested verification
1313 					resMessages2 = genMessages(false, "Filtered run, pushed two debug groups");
1314 					verify(refMessages, resMessages2, filter2full);
1315 
1316 					gl.popDebugGroup();
1317 				}
1318 
1319 				// First restore verification
1320 				resMessages1 = genMessages(false, "Filtered run, popped second debug group");
1321 				verify(refMessages, resMessages1, filter1full);
1322 
1323 				gl.popDebugGroup();
1324 			}
1325 
1326 			// restore verification
1327 			resMessages0 = genMessages(false, "Filtered run, popped first debug group");
1328 			verify(refMessages, resMessages0, filter0);
1329 
1330 			if (!isDebugContext() && refMessages.empty())
1331 				m_results.addResult(QP_TEST_RESULT_QUALITY_WARNING, "Verification accuracy is lacking without a debug context");
1332 		}
1333 	}
1334 	catch (...)
1335 	{
1336 		gl.disable(GL_DEBUG_OUTPUT);
1337 		gl.debugMessageCallback(DE_NULL, DE_NULL);
1338 		throw;
1339 	}
1340 
1341 	gl.disable(GL_DEBUG_OUTPUT);
1342 	gl.debugMessageCallback(DE_NULL, DE_NULL);
1343 	m_results.setTestContextResult(m_testCtx);
1344 	return STOP;
1345 }
1346 
1347 // Basic grouping functionality
1348 class GroupCase : public BaseCase
1349 {
1350 public:
1351 							GroupCase	(Context&				ctx,
1352 										 const char*			name,
1353 										 const char*			desc);
~GroupCase()1354 	virtual					~GroupCase	() {}
1355 
1356 	virtual IterateResult	iterate		(void);
1357 
1358 private:
1359 	virtual void			callback	(GLenum source, GLenum type, GLuint id, GLenum severity, const string& message);
1360 
1361 	MessageData				m_lastMessage;
1362 };
1363 
GroupCase(Context & ctx,const char * name,const char * desc)1364 GroupCase::GroupCase (Context&				ctx,
1365 					  const char*			name,
1366 					  const char*			desc)
1367 	: BaseCase(ctx, name, desc)
1368 {
1369 }
1370 
iterate(void)1371 GroupCase::IterateResult GroupCase::iterate (void)
1372 {
1373 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
1374 
1375 	const glw::Functions&	gl		= m_context.getRenderContext().getFunctions();
1376 	tcu::TestLog&			log		= m_testCtx.getLog();
1377 	glu::CallLogWrapper		wrapper	(gl, log);
1378 
1379 	gl.enable(GL_DEBUG_OUTPUT);
1380 	gl.enable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1381 	gl.debugMessageControl(GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, false); // disable all
1382 	gl.debugMessageControl(GL_DEBUG_SOURCE_API, GL_DEBUG_TYPE_ERROR, GL_DONT_CARE, 0, DE_NULL, true); // enable API errors
1383 	gl.debugMessageControl(GL_DEBUG_SOURCE_APPLICATION, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, true); // enable application messages
1384 	gl.debugMessageControl(GL_DEBUG_SOURCE_THIRD_PARTY, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, true); // enable third party messages
1385 	gl.debugMessageCallback(callbackHandle, this);
1386 
1387 	wrapper.enableLogging(true);
1388 	wrapper.glPushDebugGroup(GL_DEBUG_SOURCE_APPLICATION, 1234, -1, "Pushed debug stack");
1389 	verifyMessage(m_lastMessage, GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PUSH_GROUP, 1234, GL_DEBUG_SEVERITY_NOTIFICATION);
1390 	wrapper.glPopDebugGroup();
1391 	verifyMessage(m_lastMessage, GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_POP_GROUP, 1234, GL_DEBUG_SEVERITY_NOTIFICATION);
1392 
1393 	wrapper.glPushDebugGroup(GL_DEBUG_SOURCE_THIRD_PARTY, 4231, -1, "Pushed debug stack");
1394 	verifyMessage(m_lastMessage, GL_DEBUG_SOURCE_THIRD_PARTY, GL_DEBUG_TYPE_PUSH_GROUP, 4231, GL_DEBUG_SEVERITY_NOTIFICATION);
1395 	wrapper.glPopDebugGroup();
1396 	verifyMessage(m_lastMessage, GL_DEBUG_SOURCE_THIRD_PARTY, GL_DEBUG_TYPE_POP_GROUP, 4231, GL_DEBUG_SEVERITY_NOTIFICATION);
1397 
1398 	gl.debugMessageCallback(DE_NULL, DE_NULL);
1399 	gl.disable(GL_DEBUG_OUTPUT);
1400 
1401 	m_results.setTestContextResult(m_testCtx);
1402 
1403 	return STOP;
1404 }
1405 
callback(GLenum source,GLenum type,GLuint id,GLenum severity,const string & message)1406 void GroupCase::callback (GLenum source, GLenum type, GLuint id, GLenum severity, const string& message)
1407 {
1408 	m_lastMessage = MessageData(MessageID(source, type, id), severity, message);
1409 }
1410 
1411 // Asynchronous debug output
1412 class AsyncCase : public BaseCase
1413 {
1414 public:
1415 										AsyncCase			(Context&							ctx,
1416 															 const char*						name,
1417 															 const char*						desc,
1418 															 const vector<TestFunctionWrapper>&	errorFuncs,
1419 															 bool								useCallbacks);
~AsyncCase(void)1420 	virtual								~AsyncCase			(void) {}
1421 
1422 	virtual IterateResult				iterate				(void);
1423 
1424 	virtual void						expectMessage		(glw::GLenum source, glw::GLenum type);
1425 
1426 private:
1427 	struct MessageCount
1428 	{
1429 		int received;
1430 		int expected;
1431 
MessageCountdeqp::gles31::Functional::__anon0b3d5a1f0111::AsyncCase::MessageCount1432 		MessageCount(void) : received(0), expected(0) {}
1433 	};
1434 	typedef map<MessageID, MessageCount> MessageCounter;
1435 
1436 	enum VerifyState
1437 	{
1438 		VERIFY_PASS = 0,
1439 		VERIFY_MINIMUM,
1440 		VERIFY_FAIL,
1441 
1442 		VERIFY_LAST
1443 	};
1444 
1445 	virtual void						callback			(glw::GLenum source, glw::GLenum type, glw::GLuint id, glw::GLenum severity, const std::string& message);
1446 	VerifyState							verify				(bool uselog);
1447 	void								fetchLogMessages	(void);
1448 
1449 	const vector<TestFunctionWrapper>	m_errorFuncs;
1450 	const bool							m_useCallbacks;
1451 
1452 	MessageCounter						m_counts;
1453 
1454 	de::Mutex							m_mutex;
1455 };
1456 
AsyncCase(Context & ctx,const char * name,const char * desc,const vector<TestFunctionWrapper> & errorFuncs,bool useCallbacks)1457 AsyncCase::AsyncCase (Context&								ctx,
1458 					  const char*							name,
1459 					  const char*							desc,
1460 					  const vector<TestFunctionWrapper>&	errorFuncs,
1461 					  bool									useCallbacks)
1462 	: BaseCase			(ctx, name, desc)
1463 	, m_errorFuncs		(errorFuncs)
1464 	, m_useCallbacks	(useCallbacks)
1465 {
1466 }
1467 
iterate(void)1468 AsyncCase::IterateResult AsyncCase::iterate (void)
1469 {
1470 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
1471 
1472 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
1473 	tcu::TestLog&			log			= m_testCtx.getLog();
1474 	DebugMessageTestContext	context		= DebugMessageTestContext(*this, m_context.getRenderContext(), m_context.getContextInfo(), log, m_results, true);
1475 	const int				maxWait		= 10000; // ms
1476 	const int				warnWait	= 100;
1477 
1478 	// Clear log from earlier messages
1479 	{
1480 		GLint numMessages = 0;
1481 		gl.getIntegerv(GL_DEBUG_LOGGED_MESSAGES, &numMessages);
1482 		gl.getDebugMessageLog(numMessages, 0, DE_NULL, DE_NULL, DE_NULL, DE_NULL, DE_NULL, DE_NULL);
1483 	}
1484 
1485 	gl.enable(GL_DEBUG_OUTPUT);
1486 	gl.enable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1487 	gl.debugMessageControl(GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, 0, DE_NULL, false);
1488 
1489 	// Some messages could be dependent on the value of DEBUG_OUTPUT_SYNCHRONOUS so only use API errors which should be generated in all cases
1490 	gl.debugMessageControl(GL_DEBUG_SOURCE_API, GL_DEBUG_TYPE_ERROR, GL_DONT_CARE, 0, DE_NULL, true);
1491 
1492 	if (m_useCallbacks) // will use log otherwise
1493 		gl.debugMessageCallback(callbackHandle, this);
1494 	else
1495 		gl.debugMessageCallback(DE_NULL, DE_NULL);
1496 
1497 	// Reference run (synchoronous)
1498 	{
1499 		tcu::ScopedLogSection section(log, "reference run", "Reference run (synchronous)");
1500 
1501 		for (int ndx = 0; ndx < int(m_errorFuncs.size()); ndx++)
1502 			m_errorFuncs[ndx].call(context);
1503 	}
1504 
1505 	if (m_counts.empty())
1506 	{
1507 		if (!isDebugContext())
1508 			m_results.addResult(QP_TEST_RESULT_QUALITY_WARNING, "Need debug context to guarantee implementation behaviour (see command line options)");
1509 
1510 		log << TestLog::Message << "Reference run produced no messages, nothing to verify" << TestLog::EndMessage;
1511 
1512 		gl.debugMessageCallback(DE_NULL, DE_NULL);
1513 		gl.disable(GL_DEBUG_OUTPUT);
1514 
1515 		m_results.setTestContextResult(m_testCtx);
1516 		return STOP;
1517 	}
1518 
1519 	for (MessageCounter::iterator itr = m_counts.begin(); itr != m_counts.end(); itr++)
1520 	{
1521 		itr->second.expected = itr->second.received;
1522 		itr->second.received = 0;
1523 	}
1524 
1525 	gl.disable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1526 
1527 	// Result run (async)
1528 	for (int ndx = 0; ndx < int(m_errorFuncs.size()); ndx++)
1529 		m_errorFuncs[ndx].call(context);
1530 
1531 	// Repatedly try verification, new results may be added to m_receivedMessages at any time
1532 	{
1533 		tcu::ScopedLogSection	section			(log, "result run", "Result run (asynchronous)");
1534 		VerifyState				lastTimelyState = VERIFY_FAIL;
1535 
1536 		for (int waited = 0;;)
1537 		{
1538 			const VerifyState	pass = verify(false);
1539 			const int			wait = de::max(50, waited>>2);
1540 
1541 			// Pass (possibly due to time limit)
1542 			if (pass == VERIFY_PASS || (pass == VERIFY_MINIMUM && waited >= maxWait))
1543 			{
1544 				verify(true); // log
1545 
1546 				// State changed late
1547 				if (waited >= warnWait && lastTimelyState != pass)
1548 					m_results.addResult(QP_TEST_RESULT_QUALITY_WARNING, "Async messages were returned to application somewhat slowly");
1549 
1550 				log << TestLog::Message << "Passed after ~" << waited << "ms of waiting" << TestLog::EndMessage;
1551 				break;
1552 			}
1553 			// fail
1554 			else if (waited >= maxWait)
1555 			{
1556 				verify(true); // log
1557 
1558 				log << TestLog::Message << "Waited for ~" << waited << "ms without getting all expected messages" << TestLog::EndMessage;
1559 				m_results.addResult(QP_TEST_RESULT_FAIL, "Async messages were not returned to application within a reasonable timeframe");
1560 				break;
1561 			}
1562 
1563 			if (waited < warnWait)
1564 				lastTimelyState = pass;
1565 
1566 			deSleep(wait);
1567 			waited += wait;
1568 
1569 			if (!m_useCallbacks)
1570 				fetchLogMessages();
1571 		}
1572 	}
1573 
1574 	gl.debugMessageCallback(DE_NULL, DE_NULL);
1575 
1576 	gl.disable(GL_DEBUG_OUTPUT);
1577 	m_results.setTestContextResult(m_testCtx);
1578 
1579 	return STOP;
1580 }
1581 
expectMessage(GLenum source,GLenum type)1582 void AsyncCase::expectMessage (GLenum source, GLenum type)
1583 {
1584 	// Good time to clean up the queue as this should be called after most messages are generated
1585 	if (!m_useCallbacks)
1586 		fetchLogMessages();
1587 
1588 	DE_UNREF(source);
1589 	DE_UNREF(type);
1590 }
1591 
callback(GLenum source,GLenum type,GLuint id,GLenum severity,const string & message)1592 void AsyncCase::callback (GLenum source, GLenum type, GLuint id, GLenum severity, const string& message)
1593 {
1594 	DE_ASSERT(m_useCallbacks);
1595 	DE_UNREF(severity);
1596 	DE_UNREF(message);
1597 
1598 	de::ScopedLock lock(m_mutex);
1599 
1600 	m_counts[MessageID(source, type, id)].received++;
1601 }
1602 
1603 // Note that we can never guarantee getting all messages back when using logs/fetching as the GL may create more than its log size limit during an arbitrary period of time
fetchLogMessages(void)1604 void AsyncCase::fetchLogMessages (void)
1605 {
1606 	const glw::Functions&	gl		= m_context.getRenderContext().getFunctions();
1607 	GLint					numMsg	= 0;
1608 
1609 	gl.getIntegerv(GL_DEBUG_LOGGED_MESSAGES, &numMsg);
1610 
1611 	for(int msgNdx = 0; msgNdx < numMsg; msgNdx++)
1612 	{
1613 		int			msgLen = 0;
1614 		MessageData msg;
1615 
1616 		gl.getIntegerv(GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH, &msgLen);
1617 
1618 		TCU_CHECK_MSG(msgLen >= 0, "Negative message length");
1619 		TCU_CHECK_MSG(msgLen < 100000, "Excessively long message");
1620 
1621 		msg.message.resize(msgLen);
1622 		gl.getDebugMessageLog(1, msgLen, &msg.id.source, &msg.id.type, &msg.id.id, &msg.severity, &msgLen, &msg.message[0]);
1623 
1624 		{
1625 			const de::ScopedLock lock(m_mutex); // Don't block during API call
1626 
1627 			m_counts[MessageID(msg.id)].received++;
1628 		}
1629 	}
1630 }
1631 
verify(bool uselog)1632 AsyncCase::VerifyState AsyncCase::verify (bool uselog)
1633 {
1634 	using std::map;
1635 
1636 	VerifyState			retval		= VERIFY_PASS;
1637 	TestLog&			log			= m_testCtx.getLog();
1638 
1639 	const de::ScopedLock lock(m_mutex);
1640 
1641 	for (map<MessageID, MessageCount>::const_iterator itr = m_counts.begin(); itr != m_counts.end(); itr++)
1642 	{
1643 		const MessageID&	id			= itr->first;
1644 
1645 		const int			refCount	= itr->second.expected;
1646 		const int			resCount	= itr->second.received;
1647 		const bool			enabled		= true;
1648 
1649 		VerificationResult	result		= verifyMessageCount(id, GL_DONT_CARE, refCount, resCount, enabled);
1650 
1651 		if (uselog)
1652 			log << TestLog::Message << result.logMessage << TestLog::EndMessage;
1653 
1654 		if (result.result == QP_TEST_RESULT_FAIL)
1655 			retval = VERIFY_FAIL;
1656 		else if (result.result != QP_TEST_RESULT_PASS && retval == VERIFY_PASS)
1657 			retval = VERIFY_MINIMUM;
1658 	}
1659 
1660 	return retval;
1661 }
1662 
1663 // Tests debug labels
1664 class LabelCase : public TestCase
1665 {
1666 public:
1667 							LabelCase	(Context&				ctx,
1668 										 const char*			name,
1669 										 const char*			desc,
1670 										 GLenum					identifier);
~LabelCase(void)1671 	virtual					~LabelCase	(void) {}
1672 
1673 	virtual IterateResult	iterate		(void);
1674 
1675 private:
1676 	GLenum					m_identifier;
1677 };
1678 
LabelCase(Context & ctx,const char * name,const char * desc,GLenum identifier)1679 LabelCase::LabelCase (Context&		ctx,
1680 					  const char*			name,
1681 					  const char*			desc,
1682 					  GLenum				identifier)
1683 	: TestCase		(ctx, name, desc)
1684 	, m_identifier	(identifier)
1685 {
1686 }
1687 
iterate(void)1688 LabelCase::IterateResult LabelCase::iterate (void)
1689 {
1690 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
1691 
1692 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
1693 	const char*	const		msg			= "This is a debug label";
1694 	GLuint					object		= 0;
1695 	int						outlen		= -1;
1696 	char					buffer[64];
1697 
1698 	switch(m_identifier)
1699 	{
1700 		case GL_BUFFER:
1701 			gl.genBuffers(1, &object);
1702 			gl.bindBuffer(GL_ARRAY_BUFFER, object);
1703 			gl.bindBuffer(GL_ARRAY_BUFFER, 0);
1704 			break;
1705 
1706 		case GL_SHADER:
1707 			object = gl.createShader(GL_FRAGMENT_SHADER);
1708 			break;
1709 
1710 		case GL_PROGRAM:
1711 			object = gl.createProgram();
1712 			break;
1713 
1714 		case GL_QUERY:
1715 			gl.genQueries(1, &object);
1716 			gl.beginQuery(GL_ANY_SAMPLES_PASSED, object); // Create
1717 			gl.endQuery(GL_ANY_SAMPLES_PASSED); // Cleanup
1718 			break;
1719 
1720 		case GL_PROGRAM_PIPELINE:
1721 			gl.genProgramPipelines(1, &object);
1722 			gl.bindProgramPipeline(object); // Create
1723 			gl.bindProgramPipeline(0); // Cleanup
1724 			break;
1725 
1726 		case GL_TRANSFORM_FEEDBACK:
1727 			gl.genTransformFeedbacks(1, &object);
1728 			gl.bindTransformFeedback(GL_TRANSFORM_FEEDBACK, object);
1729 			gl.bindTransformFeedback(GL_TRANSFORM_FEEDBACK, 0);
1730 			break;
1731 
1732 		case GL_SAMPLER:
1733 			gl.genSamplers(1, &object);
1734 			gl.bindSampler(0, object);
1735 			gl.bindSampler(0, 0);
1736 			break;
1737 
1738 		case GL_TEXTURE:
1739 			gl.genTextures(1, &object);
1740 			gl.bindTexture(GL_TEXTURE_2D, object);
1741 			gl.bindTexture(GL_TEXTURE_2D, 0);
1742 			break;
1743 
1744 		case GL_RENDERBUFFER:
1745 			gl.genRenderbuffers(1, &object);
1746 			gl.bindRenderbuffer(GL_RENDERBUFFER, object);
1747 			gl.bindRenderbuffer(GL_RENDERBUFFER, 0);
1748 			break;
1749 
1750 		case GL_FRAMEBUFFER:
1751 			gl.genFramebuffers(1, &object);
1752 			gl.bindFramebuffer(GL_DRAW_FRAMEBUFFER, object);
1753 			gl.bindFramebuffer(GL_DRAW_FRAMEBUFFER, m_context.getRenderContext().getDefaultFramebuffer());
1754 			break;
1755 
1756 		default:
1757 			DE_FATAL("Invalid identifier");
1758 	}
1759 
1760 	gl.objectLabel(m_identifier, object, -1, msg);
1761 
1762 	deMemset(buffer, 'X', sizeof(buffer));
1763 	gl.getObjectLabel(m_identifier, object, sizeof(buffer), &outlen, buffer);
1764 
1765 	if (outlen == 0)
1766 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Failed to query debug label from object");
1767 	else if (deStringEqual(msg, buffer))
1768 	{
1769 		m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
1770 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
1771 	}
1772 	else
1773 	{
1774 		buffer[63] = '\0'; // make sure buffer is null terminated before printing
1775 		m_testCtx.getLog() << TestLog::Message << "Query returned wrong string: expected \"" << msg << "\" but got \"" << buffer << "\"" << TestLog::EndMessage;
1776 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Query returned wrong label");
1777 	}
1778 
1779 	switch(m_identifier)
1780 	{
1781 		case GL_BUFFER:				gl.deleteBuffers(1, &object);				break;
1782 		case GL_SHADER:				gl.deleteShader(object);					break;
1783 		case GL_PROGRAM:			gl.deleteProgram(object);					break;
1784 		case GL_QUERY:				gl.deleteQueries(1, &object);				break;
1785 		case GL_PROGRAM_PIPELINE:	gl.deleteProgramPipelines(1, &object);		break;
1786 		case GL_TRANSFORM_FEEDBACK:	gl.deleteTransformFeedbacks(1, &object);	break;
1787 		case GL_SAMPLER:			gl.deleteSamplers(1, &object);				break;
1788 		case GL_TEXTURE:			gl.deleteTextures(1, &object);				break;
1789 		case GL_RENDERBUFFER:		gl.deleteRenderbuffers(1, &object);			break;
1790 		case GL_FRAMEBUFFER:		gl.deleteFramebuffers(1, &object);			break;
1791 
1792 		default:
1793 			DE_FATAL("Invalid identifier");
1794 	}
1795 
1796 	return STOP;
1797 }
1798 
1799 
DebugMessageTestContext(BaseCase & host,glu::RenderContext & renderCtx,const glu::ContextInfo & ctxInfo,tcu::TestLog & log,tcu::ResultCollector & results,bool enableLog)1800 DebugMessageTestContext::DebugMessageTestContext (BaseCase&					host,
1801 												  glu::RenderContext&		renderCtx,
1802 												  const glu::ContextInfo&	ctxInfo,
1803 												  tcu::TestLog&				log,
1804 												  tcu::ResultCollector&		results,
1805 												  bool						enableLog)
1806 	: NegativeTestContext	(host, renderCtx, ctxInfo, log, results, enableLog)
1807 	, m_debugHost			(host)
1808 {
1809 }
1810 
~DebugMessageTestContext(void)1811 DebugMessageTestContext::~DebugMessageTestContext (void)
1812 {
1813 }
1814 
expectMessage(GLenum source,GLenum type)1815 void DebugMessageTestContext::expectMessage (GLenum source, GLenum type)
1816 {
1817 	m_debugHost.expectMessage(source, type);
1818 }
1819 
1820 class SyncLabelCase : public TestCase
1821 {
1822 public:
1823 							SyncLabelCase	(Context& ctx, const char* name, const char* desc);
1824 	virtual IterateResult	iterate			(void);
1825 };
1826 
SyncLabelCase(Context & ctx,const char * name,const char * desc)1827 SyncLabelCase::SyncLabelCase (Context& ctx, const char* name, const char* desc)
1828 	: TestCase(ctx, name, desc)
1829 {
1830 }
1831 
iterate(void)1832 SyncLabelCase::IterateResult SyncLabelCase::iterate (void)
1833 {
1834 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
1835 
1836 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
1837 	const char*	const		msg			= "This is a debug label";
1838 	int						outlen		= -1;
1839 	char					buffer[64];
1840 
1841 	glw::GLsync				sync		= gl.fenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
1842 	GLU_EXPECT_NO_ERROR(gl.getError(), "fenceSync");
1843 
1844 	gl.objectPtrLabel(sync, -1, msg);
1845 
1846 	deMemset(buffer, 'X', sizeof(buffer));
1847 	gl.getObjectPtrLabel(sync, sizeof(buffer), &outlen, buffer);
1848 
1849 	if (outlen == 0)
1850 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Failed to query debug label from object");
1851 	else if (deStringEqual(msg, buffer))
1852 	{
1853 		m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
1854 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
1855 	}
1856 	else
1857 	{
1858 		buffer[63] = '\0'; // make sure buffer is null terminated before printing
1859 		m_testCtx.getLog() << TestLog::Message << "Query returned wrong string: expected \"" << msg << "\" but got \"" << buffer << "\"" << TestLog::EndMessage;
1860 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Query returned wrong label");
1861 	}
1862 
1863 	gl.deleteSync(sync);
1864 
1865 	return STOP;
1866 }
1867 
1868 class InitialLabelCase : public TestCase
1869 {
1870 public:
1871 							InitialLabelCase	(Context& ctx, const char* name, const char* desc);
1872 	virtual IterateResult	iterate				(void);
1873 };
1874 
InitialLabelCase(Context & ctx,const char * name,const char * desc)1875 InitialLabelCase::InitialLabelCase (Context& ctx, const char* name, const char* desc)
1876 	: TestCase(ctx, name, desc)
1877 {
1878 }
1879 
iterate(void)1880 InitialLabelCase::IterateResult InitialLabelCase::iterate (void)
1881 {
1882 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
1883 
1884 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
1885 	tcu::ResultCollector	result		(m_testCtx.getLog(), " // ERROR: ");
1886 	int						outlen		= -1;
1887 	GLuint					shader;
1888 	glw::GLsync				sync;
1889 	char					buffer[64];
1890 
1891 	sync = gl.fenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
1892 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "fenceSync");
1893 
1894 	shader = gl.createShader(GL_FRAGMENT_SHADER);
1895 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "createShader");
1896 
1897 	{
1898 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Shader", "Shader object");
1899 		m_testCtx.getLog() << TestLog::Message << "Querying initial value" << TestLog::EndMessage;
1900 
1901 		buffer[0] = 'X';
1902 		outlen = -1;
1903 		gl.getObjectLabel(GL_SHADER, shader, sizeof(buffer), &outlen, buffer);
1904 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
1905 
1906 		if (outlen != 0)
1907 			result.fail("'length' was not zero, got " + de::toString(outlen));
1908 		else if (buffer[0] != '\0')
1909 			result.fail("label was not null terminated");
1910 		else
1911 			m_testCtx.getLog() << TestLog::Message << "Got 0-sized null-terminated string." << TestLog::EndMessage;
1912 	}
1913 
1914 	{
1915 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Sync", "Sync object");
1916 		m_testCtx.getLog() << TestLog::Message << "Querying initial value" << TestLog::EndMessage;
1917 
1918 		buffer[0] = 'X';
1919 		outlen = -1;
1920 		gl.getObjectPtrLabel(sync, sizeof(buffer), &outlen, buffer);
1921 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
1922 
1923 		if (outlen != 0)
1924 			result.fail("'length' was not zero, got " + de::toString(outlen));
1925 		else if (buffer[0] != '\0')
1926 			result.fail("label was not null terminated");
1927 		else
1928 			m_testCtx.getLog() << TestLog::Message << "Got 0-sized null-terminated string." << TestLog::EndMessage;
1929 	}
1930 
1931 	gl.deleteShader(shader);
1932 	gl.deleteSync(sync);
1933 
1934 	result.setTestContextResult(m_testCtx);
1935 	return STOP;
1936 }
1937 
1938 class ClearLabelCase : public TestCase
1939 {
1940 public:
1941 							ClearLabelCase		(Context& ctx, const char* name, const char* desc);
1942 	virtual IterateResult	iterate				(void);
1943 };
1944 
ClearLabelCase(Context & ctx,const char * name,const char * desc)1945 ClearLabelCase::ClearLabelCase (Context& ctx, const char* name, const char* desc)
1946 	: TestCase(ctx, name, desc)
1947 {
1948 }
1949 
iterate(void)1950 ClearLabelCase::IterateResult ClearLabelCase::iterate (void)
1951 {
1952 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
1953 
1954 	static const struct
1955 	{
1956 		const char*	description;
1957 		int			length;
1958 	} s_clearMethods[] =
1959 	{
1960 		{ " with NULL label and 0 length",			0	},
1961 		{ " with NULL label and 1 length",			1	},
1962 		{ " with NULL label and negative length",	-1	},
1963 	};
1964 
1965 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
1966 	tcu::ResultCollector	result		(m_testCtx.getLog(), " // ERROR: ");
1967 	const char*	const		msg			= "This is a debug label";
1968 	int						outlen		= -1;
1969 	GLuint					shader;
1970 	glw::GLsync				sync;
1971 	char					buffer[64];
1972 
1973 	sync = gl.fenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
1974 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "fenceSync");
1975 
1976 	shader = gl.createShader(GL_FRAGMENT_SHADER);
1977 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "createShader");
1978 
1979 	{
1980 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Shader", "Shader object");
1981 
1982 		for (int methodNdx = 0; methodNdx < DE_LENGTH_OF_ARRAY(s_clearMethods); ++methodNdx)
1983 		{
1984 			m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\"" << TestLog::EndMessage;
1985 			gl.objectLabel(GL_SHADER, shader, -2,  msg);
1986 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectLabel");
1987 
1988 			m_testCtx.getLog() << TestLog::Message << "Clearing label " << s_clearMethods[methodNdx].description << TestLog::EndMessage;
1989 			gl.objectLabel(GL_SHADER, shader, s_clearMethods[methodNdx].length, DE_NULL);
1990 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectLabel");
1991 
1992 			m_testCtx.getLog() << TestLog::Message << "Querying label" << TestLog::EndMessage;
1993 			buffer[0] = 'X';
1994 			outlen = -1;
1995 			gl.getObjectLabel(GL_SHADER, shader, sizeof(buffer), &outlen, buffer);
1996 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
1997 
1998 			if (outlen != 0)
1999 				result.fail("'length' was not zero, got " + de::toString(outlen));
2000 			else if (buffer[0] != '\0')
2001 				result.fail("label was not null terminated");
2002 			else
2003 				m_testCtx.getLog() << TestLog::Message << "Got 0-sized null-terminated string." << TestLog::EndMessage;
2004 		}
2005 	}
2006 
2007 	{
2008 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Sync", "Sync object");
2009 
2010 		for (int methodNdx = 0; methodNdx < DE_LENGTH_OF_ARRAY(s_clearMethods); ++methodNdx)
2011 		{
2012 			m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\"" << TestLog::EndMessage;
2013 			gl.objectPtrLabel(sync, -2, msg);
2014 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectPtrLabel");
2015 
2016 			m_testCtx.getLog() << TestLog::Message << "Clearing label " << s_clearMethods[methodNdx].description << TestLog::EndMessage;
2017 			gl.objectPtrLabel(sync, s_clearMethods[methodNdx].length, DE_NULL);
2018 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectPtrLabel");
2019 
2020 			m_testCtx.getLog() << TestLog::Message << "Querying label" << TestLog::EndMessage;
2021 			buffer[0] = 'X';
2022 			outlen = -1;
2023 			gl.getObjectPtrLabel(sync, sizeof(buffer), &outlen, buffer);
2024 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2025 
2026 			if (outlen != 0)
2027 				result.fail("'length' was not zero, got " + de::toString(outlen));
2028 			else if (buffer[0] != '\0')
2029 				result.fail("label was not null terminated");
2030 			else
2031 				m_testCtx.getLog() << TestLog::Message << "Got 0-sized null-terminated string." << TestLog::EndMessage;
2032 		}
2033 	}
2034 
2035 	gl.deleteShader(shader);
2036 	gl.deleteSync(sync);
2037 
2038 	result.setTestContextResult(m_testCtx);
2039 	return STOP;
2040 }
2041 
2042 class SpecifyWithLengthCase : public TestCase
2043 {
2044 public:
2045 							SpecifyWithLengthCase	(Context& ctx, const char* name, const char* desc);
2046 	virtual IterateResult	iterate					(void);
2047 };
2048 
SpecifyWithLengthCase(Context & ctx,const char * name,const char * desc)2049 SpecifyWithLengthCase::SpecifyWithLengthCase (Context& ctx, const char* name, const char* desc)
2050 	: TestCase(ctx, name, desc)
2051 {
2052 }
2053 
iterate(void)2054 SpecifyWithLengthCase::IterateResult SpecifyWithLengthCase::iterate (void)
2055 {
2056 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2057 
2058 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
2059 	tcu::ResultCollector	result		(m_testCtx.getLog(), " // ERROR: ");
2060 	const char*	const		msg			= "This is a debug label";
2061 	const char*	const		clipMsg		= "This is a de";
2062 	int						outlen		= -1;
2063 	GLuint					shader;
2064 	glw::GLsync				sync;
2065 	char					buffer[64];
2066 
2067 	sync = gl.fenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
2068 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "fenceSync");
2069 
2070 	shader = gl.createShader(GL_FRAGMENT_SHADER);
2071 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "createShader");
2072 
2073 	{
2074 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Shader", "Shader object");
2075 
2076 		m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\" with length 12" << TestLog::EndMessage;
2077 		gl.objectLabel(GL_SHADER, shader, 12, msg);
2078 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectLabel");
2079 
2080 		m_testCtx.getLog() << TestLog::Message << "Querying label" << TestLog::EndMessage;
2081 		deMemset(buffer, 'X', sizeof(buffer));
2082 		gl.getObjectLabel(GL_SHADER, shader, sizeof(buffer), &outlen, buffer);
2083 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2084 
2085 		if (outlen != 12)
2086 			result.fail("'length' was not 12, got " + de::toString(outlen));
2087 		else if (deStringEqual(clipMsg, buffer))
2088 		{
2089 			m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2090 		}
2091 		else
2092 		{
2093 			buffer[63] = '\0'; // make sure buffer is null terminated before printing
2094 			m_testCtx.getLog() << TestLog::Message << "Query returned wrong string: expected \"" << clipMsg << "\" but got \"" << buffer << "\"" << TestLog::EndMessage;
2095 			result.fail("Query returned wrong label");
2096 		}
2097 	}
2098 
2099 	{
2100 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Sync", "Sync object");
2101 
2102 		m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\" with length 12" << TestLog::EndMessage;
2103 		gl.objectPtrLabel(sync, 12, msg);
2104 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectPtrLabel");
2105 
2106 		m_testCtx.getLog() << TestLog::Message << "Querying label" << TestLog::EndMessage;
2107 		deMemset(buffer, 'X', sizeof(buffer));
2108 		gl.getObjectPtrLabel(sync, sizeof(buffer), &outlen, buffer);
2109 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2110 
2111 		if (outlen != 12)
2112 			result.fail("'length' was not 12, got " + de::toString(outlen));
2113 		else if (deStringEqual(clipMsg, buffer))
2114 		{
2115 			m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2116 		}
2117 		else
2118 		{
2119 			buffer[63] = '\0'; // make sure buffer is null terminated before printing
2120 			m_testCtx.getLog() << TestLog::Message << "Query returned wrong string: expected \"" << clipMsg << "\" but got \"" << buffer << "\"" << TestLog::EndMessage;
2121 			result.fail("Query returned wrong label");
2122 		}
2123 	}
2124 
2125 	{
2126 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "ZeroSized", "ZeroSized");
2127 
2128 		m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\" with length 0" << TestLog::EndMessage;
2129 		gl.objectLabel(GL_SHADER, shader, 0, msg);
2130 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectLabel");
2131 
2132 		m_testCtx.getLog() << TestLog::Message << "Querying label" << TestLog::EndMessage;
2133 		deMemset(buffer, 'X', sizeof(buffer));
2134 		gl.getObjectLabel(GL_SHADER, shader, sizeof(buffer), &outlen, buffer);
2135 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2136 
2137 		if (outlen != 0)
2138 			result.fail("'length' was not zero, got " + de::toString(outlen));
2139 		else if (buffer[0] != '\0')
2140 			result.fail("label was not null terminated");
2141 		else
2142 			m_testCtx.getLog() << TestLog::Message << "Got 0-sized null-terminated string." << TestLog::EndMessage;
2143 	}
2144 
2145 	gl.deleteShader(shader);
2146 	gl.deleteSync(sync);
2147 
2148 	result.setTestContextResult(m_testCtx);
2149 	return STOP;
2150 }
2151 
2152 class BufferLimitedLabelCase : public TestCase
2153 {
2154 public:
2155 							BufferLimitedLabelCase	(Context& ctx, const char* name, const char* desc);
2156 	virtual IterateResult	iterate					(void);
2157 };
2158 
BufferLimitedLabelCase(Context & ctx,const char * name,const char * desc)2159 BufferLimitedLabelCase::BufferLimitedLabelCase (Context& ctx, const char* name, const char* desc)
2160 	: TestCase(ctx, name, desc)
2161 {
2162 }
2163 
iterate(void)2164 BufferLimitedLabelCase::IterateResult BufferLimitedLabelCase::iterate (void)
2165 {
2166 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2167 
2168 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
2169 	tcu::ResultCollector	result		(m_testCtx.getLog(), " // ERROR: ");
2170 	const char*	const		msg			= "This is a debug label";
2171 	int						outlen		= -1;
2172 	GLuint					shader;
2173 	glw::GLsync				sync;
2174 	char					buffer[64];
2175 
2176 	sync = gl.fenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
2177 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "fenceSync");
2178 
2179 	shader = gl.createShader(GL_FRAGMENT_SHADER);
2180 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "createShader");
2181 
2182 	{
2183 		const tcu::ScopedLogSection superSection(m_testCtx.getLog(), "Shader", "Shader object");
2184 
2185 		m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\"" << TestLog::EndMessage;
2186 		gl.objectLabel(GL_SHADER, shader, -1, msg);
2187 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectLabel");
2188 
2189 		{
2190 			const tcu::ScopedLogSection section(m_testCtx.getLog(), "QueryAll", "Query All");
2191 
2192 			m_testCtx.getLog() << TestLog::Message << "Querying whole label, buffer size = 22" << TestLog::EndMessage;
2193 			deMemset(buffer, 'X', sizeof(buffer));
2194 			gl.getObjectLabel(GL_SHADER, shader, 22, &outlen, buffer);
2195 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2196 
2197 			if (outlen != 21)
2198 				result.fail("'length' was not 21, got " + de::toString(outlen));
2199 			else if (buffer[outlen] != '\0')
2200 				result.fail("Buffer was not null-terminated");
2201 			else if (buffer[outlen+1] != 'X')
2202 				result.fail("Query wrote over buffer bound");
2203 			else if (!deStringEqual(msg, buffer))
2204 			{
2205 				buffer[63] = '\0'; // make sure buffer is null terminated before printing
2206 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2207 				result.fail("Query returned wrong label");
2208 			}
2209 			else
2210 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2211 		}
2212 		{
2213 			const tcu::ScopedLogSection section(m_testCtx.getLog(), "QueryAllNoSize", "Query all without size");
2214 
2215 			m_testCtx.getLog() << TestLog::Message << "Querying whole label, buffer size = 22" << TestLog::EndMessage;
2216 			deMemset(buffer, 'X', sizeof(buffer));
2217 			gl.getObjectLabel(GL_SHADER, shader, 22, DE_NULL, buffer);
2218 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2219 
2220 			buffer[63] = '\0'; // make sure buffer is null terminated before strlen
2221 
2222 			if (strlen(buffer) != 21)
2223 				result.fail("Buffer length was not 21");
2224 			else if (buffer[21] != '\0')
2225 				result.fail("Buffer was not null-terminated");
2226 			else if (buffer[22] != 'X')
2227 				result.fail("Query wrote over buffer bound");
2228 			else if (!deStringEqual(msg, buffer))
2229 			{
2230 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2231 				result.fail("Query returned wrong label");
2232 			}
2233 			else
2234 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2235 		}
2236 		{
2237 			const tcu::ScopedLogSection section(m_testCtx.getLog(), "QueryLess", "Query substring");
2238 
2239 			m_testCtx.getLog() << TestLog::Message << "Querying whole label, buffer size = 2" << TestLog::EndMessage;
2240 			deMemset(buffer, 'X', sizeof(buffer));
2241 			gl.getObjectLabel(GL_SHADER, shader, 2, &outlen, buffer);
2242 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2243 
2244 			if (outlen != 1)
2245 				result.fail("'length' was not 1, got " + de::toString(outlen));
2246 			else if (buffer[outlen] != '\0')
2247 				result.fail("Buffer was not null-terminated");
2248 			else if (buffer[outlen+1] != 'X')
2249 				result.fail("Query wrote over buffer bound");
2250 			else if (!deStringBeginsWith(msg, buffer))
2251 			{
2252 				buffer[63] = '\0'; // make sure buffer is null terminated before printing
2253 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2254 				result.fail("Query returned wrong label");
2255 			}
2256 			else
2257 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2258 		}
2259 		{
2260 			const tcu::ScopedLogSection section(m_testCtx.getLog(), "QueryNone", "Query one character");
2261 
2262 			m_testCtx.getLog() << TestLog::Message << "Querying whole label, buffer size = 1" << TestLog::EndMessage;
2263 			deMemset(buffer, 'X', sizeof(buffer));
2264 			gl.getObjectLabel(GL_SHADER, shader, 1, &outlen, buffer);
2265 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2266 
2267 			if (outlen != 0)
2268 				result.fail("'length' was not 0, got " + de::toString(outlen));
2269 			else if (buffer[outlen] != '\0')
2270 				result.fail("Buffer was not null-terminated");
2271 			else if (buffer[outlen+1] != 'X')
2272 				result.fail("Query wrote over buffer bound");
2273 			else
2274 				m_testCtx.getLog() << TestLog::Message << "Query returned zero-sized null-terminated string" << TestLog::EndMessage;
2275 		}
2276 	}
2277 
2278 	{
2279 		const tcu::ScopedLogSection superSection(m_testCtx.getLog(), "Sync", "Sync object");
2280 
2281 		m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\"" << TestLog::EndMessage;
2282 		gl.objectPtrLabel(sync, -1, msg);
2283 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectPtrLabel");
2284 
2285 		{
2286 			const tcu::ScopedLogSection section(m_testCtx.getLog(), "QueryAll", "Query All");
2287 
2288 			m_testCtx.getLog() << TestLog::Message << "Querying whole label, buffer size = 22" << TestLog::EndMessage;
2289 			deMemset(buffer, 'X', sizeof(buffer));
2290 			gl.getObjectPtrLabel(sync, 22, &outlen, buffer);
2291 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2292 
2293 			if (outlen != 21)
2294 				result.fail("'length' was not 21, got " + de::toString(outlen));
2295 			else if (buffer[outlen] != '\0')
2296 				result.fail("Buffer was not null-terminated");
2297 			else if (buffer[outlen+1] != 'X')
2298 				result.fail("Query wrote over buffer bound");
2299 			else if (!deStringEqual(msg, buffer))
2300 			{
2301 				buffer[63] = '\0'; // make sure buffer is null terminated before printing
2302 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2303 				result.fail("Query returned wrong label");
2304 			}
2305 			else
2306 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2307 		}
2308 		{
2309 			const tcu::ScopedLogSection section(m_testCtx.getLog(), "QueryAllNoSize", "Query all without size");
2310 
2311 			m_testCtx.getLog() << TestLog::Message << "Querying whole label, buffer size = 22" << TestLog::EndMessage;
2312 			deMemset(buffer, 'X', sizeof(buffer));
2313 			gl.getObjectPtrLabel(sync, 22, DE_NULL, buffer);
2314 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2315 
2316 			buffer[63] = '\0'; // make sure buffer is null terminated before strlen
2317 
2318 			if (strlen(buffer) != 21)
2319 				result.fail("Buffer length was not 21");
2320 			else if (buffer[21] != '\0')
2321 				result.fail("Buffer was not null-terminated");
2322 			else if (buffer[22] != 'X')
2323 				result.fail("Query wrote over buffer bound");
2324 			else if (!deStringEqual(msg, buffer))
2325 			{
2326 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2327 				result.fail("Query returned wrong label");
2328 			}
2329 			else
2330 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2331 		}
2332 		{
2333 			const tcu::ScopedLogSection section(m_testCtx.getLog(), "QueryLess", "Query substring");
2334 
2335 			m_testCtx.getLog() << TestLog::Message << "Querying whole label, buffer size = 2" << TestLog::EndMessage;
2336 			deMemset(buffer, 'X', sizeof(buffer));
2337 			gl.getObjectPtrLabel(sync, 2, &outlen, buffer);
2338 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2339 
2340 			if (outlen != 1)
2341 				result.fail("'length' was not 1, got " + de::toString(outlen));
2342 			else if (buffer[outlen] != '\0')
2343 				result.fail("Buffer was not null-terminated");
2344 			else if (buffer[outlen+1] != 'X')
2345 				result.fail("Query wrote over buffer bound");
2346 			else if (!deStringBeginsWith(msg, buffer))
2347 			{
2348 				buffer[63] = '\0'; // make sure buffer is null terminated before printing
2349 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2350 				result.fail("Query returned wrong label");
2351 			}
2352 			else
2353 				m_testCtx.getLog() << TestLog::Message << "Query returned string: \"" << buffer << "\"" << TestLog::EndMessage;
2354 		}
2355 		{
2356 			const tcu::ScopedLogSection section(m_testCtx.getLog(), "QueryNone", "Query one character");
2357 
2358 			m_testCtx.getLog() << TestLog::Message << "Querying whole label, buffer size = 1" << TestLog::EndMessage;
2359 			deMemset(buffer, 'X', sizeof(buffer));
2360 			gl.getObjectPtrLabel(sync, 1, &outlen, buffer);
2361 			GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2362 
2363 			if (outlen != 0)
2364 				result.fail("'length' was not 0, got " + de::toString(outlen));
2365 			else if (buffer[outlen] != '\0')
2366 				result.fail("Buffer was not null-terminated");
2367 			else if (buffer[outlen+1] != 'X')
2368 				result.fail("Query wrote over buffer bound");
2369 			else
2370 				m_testCtx.getLog() << TestLog::Message << "Query returned zero-sized null-terminated string" << TestLog::EndMessage;
2371 		}
2372 	}
2373 
2374 	gl.deleteShader(shader);
2375 	gl.deleteSync(sync);
2376 
2377 	result.setTestContextResult(m_testCtx);
2378 	return STOP;
2379 }
2380 
2381 class LabelMaxSizeCase : public TestCase
2382 {
2383 public:
2384 							LabelMaxSizeCase	(Context& ctx, const char* name, const char* desc);
2385 	virtual IterateResult	iterate				(void);
2386 };
2387 
LabelMaxSizeCase(Context & ctx,const char * name,const char * desc)2388 LabelMaxSizeCase::LabelMaxSizeCase (Context& ctx, const char* name, const char* desc)
2389 	: TestCase(ctx, name, desc)
2390 {
2391 }
2392 
iterate(void)2393 LabelMaxSizeCase::IterateResult LabelMaxSizeCase::iterate (void)
2394 {
2395 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2396 
2397 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
2398 	tcu::ResultCollector	result		(m_testCtx.getLog(), " // ERROR: ");
2399 	int						maxLabelLen	= -1;
2400 	int						outlen		= -1;
2401 	GLuint					shader;
2402 	glw::GLsync				sync;
2403 
2404 	gl.getIntegerv(GL_MAX_LABEL_LENGTH, &maxLabelLen);
2405 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "GL_MAX_LABEL_LENGTH");
2406 
2407 	m_testCtx.getLog() << TestLog::Message << "GL_MAX_LABEL_LENGTH = " << maxLabelLen << TestLog::EndMessage;
2408 
2409 	if (maxLabelLen < 256)
2410 		throw tcu::TestError("maxLabelLen was less than required (256)");
2411 	if (maxLabelLen > 8192)
2412 	{
2413 		m_testCtx.getLog()
2414 			<< TestLog::Message
2415 			<< "GL_MAX_LABEL_LENGTH is very large. Application having larger labels is unlikely, skipping test."
2416 			<< TestLog::EndMessage;
2417 		m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
2418 		return STOP;
2419 	}
2420 
2421 	sync = gl.fenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
2422 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "fenceSync");
2423 
2424 	shader = gl.createShader(GL_FRAGMENT_SHADER);
2425 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "createShader");
2426 
2427 	{
2428 		const tcu::ScopedLogSection	section		(m_testCtx.getLog(), "Shader", "Shader object");
2429 		std::vector<char>			buffer		(maxLabelLen, 'X');
2430 		std::vector<char>			readBuffer	(maxLabelLen, 'X');
2431 
2432 		buffer[maxLabelLen-1] = '\0';
2433 
2434 		m_testCtx.getLog() << TestLog::Message << "Setting max length label, with implicit size. (length = -1)" << TestLog::EndMessage;
2435 		gl.objectLabel(GL_SHADER, shader, -1,  &buffer[0]);
2436 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectLabel");
2437 
2438 		m_testCtx.getLog() << TestLog::Message << "Querying label back" << TestLog::EndMessage;
2439 		outlen = -1;
2440 		gl.getObjectLabel(GL_SHADER, shader, maxLabelLen, &outlen, &readBuffer[0]);
2441 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2442 
2443 		if (outlen != maxLabelLen-1)
2444 			result.fail("'length' was not " + de::toString(maxLabelLen-1) + ", got " + de::toString(outlen));
2445 		else if (readBuffer[outlen] != '\0')
2446 			result.fail("Buffer was not null-terminated");
2447 
2448 		m_testCtx.getLog() << TestLog::Message << "Setting max length label, with explicit size. (length = " << (maxLabelLen-1) << ")" << TestLog::EndMessage;
2449 		gl.objectLabel(GL_SHADER, shader, maxLabelLen-1,  &buffer[0]);
2450 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectLabel");
2451 
2452 		m_testCtx.getLog() << TestLog::Message << "Querying label back" << TestLog::EndMessage;
2453 		outlen = -1;
2454 		readBuffer[maxLabelLen-1] = 'X';
2455 		gl.getObjectLabel(GL_SHADER, shader, maxLabelLen, &outlen, &readBuffer[0]);
2456 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2457 
2458 		if (outlen != maxLabelLen-1)
2459 			result.fail("'length' was not " + de::toString(maxLabelLen-1) + ", got " + de::toString(outlen));
2460 		else if (readBuffer[outlen] != '\0')
2461 			result.fail("Buffer was not null-terminated");
2462 	}
2463 
2464 	{
2465 		const tcu::ScopedLogSection section		(m_testCtx.getLog(), "Sync", "Sync object");
2466 		std::vector<char>			buffer		(maxLabelLen, 'X');
2467 		std::vector<char>			readBuffer	(maxLabelLen, 'X');
2468 
2469 		buffer[maxLabelLen-1] = '\0';
2470 
2471 		m_testCtx.getLog() << TestLog::Message << "Setting max length label, with implicit size. (length = -1)" << TestLog::EndMessage;
2472 		gl.objectPtrLabel(sync, -1,  &buffer[0]);
2473 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectPtrLabel");
2474 
2475 		m_testCtx.getLog() << TestLog::Message << "Querying label back" << TestLog::EndMessage;
2476 		outlen = -1;
2477 		gl.getObjectPtrLabel(sync, maxLabelLen, &outlen, &readBuffer[0]);
2478 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2479 
2480 		if (outlen != maxLabelLen-1)
2481 			result.fail("'length' was not " + de::toString(maxLabelLen-1) + ", got " + de::toString(outlen));
2482 		else if (readBuffer[outlen] != '\0')
2483 			result.fail("Buffer was not null-terminated");
2484 
2485 		m_testCtx.getLog() << TestLog::Message << "Setting max length label, with explicit size. (length = " << (maxLabelLen-1) << ")" << TestLog::EndMessage;
2486 		gl.objectPtrLabel(sync, maxLabelLen-1,  &buffer[0]);
2487 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectPtrLabel");
2488 
2489 		m_testCtx.getLog() << TestLog::Message << "Querying label back" << TestLog::EndMessage;
2490 		outlen = -1;
2491 		readBuffer[maxLabelLen-1] = 'X';
2492 		gl.getObjectPtrLabel(sync, maxLabelLen, &outlen, &readBuffer[0]);
2493 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2494 
2495 		if (outlen != maxLabelLen-1)
2496 			result.fail("'length' was not " + de::toString(maxLabelLen-1) + ", got " + de::toString(outlen));
2497 		else if (readBuffer[outlen] != '\0')
2498 			result.fail("Buffer was not null-terminated");
2499 	}
2500 
2501 	gl.deleteShader(shader);
2502 	gl.deleteSync(sync);
2503 
2504 	result.setTestContextResult(m_testCtx);
2505 	return STOP;
2506 }
2507 
2508 class LabelLengthCase : public TestCase
2509 {
2510 public:
2511 							LabelLengthCase	(Context& ctx, const char* name, const char* desc);
2512 	virtual IterateResult	iterate			(void);
2513 };
2514 
LabelLengthCase(Context & ctx,const char * name,const char * desc)2515 LabelLengthCase::LabelLengthCase (Context& ctx, const char* name, const char* desc)
2516 	: TestCase(ctx, name, desc)
2517 {
2518 }
2519 
iterate(void)2520 LabelLengthCase::IterateResult LabelLengthCase::iterate (void)
2521 {
2522 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2523 
2524 	const glw::Functions&	gl			= m_context.getRenderContext().getFunctions();
2525 	tcu::ResultCollector	result		(m_testCtx.getLog(), " // ERROR: ");
2526 	const char*	const		msg			= "This is a debug label";
2527 	int						outlen		= -1;
2528 	GLuint					shader;
2529 	glw::GLsync				sync;
2530 
2531 	sync = gl.fenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
2532 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "fenceSync");
2533 
2534 	shader = gl.createShader(GL_FRAGMENT_SHADER);
2535 	GLS_COLLECT_GL_ERROR(result, gl.getError(), "createShader");
2536 
2537 	{
2538 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Shader", "Shader object");
2539 
2540 		m_testCtx.getLog() << TestLog::Message << "Querying label length" << TestLog::EndMessage;
2541 		outlen = -1;
2542 		gl.getObjectLabel(GL_SHADER, shader, 0, &outlen, DE_NULL);
2543 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2544 
2545 		if (outlen != 0)
2546 			result.fail("'length' was not 0, got " + de::toString(outlen));
2547 		else
2548 			m_testCtx.getLog() << TestLog::Message << "Query returned length: " << outlen << TestLog::EndMessage;
2549 
2550 		m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\"" << TestLog::EndMessage;
2551 		gl.objectLabel(GL_SHADER, shader, -1, msg);
2552 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectLabel");
2553 
2554 		m_testCtx.getLog() << TestLog::Message << "Querying label length" << TestLog::EndMessage;
2555 		outlen = -1;
2556 		gl.getObjectLabel(GL_SHADER, shader, 0, &outlen, DE_NULL);
2557 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectLabel");
2558 
2559 		if (outlen != 21)
2560 			result.fail("'length' was not 21, got " + de::toString(outlen));
2561 		else
2562 			m_testCtx.getLog() << TestLog::Message << "Query returned length: " << outlen << TestLog::EndMessage;
2563 	}
2564 
2565 	{
2566 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Sync", "Sync object");
2567 
2568 		m_testCtx.getLog() << TestLog::Message << "Querying label length" << TestLog::EndMessage;
2569 		outlen = -1;
2570 		gl.getObjectPtrLabel(sync, 0, &outlen, DE_NULL);
2571 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2572 
2573 		if (outlen != 0)
2574 			result.fail("'length' was not 0, got " + de::toString(outlen));
2575 		else
2576 			m_testCtx.getLog() << TestLog::Message << "Query returned length: " << outlen << TestLog::EndMessage;
2577 
2578 		m_testCtx.getLog() << TestLog::Message << "Setting label to string: \"" << msg << "\"" << TestLog::EndMessage;
2579 		gl.objectPtrLabel(sync, -1, msg);
2580 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "objectPtrLabel");
2581 
2582 		m_testCtx.getLog() << TestLog::Message << "Querying label length" << TestLog::EndMessage;
2583 		outlen = -1;
2584 		gl.getObjectPtrLabel(sync, 0, &outlen, DE_NULL);
2585 		GLS_COLLECT_GL_ERROR(result, gl.getError(), "getObjectPtrLabel");
2586 
2587 		if (outlen != 21)
2588 			result.fail("'length' was not 21, got " + de::toString(outlen));
2589 		else
2590 			m_testCtx.getLog() << TestLog::Message << "Query returned length: " << outlen << TestLog::EndMessage;
2591 	}
2592 
2593 	gl.deleteShader(shader);
2594 	gl.deleteSync(sync);
2595 
2596 	result.setTestContextResult(m_testCtx);
2597 	return STOP;
2598 }
2599 
2600 class LimitQueryCase : public TestCase
2601 {
2602 public:
2603 											LimitQueryCase	(Context&						context,
2604 															 const char*					name,
2605 															 const char*					description,
2606 															 glw::GLenum					target,
2607 															 int							limit,
2608 															 gls::StateQueryUtil::QueryType	type);
2609 
2610 	IterateResult							iterate			(void);
2611 private:
2612 	const gls::StateQueryUtil::QueryType	m_type;
2613 	const int								m_limit;
2614 	const glw::GLenum						m_target;
2615 };
2616 
LimitQueryCase(Context & context,const char * name,const char * description,glw::GLenum target,int limit,gls::StateQueryUtil::QueryType type)2617 LimitQueryCase::LimitQueryCase (Context&						context,
2618 								const char*						name,
2619 								const char*						description,
2620 								glw::GLenum						target,
2621 								int								limit,
2622 								gls::StateQueryUtil::QueryType	type)
2623 	: TestCase	(context, name, description)
2624 	, m_type	(type)
2625 	, m_limit	(limit)
2626 	, m_target	(target)
2627 {
2628 }
2629 
iterate(void)2630 LimitQueryCase::IterateResult LimitQueryCase::iterate (void)
2631 {
2632 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2633 
2634 	glu::CallLogWrapper		gl		(m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
2635 	tcu::ResultCollector	result	(m_testCtx.getLog(), " // ERROR: ");
2636 
2637 	gl.enableLogging(true);
2638 	gls::StateQueryUtil::verifyStateIntegerMin(result, gl, m_target, m_limit, m_type);
2639 
2640 	result.setTestContextResult(m_testCtx);
2641 	return STOP;
2642 }
2643 
2644 class IsEnabledCase : public TestCase
2645 {
2646 public:
2647 	enum InitialValue
2648 	{
2649 		INITIAL_CTX_IS_DEBUG = 0,
2650 		INITIAL_FALSE,
2651 	};
2652 
2653 											IsEnabledCase	(Context&						context,
2654 															 const char*					name,
2655 															 const char*					description,
2656 															 glw::GLenum					target,
2657 															 InitialValue					initial,
2658 															 gls::StateQueryUtil::QueryType	type);
2659 
2660 	IterateResult							iterate			(void);
2661 private:
2662 	const gls::StateQueryUtil::QueryType	m_type;
2663 	const glw::GLenum						m_target;
2664 	const InitialValue						m_initial;
2665 };
2666 
IsEnabledCase(Context & context,const char * name,const char * description,glw::GLenum target,InitialValue initial,gls::StateQueryUtil::QueryType type)2667 IsEnabledCase::IsEnabledCase (Context&							context,
2668 							  const char*						name,
2669 							  const char*						description,
2670 							  glw::GLenum						target,
2671 							  InitialValue						initial,
2672 							  gls::StateQueryUtil::QueryType	type)
2673 	: TestCase	(context, name, description)
2674 	, m_type	(type)
2675 	, m_target	(target)
2676 	, m_initial	(initial)
2677 {
2678 }
2679 
iterate(void)2680 IsEnabledCase::IterateResult IsEnabledCase::iterate (void)
2681 {
2682 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2683 
2684 	glu::CallLogWrapper		gl		(m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
2685 	tcu::ResultCollector	result	(m_testCtx.getLog(), " // ERROR: ");
2686 	bool					initial;
2687 
2688 	gl.enableLogging(true);
2689 
2690 	if (m_initial == INITIAL_FALSE)
2691 		initial = false;
2692 	else
2693 	{
2694 		DE_ASSERT(m_initial == INITIAL_CTX_IS_DEBUG);
2695 		initial = (m_context.getRenderContext().getType().getFlags() & glu::CONTEXT_DEBUG) != 0;
2696 	}
2697 
2698 	// check inital value
2699 	gls::StateQueryUtil::verifyStateBoolean(result, gl, m_target, initial, m_type);
2700 
2701 	// check toggle
2702 
2703 	gl.glEnable(m_target);
2704 	GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glEnable");
2705 
2706 	gls::StateQueryUtil::verifyStateBoolean(result, gl, m_target, true, m_type);
2707 
2708 	gl.glDisable(m_target);
2709 	GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDisable");
2710 
2711 	gls::StateQueryUtil::verifyStateBoolean(result, gl, m_target, false, m_type);
2712 
2713 	result.setTestContextResult(m_testCtx);
2714 	return STOP;
2715 }
2716 
2717 class PositiveIntegerCase : public TestCase
2718 {
2719 public:
2720 											PositiveIntegerCase	(Context&						context,
2721 																 const char*					name,
2722 																 const char*					description,
2723 																 glw::GLenum					target,
2724 																 gls::StateQueryUtil::QueryType	type);
2725 
2726 	IterateResult							iterate			(void);
2727 private:
2728 	const gls::StateQueryUtil::QueryType	m_type;
2729 	const glw::GLenum						m_target;
2730 };
2731 
PositiveIntegerCase(Context & context,const char * name,const char * description,glw::GLenum target,gls::StateQueryUtil::QueryType type)2732 PositiveIntegerCase::PositiveIntegerCase (Context&							context,
2733 										  const char*						name,
2734 										  const char*						description,
2735 										  glw::GLenum						target,
2736 										  gls::StateQueryUtil::QueryType	type)
2737 	: TestCase	(context, name, description)
2738 	, m_type	(type)
2739 	, m_target	(target)
2740 {
2741 }
2742 
iterate(void)2743 PositiveIntegerCase::IterateResult PositiveIntegerCase::iterate (void)
2744 {
2745 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2746 
2747 	glu::CallLogWrapper		gl		(m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
2748 	tcu::ResultCollector	result	(m_testCtx.getLog(), " // ERROR: ");
2749 
2750 	gl.enableLogging(true);
2751 	gls::StateQueryUtil::verifyStateIntegerMin(result, gl, m_target, 0, m_type);
2752 
2753 	result.setTestContextResult(m_testCtx);
2754 	return STOP;
2755 }
2756 
2757 class GroupStackDepthQueryCase : public TestCase
2758 {
2759 public:
2760 											GroupStackDepthQueryCase	(Context&						context,
2761 																		 const char*					name,
2762 																		 const char*					description,
2763 																		 gls::StateQueryUtil::QueryType	type);
2764 
2765 	IterateResult							iterate			(void);
2766 private:
2767 	const gls::StateQueryUtil::QueryType	m_type;
2768 };
2769 
GroupStackDepthQueryCase(Context & context,const char * name,const char * description,gls::StateQueryUtil::QueryType type)2770 GroupStackDepthQueryCase::GroupStackDepthQueryCase (Context&						context,
2771 													const char*						name,
2772 													const char*						description,
2773 													gls::StateQueryUtil::QueryType	type)
2774 	: TestCase	(context, name, description)
2775 	, m_type	(type)
2776 {
2777 }
2778 
iterate(void)2779 GroupStackDepthQueryCase::IterateResult GroupStackDepthQueryCase::iterate (void)
2780 {
2781 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2782 
2783 	glu::CallLogWrapper		gl		(m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
2784 	tcu::ResultCollector	result	(m_testCtx.getLog(), " // ERROR: ");
2785 
2786 	gl.enableLogging(true);
2787 
2788 	{
2789 		const tcu::ScopedLogSection	section(m_testCtx.getLog(), "Initial", "Initial");
2790 
2791 		gls::StateQueryUtil::verifyStateInteger(result, gl, GL_DEBUG_GROUP_STACK_DEPTH, 1, m_type);
2792 	}
2793 
2794 	{
2795 		const tcu::ScopedLogSection	section(m_testCtx.getLog(), "Scoped", "Scoped");
2796 
2797 		gl.glPushDebugGroup(GL_DEBUG_SOURCE_APPLICATION, 1, -1, "Application group 1");
2798 		gls::StateQueryUtil::verifyStateInteger(result, gl, GL_DEBUG_GROUP_STACK_DEPTH, 2, m_type);
2799 		gl.glPopDebugGroup();
2800 	}
2801 
2802 	result.setTestContextResult(m_testCtx);
2803 	return STOP;
2804 }
2805 
dummyCallback(GLenum,GLenum,GLuint,GLenum,GLsizei,const char *,const void *)2806 extern "C" void GLW_APIENTRY dummyCallback(GLenum, GLenum, GLuint, GLenum, GLsizei, const char*, const void*)
2807 {
2808 	// dummy
2809 }
2810 
2811 class DebugCallbackFunctionCase : public TestCase
2812 {
2813 public:
2814 					DebugCallbackFunctionCase	(Context& context, const char* name, const char* description);
2815 	IterateResult	iterate						(void);
2816 };
2817 
DebugCallbackFunctionCase(Context & context,const char * name,const char * description)2818 DebugCallbackFunctionCase::DebugCallbackFunctionCase (Context& context, const char* name, const char* description)
2819 	: TestCase	(context, name, description)
2820 {
2821 }
2822 
iterate(void)2823 DebugCallbackFunctionCase::IterateResult DebugCallbackFunctionCase::iterate (void)
2824 {
2825 	using namespace gls::StateQueryUtil;
2826 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2827 
2828 	glu::CallLogWrapper		gl		(m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
2829 	tcu::ResultCollector	result	(m_testCtx.getLog(), " // ERROR: ");
2830 
2831 	gl.enableLogging(true);
2832 
2833 	{
2834 		const tcu::ScopedLogSection	section(m_testCtx.getLog(), "Initial", "Initial");
2835 
2836 		verifyStatePointer(result, gl, GL_DEBUG_CALLBACK_FUNCTION, 0, QUERY_POINTER);
2837 	}
2838 
2839 	{
2840 		const tcu::ScopedLogSection	section(m_testCtx.getLog(), "Set", "Set");
2841 
2842 		gl.glDebugMessageCallback(dummyCallback, DE_NULL);
2843 		verifyStatePointer(result, gl, GL_DEBUG_CALLBACK_FUNCTION, (const void*)dummyCallback, QUERY_POINTER);
2844 	}
2845 
2846 	result.setTestContextResult(m_testCtx);
2847 	return STOP;
2848 }
2849 
2850 class DebugCallbackUserParamCase : public TestCase
2851 {
2852 public:
2853 					DebugCallbackUserParamCase	(Context& context, const char* name, const char* description);
2854 	IterateResult	iterate						(void);
2855 };
2856 
DebugCallbackUserParamCase(Context & context,const char * name,const char * description)2857 DebugCallbackUserParamCase::DebugCallbackUserParamCase (Context& context, const char* name, const char* description)
2858 	: TestCase	(context, name, description)
2859 {
2860 }
2861 
iterate(void)2862 DebugCallbackUserParamCase::IterateResult DebugCallbackUserParamCase::iterate (void)
2863 {
2864 	using namespace gls::StateQueryUtil;
2865 
2866 	TCU_CHECK_AND_THROW(NotSupportedError, isKHRDebugSupported(m_context), "GL_KHR_debug is not supported");
2867 
2868 	glu::CallLogWrapper		gl		(m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
2869 	tcu::ResultCollector	result	(m_testCtx.getLog(), " // ERROR: ");
2870 
2871 	gl.enableLogging(true);
2872 
2873 	{
2874 		const tcu::ScopedLogSection	section(m_testCtx.getLog(), "Initial", "Initial");
2875 
2876 		verifyStatePointer(result, gl, GL_DEBUG_CALLBACK_USER_PARAM, 0, QUERY_POINTER);
2877 	}
2878 
2879 	{
2880 		const tcu::ScopedLogSection	section	(m_testCtx.getLog(), "Set", "Set");
2881 		const void*					param	= (void*)(int*)0x123;
2882 
2883 		gl.glDebugMessageCallback(dummyCallback, param);
2884 		verifyStatePointer(result, gl, GL_DEBUG_CALLBACK_USER_PARAM, param, QUERY_POINTER);
2885 	}
2886 
2887 	result.setTestContextResult(m_testCtx);
2888 	return STOP;
2889 }
2890 
2891 } // anonymous
2892 
DebugTests(Context & context)2893 DebugTests::DebugTests (Context& context)
2894 	: TestCaseGroup(context, "debug", "Debug tests")
2895 {
2896 }
2897 
2898 enum CaseType
2899 {
2900 	CASETYPE_CALLBACK = 0,
2901 	CASETYPE_LOG,
2902 	CASETYPE_GETERROR,
2903 
2904 	CASETYPE_LAST
2905 };
2906 
createCase(CaseType type,Context & ctx,const char * name,const char * desc,TestFunctionWrapper function)2907 tcu::TestNode* createCase (CaseType type, Context& ctx, const char* name, const char* desc, TestFunctionWrapper function)
2908 {
2909 	switch(type)
2910 	{
2911 		case CASETYPE_CALLBACK: return new CallbackErrorCase(ctx, name, desc, function);
2912 		case CASETYPE_LOG:		return new LogErrorCase(ctx, name, desc, function);
2913 		case CASETYPE_GETERROR: return new GetErrorCase(ctx, name, desc, function);
2914 
2915 		default:
2916 			DE_FATAL("Invalid type");
2917 	}
2918 
2919 	return DE_NULL;
2920 }
2921 
createChildCases(CaseType type,Context & ctx,const char * name,const char * desc,const vector<FunctionContainer> & funcs)2922 tcu::TestCaseGroup* createChildCases (CaseType type, Context& ctx, const char* name, const char* desc, const vector<FunctionContainer>& funcs)
2923 {
2924 	tcu::TestCaseGroup* host = new tcu::TestCaseGroup(ctx.getTestContext(), name, desc);
2925 
2926 	for (size_t ndx = 0; ndx < funcs.size(); ndx++)
2927 			host->addChild(createCase(type, ctx, funcs[ndx].name, funcs[ndx].desc, funcs[ndx].function));
2928 
2929 	return host;
2930 }
2931 
wrapCoreFunctions(const vector<NegativeTestShared::FunctionContainer> & fns)2932 vector<FunctionContainer> wrapCoreFunctions (const vector<NegativeTestShared::FunctionContainer>& fns)
2933 {
2934 	vector<FunctionContainer> retVal;
2935 
2936 	retVal.resize(fns.size());
2937 	for (int ndx = 0; ndx < (int)fns.size(); ++ndx)
2938 	{
2939 		retVal[ndx].function = TestFunctionWrapper(fns[ndx].function);
2940 		retVal[ndx].name = fns[ndx].name;
2941 		retVal[ndx].desc = fns[ndx].desc;
2942 	}
2943 
2944 	return retVal;
2945 }
2946 
init(void)2947 void DebugTests::init (void)
2948 {
2949 	const vector<FunctionContainer> bufferFuncs				 = wrapCoreFunctions(NegativeTestShared::getNegativeBufferApiTestFunctions());
2950 	const vector<FunctionContainer> textureFuncs			 = wrapCoreFunctions(NegativeTestShared::getNegativeTextureApiTestFunctions());
2951 	const vector<FunctionContainer> shaderFuncs				 = wrapCoreFunctions(NegativeTestShared::getNegativeShaderApiTestFunctions());
2952 	const vector<FunctionContainer> fragmentFuncs			 = wrapCoreFunctions(NegativeTestShared::getNegativeFragmentApiTestFunctions());
2953 	const vector<FunctionContainer> vaFuncs					 = wrapCoreFunctions(NegativeTestShared::getNegativeVertexArrayApiTestFunctions());
2954 	const vector<FunctionContainer> stateFuncs				 = wrapCoreFunctions(NegativeTestShared::getNegativeStateApiTestFunctions());
2955 	const vector<FunctionContainer> tessellationFuncs		 = wrapCoreFunctions(NegativeTestShared::getNegativeTessellationTestFunctions());
2956 	const vector<FunctionContainer> atomicCounterFuncs		 = wrapCoreFunctions(NegativeTestShared::getNegativeAtomicCounterTestFunctions());
2957 	const vector<FunctionContainer> imageLoadFuncs			 = wrapCoreFunctions(NegativeTestShared::getNegativeShaderImageLoadTestFunctions());
2958 	const vector<FunctionContainer> imageStoreFuncs			 = wrapCoreFunctions(NegativeTestShared::getNegativeShaderImageStoreTestFunctions());
2959 	const vector<FunctionContainer> imageAtomicFuncs		 = wrapCoreFunctions(NegativeTestShared::getNegativeShaderImageAtomicTestFunctions());
2960 	const vector<FunctionContainer> imageAtomicExchangeFuncs = wrapCoreFunctions(NegativeTestShared::getNegativeShaderImageAtomicExchangeTestFunctions());
2961 	const vector<FunctionContainer> shaderFunctionFuncs		 = wrapCoreFunctions(NegativeTestShared::getNegativeShaderFunctionTestFunctions());
2962 	const vector<FunctionContainer> shaderDirectiveFuncs	 = wrapCoreFunctions(NegativeTestShared::getNegativeShaderDirectiveTestFunctions());
2963 	const vector<FunctionContainer> ssboBlockFuncs			 = wrapCoreFunctions(NegativeTestShared::getNegativeSSBOBlockTestFunctions());
2964 	const vector<FunctionContainer> preciseFuncs			 = wrapCoreFunctions(NegativeTestShared::getNegativePreciseTestFunctions());
2965 	const vector<FunctionContainer> advancedBlendFuncs		 = wrapCoreFunctions(NegativeTestShared::getNegativeAdvancedBlendEquationTestFunctions());
2966 	const vector<FunctionContainer> shaderStorageFuncs		 = wrapCoreFunctions(NegativeTestShared::getNegativeShaderStorageTestFunctions());
2967 	const vector<FunctionContainer> sampleVariablesFuncs	 = wrapCoreFunctions(NegativeTestShared::getNegativeSampleVariablesTestFunctions());
2968 	const vector<FunctionContainer> computeFuncs			 = wrapCoreFunctions(NegativeTestShared::getNegativeComputeTestFunctions());
2969 	const vector<FunctionContainer> framebufferFetchFuncs    = wrapCoreFunctions(NegativeTestShared::getNegativeShaderFramebufferFetchTestFunctions());
2970 	const vector<FunctionContainer> externalFuncs			 = getUserMessageFuncs();
2971 
2972 	{
2973 		using namespace gls::StateQueryUtil;
2974 
2975 		tcu::TestCaseGroup* const queries = new tcu::TestCaseGroup(m_testCtx, "state_query", "State query");
2976 
2977 		static const struct
2978 		{
2979 			const char*	name;
2980 			const char*	targetName;
2981 			glw::GLenum	target;
2982 			int			limit;
2983 		} limits[] =
2984 		{
2985 			{ "max_debug_message_length",		"MAX_DEBUG_MESSAGE_LENGTH",		GL_MAX_DEBUG_MESSAGE_LENGTH,	1	},
2986 			{ "max_debug_logged_messages",		"MAX_DEBUG_LOGGED_MESSAGES",	GL_MAX_DEBUG_LOGGED_MESSAGES,	1	},
2987 			{ "max_debug_group_stack_depth",	"MAX_DEBUG_GROUP_STACK_DEPTH",	GL_MAX_DEBUG_GROUP_STACK_DEPTH,	64	},
2988 			{ "max_label_length",				"MAX_LABEL_LENGTH",				GL_MAX_LABEL_LENGTH,			256	},
2989 		};
2990 
2991 		addChild(queries);
2992 
2993 		#define FOR_ALL_TYPES(X) \
2994 			do \
2995 			{ \
2996 				{ \
2997 					const char* const	postfix = "_getboolean"; \
2998 					const QueryType		queryType = QUERY_BOOLEAN; \
2999 					X; \
3000 				} \
3001 				{ \
3002 					const char* const	postfix = "_getinteger"; \
3003 					const QueryType		queryType = QUERY_INTEGER; \
3004 					X; \
3005 				} \
3006 				{ \
3007 					const char* const	postfix = "_getinteger64"; \
3008 					const QueryType		queryType = QUERY_INTEGER64; \
3009 					X; \
3010 				} \
3011 				{ \
3012 					const char* const	postfix = "_getfloat"; \
3013 					const QueryType		queryType = QUERY_FLOAT; \
3014 					X; \
3015 				} \
3016 			} \
3017 			while (deGetFalse())
3018 		#define FOR_ALL_ENABLE_TYPES(X) \
3019 			do \
3020 			{ \
3021 				{ \
3022 					const char* const	postfix = "_isenabled"; \
3023 					const QueryType		queryType = QUERY_ISENABLED; \
3024 					X; \
3025 				} \
3026 				FOR_ALL_TYPES(X); \
3027 			} \
3028 			while (deGetFalse())
3029 
3030 		for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(limits); ++ndx)
3031 		{
3032 			FOR_ALL_TYPES(queries->addChild(new LimitQueryCase(m_context,
3033 															   (std::string(limits[ndx].name) + postfix).c_str(),
3034 															   (std::string("Test ") + limits[ndx].targetName).c_str(),
3035 															   limits[ndx].target, limits[ndx].limit, queryType)));
3036 		}
3037 
3038 		FOR_ALL_ENABLE_TYPES(queries->addChild(new IsEnabledCase	(m_context, (std::string("debug_output") + postfix).c_str(),						"Test DEBUG_OUTPUT",						GL_DEBUG_OUTPUT,				IsEnabledCase::INITIAL_CTX_IS_DEBUG,	queryType)));
3039 		FOR_ALL_ENABLE_TYPES(queries->addChild(new IsEnabledCase	(m_context, (std::string("debug_output_synchronous") + postfix).c_str(),			"Test DEBUG_OUTPUT_SYNCHRONOUS",			GL_DEBUG_OUTPUT_SYNCHRONOUS,	IsEnabledCase::INITIAL_FALSE,			queryType)));
3040 
3041 		FOR_ALL_TYPES(queries->addChild(new PositiveIntegerCase		(m_context, (std::string("debug_logged_messages") + postfix).c_str(),				"Test DEBUG_LOGGED_MESSAGES",				GL_DEBUG_LOGGED_MESSAGES,				queryType)));
3042 		FOR_ALL_TYPES(queries->addChild(new PositiveIntegerCase		(m_context, (std::string("debug_next_logged_message_length") + postfix).c_str(),	"Test DEBUG_NEXT_LOGGED_MESSAGE_LENGTH",	GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH,	queryType)));
3043 		FOR_ALL_TYPES(queries->addChild(new GroupStackDepthQueryCase(m_context, (std::string("debug_group_stack_depth") + postfix).c_str(),				"Test DEBUG_GROUP_STACK_DEPTH",				queryType)));
3044 
3045 		queries->addChild(new DebugCallbackFunctionCase	(m_context, "debug_callback_function_getpointer",	"Test DEBUG_CALLBACK_FUNCTION"));
3046 		queries->addChild(new DebugCallbackUserParamCase(m_context, "debug_callback_user_param_getpointer", "Test DEBUG_CALLBACK_USER_PARAM"));
3047 
3048 		#undef FOR_ALL_TYPES
3049 		#undef FOR_ALL_ENABLE_TYPES
3050 	}
3051 
3052 	{
3053 		tcu::TestCaseGroup* const	negative	= new tcu::TestCaseGroup(m_testCtx, "negative_coverage", "API error coverage with various reporting methods");
3054 
3055 		addChild(negative);
3056 		{
3057 			tcu::TestCaseGroup* const	host	= new tcu::TestCaseGroup(m_testCtx, "callbacks", "Reporting of standard API errors via callback");
3058 
3059 			negative->addChild(host);
3060 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "buffer",						"Negative Buffer API Cases",						bufferFuncs));
3061 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "texture",					"Negative Texture API Cases",						textureFuncs));
3062 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "shader",						"Negative Shader API Cases",						shaderFuncs));
3063 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "fragment",					"Negative Fragment API Cases",						fragmentFuncs));
3064 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "vertex_array",				"Negative Vertex Array API Cases",					vaFuncs));
3065 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "state",						"Negative GL State API Cases",						stateFuncs));
3066 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "atomic_counter",				"Negative Atomic Counter API Cases",				atomicCounterFuncs));
3067 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "shader_image_load",			"Negative Shader Image Load API Cases",				imageLoadFuncs));
3068 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "shader_image_store",			"Negative Shader Image Store API Cases",			imageStoreFuncs));
3069 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "shader_image_atomic",		"Negative Shader Image Atomic API Cases",			imageAtomicFuncs));
3070 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "shader_image_exchange",		"Negative Shader Image Atomic Exchange API Cases",	imageAtomicExchangeFuncs));
3071 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "shader_function",			"Negative Shader Function Cases",					shaderFunctionFuncs));
3072 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "shader_directive",			"Negative Shader Directive Cases",					shaderDirectiveFuncs));
3073 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "ssbo_block",					"Negative SSBO Block Cases",						ssboBlockFuncs));
3074 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "precise",					"Negative Precise Cases",							preciseFuncs));
3075 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "advanced_blend",				"Negative Advanced Blend Equation Cases",			advancedBlendFuncs));
3076 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "shader_storage",				"Negative Shader Storage Cases",					shaderStorageFuncs));
3077 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "tessellation",				"Negative Tessellation Cases",						tessellationFuncs));
3078 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "oes_sample_variables",		"Negative Sample Variables Cases",					sampleVariablesFuncs));
3079 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "compute",					"Negative Compute Cases",							computeFuncs));
3080 			host->addChild(createChildCases(CASETYPE_CALLBACK, m_context, "framebuffer_fetch",			"Negative Framebuffer Fetch Cases",					framebufferFetchFuncs));
3081 		}
3082 
3083 		{
3084 			tcu::TestCaseGroup* const	host	= new tcu::TestCaseGroup(m_testCtx, "log", "Reporting of standard API errors via log");
3085 
3086 			negative->addChild(host);
3087 
3088 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "buffer",					"Negative Buffer API Cases",						bufferFuncs));
3089 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "texture",					"Negative Texture API Cases",						textureFuncs));
3090 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "shader",					"Negative Shader API Cases",						shaderFuncs));
3091 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "fragment",				"Negative Fragment API Cases",						fragmentFuncs));
3092 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "vertex_array",			"Negative Vertex Array API Cases",					vaFuncs));
3093 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "state",					"Negative GL State API Cases",						stateFuncs));
3094 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "atomic_counter",			"Negative Atomic Counter API Cases",				atomicCounterFuncs));
3095 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "shader_image_load",		"Negative Shader Image Load API Cases",				imageLoadFuncs));
3096 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "shader_image_store",		"Negative Shader Image Store API Cases",			imageStoreFuncs));
3097 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "shader_image_atomic",		"Negative Shader Image Atomic API Cases",			imageAtomicFuncs));
3098 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "shader_image_exchange",	"Negative Shader Image Atomic Exchange API Cases",	imageAtomicExchangeFuncs));
3099 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "shader_function",			"Negative Shader Function Cases",					shaderFunctionFuncs));
3100 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "shader_directive",		"Negative Shader Directive Cases",					shaderDirectiveFuncs));
3101 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "ssbo_block",				"Negative SSBO Block Cases",						ssboBlockFuncs));
3102 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "precise",					"Negative Precise Cases",							preciseFuncs));
3103 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "advanced_blend",			"Negative Advanced Blend Equation Cases",			advancedBlendFuncs));
3104 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "shader_storage",			"Negative Shader Storage Cases",					shaderStorageFuncs));
3105 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "tessellation",			"Negative Tessellation Cases",						tessellationFuncs));
3106 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "oes_sample_variables",	"Negative Sample Variables Cases",					sampleVariablesFuncs));
3107 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "compute",					"Negative Compute Cases",							computeFuncs));
3108 			host->addChild(createChildCases(CASETYPE_LOG, m_context, "framebuffer_fetch",		"Negative Framebuffer Fetch Cases",					framebufferFetchFuncs));
3109 		}
3110 
3111 		{
3112 			tcu::TestCaseGroup* const	host	= new tcu::TestCaseGroup(m_testCtx, "get_error", "Reporting of standard API errors via glGetError");
3113 
3114 			negative->addChild(host);
3115 
3116 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "buffer",						"Negative Buffer API Cases",						bufferFuncs));
3117 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "texture",					"Negative Texture API Cases",						textureFuncs));
3118 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "shader",						"Negative Shader API Cases",						shaderFuncs));
3119 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "fragment",					"Negative Fragment API Cases",						fragmentFuncs));
3120 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "vertex_array",				"Negative Vertex Array API Cases",					vaFuncs));
3121 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "state",						"Negative GL State API Cases",						stateFuncs));
3122 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "atomic_counter",				"Negative Atomic Counter API Cases",				atomicCounterFuncs));
3123 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "shader_image_load",			"Negative Shader Image Load API Cases",				imageLoadFuncs));
3124 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "shader_image_store",			"Negative Shader Image Store API Cases",			imageStoreFuncs));
3125 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "shader_image_atomic",		"Negative Shader Image Atomic API Cases",			imageAtomicFuncs));
3126 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "shader_image_exchange",		"Negative Shader Image Atomic Exchange API Cases",	imageAtomicExchangeFuncs));
3127 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "shader_function",			"Negative Shader Function Cases",					shaderFunctionFuncs));
3128 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "shader_directive",			"Negative Shader Directive Cases",					shaderDirectiveFuncs));
3129 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "ssbo_block",					"Negative SSBO Block Cases",						ssboBlockFuncs));
3130 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "precise",					"Negative Precise Cases",							preciseFuncs));
3131 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "advanced_blend",				"Negative Advanced Blend Equation Cases",			advancedBlendFuncs));
3132 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "shader_storage",				"Negative Shader Storage Cases",					shaderStorageFuncs));
3133 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "tessellation",				"Negative Tessellation Cases",						tessellationFuncs));
3134 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "oes_sample_variables",		"Negative Sample Variables Cases",					sampleVariablesFuncs));
3135 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "compute",					"Negative Compute Cases",							computeFuncs));
3136 			host->addChild(createChildCases(CASETYPE_GETERROR, m_context, "framebuffer_fetch",			"Negative Framebuffer Fetch Cases",					framebufferFetchFuncs));
3137 		}
3138 	}
3139 
3140 	{
3141 		tcu::TestCaseGroup* const host = createChildCases(CASETYPE_CALLBACK, m_context, "externally_generated", "Externally Generated Messages", externalFuncs);
3142 
3143 		host->addChild(new GroupCase(m_context, "push_pop_consistency", "Push/pop message generation with full message output checking"));
3144 
3145 		addChild(host);
3146 	}
3147 
3148 	{
3149 		vector<FunctionContainer>	containers;
3150 		vector<TestFunctionWrapper>	allFuncs;
3151 
3152 		de::Random					rng			(0x53941903 ^ m_context.getTestContext().getCommandLine().getBaseSeed());
3153 
3154 		containers.insert(containers.end(), bufferFuncs.begin(), bufferFuncs.end());
3155 		containers.insert(containers.end(), textureFuncs.begin(), textureFuncs.end());
3156 		containers.insert(containers.end(), externalFuncs.begin(), externalFuncs.end());
3157 
3158 		for (size_t ndx = 0; ndx < containers.size(); ndx++)
3159 			allFuncs.push_back(containers[ndx].function);
3160 
3161 		rng.shuffle(allFuncs.begin(), allFuncs.end());
3162 
3163 		{
3164 			tcu::TestCaseGroup* const	filtering				= new tcu::TestCaseGroup(m_testCtx, "error_filters", "Filtering of reported errors");
3165 			const int					errorFuncsPerCase		= 4;
3166 			const int					maxFilteringCaseCount	= 32;
3167 			const int					caseCount				= (int(allFuncs.size()) + errorFuncsPerCase-1) / errorFuncsPerCase;
3168 
3169 			addChild(filtering);
3170 
3171 			for (int caseNdx = 0; caseNdx < de::min(caseCount, maxFilteringCaseCount); caseNdx++)
3172 			{
3173 				const int					start		= caseNdx*errorFuncsPerCase;
3174 				const int					end			= de::min((caseNdx+1)*errorFuncsPerCase, int(allFuncs.size()));
3175 				const string				name		= "case_" + de::toString(caseNdx);
3176 				vector<TestFunctionWrapper>	funcs		(allFuncs.begin()+start, allFuncs.begin()+end);
3177 
3178 				// These produce lots of different message types, thus always include at least one when testing filtering
3179 				funcs.insert(funcs.end(), externalFuncs[caseNdx%externalFuncs.size()].function);
3180 
3181 				filtering->addChild(new FilterCase(m_context, name.c_str(), "DebugMessageControl usage", funcs));
3182 			}
3183 		}
3184 
3185 		{
3186 			tcu::TestCaseGroup* const	groups					= new tcu::TestCaseGroup(m_testCtx, "error_groups", "Filtering of reported errors with use of Error Groups");
3187 			const int					errorFuncsPerCase		= 4;
3188 			const int					maxFilteringCaseCount	= 16;
3189 			const int					caseCount				= (int(allFuncs.size()) + errorFuncsPerCase-1) / errorFuncsPerCase;
3190 
3191 			addChild(groups);
3192 
3193 			for (int caseNdx = 0; caseNdx < caseCount && caseNdx < maxFilteringCaseCount; caseNdx++)
3194 			{
3195 				const int					start		= caseNdx*errorFuncsPerCase;
3196 				const int					end			= de::min((caseNdx+1)*errorFuncsPerCase, int(allFuncs.size()));
3197 				const string				name		= ("case_" + de::toString(caseNdx)).c_str();
3198 				vector<TestFunctionWrapper>	funcs		(&allFuncs[0]+start, &allFuncs[0]+end);
3199 
3200 				// These produce lots of different message types, thus always include at least one when testing filtering
3201 				funcs.insert(funcs.end(), externalFuncs[caseNdx%externalFuncs.size()].function);
3202 
3203 				groups->addChild(new GroupFilterCase(m_context, name.c_str(), "Debug Group usage", funcs));
3204 			}
3205 		}
3206 
3207 		{
3208 			tcu::TestCaseGroup* const	async				= new tcu::TestCaseGroup(m_testCtx, "async", "Asynchronous message generation");
3209 			const int					errorFuncsPerCase	= 2;
3210 			const int					maxAsyncCaseCount	= 16;
3211 			const int					caseCount			= (int(allFuncs.size()) + errorFuncsPerCase-1) / errorFuncsPerCase;
3212 
3213 			addChild(async);
3214 
3215 			for (int caseNdx = 0; caseNdx < caseCount && caseNdx < maxAsyncCaseCount; caseNdx++)
3216 			{
3217 				const int					start		= caseNdx*errorFuncsPerCase;
3218 				const int					end			= de::min((caseNdx+1)*errorFuncsPerCase, int(allFuncs.size()));
3219 				const string				name		= ("case_" + de::toString(caseNdx)).c_str();
3220 				vector<TestFunctionWrapper>	funcs		(&allFuncs[0]+start, &allFuncs[0]+end);
3221 
3222 				if (caseNdx&0x1)
3223 					async->addChild(new AsyncCase(m_context, (name+"_callback").c_str(), "Async message generation", funcs, true));
3224 				else
3225 					async->addChild(new AsyncCase(m_context, (name+"_log").c_str(), "Async message generation", funcs, false));
3226 			}
3227 		}
3228 	}
3229 
3230 	{
3231 		tcu::TestCaseGroup* const labels = new tcu::TestCaseGroup(m_testCtx, "object_labels", "Labeling objects");
3232 
3233 		const struct
3234 		{
3235 			GLenum		identifier;
3236 			const char*	name;
3237 			const char* desc;
3238 		} cases[] =
3239 		{
3240 			{ GL_BUFFER,				"buffer",				"Debug label on a buffer object"				},
3241 			{ GL_SHADER,				"shader",				"Debug label on a shader object"				},
3242 			{ GL_PROGRAM,				"program",				"Debug label on a program object"				},
3243 			{ GL_QUERY,					"query",				"Debug label on a query object"					},
3244 			{ GL_PROGRAM_PIPELINE,		"program_pipeline",		"Debug label on a program pipeline object"		},
3245 			{ GL_TRANSFORM_FEEDBACK,	"transform_feedback",	"Debug label on a transform feedback object"	},
3246 			{ GL_SAMPLER,				"sampler",				"Debug label on a sampler object"				},
3247 			{ GL_TEXTURE,				"texture",				"Debug label on a texture object"				},
3248 			{ GL_RENDERBUFFER,			"renderbuffer",			"Debug label on a renderbuffer object"			},
3249 			{ GL_FRAMEBUFFER,			"framebuffer",			"Debug label on a framebuffer object"			},
3250 		};
3251 
3252 		addChild(labels);
3253 
3254 		labels->addChild(new InitialLabelCase		(m_context, "initial",				"Debug label initial value"));
3255 		labels->addChild(new ClearLabelCase			(m_context, "clearing",				"Debug label clearing"));
3256 		labels->addChild(new SpecifyWithLengthCase	(m_context, "specify_with_length",	"Debug label specified with length"));
3257 		labels->addChild(new BufferLimitedLabelCase	(m_context, "buffer_limited_query",	"Debug label query to too short buffer"));
3258 		labels->addChild(new LabelMaxSizeCase		(m_context, "max_label_length",		"Max sized debug label"));
3259 		labels->addChild(new LabelLengthCase		(m_context, "query_length_only",	"Query debug label length"));
3260 
3261 		for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(cases); ndx++)
3262 			labels->addChild(new LabelCase(m_context, cases[ndx].name, cases[ndx].desc, cases[ndx].identifier));
3263 		labels->addChild(new SyncLabelCase(m_context, "sync", "Debug label on a sync object"));
3264 	}
3265 }
3266 
3267 } // Functional
3268 } // gles31
3269 } // deqp
3270