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 Program interface query test case
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es31fProgramInterfaceQueryTestCase.hpp"
25 #include "es31fProgramInterfaceDefinitionUtil.hpp"
26 #include "tcuTestLog.hpp"
27 #include "gluVarTypeUtil.hpp"
28 #include "gluStrUtil.hpp"
29 #include "gluContextInfo.hpp"
30 #include "gluShaderProgram.hpp"
31 #include "glwFunctions.hpp"
32 #include "glwEnums.hpp"
33 #include "deString.h"
34 #include "deStringUtil.hpp"
35 #include "deSTLUtil.hpp"
36 
37 namespace deqp
38 {
39 namespace gles31
40 {
41 namespace Functional
42 {
43 namespace
44 {
45 
46 using ProgramInterfaceDefinition::VariablePathComponent;
47 using ProgramInterfaceDefinition::VariableSearchFilter;
48 
getProgramDefaultBlockInterfaceFromStorage(glu::Storage storage)49 static glw::GLenum getProgramDefaultBlockInterfaceFromStorage (glu::Storage storage)
50 {
51 	switch (storage)
52 	{
53 		case glu::STORAGE_IN:
54 		case glu::STORAGE_PATCH_IN:
55 			return GL_PROGRAM_INPUT;
56 
57 		case glu::STORAGE_OUT:
58 		case glu::STORAGE_PATCH_OUT:
59 			return GL_PROGRAM_OUTPUT;
60 
61 		case glu::STORAGE_UNIFORM:
62 			return GL_UNIFORM;
63 
64 		default:
65 			DE_ASSERT(false);
66 			return 0;
67 	}
68 }
69 
isBufferBackedInterfaceBlockStorage(glu::Storage storage)70 static bool isBufferBackedInterfaceBlockStorage (glu::Storage storage)
71 {
72 	return storage == glu::STORAGE_BUFFER || storage == glu::STORAGE_UNIFORM;
73 }
74 
getRequiredExtensionForStage(glu::ShaderType stage)75 const char* getRequiredExtensionForStage (glu::ShaderType stage)
76 {
77 	switch (stage)
78 	{
79 		case glu::SHADERTYPE_COMPUTE:
80 		case glu::SHADERTYPE_VERTEX:
81 		case glu::SHADERTYPE_FRAGMENT:
82 			return DE_NULL;
83 
84 		case glu::SHADERTYPE_GEOMETRY:
85 			return "GL_EXT_geometry_shader";
86 
87 		case glu::SHADERTYPE_TESSELLATION_CONTROL:
88 		case glu::SHADERTYPE_TESSELLATION_EVALUATION:
89 			return "GL_EXT_tessellation_shader";
90 
91 		default:
92 			DE_ASSERT(false);
93 			return DE_NULL;
94 	}
95 }
96 
getTypeSize(glu::DataType type)97 static int getTypeSize (glu::DataType type)
98 {
99 	if (type == glu::TYPE_FLOAT)
100 		return 4;
101 	else if (type == glu::TYPE_INT || type == glu::TYPE_UINT)
102 		return 4;
103 	else if (type == glu::TYPE_BOOL)
104 		return 4; // uint
105 
106 	DE_ASSERT(false);
107 	return 0;
108 }
109 
getVarTypeSize(const glu::VarType & type)110 static int getVarTypeSize (const glu::VarType& type)
111 {
112 	if (type.isBasicType())
113 	{
114 		// return in basic machine units
115 		return glu::getDataTypeScalarSize(type.getBasicType()) * getTypeSize(glu::getDataTypeScalarType(type.getBasicType()));
116 	}
117 	else if (type.isStructType())
118 	{
119 		int size = 0;
120 		for (int ndx = 0; ndx < type.getStructPtr()->getNumMembers(); ++ndx)
121 			size += getVarTypeSize(type.getStructPtr()->getMember(ndx).getType());
122 		return size;
123 	}
124 	else if (type.isArrayType())
125 	{
126 		// unsized arrays are handled as if they had only one element
127 		if (type.getArraySize() == glu::VarType::UNSIZED_ARRAY)
128 			return getVarTypeSize(type.getElementType());
129 		else
130 			return type.getArraySize() * getVarTypeSize(type.getElementType());
131 	}
132 	else
133 	{
134 		DE_ASSERT(false);
135 		return 0;
136 	}
137 }
138 
getMatrixOrderFromPath(const std::vector<VariablePathComponent> & path)139 static glu::MatrixOrder getMatrixOrderFromPath (const std::vector<VariablePathComponent>& path)
140 {
141 	glu::MatrixOrder order = glu::MATRIXORDER_LAST;
142 
143 	// inherit majority
144 	for (int pathNdx = 0; pathNdx < (int)path.size(); ++pathNdx)
145 	{
146 		glu::MatrixOrder matOrder;
147 
148 		if (path[pathNdx].isInterfaceBlock())
149 			matOrder = path[pathNdx].getInterfaceBlock()->layout.matrixOrder;
150 		else if (path[pathNdx].isDeclaration())
151 			matOrder = path[pathNdx].getDeclaration()->layout.matrixOrder;
152 		else if (path[pathNdx].isVariableType())
153 			matOrder = glu::MATRIXORDER_LAST;
154 		else
155 		{
156 			DE_ASSERT(false);
157 			return glu::MATRIXORDER_LAST;
158 		}
159 
160 		if (matOrder != glu::MATRIXORDER_LAST)
161 			order = matOrder;
162 	}
163 
164 	return order;
165 }
166 
167 class PropValidator
168 {
169 public:
170 									PropValidator					(Context& context, ProgramResourcePropFlags validationProp, const char* requiredExtension);
171 
172 	virtual std::string				getHumanReadablePropertyString	(glw::GLint propVal) const;
173 	virtual void					validate						(const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const = 0;
174 
175 	bool							isSupported						(void) const;
176 	bool							isSelected						(deUint32 caseFlags) const;
177 
178 protected:
179 	void							setError						(const std::string& err) const;
180 
181 	tcu::TestContext&				m_testCtx;
182 	const glu::RenderContext&		m_renderContext;
183 
184 private:
185 	const glu::ContextInfo&			m_contextInfo;
186 	const char*						m_extension;
187 	const ProgramResourcePropFlags	m_validationProp;
188 };
189 
PropValidator(Context & context,ProgramResourcePropFlags validationProp,const char * requiredExtension)190 PropValidator::PropValidator (Context& context, ProgramResourcePropFlags validationProp, const char* requiredExtension)
191 	: m_testCtx			(context.getTestContext())
192 	, m_renderContext	(context.getRenderContext())
193 	, m_contextInfo		(context.getContextInfo())
194 	, m_extension		(requiredExtension)
195 	, m_validationProp	(validationProp)
196 {
197 }
198 
getHumanReadablePropertyString(glw::GLint propVal) const199 std::string PropValidator::getHumanReadablePropertyString (glw::GLint propVal) const
200 {
201 	return de::toString(propVal);
202 }
203 
isSupported(void) const204 bool PropValidator::isSupported (void) const
205 {
206 	if(glu::contextSupports(m_renderContext.getType(), glu::ApiType::es(3, 2)) ||
207 	   glu::contextSupports(m_renderContext.getType(), glu::ApiType::core(4, 5)))
208 		return true;
209 	return m_extension == DE_NULL || m_contextInfo.isExtensionSupported(m_extension);
210 }
211 
isSelected(deUint32 caseFlags) const212 bool PropValidator::isSelected (deUint32 caseFlags) const
213 {
214 	return (caseFlags & (deUint32)m_validationProp) != 0;
215 }
216 
setError(const std::string & err) const217 void PropValidator::setError (const std::string& err) const
218 {
219 	// don't overwrite earlier errors
220 	if (m_testCtx.getTestResult() == QP_TEST_RESULT_PASS)
221 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, err.c_str());
222 }
223 
224 class SingleVariableValidator : public PropValidator
225 {
226 public:
227 					SingleVariableValidator	(Context& context, ProgramResourcePropFlags validationProp, glw::GLuint programID, const VariableSearchFilter& filter, const char* requiredExtension);
228 
229 	void			validate				(const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
230 	virtual void	validateSingleVariable	(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const = 0;
231 	virtual void	validateBuiltinVariable	(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
232 
233 protected:
234 	const VariableSearchFilter	m_filter;
235 	const glw::GLuint			m_programID;
236 };
237 
SingleVariableValidator(Context & context,ProgramResourcePropFlags validationProp,glw::GLuint programID,const VariableSearchFilter & filter,const char * requiredExtension)238 SingleVariableValidator::SingleVariableValidator (Context& context, ProgramResourcePropFlags validationProp, glw::GLuint programID, const VariableSearchFilter& filter, const char* requiredExtension)
239 	: PropValidator	(context, validationProp, requiredExtension)
240 	, m_filter		(filter)
241 	, m_programID	(programID)
242 {
243 }
244 
validate(const ProgramInterfaceDefinition::Program * program,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const245 void SingleVariableValidator::validate (const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
246 {
247 	std::vector<VariablePathComponent> path;
248 
249 	if (findProgramVariablePathByPathName(path, program, resource, m_filter))
250 	{
251 		const glu::VarType* variable = (path.back().isVariableType()) ? (path.back().getVariableType()) : (DE_NULL);
252 
253 		if (!variable || !variable->isBasicType())
254 		{
255 			m_testCtx.getLog() << tcu::TestLog::Message << "Error, resource name \"" << resource << "\" refers to a non-basic type." << tcu::TestLog::EndMessage;
256 			setError("resource not basic type");
257 		}
258 		else
259 			validateSingleVariable(path, resource, propValue, implementationName);
260 
261 		// finding matching variable in any shader is sufficient
262 		return;
263 	}
264 	else if (deStringBeginsWith(resource.c_str(), "gl_"))
265 	{
266 		// special case for builtins
267 		validateBuiltinVariable(resource, propValue, implementationName);
268 		return;
269 	}
270 
271 	// we are only supplied good names, generated by ourselves
272 	DE_ASSERT(false);
273 	throw tcu::InternalError("Resource name consistency error");
274 }
275 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const276 void SingleVariableValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
277 {
278 	DE_UNREF(resource);
279 	DE_UNREF(propValue);
280 	DE_UNREF(implementationName);
281 	DE_ASSERT(false);
282 }
283 
284 class SingleBlockValidator : public PropValidator
285 {
286 public:
287 								SingleBlockValidator	(Context& context, ProgramResourcePropFlags validationProp, glw::GLuint programID, const VariableSearchFilter& filter, const char* requiredExtension);
288 
289 	void						validate				(const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
290 	virtual void				validateSingleBlock		(const glu::InterfaceBlock& block, const std::vector<int>& instanceIndex, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const = 0;
291 
292 protected:
293 	const VariableSearchFilter	m_filter;
294 	const glw::GLuint			m_programID;
295 };
296 
SingleBlockValidator(Context & context,ProgramResourcePropFlags validationProp,glw::GLuint programID,const VariableSearchFilter & filter,const char * requiredExtension)297 SingleBlockValidator::SingleBlockValidator (Context& context, ProgramResourcePropFlags validationProp, glw::GLuint programID, const VariableSearchFilter& filter, const char* requiredExtension)
298 	: PropValidator	(context, validationProp, requiredExtension)
299 	, m_filter		(filter)
300 	, m_programID	(programID)
301 {
302 }
303 
validate(const ProgramInterfaceDefinition::Program * program,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const304 void SingleBlockValidator::validate (const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
305 {
306 	glu::VarTokenizer	tokenizer		(resource.c_str());
307 	const std::string	blockName		= tokenizer.getIdentifier();
308 	std::vector<int>	instanceIndex;
309 
310 	tokenizer.advance();
311 
312 	// array index
313 	while (tokenizer.getToken() == glu::VarTokenizer::TOKEN_LEFT_BRACKET)
314 	{
315 		tokenizer.advance();
316 		DE_ASSERT(tokenizer.getToken() == glu::VarTokenizer::TOKEN_NUMBER);
317 
318 		instanceIndex.push_back(tokenizer.getNumber());
319 
320 		tokenizer.advance();
321 		DE_ASSERT(tokenizer.getToken() == glu::VarTokenizer::TOKEN_RIGHT_BRACKET);
322 
323 		tokenizer.advance();
324 	}
325 
326 	// no trailing garbage
327 	DE_ASSERT(tokenizer.getToken() == glu::VarTokenizer::TOKEN_END);
328 
329 	for (int shaderNdx = 0; shaderNdx < (int)program->getShaders().size(); ++shaderNdx)
330 	{
331 		const ProgramInterfaceDefinition::Shader* const shader = program->getShaders()[shaderNdx];
332 		if (!m_filter.matchesFilter(shader))
333 			continue;
334 
335 		for (int blockNdx = 0; blockNdx < (int)shader->getDefaultBlock().interfaceBlocks.size(); ++blockNdx)
336 		{
337 			const glu::InterfaceBlock& block = shader->getDefaultBlock().interfaceBlocks[blockNdx];
338 
339 			if (m_filter.matchesFilter(block) && block.interfaceName == blockName)
340 			{
341 				// dimensions match
342 				DE_ASSERT(instanceIndex.size() == block.dimensions.size());
343 
344 				validateSingleBlock(block, instanceIndex, resource, propValue, implementationName);
345 				return;
346 			}
347 		}
348 	}
349 
350 	// we are only supplied good names, generated by ourselves
351 	DE_ASSERT(false);
352 	throw tcu::InternalError("Resource name consistency error");
353 }
354 
355 class TypeValidator : public SingleVariableValidator
356 {
357 public:
358 				TypeValidator					(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
359 
360 	std::string	getHumanReadablePropertyString	(glw::GLint propVal) const;
361 	void		validateSingleVariable			(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
362 	void		validateBuiltinVariable			(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
363 };
364 
TypeValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)365 TypeValidator::TypeValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
366 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_TYPE, programID, filter, DE_NULL)
367 {
368 }
369 
getHumanReadablePropertyString(glw::GLint propVal) const370 std::string TypeValidator::getHumanReadablePropertyString (glw::GLint propVal) const
371 {
372 	return de::toString(glu::getShaderVarTypeStr(propVal));
373 }
374 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const375 void TypeValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
376 {
377 	const glu::VarType* variable = path.back().getVariableType();
378 
379 	DE_UNREF(resource);
380 	DE_UNREF(implementationName);
381 
382 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying type, expecting " << glu::getDataTypeName(variable->getBasicType()) << tcu::TestLog::EndMessage;
383 
384 	if (variable->getBasicType() != glu::getDataTypeFromGLType(propValue))
385 	{
386 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << glu::getDataTypeName(glu::getDataTypeFromGLType(propValue)) << tcu::TestLog::EndMessage;
387 		setError("resource type invalid");
388 	}
389 }
390 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const391 void TypeValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
392 {
393 	DE_UNREF(implementationName);
394 
395 	static const struct
396 	{
397 		const char*		name;
398 		glu::DataType	type;
399 	} builtins[] =
400 	{
401 		{ "gl_Position",				glu::TYPE_FLOAT_VEC4	},
402 		{ "gl_FragCoord",				glu::TYPE_FLOAT_VEC4	},
403 		{ "gl_PerVertex.gl_Position",	glu::TYPE_FLOAT_VEC4	},
404 		{ "gl_VertexID",				glu::TYPE_INT			},
405 		{ "gl_InvocationID",			glu::TYPE_INT			},
406 		{ "gl_NumWorkGroups",			glu::TYPE_UINT_VEC3		},
407 		{ "gl_FragDepth",				glu::TYPE_FLOAT			},
408 		{ "gl_TessLevelOuter[0]",		glu::TYPE_FLOAT			},
409 		{ "gl_TessLevelInner[0]",		glu::TYPE_FLOAT			},
410 	};
411 
412 	for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(builtins); ++ndx)
413 	{
414 		if (resource == builtins[ndx].name)
415 		{
416 			m_testCtx.getLog() << tcu::TestLog::Message << "Verifying type, expecting " << glu::getDataTypeName(builtins[ndx].type) << tcu::TestLog::EndMessage;
417 
418 			if (glu::getDataTypeFromGLType(propValue) != builtins[ndx].type)
419 			{
420 				m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << glu::getDataTypeName(glu::getDataTypeFromGLType(propValue)) << tcu::TestLog::EndMessage;
421 				setError("resource type invalid");
422 			}
423 			return;
424 		}
425 	}
426 
427 	DE_ASSERT(false);
428 }
429 
430 class ArraySizeValidator : public SingleVariableValidator
431 {
432 public:
433 				ArraySizeValidator				(Context& context, glw::GLuint programID, int unsizedArraySize, const VariableSearchFilter& filter);
434 
435 	void		validateSingleVariable			(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
436 	void		validateBuiltinVariable			(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
437 
438 private:
439 	const int	m_unsizedArraySize;
440 };
441 
ArraySizeValidator(Context & context,glw::GLuint programID,int unsizedArraySize,const VariableSearchFilter & filter)442 ArraySizeValidator::ArraySizeValidator (Context& context, glw::GLuint programID, int unsizedArraySize, const VariableSearchFilter& filter)
443 	: SingleVariableValidator	(context, PROGRAMRESOURCEPROP_ARRAY_SIZE, programID, filter, DE_NULL)
444 	, m_unsizedArraySize		(unsizedArraySize)
445 {
446 }
447 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const448 void ArraySizeValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
449 {
450 	const VariablePathComponent		nullComponent;
451 	const VariablePathComponent&	enclosingcomponent	= (path.size() > 1) ? (path[path.size()-2]) : (nullComponent);
452 
453 	const bool						isArray				= enclosingcomponent.isVariableType() && enclosingcomponent.getVariableType()->isArrayType();
454 	const bool						inUnsizedArray		= isArray && (enclosingcomponent.getVariableType()->getArraySize() == glu::VarType::UNSIZED_ARRAY);
455 	const int						arraySize			= (!isArray) ? (1) : (inUnsizedArray) ? (m_unsizedArraySize) : (enclosingcomponent.getVariableType()->getArraySize());
456 
457 	DE_ASSERT(arraySize >= 0);
458 	DE_UNREF(resource);
459 	DE_UNREF(implementationName);
460 
461 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying array size, expecting " << arraySize << tcu::TestLog::EndMessage;
462 
463 	if (arraySize != propValue)
464 	{
465 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
466 		setError("resource array size invalid");
467 	}
468 }
469 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const470 void ArraySizeValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
471 {
472 	DE_UNREF(implementationName);
473 
474 	static const struct
475 	{
476 		const char*		name;
477 		int				arraySize;
478 	} builtins[] =
479 	{
480 		{ "gl_Position",				1	},
481 		{ "gl_VertexID",				1	},
482 		{ "gl_FragCoord",				1	},
483 		{ "gl_PerVertex.gl_Position",	1	},
484 		{ "gl_InvocationID",			1	},
485 		{ "gl_NumWorkGroups",			1	},
486 		{ "gl_FragDepth",				1	},
487 		{ "gl_TessLevelOuter[0]",		4	},
488 		{ "gl_TessLevelInner[0]",		2	},
489 	};
490 
491 	for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(builtins); ++ndx)
492 	{
493 		if (resource == builtins[ndx].name)
494 		{
495 			m_testCtx.getLog() << tcu::TestLog::Message << "Verifying array size, expecting " << builtins[ndx].arraySize << tcu::TestLog::EndMessage;
496 
497 			if (propValue != builtins[ndx].arraySize)
498 			{
499 				m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
500 				setError("resource array size invalid");
501 			}
502 			return;
503 		}
504 	}
505 
506 	DE_ASSERT(false);
507 }
508 
509 class ArrayStrideValidator : public SingleVariableValidator
510 {
511 public:
512 				ArrayStrideValidator			(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
513 
514 	void		validateSingleVariable			(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
515 };
516 
ArrayStrideValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)517 ArrayStrideValidator::ArrayStrideValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
518 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_ARRAY_STRIDE, programID, filter, DE_NULL)
519 {
520 }
521 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const522 void ArrayStrideValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
523 {
524 	const VariablePathComponent		nullComponent;
525 	const VariablePathComponent&	component			= path.back();
526 	const VariablePathComponent&	enclosingcomponent	= (path.size() > 1) ? (path[path.size()-2]) : (nullComponent);
527 	const VariablePathComponent&	firstComponent		= path.front();
528 
529 	const bool						isBufferBlock		= firstComponent.isInterfaceBlock() && isBufferBackedInterfaceBlockStorage(firstComponent.getInterfaceBlock()->storage);
530 	const bool						isArray				= enclosingcomponent.isVariableType() && enclosingcomponent.getVariableType()->isArrayType();
531 	const bool						isAtomicCounter		= glu::isDataTypeAtomicCounter(component.getVariableType()->getBasicType()); // atomic counters are buffer backed with a stride of 4 basic machine units
532 
533 	DE_UNREF(resource);
534 	DE_UNREF(implementationName);
535 
536 	// Layout tests will verify layouts of buffer backed arrays properly. Here we just check values are greater or equal to the element size
537 	if (isBufferBlock && isArray)
538 	{
539 		const int elementSize = glu::getDataTypeScalarSize(component.getVariableType()->getBasicType()) * getTypeSize(glu::getDataTypeScalarType(component.getVariableType()->getBasicType()));
540 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying array stride, expecting greater or equal to " << elementSize << tcu::TestLog::EndMessage;
541 
542 		if (propValue < elementSize)
543 		{
544 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
545 			setError("resource array stride invalid");
546 		}
547 	}
548 	else
549 	{
550 		// Atomics are buffer backed with stride of 4 even though they are not in an interface block
551 		const int arrayStride = (isAtomicCounter && isArray) ? (4) : (!isBufferBlock && !isAtomicCounter) ? (-1) : (0);
552 
553 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying array stride, expecting " << arrayStride << tcu::TestLog::EndMessage;
554 
555 		if (arrayStride != propValue)
556 		{
557 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
558 			setError("resource array stride invalid");
559 		}
560 	}
561 }
562 
563 class BlockIndexValidator : public SingleVariableValidator
564 {
565 public:
566 				BlockIndexValidator				(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
567 
568 	void		validateSingleVariable			(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
569 };
570 
BlockIndexValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)571 BlockIndexValidator::BlockIndexValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
572 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_BLOCK_INDEX, programID, filter, DE_NULL)
573 {
574 }
575 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const576 void BlockIndexValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
577 {
578 	const VariablePathComponent& firstComponent = path.front();
579 
580 	DE_UNREF(resource);
581 	DE_UNREF(implementationName);
582 
583 	if (!firstComponent.isInterfaceBlock())
584 	{
585 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying block index, expecting -1" << tcu::TestLog::EndMessage;
586 
587 		if (propValue != -1)
588 		{
589 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
590 			setError("resource block index invalid");
591 		}
592 	}
593 	else
594 	{
595 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying block index, expecting a valid block index" << tcu::TestLog::EndMessage;
596 
597 		if (propValue == -1)
598 		{
599 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
600 			setError("resource block index invalid");
601 		}
602 		else
603 		{
604 			const glw::Functions&	gl			= m_renderContext.getFunctions();
605 			const glw::GLenum		interface	= (firstComponent.getInterfaceBlock()->storage == glu::STORAGE_UNIFORM) ? (GL_UNIFORM_BLOCK) :
606 												  (firstComponent.getInterfaceBlock()->storage == glu::STORAGE_BUFFER) ? (GL_SHADER_STORAGE_BLOCK) :
607 												  (0);
608 			glw::GLint				written		= 0;
609 			std::vector<char>		nameBuffer	(firstComponent.getInterfaceBlock()->interfaceName.size() + 3 * firstComponent.getInterfaceBlock()->dimensions.size() + 2, '\0'); // +3 for appended "[N]", +1 for '\0' and +1 just for safety
610 
611 			gl.getProgramResourceName(m_programID, interface, propValue, (int)nameBuffer.size() - 1, &written, &nameBuffer[0]);
612 			GLU_EXPECT_NO_ERROR(gl.getError(), "query block name");
613 			TCU_CHECK(written < (int)nameBuffer.size());
614 			TCU_CHECK(nameBuffer.back() == '\0');
615 
616 			{
617 				const std::string	blockName		(&nameBuffer[0], written);
618 				std::ostringstream	expectedName;
619 
620 				expectedName << firstComponent.getInterfaceBlock()->interfaceName;
621 				for (int dimensionNdx = 0; dimensionNdx < (int)firstComponent.getInterfaceBlock()->dimensions.size(); ++dimensionNdx)
622 					expectedName << "[0]";
623 
624 				m_testCtx.getLog() << tcu::TestLog::Message << "Block name with index " << propValue << " is \"" << blockName << "\"" << tcu::TestLog::EndMessage;
625 				if (blockName != expectedName.str())
626 				{
627 					m_testCtx.getLog() << tcu::TestLog::Message << "\tError, expected " << expectedName.str() << tcu::TestLog::EndMessage;
628 					setError("resource block index invalid");
629 				}
630 			}
631 		}
632 	}
633 }
634 
635 class IsRowMajorValidator : public SingleVariableValidator
636 {
637 public:
638 				IsRowMajorValidator				(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
639 
640 	std::string getHumanReadablePropertyString	(glw::GLint propVal) const;
641 	void		validateSingleVariable			(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
642 };
643 
IsRowMajorValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)644 IsRowMajorValidator::IsRowMajorValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
645 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_MATRIX_ROW_MAJOR, programID, filter, DE_NULL)
646 {
647 }
648 
getHumanReadablePropertyString(glw::GLint propVal) const649 std::string IsRowMajorValidator::getHumanReadablePropertyString (glw::GLint propVal) const
650 {
651 	return de::toString(glu::getBooleanStr(propVal));
652 }
653 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const654 void IsRowMajorValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
655 {
656 	const VariablePathComponent&	component			= path.back();
657 	const VariablePathComponent&	firstComponent		= path.front();
658 
659 	const bool						isBufferBlock		= firstComponent.isInterfaceBlock() && isBufferBackedInterfaceBlockStorage(firstComponent.getInterfaceBlock()->storage);
660 	const bool						isMatrix			= glu::isDataTypeMatrix(component.getVariableType()->getBasicType());
661 	const int						expected			= (isBufferBlock && isMatrix && getMatrixOrderFromPath(path) == glu::MATRIXORDER_ROW_MAJOR) ? (1) : (0);
662 
663 	DE_UNREF(resource);
664 	DE_UNREF(implementationName);
665 
666 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying matrix order, expecting IS_ROW_MAJOR = " << expected << tcu::TestLog::EndMessage;
667 
668 	if (propValue != expected)
669 	{
670 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
671 		setError("resource matrix order invalid");
672 	}
673 }
674 
675 class MatrixStrideValidator : public SingleVariableValidator
676 {
677 public:
678 				MatrixStrideValidator			(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
679 
680 	void		validateSingleVariable			(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
681 };
682 
MatrixStrideValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)683 MatrixStrideValidator::MatrixStrideValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
684 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_MATRIX_STRIDE, programID, filter, DE_NULL)
685 {
686 }
687 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const688 void MatrixStrideValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
689 {
690 	const VariablePathComponent&	component			= path.back();
691 	const VariablePathComponent&	firstComponent		= path.front();
692 
693 	const bool						isBufferBlock		= firstComponent.isInterfaceBlock() && isBufferBackedInterfaceBlockStorage(firstComponent.getInterfaceBlock()->storage);
694 	const bool						isMatrix			= glu::isDataTypeMatrix(component.getVariableType()->getBasicType());
695 
696 	DE_UNREF(resource);
697 	DE_UNREF(implementationName);
698 
699 	// Layout tests will verify layouts of buffer backed arrays properly. Here we just check the stride is is greater or equal to the row/column size
700 	if (isBufferBlock && isMatrix)
701 	{
702 		const bool	columnMajor			= getMatrixOrderFromPath(path) != glu::MATRIXORDER_ROW_MAJOR;
703 		const int	numMajorElements	= (columnMajor) ? (glu::getDataTypeMatrixNumRows(component.getVariableType()->getBasicType())) : (glu::getDataTypeMatrixNumColumns(component.getVariableType()->getBasicType()));
704 		const int	majorSize			= numMajorElements * getTypeSize(glu::getDataTypeScalarType(component.getVariableType()->getBasicType()));
705 
706 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying matrix stride, expecting greater or equal to " << majorSize << tcu::TestLog::EndMessage;
707 
708 		if (propValue < majorSize)
709 		{
710 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
711 			setError("resource matrix stride invalid");
712 		}
713 	}
714 	else
715 	{
716 		const int matrixStride = (!isBufferBlock && !glu::isDataTypeAtomicCounter(component.getVariableType()->getBasicType())) ? (-1) : (0);
717 
718 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying matrix stride, expecting " << matrixStride << tcu::TestLog::EndMessage;
719 
720 		if (matrixStride != propValue)
721 		{
722 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
723 			setError("resource matrix stride invalid");
724 		}
725 	}
726 }
727 
728 class AtomicCounterBufferIndexVerifier : public SingleVariableValidator
729 {
730 public:
731 				AtomicCounterBufferIndexVerifier	(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
732 
733 	void		validateSingleVariable				(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
734 };
735 
AtomicCounterBufferIndexVerifier(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)736 AtomicCounterBufferIndexVerifier::AtomicCounterBufferIndexVerifier (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
737 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_ATOMIC_COUNTER_BUFFER_INDEX, programID, filter, DE_NULL)
738 {
739 }
740 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const741 void AtomicCounterBufferIndexVerifier::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
742 {
743 	DE_UNREF(resource);
744 	DE_UNREF(implementationName);
745 
746 	if (!glu::isDataTypeAtomicCounter(path.back().getVariableType()->getBasicType()))
747 	{
748 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying atomic counter buffer index, expecting -1" << tcu::TestLog::EndMessage;
749 
750 		if (propValue != -1)
751 		{
752 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
753 			setError("resource atomic counter buffer index invalid");
754 		}
755 	}
756 	else
757 	{
758 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying atomic counter buffer index, expecting a valid index" << tcu::TestLog::EndMessage;
759 
760 		if (propValue == -1)
761 		{
762 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
763 			setError("resource atomic counter buffer index invalid");
764 		}
765 		else
766 		{
767 			const glw::Functions&	gl					= m_renderContext.getFunctions();
768 			glw::GLint				numActiveResources	= 0;
769 
770 			gl.getProgramInterfaceiv(m_programID, GL_ATOMIC_COUNTER_BUFFER, GL_ACTIVE_RESOURCES, &numActiveResources);
771 			GLU_EXPECT_NO_ERROR(gl.getError(), "getProgramInterfaceiv(..., GL_ATOMIC_COUNTER_BUFFER, GL_ACTIVE_RESOURCES, ...)");
772 
773 			if (propValue >= numActiveResources)
774 			{
775 				m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << ", GL_ACTIVE_RESOURCES = " << numActiveResources << tcu::TestLog::EndMessage;
776 				setError("resource atomic counter buffer index invalid");
777 			}
778 		}
779 	}
780 }
781 
782 class LocationValidator : public SingleVariableValidator
783 {
784 public:
785 				LocationValidator		(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
786 
787 	void		validateSingleVariable	(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
788 	void		validateBuiltinVariable	(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
789 };
790 
LocationValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)791 LocationValidator::LocationValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
792 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_LOCATION, programID, filter, DE_NULL)
793 {
794 }
795 
getVariableLocationLength(const glu::VarType & type)796 static int getVariableLocationLength (const glu::VarType& type)
797 {
798 	if (type.isBasicType())
799 	{
800 		if (glu::isDataTypeMatrix(type.getBasicType()))
801 			return glu::getDataTypeMatrixNumColumns(type.getBasicType());
802 		else
803 			return 1;
804 	}
805 	else if (type.isStructType())
806 	{
807 		int size = 0;
808 		for (int ndx = 0; ndx < type.getStructPtr()->getNumMembers(); ++ndx)
809 			size += getVariableLocationLength(type.getStructPtr()->getMember(ndx).getType());
810 		return size;
811 	}
812 	else if (type.isArrayType())
813 		return type.getArraySize() * getVariableLocationLength(type.getElementType());
814 	else
815 	{
816 		DE_ASSERT(false);
817 		return 0;
818 	}
819 }
820 
getIOSubVariableLocation(const std::vector<VariablePathComponent> & path,int startNdx,int currentLocation)821 static int getIOSubVariableLocation (const std::vector<VariablePathComponent>& path, int startNdx, int currentLocation)
822 {
823 	if (currentLocation == -1)
824 		return -1;
825 
826 	if (path[startNdx].getVariableType()->isBasicType())
827 		return currentLocation;
828 	else if (path[startNdx].getVariableType()->isArrayType())
829 		return getIOSubVariableLocation(path, startNdx+1, currentLocation);
830 	else if (path[startNdx].getVariableType()->isStructType())
831 	{
832 		for (int ndx = 0; ndx < path[startNdx].getVariableType()->getStructPtr()->getNumMembers(); ++ndx)
833 		{
834 			if (&path[startNdx].getVariableType()->getStructPtr()->getMember(ndx).getType() == path[startNdx + 1].getVariableType())
835 				return getIOSubVariableLocation(path, startNdx + 1, currentLocation);
836 
837 			if (currentLocation != -1)
838 				currentLocation += getVariableLocationLength(path[startNdx].getVariableType()->getStructPtr()->getMember(ndx).getType());
839 		}
840 
841 		// could not find member, never happens
842 		DE_ASSERT(false);
843 		return -1;
844 	}
845 	else
846 	{
847 		DE_ASSERT(false);
848 		return -1;
849 	}
850 }
851 
getIOBlockVariableLocation(const std::vector<VariablePathComponent> & path)852 static int getIOBlockVariableLocation (const std::vector<VariablePathComponent>& path)
853 {
854 	const glu::InterfaceBlock*	block			= path.front().getInterfaceBlock();
855 	int							currentLocation	= block->layout.location;
856 
857 	// Find the block member
858 	for (int memberNdx = 0; memberNdx < (int)block->variables.size(); ++memberNdx)
859 	{
860 		if (block->variables[memberNdx].layout.location != -1)
861 			currentLocation = block->variables[memberNdx].layout.location;
862 
863 		if (&block->variables[memberNdx] == path[1].getDeclaration())
864 			break;
865 
866 		// unspecified + unspecified = unspecified
867 		if (currentLocation != -1)
868 			currentLocation += getVariableLocationLength(block->variables[memberNdx].varType);
869 	}
870 
871 	// Find subtype location in the complex type
872 	return getIOSubVariableLocation(path, 2, currentLocation);
873 }
874 
getExplicitLocationFromPath(const std::vector<VariablePathComponent> & path)875 static int getExplicitLocationFromPath (const std::vector<VariablePathComponent>& path)
876 {
877 	const glu::VariableDeclaration* varDecl = (path[0].isInterfaceBlock()) ? (path[1].getDeclaration()) : (path[0].getDeclaration());
878 
879 	if (path.front().isInterfaceBlock() && path.front().getInterfaceBlock()->storage == glu::STORAGE_UNIFORM)
880 	{
881 		// inside uniform block
882 		return -1;
883 	}
884 	else if (path.front().isInterfaceBlock() && (path.front().getInterfaceBlock()->storage == glu::STORAGE_IN		||
885 												 path.front().getInterfaceBlock()->storage == glu::STORAGE_OUT		||
886 												 path.front().getInterfaceBlock()->storage == glu::STORAGE_PATCH_IN	||
887 												 path.front().getInterfaceBlock()->storage == glu::STORAGE_PATCH_OUT))
888 	{
889 		// inside ioblock
890 		return getIOBlockVariableLocation(path);
891 	}
892 	else if (varDecl->storage == glu::STORAGE_UNIFORM)
893 	{
894 		// default block uniform
895 		return varDecl->layout.location;
896 	}
897 	else if (varDecl->storage == glu::STORAGE_IN		||
898 			 varDecl->storage == glu::STORAGE_OUT		||
899 			 varDecl->storage == glu::STORAGE_PATCH_IN	||
900 			 varDecl->storage == glu::STORAGE_PATCH_OUT)
901 	{
902 		// default block input/output
903 		return getIOSubVariableLocation(path, 1, varDecl->layout.location);
904 	}
905 	else
906 	{
907 		DE_ASSERT(false);
908 		return -1;
909 	}
910 }
911 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const912 void LocationValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
913 {
914 	const bool			isAtomicCounterUniform	= glu::isDataTypeAtomicCounter(path.back().getVariableType()->getBasicType());
915 	const bool			isUniformBlockVariable	= path.front().isInterfaceBlock() && path.front().getInterfaceBlock()->storage == glu::STORAGE_UNIFORM;
916 	const bool			isVertexShader			= m_filter.getShaderTypeBits() == (1u << glu::SHADERTYPE_VERTEX);
917 	const bool			isFragmentShader		= m_filter.getShaderTypeBits() == (1u << glu::SHADERTYPE_FRAGMENT);
918 	const glu::Storage	storage					= (path.front().isInterfaceBlock()) ? (path.front().getInterfaceBlock()->storage) : (path.front().getDeclaration()->storage);
919 	const bool			isInputVariable			= (storage == glu::STORAGE_IN || storage == glu::STORAGE_PATCH_IN);
920 	const bool			isOutputVariable		= (storage == glu::STORAGE_OUT || storage == glu::STORAGE_PATCH_OUT);
921 	const int			explicitLayoutLocation	= getExplicitLocationFromPath(path);
922 
923 	bool				expectLocation;
924 	std::string			reasonStr;
925 
926 	DE_UNREF(resource);
927 
928 	if (isAtomicCounterUniform)
929 	{
930 		expectLocation = false;
931 		reasonStr = "Atomic counter uniforms have effective location of -1";
932 	}
933 	else if (isUniformBlockVariable)
934 	{
935 		expectLocation = false;
936 		reasonStr = "Uniform block variables have effective location of -1";
937 	}
938 	else if (isInputVariable && !isVertexShader && explicitLayoutLocation == -1)
939 	{
940 		expectLocation = false;
941 		reasonStr = "Inputs (except for vertex shader inputs) not declared with a location layout qualifier have effective location of -1";
942 	}
943 	else if (isOutputVariable && !isFragmentShader && explicitLayoutLocation == -1)
944 	{
945 		expectLocation = false;
946 		reasonStr = "Outputs (except for fragment shader outputs) not declared with a location layout qualifier have effective location of -1";
947 	}
948 	else
949 	{
950 		expectLocation = true;
951 	}
952 
953 	if (!expectLocation)
954 	{
955 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying uniform location, expecting -1. (" << reasonStr << ")" << tcu::TestLog::EndMessage;
956 
957 		if (propValue != -1)
958 		{
959 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
960 			setError("resource location invalid");
961 		}
962 	}
963 	else
964 	{
965 		bool locationOk;
966 
967 		if (explicitLayoutLocation == -1)
968 		{
969 			m_testCtx.getLog() << tcu::TestLog::Message << "Verifying location, expecting a valid location" << tcu::TestLog::EndMessage;
970 			locationOk = (propValue != -1);
971 		}
972 		else
973 		{
974 			m_testCtx.getLog() << tcu::TestLog::Message << "Verifying location, expecting " << explicitLayoutLocation << tcu::TestLog::EndMessage;
975 			locationOk = (propValue == explicitLayoutLocation);
976 		}
977 
978 		if (!locationOk)
979 		{
980 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
981 			setError("resource location invalid");
982 		}
983 		else
984 		{
985 			const VariablePathComponent		nullComponent;
986 			const VariablePathComponent&	enclosingcomponent	= (path.size() > 1) ? (path[path.size()-2]) : (nullComponent);
987 			const bool						isArray				= enclosingcomponent.isVariableType() && enclosingcomponent.getVariableType()->isArrayType();
988 
989 			const glw::Functions&			gl					= m_renderContext.getFunctions();
990 			const glw::GLenum				interface			= getProgramDefaultBlockInterfaceFromStorage(storage);
991 
992 			m_testCtx.getLog() << tcu::TestLog::Message << "Comparing location to the values returned by GetProgramResourceLocation" << tcu::TestLog::EndMessage;
993 
994 			// Test all bottom-level array elements
995 			if (isArray)
996 			{
997 				const std::string arrayResourceName = (implementationName.size() > 3) ? (implementationName.substr(0, implementationName.size() - 3)) : (""); // chop "[0]"
998 
999 				for (int arrayElementNdx = 0; arrayElementNdx < enclosingcomponent.getVariableType()->getArraySize(); ++arrayElementNdx)
1000 				{
1001 					const std::string	elementResourceName	= arrayResourceName + "[" + de::toString(arrayElementNdx) + "]";
1002 					const glw::GLint	location			= gl.getProgramResourceLocation(m_programID, interface, elementResourceName.c_str());
1003 
1004 					if (location != propValue+arrayElementNdx)
1005 					{
1006 						m_testCtx.getLog()
1007 							<< tcu::TestLog::Message
1008 							<< "\tError, getProgramResourceLocation (resource=\"" << elementResourceName << "\") returned location " << location
1009 							<< ", expected " << (propValue+arrayElementNdx)
1010 							<< tcu::TestLog::EndMessage;
1011 						setError("resource location invalid");
1012 					}
1013 					else
1014 						m_testCtx.getLog() << tcu::TestLog::Message << "\tLocation of \"" << elementResourceName << "\":\t" << location << tcu::TestLog::EndMessage;
1015 				}
1016 			}
1017 			else
1018 			{
1019 				const glw::GLint location = gl.getProgramResourceLocation(m_programID, interface, implementationName.c_str());
1020 
1021 				if (location != propValue)
1022 				{
1023 					m_testCtx.getLog() << tcu::TestLog::Message << "\tError, getProgramResourceLocation returned location " << location << ", expected " << propValue << tcu::TestLog::EndMessage;
1024 					setError("resource location invalid");
1025 				}
1026 			}
1027 
1028 		}
1029 	}
1030 }
1031 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1032 void LocationValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1033 {
1034 	DE_UNREF(resource);
1035 	DE_UNREF(implementationName);
1036 
1037 	// built-ins have no location
1038 
1039 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying location, expecting -1" << tcu::TestLog::EndMessage;
1040 
1041 	if (propValue != -1)
1042 	{
1043 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
1044 		setError("resource location invalid");
1045 	}
1046 }
1047 
1048 class VariableNameLengthValidator : public SingleVariableValidator
1049 {
1050 public:
1051 				VariableNameLengthValidator	(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
1052 
1053 	void		validateSingleVariable		(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1054 	void		validateBuiltinVariable		(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1055 	void		validateNameLength			(const std::string& implementationName, glw::GLint propValue) const;
1056 };
1057 
VariableNameLengthValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)1058 VariableNameLengthValidator::VariableNameLengthValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
1059 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_NAME_LENGTH, programID, filter, DE_NULL)
1060 {
1061 }
1062 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1063 void VariableNameLengthValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1064 {
1065 	DE_UNREF(path);
1066 	DE_UNREF(resource);
1067 	validateNameLength(implementationName, propValue);
1068 }
1069 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1070 void VariableNameLengthValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1071 {
1072 	DE_UNREF(resource);
1073 	validateNameLength(implementationName, propValue);
1074 }
1075 
validateNameLength(const std::string & implementationName,glw::GLint propValue) const1076 void VariableNameLengthValidator::validateNameLength (const std::string& implementationName, glw::GLint propValue) const
1077 {
1078 	const int expected = (int)implementationName.length() + 1; // includes null byte
1079 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying name length, expecting " << expected << " (" << (int)implementationName.length() << " for \"" << implementationName << "\" + 1 byte for terminating null character)" << tcu::TestLog::EndMessage;
1080 
1081 	if (propValue != expected)
1082 	{
1083 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid name length, got " << propValue << tcu::TestLog::EndMessage;
1084 		setError("name length invalid");
1085 	}
1086 }
1087 
1088 class OffsetValidator : public SingleVariableValidator
1089 {
1090 public:
1091 				OffsetValidator			(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
1092 
1093 	void		validateSingleVariable	(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1094 };
1095 
OffsetValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)1096 OffsetValidator::OffsetValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
1097 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_OFFSET, programID, filter, DE_NULL)
1098 {
1099 }
1100 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1101 void OffsetValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1102 {
1103 	const bool isAtomicCounterUniform		= glu::isDataTypeAtomicCounter(path.back().getVariableType()->getBasicType());
1104 	const bool isBufferBackedBlockStorage	= path.front().isInterfaceBlock() && isBufferBackedInterfaceBlockStorage(path.front().getInterfaceBlock()->storage);
1105 
1106 	DE_UNREF(resource);
1107 	DE_UNREF(implementationName);
1108 
1109 	if (!isAtomicCounterUniform && !isBufferBackedBlockStorage)
1110 	{
1111 		// Not buffer backed
1112 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying offset, expecting -1" << tcu::TestLog::EndMessage;
1113 
1114 		if (propValue != -1)
1115 		{
1116 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid offset, got " << propValue << tcu::TestLog::EndMessage;
1117 			setError("offset invalid");
1118 		}
1119 	}
1120 	else
1121 	{
1122 		// Expect a valid offset
1123 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying offset, expecting a valid offset" << tcu::TestLog::EndMessage;
1124 
1125 		if (propValue < 0)
1126 		{
1127 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid offset, got " << propValue << tcu::TestLog::EndMessage;
1128 			setError("offset invalid");
1129 		}
1130 	}
1131 }
1132 
1133 class VariableReferencedByShaderValidator : public PropValidator
1134 {
1135 public:
1136 								VariableReferencedByShaderValidator	(Context& context, glu::ShaderType shaderType, const VariableSearchFilter& searchFilter);
1137 
1138 	std::string					getHumanReadablePropertyString		(glw::GLint propVal) const;
1139 	void						validate							(const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1140 
1141 private:
1142 	const VariableSearchFilter	m_filter;
1143 	const glu::ShaderType		m_shaderType;
1144 };
1145 
VariableReferencedByShaderValidator(Context & context,glu::ShaderType shaderType,const VariableSearchFilter & searchFilter)1146 VariableReferencedByShaderValidator::VariableReferencedByShaderValidator (Context& context, glu::ShaderType shaderType, const VariableSearchFilter& searchFilter)
1147 	: PropValidator	(context, PROGRAMRESOURCEPROP_REFERENCED_BY_SHADER, getRequiredExtensionForStage(shaderType))
1148 	, m_filter		(VariableSearchFilter::logicalAnd(VariableSearchFilter::createShaderTypeFilter(shaderType), searchFilter))
1149 	, m_shaderType	(shaderType)
1150 {
1151 	DE_ASSERT(m_shaderType < glu::SHADERTYPE_LAST);
1152 }
1153 
getHumanReadablePropertyString(glw::GLint propVal) const1154 std::string VariableReferencedByShaderValidator::getHumanReadablePropertyString (glw::GLint propVal) const
1155 {
1156 	return de::toString(glu::getBooleanStr(propVal));
1157 }
1158 
validate(const ProgramInterfaceDefinition::Program * program,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1159 void VariableReferencedByShaderValidator::validate (const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1160 {
1161 	DE_UNREF(implementationName);
1162 
1163 	std::vector<VariablePathComponent>	dummyPath;
1164 	const bool							referencedByShader = findProgramVariablePathByPathName(dummyPath, program, resource, m_filter);
1165 
1166 	m_testCtx.getLog()
1167 		<< tcu::TestLog::Message
1168 		<< "Verifying referenced by " << glu::getShaderTypeName(m_shaderType) << " shader, expecting "
1169 		<< ((referencedByShader) ? ("GL_TRUE") : ("GL_FALSE"))
1170 		<< tcu::TestLog::EndMessage;
1171 
1172 	if (propValue != ((referencedByShader) ? (1) : (0)))
1173 	{
1174 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid referenced_by_" << glu::getShaderTypeName(m_shaderType) << ", got " << propValue << tcu::TestLog::EndMessage;
1175 		setError("referenced_by_" + std::string(glu::getShaderTypeName(m_shaderType)) + " invalid");
1176 	}
1177 }
1178 
1179 class BlockNameLengthValidator : public SingleBlockValidator
1180 {
1181 public:
1182 			BlockNameLengthValidator	(Context& context, const glw::GLuint programID, const VariableSearchFilter& filter);
1183 
1184 	void	validateSingleBlock			(const glu::InterfaceBlock& block, const std::vector<int>& instanceIndex, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1185 };
1186 
BlockNameLengthValidator(Context & context,const glw::GLuint programID,const VariableSearchFilter & filter)1187 BlockNameLengthValidator::BlockNameLengthValidator (Context& context, const glw::GLuint programID, const VariableSearchFilter& filter)
1188 	: SingleBlockValidator(context, PROGRAMRESOURCEPROP_NAME_LENGTH, programID, filter, DE_NULL)
1189 {
1190 }
1191 
validateSingleBlock(const glu::InterfaceBlock & block,const std::vector<int> & instanceIndex,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1192 void BlockNameLengthValidator::validateSingleBlock (const glu::InterfaceBlock& block, const std::vector<int>& instanceIndex, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1193 {
1194 	DE_UNREF(instanceIndex);
1195 	DE_UNREF(block);
1196 	DE_UNREF(resource);
1197 
1198 	const int expected = (int)implementationName.length() + 1; // includes null byte
1199 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying name length, expecting " << expected << " (" << (int)implementationName.length() << " for \"" << implementationName << "\" + 1 byte for terminating null character)" << tcu::TestLog::EndMessage;
1200 
1201 	if (propValue != expected)
1202 	{
1203 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid name length, got " << propValue << tcu::TestLog::EndMessage;
1204 		setError("name length invalid");
1205 	}
1206 }
1207 
1208 class BufferBindingValidator : public SingleBlockValidator
1209 {
1210 public:
1211 			BufferBindingValidator	(Context& context, const glw::GLuint programID, const VariableSearchFilter& filter);
1212 
1213 	void	validateSingleBlock		(const glu::InterfaceBlock& block, const std::vector<int>& instanceIndex, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1214 };
1215 
BufferBindingValidator(Context & context,const glw::GLuint programID,const VariableSearchFilter & filter)1216 BufferBindingValidator::BufferBindingValidator (Context& context, const glw::GLuint programID, const VariableSearchFilter& filter)
1217 	: SingleBlockValidator(context, PROGRAMRESOURCEPROP_BUFFER_BINDING, programID, filter, DE_NULL)
1218 {
1219 }
1220 
validateSingleBlock(const glu::InterfaceBlock & block,const std::vector<int> & instanceIndex,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1221 void BufferBindingValidator::validateSingleBlock (const glu::InterfaceBlock& block, const std::vector<int>& instanceIndex, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1222 {
1223 	DE_UNREF(resource);
1224 	DE_UNREF(implementationName);
1225 
1226 	if (block.layout.binding != -1)
1227 	{
1228 		int flatIndex		= 0;
1229 		int dimensionSize	= 1;
1230 
1231 		for (int dimensionNdx = (int)(block.dimensions.size()) - 1; dimensionNdx >= 0; --dimensionNdx)
1232 		{
1233 			flatIndex += dimensionSize * instanceIndex[dimensionNdx];
1234 			dimensionSize *= block.dimensions[dimensionNdx];
1235 		}
1236 
1237 		const int expected = (block.dimensions.empty()) ? (block.layout.binding) : (block.layout.binding + flatIndex);
1238 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying block binding, expecting " << expected << tcu::TestLog::EndMessage;
1239 
1240 		if (propValue != expected)
1241 		{
1242 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid buffer binding, got " << propValue << tcu::TestLog::EndMessage;
1243 			setError("buffer binding invalid");
1244 		}
1245 	}
1246 	else
1247 	{
1248 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying buffer binding, expecting a valid binding" << tcu::TestLog::EndMessage;
1249 
1250 		if (propValue < 0)
1251 		{
1252 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid buffer binding, got " << propValue << tcu::TestLog::EndMessage;
1253 			setError("buffer binding invalid");
1254 		}
1255 	}
1256 }
1257 
1258 class BlockReferencedByShaderValidator : public PropValidator
1259 {
1260 public:
1261 								BlockReferencedByShaderValidator	(Context& context, glu::ShaderType shaderType, const VariableSearchFilter& searchFilter);
1262 
1263 	std::string					getHumanReadablePropertyString		(glw::GLint propVal) const;
1264 	void						validate							(const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1265 
1266 private:
1267 	const VariableSearchFilter	m_filter;
1268 	const glu::ShaderType		m_shaderType;
1269 };
1270 
BlockReferencedByShaderValidator(Context & context,glu::ShaderType shaderType,const VariableSearchFilter & searchFilter)1271 BlockReferencedByShaderValidator::BlockReferencedByShaderValidator (Context& context, glu::ShaderType shaderType, const VariableSearchFilter& searchFilter)
1272 	: PropValidator	(context, PROGRAMRESOURCEPROP_REFERENCED_BY_SHADER, getRequiredExtensionForStage(shaderType))
1273 	, m_filter		(VariableSearchFilter::logicalAnd(VariableSearchFilter::createShaderTypeFilter(shaderType), searchFilter))
1274 	, m_shaderType	(shaderType)
1275 {
1276 	DE_ASSERT(m_shaderType < glu::SHADERTYPE_LAST);
1277 }
1278 
getHumanReadablePropertyString(glw::GLint propVal) const1279 std::string BlockReferencedByShaderValidator::getHumanReadablePropertyString (glw::GLint propVal) const
1280 {
1281 	return de::toString(glu::getBooleanStr(propVal));
1282 }
1283 
validate(const ProgramInterfaceDefinition::Program * program,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1284 void BlockReferencedByShaderValidator::validate (const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1285 {
1286 	const std::string	blockName			= glu::parseVariableName(resource.c_str());
1287 	bool				referencedByShader	= false;
1288 
1289 	DE_UNREF(implementationName);
1290 
1291 	for (int shaderNdx = 0; shaderNdx < (int)program->getShaders().size(); ++shaderNdx)
1292 	{
1293 		const ProgramInterfaceDefinition::Shader* const shader = program->getShaders()[shaderNdx];
1294 		if (!m_filter.matchesFilter(shader))
1295 			continue;
1296 
1297 		for (int blockNdx = 0; blockNdx < (int)shader->getDefaultBlock().interfaceBlocks.size(); ++blockNdx)
1298 		{
1299 			const glu::InterfaceBlock& block = shader->getDefaultBlock().interfaceBlocks[blockNdx];
1300 
1301 			if (m_filter.matchesFilter(block) && block.interfaceName == blockName)
1302 				referencedByShader = true;
1303 		}
1304 	}
1305 
1306 	m_testCtx.getLog()
1307 		<< tcu::TestLog::Message
1308 		<< "Verifying referenced by " << glu::getShaderTypeName(m_shaderType) << " shader, expecting "
1309 		<< ((referencedByShader) ? ("GL_TRUE") : ("GL_FALSE"))
1310 		<< tcu::TestLog::EndMessage;
1311 
1312 	if (propValue != ((referencedByShader) ? (1) : (0)))
1313 	{
1314 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid referenced_by_" << glu::getShaderTypeName(m_shaderType) << ", got " << propValue << tcu::TestLog::EndMessage;
1315 		setError("referenced_by_" + std::string(glu::getShaderTypeName(m_shaderType)) + " invalid");
1316 	}
1317 }
1318 
1319 class TopLevelArraySizeValidator : public SingleVariableValidator
1320 {
1321 public:
1322 				TopLevelArraySizeValidator	(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
1323 
1324 	void		validateSingleVariable		(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1325 };
1326 
TopLevelArraySizeValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)1327 TopLevelArraySizeValidator::TopLevelArraySizeValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
1328 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_TOP_LEVEL_ARRAY_SIZE, programID, filter, DE_NULL)
1329 {
1330 }
1331 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1332 void TopLevelArraySizeValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1333 {
1334 	int			expected;
1335 	std::string	reason;
1336 
1337 	DE_ASSERT(path.front().isInterfaceBlock() && path.front().getInterfaceBlock()->storage == glu::STORAGE_BUFFER);
1338 	DE_UNREF(resource);
1339 	DE_UNREF(implementationName);
1340 
1341 	if (!path[1].getDeclaration()->varType.isArrayType())
1342 	{
1343 		expected = 1;
1344 		reason = "Top-level block member is not an array";
1345 	}
1346 	else if (path[1].getDeclaration()->varType.getElementType().isBasicType())
1347 	{
1348 		expected = 1;
1349 		reason = "Top-level block member is not an array of an aggregate type";
1350 	}
1351 	else if (path[1].getDeclaration()->varType.getArraySize() == glu::VarType::UNSIZED_ARRAY)
1352 	{
1353 		expected = 0;
1354 		reason = "Top-level block member is an unsized top-level array";
1355 	}
1356 	else
1357 	{
1358 		expected = path[1].getDeclaration()->varType.getArraySize();
1359 		reason = "Top-level block member is a sized top-level array";
1360 	}
1361 
1362 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying top level array size, expecting " << expected << ". (" << reason << ")." << tcu::TestLog::EndMessage;
1363 
1364 	if (propValue != expected)
1365 	{
1366 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid top level array size, got " << propValue << tcu::TestLog::EndMessage;
1367 		setError("top level array size invalid");
1368 	}
1369 }
1370 
1371 class TopLevelArrayStrideValidator : public SingleVariableValidator
1372 {
1373 public:
1374 				TopLevelArrayStrideValidator	(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
1375 
1376 	void		validateSingleVariable			(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1377 };
1378 
TopLevelArrayStrideValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)1379 TopLevelArrayStrideValidator::TopLevelArrayStrideValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
1380 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_TOP_LEVEL_ARRAY_STRIDE, programID, filter, DE_NULL)
1381 {
1382 }
1383 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1384 void TopLevelArrayStrideValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1385 {
1386 	DE_ASSERT(path.front().isInterfaceBlock() && path.front().getInterfaceBlock()->storage == glu::STORAGE_BUFFER);
1387 	DE_UNREF(resource);
1388 	DE_UNREF(implementationName);
1389 
1390 	if (!path[1].getDeclaration()->varType.isArrayType())
1391 	{
1392 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying top level array stride, expecting 0. (Top-level block member is not an array)." << tcu::TestLog::EndMessage;
1393 
1394 		if (propValue != 0)
1395 		{
1396 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, top level array stride, got " << propValue << tcu::TestLog::EndMessage;
1397 			setError("top level array stride invalid");
1398 		}
1399 	}
1400 	else if (path[1].getDeclaration()->varType.getElementType().isBasicType())
1401 	{
1402 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying top level array stride, expecting 0. (Top-level block member is not an array of an aggregate type)." << tcu::TestLog::EndMessage;
1403 
1404 		if (propValue != 0)
1405 		{
1406 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, top level array stride, got " << propValue << tcu::TestLog::EndMessage;
1407 			setError("top level array stride invalid");
1408 		}
1409 	}
1410 	else
1411 	{
1412 		const int minimumStride = getVarTypeSize(path[1].getDeclaration()->varType.getElementType());
1413 
1414 		m_testCtx.getLog() << tcu::TestLog::Message << "Verifying top level array stride, expecting greater or equal to " << minimumStride << "." << tcu::TestLog::EndMessage;
1415 
1416 		if (propValue < minimumStride)
1417 		{
1418 			m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid top level array stride, got " << propValue << tcu::TestLog::EndMessage;
1419 			setError("top level array stride invalid");
1420 		}
1421 	}
1422 }
1423 
1424 class TransformFeedbackResourceValidator : public PropValidator
1425 {
1426 public:
1427 					TransformFeedbackResourceValidator	(Context& context, ProgramResourcePropFlags validationProp);
1428 
1429 	void			validate							(const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1430 
1431 private:
1432 	virtual void	validateBuiltinVariable				(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const = 0;
1433 	virtual void	validateSingleVariable				(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const = 0;
1434 };
1435 
1436 
TransformFeedbackResourceValidator(Context & context,ProgramResourcePropFlags validationProp)1437 TransformFeedbackResourceValidator::TransformFeedbackResourceValidator (Context& context, ProgramResourcePropFlags validationProp)
1438 	: PropValidator(context, validationProp, DE_NULL)
1439 {
1440 }
1441 
validate(const ProgramInterfaceDefinition::Program * program,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1442 void TransformFeedbackResourceValidator::validate (const ProgramInterfaceDefinition::Program* program, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1443 {
1444 	if (deStringBeginsWith(resource.c_str(), "gl_"))
1445 	{
1446 		validateBuiltinVariable(resource, propValue, implementationName);
1447 	}
1448 	else
1449 	{
1450 		// Check resource name is a xfb output. (sanity check)
1451 #if defined(DE_DEBUG)
1452 		bool generatorFound = false;
1453 
1454 		// Check the resource name is a valid transform feedback resource and find the name generating resource
1455 		for (int varyingNdx = 0; varyingNdx < (int)program->getTransformFeedbackVaryings().size(); ++varyingNdx)
1456 		{
1457 			const std::string					varyingName = program->getTransformFeedbackVaryings()[varyingNdx];
1458 			std::vector<VariablePathComponent>	path;
1459 			std::vector<std::string>			resources;
1460 
1461 			if (!findProgramVariablePathByPathName(path, program, varyingName, VariableSearchFilter::createShaderTypeStorageFilter(getProgramTransformFeedbackStage(program), glu::STORAGE_OUT)))
1462 			{
1463 				// program does not contain feedback varying, not valid program
1464 				DE_ASSERT(false);
1465 				return;
1466 			}
1467 
1468 			generateVariableTypeResourceNames(resources, varyingName, *path.back().getVariableType(), RESOURCE_NAME_GENERATION_FLAG_TRANSFORM_FEEDBACK_VARIABLE);
1469 
1470 			if (de::contains(resources.begin(), resources.end(), resource))
1471 			{
1472 				generatorFound = true;
1473 				break;
1474 			}
1475 		}
1476 
1477 		// resource name was not found, should never happen
1478 		DE_ASSERT(generatorFound);
1479 		DE_UNREF(generatorFound);
1480 #endif
1481 
1482 		// verify resource
1483 		{
1484 			std::vector<VariablePathComponent> path;
1485 
1486 			if (!findProgramVariablePathByPathName(path, program, resource, VariableSearchFilter::createShaderTypeStorageFilter(getProgramTransformFeedbackStage(program), glu::STORAGE_OUT)))
1487 				DE_ASSERT(false);
1488 
1489 			validateSingleVariable(path, resource, propValue, implementationName);
1490 		}
1491 	}
1492 }
1493 
1494 class TransformFeedbackArraySizeValidator : public TransformFeedbackResourceValidator
1495 {
1496 public:
1497 				TransformFeedbackArraySizeValidator	(Context& context);
1498 
1499 	void		validateBuiltinVariable				(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1500 	void		validateSingleVariable				(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1501 };
1502 
TransformFeedbackArraySizeValidator(Context & context)1503 TransformFeedbackArraySizeValidator::TransformFeedbackArraySizeValidator (Context& context)
1504 	: TransformFeedbackResourceValidator(context, PROGRAMRESOURCEPROP_ARRAY_SIZE)
1505 {
1506 }
1507 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1508 void TransformFeedbackArraySizeValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1509 {
1510 	DE_UNREF(implementationName);
1511 
1512 	int arraySize = 0;
1513 
1514 	if (resource == "gl_Position")
1515 		arraySize = 1;
1516 	else
1517 		DE_ASSERT(false);
1518 
1519 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying array size, expecting " << arraySize << tcu::TestLog::EndMessage;
1520 	if (arraySize != propValue)
1521 	{
1522 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
1523 		setError("resource array size invalid");
1524 	}
1525 }
1526 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1527 void TransformFeedbackArraySizeValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1528 {
1529 	DE_UNREF(resource);
1530 	DE_UNREF(implementationName);
1531 
1532 	const int arraySize = (path.back().getVariableType()->isArrayType()) ? (path.back().getVariableType()->getArraySize()) : (1);
1533 
1534 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying array size, expecting " << arraySize << tcu::TestLog::EndMessage;
1535 	if (arraySize != propValue)
1536 	{
1537 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
1538 		setError("resource array size invalid");
1539 	}
1540 }
1541 
1542 class TransformFeedbackNameLengthValidator : public TransformFeedbackResourceValidator
1543 {
1544 public:
1545 				TransformFeedbackNameLengthValidator	(Context& context);
1546 
1547 private:
1548 	void		validateBuiltinVariable					(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1549 	void		validateSingleVariable					(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1550 	void		validateVariable						(const std::string& implementationName, glw::GLint propValue) const;
1551 };
1552 
TransformFeedbackNameLengthValidator(Context & context)1553 TransformFeedbackNameLengthValidator::TransformFeedbackNameLengthValidator (Context& context)
1554 	: TransformFeedbackResourceValidator(context, PROGRAMRESOURCEPROP_NAME_LENGTH)
1555 {
1556 }
1557 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1558 void TransformFeedbackNameLengthValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1559 {
1560 	DE_UNREF(resource);
1561 	validateVariable(implementationName, propValue);
1562 }
1563 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1564 void TransformFeedbackNameLengthValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1565 {
1566 	DE_UNREF(path);
1567 	DE_UNREF(resource);
1568 	validateVariable(implementationName, propValue);
1569 }
1570 
validateVariable(const std::string & implementationName,glw::GLint propValue) const1571 void TransformFeedbackNameLengthValidator::validateVariable (const std::string& implementationName, glw::GLint propValue) const
1572 {
1573 	const int expected = (int)implementationName.length() + 1; // includes null byte
1574 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying name length, expecting " << expected << " (" << (int)implementationName.length() << " for \"" << implementationName << "\" + 1 byte for terminating null character)" << tcu::TestLog::EndMessage;
1575 
1576 	if (propValue != expected)
1577 	{
1578 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, invalid name length, got " << propValue << tcu::TestLog::EndMessage;
1579 		setError("name length invalid");
1580 	}
1581 }
1582 
1583 class TransformFeedbackTypeValidator : public TransformFeedbackResourceValidator
1584 {
1585 public:
1586 				TransformFeedbackTypeValidator		(Context& context);
1587 
1588 	void		validateBuiltinVariable				(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1589 	void		validateSingleVariable				(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1590 };
1591 
TransformFeedbackTypeValidator(Context & context)1592 TransformFeedbackTypeValidator::TransformFeedbackTypeValidator (Context& context)
1593 	: TransformFeedbackResourceValidator(context, PROGRAMRESOURCEPROP_TYPE)
1594 {
1595 }
1596 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1597 void TransformFeedbackTypeValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1598 {
1599 	DE_UNREF(implementationName);
1600 
1601 	glu::DataType varType = glu::TYPE_INVALID;
1602 
1603 	if (resource == "gl_Position")
1604 		varType = glu::TYPE_FLOAT_VEC4;
1605 	else
1606 		DE_ASSERT(false);
1607 
1608 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying type, expecting " << glu::getDataTypeName(varType) << tcu::TestLog::EndMessage;
1609 	if (glu::getDataTypeFromGLType(propValue) != varType)
1610 	{
1611 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << glu::getDataTypeName(glu::getDataTypeFromGLType(propValue)) << tcu::TestLog::EndMessage;
1612 		setError("resource type invalid");
1613 	}
1614 	return;
1615 }
1616 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1617 void TransformFeedbackTypeValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1618 {
1619 	DE_UNREF(resource);
1620 	DE_UNREF(implementationName);
1621 
1622 	// Unlike other interfaces, xfb program interface uses just variable name to refer to arrays of basic types. (Others use "variable[0]")
1623 	// Thus we might end up querying a type for an array. In this case, return the type of an array element.
1624 	const glu::VarType& variable    = *path.back().getVariableType();
1625 	const glu::VarType& elementType = (variable.isArrayType()) ? (variable.getElementType()) : (variable);
1626 
1627 	DE_ASSERT(elementType.isBasicType());
1628 
1629 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying type, expecting " << glu::getDataTypeName(elementType.getBasicType()) << tcu::TestLog::EndMessage;
1630 	if (elementType.getBasicType() != glu::getDataTypeFromGLType(propValue))
1631 	{
1632 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << glu::getDataTypeName(glu::getDataTypeFromGLType(propValue)) << tcu::TestLog::EndMessage;
1633 		setError("resource type invalid");
1634 	}
1635 }
1636 
1637 class PerPatchValidator : public SingleVariableValidator
1638 {
1639 public:
1640 				PerPatchValidator				(Context& context, glw::GLuint programID, const VariableSearchFilter& filter);
1641 
1642 	std::string getHumanReadablePropertyString	(glw::GLint propVal) const;
1643 	void		validateSingleVariable			(const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1644 	void		validateBuiltinVariable			(const std::string& resource, glw::GLint propValue, const std::string& implementationName) const;
1645 };
1646 
PerPatchValidator(Context & context,glw::GLuint programID,const VariableSearchFilter & filter)1647 PerPatchValidator::PerPatchValidator (Context& context, glw::GLuint programID, const VariableSearchFilter& filter)
1648 	: SingleVariableValidator(context, PROGRAMRESOURCEPROP_IS_PER_PATCH, programID, filter, "GL_EXT_tessellation_shader")
1649 {
1650 }
1651 
getHumanReadablePropertyString(glw::GLint propVal) const1652 std::string PerPatchValidator::getHumanReadablePropertyString (glw::GLint propVal) const
1653 {
1654 	return de::toString(glu::getBooleanStr(propVal));
1655 }
1656 
validateSingleVariable(const std::vector<VariablePathComponent> & path,const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1657 void PerPatchValidator::validateSingleVariable (const std::vector<VariablePathComponent>& path, const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1658 {
1659 	const glu::Storage	storage		= (path.front().isInterfaceBlock()) ? (path.front().getInterfaceBlock()->storage) : (path.front().getDeclaration()->storage);
1660 	const int			expected	= (storage == glu::STORAGE_PATCH_IN || storage == glu::STORAGE_PATCH_OUT) ? (1) : (0);
1661 
1662 	DE_UNREF(resource);
1663 	DE_UNREF(implementationName);
1664 
1665 	m_testCtx.getLog() << tcu::TestLog::Message << "Verifying if is per patch, expecting IS_PER_PATCH = " << expected << tcu::TestLog::EndMessage;
1666 
1667 	if (propValue != expected)
1668 	{
1669 		m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
1670 		setError("resource is per patch invalid");
1671 	}
1672 }
1673 
validateBuiltinVariable(const std::string & resource,glw::GLint propValue,const std::string & implementationName) const1674 void PerPatchValidator::validateBuiltinVariable (const std::string& resource, glw::GLint propValue, const std::string& implementationName) const
1675 {
1676 	DE_UNREF(implementationName);
1677 
1678 	static const struct
1679 	{
1680 		const char*		name;
1681 		int				isPerPatch;
1682 	} builtins[] =
1683 	{
1684 		{ "gl_Position",				0	},
1685 		{ "gl_PerVertex.gl_Position",	0	},
1686 		{ "gl_InvocationID",			0	},
1687 		{ "gl_TessLevelOuter[0]",		1	},
1688 		{ "gl_TessLevelInner[0]",		1	},
1689 	};
1690 
1691 	for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(builtins); ++ndx)
1692 	{
1693 		if (resource == builtins[ndx].name)
1694 		{
1695 			m_testCtx.getLog() << tcu::TestLog::Message << "Verifying if is per patch, expecting IS_PER_PATCH = " << builtins[ndx].isPerPatch << tcu::TestLog::EndMessage;
1696 
1697 			if (propValue != builtins[ndx].isPerPatch)
1698 			{
1699 				m_testCtx.getLog() << tcu::TestLog::Message << "\tError, got " << propValue << tcu::TestLog::EndMessage;
1700 				setError("resource is per patch invalid");
1701 			}
1702 			return;
1703 		}
1704 	}
1705 
1706 	DE_ASSERT(false);
1707 }
1708 
1709 } // anonymous
1710 
ProgramResourceQueryTestTarget(ProgramInterface interface_,deUint32 propFlags_)1711 ProgramResourceQueryTestTarget::ProgramResourceQueryTestTarget (ProgramInterface interface_, deUint32 propFlags_)
1712 	: interface(interface_)
1713 	, propFlags(propFlags_)
1714 {
1715 	switch (interface)
1716 	{
1717 		case PROGRAMINTERFACE_UNIFORM:						DE_ASSERT((propFlags & PROGRAMRESOURCEPROP_UNIFORM_INTERFACE_MASK)			== propFlags);	break;
1718 		case PROGRAMINTERFACE_UNIFORM_BLOCK:				DE_ASSERT((propFlags & PROGRAMRESOURCEPROP_UNIFORM_BLOCK_INTERFACE_MASK)	== propFlags);	break;
1719 		case PROGRAMINTERFACE_SHADER_STORAGE_BLOCK:			DE_ASSERT((propFlags & PROGRAMRESOURCEPROP_SHADER_STORAGE_BLOCK_MASK)		== propFlags);	break;
1720 		case PROGRAMINTERFACE_PROGRAM_INPUT:				DE_ASSERT((propFlags & PROGRAMRESOURCEPROP_PROGRAM_INPUT_MASK)				== propFlags);	break;
1721 		case PROGRAMINTERFACE_PROGRAM_OUTPUT:				DE_ASSERT((propFlags & PROGRAMRESOURCEPROP_PROGRAM_OUTPUT_MASK)				== propFlags);	break;
1722 		case PROGRAMINTERFACE_BUFFER_VARIABLE:				DE_ASSERT((propFlags & PROGRAMRESOURCEPROP_BUFFER_VARIABLE_MASK)			== propFlags);	break;
1723 		case PROGRAMINTERFACE_TRANSFORM_FEEDBACK_VARYING:	DE_ASSERT((propFlags & PROGRAMRESOURCEPROP_TRANSFORM_FEEDBACK_VARYING_MASK)	== propFlags);	break;
1724 
1725 		default:
1726 			DE_ASSERT(false);
1727 	}
1728 }
1729 
ProgramInterfaceQueryTestCase(Context & context,const char * name,const char * description,ProgramResourceQueryTestTarget queryTarget)1730 ProgramInterfaceQueryTestCase::ProgramInterfaceQueryTestCase (Context& context, const char* name, const char* description, ProgramResourceQueryTestTarget queryTarget)
1731 	: TestCase		(context, name, description)
1732 	, m_queryTarget	(queryTarget)
1733 {
1734 }
1735 
~ProgramInterfaceQueryTestCase(void)1736 ProgramInterfaceQueryTestCase::~ProgramInterfaceQueryTestCase (void)
1737 {
1738 }
1739 
getTargetInterface(void) const1740 ProgramInterface ProgramInterfaceQueryTestCase::getTargetInterface (void) const
1741 {
1742 	return m_queryTarget.interface;
1743 }
1744 
getGLInterfaceEnumValue(ProgramInterface interface)1745 static glw::GLenum getGLInterfaceEnumValue (ProgramInterface interface)
1746 {
1747 	switch (interface)
1748 	{
1749 		case PROGRAMINTERFACE_UNIFORM:						return GL_UNIFORM;
1750 		case PROGRAMINTERFACE_UNIFORM_BLOCK:				return GL_UNIFORM_BLOCK;
1751 		case PROGRAMINTERFACE_ATOMIC_COUNTER_BUFFER:		return GL_ATOMIC_COUNTER_BUFFER;
1752 		case PROGRAMINTERFACE_PROGRAM_INPUT:				return GL_PROGRAM_INPUT;
1753 		case PROGRAMINTERFACE_PROGRAM_OUTPUT:				return GL_PROGRAM_OUTPUT;
1754 		case PROGRAMINTERFACE_TRANSFORM_FEEDBACK_VARYING:	return GL_TRANSFORM_FEEDBACK_VARYING;
1755 		case PROGRAMINTERFACE_BUFFER_VARIABLE:				return GL_BUFFER_VARIABLE;
1756 		case PROGRAMINTERFACE_SHADER_STORAGE_BLOCK:			return GL_SHADER_STORAGE_BLOCK;
1757 		default:
1758 			DE_ASSERT(false);
1759 			return 0;
1760 	};
1761 }
1762 
isInterfaceBlockInterfaceName(const ProgramInterfaceDefinition::Program * program,ProgramInterface interface,const std::string & blockInterfaceName)1763 static bool isInterfaceBlockInterfaceName (const ProgramInterfaceDefinition::Program* program, ProgramInterface interface, const std::string& blockInterfaceName)
1764 {
1765 	deUint32 validStorageBits;
1766 	deUint32 searchStageBits;
1767 
1768 	DE_STATIC_ASSERT(glu::STORAGE_LAST < 32);
1769 	DE_STATIC_ASSERT(glu::SHADERTYPE_LAST < 32);
1770 
1771 	switch (interface)
1772 	{
1773 		case PROGRAMINTERFACE_UNIFORM_BLOCK:
1774 		case PROGRAMINTERFACE_SHADER_STORAGE_BLOCK:
1775 		case PROGRAMINTERFACE_ATOMIC_COUNTER_BUFFER:
1776 			return false;
1777 
1778 		case PROGRAMINTERFACE_PROGRAM_INPUT:
1779 			validStorageBits = (1u << glu::STORAGE_IN) | (1u << glu::STORAGE_PATCH_IN);
1780 			searchStageBits = (1u << program->getFirstStage());
1781 			break;
1782 
1783 		case PROGRAMINTERFACE_PROGRAM_OUTPUT:
1784 			validStorageBits = (1u << glu::STORAGE_OUT) | (1u << glu::STORAGE_PATCH_OUT);
1785 			searchStageBits = (1u << program->getLastStage());
1786 			break;
1787 
1788 		case PROGRAMINTERFACE_TRANSFORM_FEEDBACK_VARYING:
1789 			validStorageBits = (1u << glu::STORAGE_OUT);
1790 			searchStageBits = (1u << getProgramTransformFeedbackStage(program));
1791 			break;
1792 
1793 		case PROGRAMINTERFACE_UNIFORM:
1794 			validStorageBits = (1u << glu::STORAGE_UNIFORM);
1795 			searchStageBits = 0xFFFFFFFFu;
1796 			break;
1797 
1798 		case PROGRAMINTERFACE_BUFFER_VARIABLE:
1799 			validStorageBits = (1u << glu::STORAGE_BUFFER);
1800 			searchStageBits = 0xFFFFFFFFu;
1801 			break;
1802 
1803 		default:
1804 			DE_ASSERT(false);
1805 			return false;
1806 	}
1807 
1808 	for (int shaderNdx = 0; shaderNdx < (int)program->getShaders().size(); ++shaderNdx)
1809 	{
1810 		const ProgramInterfaceDefinition::Shader* const shader = program->getShaders()[shaderNdx];
1811 		if (((1u << shader->getType()) & searchStageBits) == 0)
1812 			continue;
1813 
1814 		for (int blockNdx = 0; blockNdx < (int)shader->getDefaultBlock().interfaceBlocks.size(); ++blockNdx)
1815 		{
1816 			const glu::InterfaceBlock& block = shader->getDefaultBlock().interfaceBlocks[blockNdx];
1817 
1818 			if (((1u << block.storage) & validStorageBits) == 0)
1819 				continue;
1820 
1821 			if (block.interfaceName == blockInterfaceName)
1822 				return true;
1823 		}
1824 	}
1825 	return false;
1826 }
1827 
getInterfaceBlockInteraceNameByMember(const ProgramInterfaceDefinition::Program * program,ProgramInterface interface,const std::string & memberName)1828 static std::string getInterfaceBlockInteraceNameByMember (const ProgramInterfaceDefinition::Program* program, ProgramInterface interface, const std::string& memberName)
1829 {
1830 	deUint32 validStorageBits;
1831 	deUint32 searchStageBits;
1832 
1833 	DE_STATIC_ASSERT(glu::STORAGE_LAST < 32);
1834 	DE_STATIC_ASSERT(glu::SHADERTYPE_LAST < 32);
1835 
1836 	switch (interface)
1837 	{
1838 		case PROGRAMINTERFACE_UNIFORM_BLOCK:
1839 		case PROGRAMINTERFACE_SHADER_STORAGE_BLOCK:
1840 		case PROGRAMINTERFACE_ATOMIC_COUNTER_BUFFER:
1841 			return "";
1842 
1843 		case PROGRAMINTERFACE_PROGRAM_INPUT:
1844 			validStorageBits = (1u << glu::STORAGE_IN) | (1u << glu::STORAGE_PATCH_IN);
1845 			searchStageBits = (1u << program->getFirstStage());
1846 			break;
1847 
1848 		case PROGRAMINTERFACE_PROGRAM_OUTPUT:
1849 			validStorageBits = (1u << glu::STORAGE_OUT) | (1u << glu::STORAGE_PATCH_OUT);
1850 			searchStageBits = (1u << program->getLastStage());
1851 			break;
1852 
1853 		case PROGRAMINTERFACE_TRANSFORM_FEEDBACK_VARYING:
1854 			validStorageBits = (1u << glu::STORAGE_OUT);
1855 			searchStageBits = (1u << getProgramTransformFeedbackStage(program));
1856 			break;
1857 
1858 		case PROGRAMINTERFACE_UNIFORM:
1859 			validStorageBits = (1u << glu::STORAGE_UNIFORM);
1860 			searchStageBits = 0xFFFFFFFFu;
1861 			break;
1862 
1863 		case PROGRAMINTERFACE_BUFFER_VARIABLE:
1864 			validStorageBits = (1u << glu::STORAGE_BUFFER);
1865 			searchStageBits = 0xFFFFFFFFu;
1866 			break;
1867 
1868 		default:
1869 			DE_ASSERT(false);
1870 			return "";
1871 	}
1872 
1873 	for (int shaderNdx = 0; shaderNdx < (int)program->getShaders().size(); ++shaderNdx)
1874 	{
1875 		const ProgramInterfaceDefinition::Shader* const shader = program->getShaders()[shaderNdx];
1876 		if (((1u << shader->getType()) & searchStageBits) == 0)
1877 			continue;
1878 
1879 		for (int blockNdx = 0; blockNdx < (int)shader->getDefaultBlock().interfaceBlocks.size(); ++blockNdx)
1880 		{
1881 			const glu::InterfaceBlock& block = shader->getDefaultBlock().interfaceBlocks[blockNdx];
1882 
1883 			if (((1u << block.storage) & validStorageBits) == 0)
1884 				continue;
1885 
1886 			for (int varNdx = 0; varNdx < (int)block.variables.size(); ++varNdx)
1887 			{
1888 				if (block.variables[varNdx].name == memberName)
1889 					return block.interfaceName;
1890 			}
1891 		}
1892 	}
1893 	return "";
1894 }
1895 
queryAndValidateProps(tcu::TestContext & testCtx,const glw::Functions & gl,glw::GLuint programID,ProgramInterface interface,const char * targetResourceName,const ProgramInterfaceDefinition::Program * programDefinition,const std::vector<glw::GLenum> & props,const std::vector<const PropValidator * > & validators)1896 static void queryAndValidateProps (tcu::TestContext&							testCtx,
1897 								   const glw::Functions&						gl,
1898 								   glw::GLuint									programID,
1899 								   ProgramInterface								interface,
1900 								   const char*									targetResourceName,
1901 								   const ProgramInterfaceDefinition::Program*	programDefinition,
1902 								   const std::vector<glw::GLenum>&				props,
1903 								   const std::vector<const PropValidator*>&		validators)
1904 {
1905 	const glw::GLenum			glInterface					= getGLInterfaceEnumValue(interface);
1906 	std::string					implementationResourceName	= targetResourceName;
1907 	glw::GLuint					resourceNdx;
1908 	glw::GLint					written						= -1;
1909 
1910 	// prefill result buffer with an invalid value. -1 might be valid sometimes, avoid it. Make buffer one larger
1911 	// to allow detection of too many return values
1912 	std::vector<glw::GLint>		propValues		(props.size() + 1, -2);
1913 
1914 	DE_ASSERT(props.size() == validators.size());
1915 
1916 	// query
1917 
1918 	resourceNdx = gl.getProgramResourceIndex(programID, glInterface, targetResourceName);
1919 	GLU_EXPECT_NO_ERROR(gl.getError(), "get resource index");
1920 
1921 	if (resourceNdx == GL_INVALID_INDEX)
1922 	{
1923 		static const struct
1924 		{
1925 			bool removeTrailingArray;	// convert from "target[0]" -> "target"
1926 			bool removeTrailingMember;	// convert from "target.member" -> "target"
1927 			bool removeIOBlock;			// convert from "InterfaceName.target" -> "target"
1928 			bool addIOBlock;			// convert from "target" -> "InterfaceName.target"
1929 			bool addIOBlockArray;		// convert from "target" -> "InterfaceName[0].target"
1930 		} recoveryStrategies[] =
1931 		{
1932 			// try one patch
1933 			{ true,		false,	false,	false,	false	},
1934 			{ false,	true,	false,	false,	false	},
1935 			{ false,	false,	true,	false,	false	},
1936 			{ false,	false,	false,	true,	false	},
1937 			{ false,	false,	false,	false,	true	},
1938 			// patch both ends
1939 			{ true,		false,	true,	false,	false	},
1940 			{ true,		false,	false,	true,	false	},
1941 			{ true,		false,	false,	false,	true	},
1942 			{ false,	true,	true,	false,	false	},
1943 			{ false,	true,	false,	true,	false	},
1944 			{ false,	true,	false,	false,	true	},
1945 		};
1946 
1947 		// The resource name generation in the GL implementations is very commonly broken. Try to
1948 		// keep the tests producing useful data even in these cases by attempting to recover from
1949 		// common naming bugs. Set test result to failure even if recovery succeeded to signal
1950 		// incorrect name generation.
1951 
1952 		testCtx.getLog() << tcu::TestLog::Message << "getProgramResourceIndex returned GL_INVALID_INDEX for \"" << targetResourceName << "\"" << tcu::TestLog::EndMessage;
1953 		testCtx.setTestResult(QP_TEST_RESULT_FAIL, "could not find target resource");
1954 
1955 		for (int strategyNdx = 0; strategyNdx < DE_LENGTH_OF_ARRAY(recoveryStrategies); ++strategyNdx)
1956 		{
1957 			const std::string	resourceName			= std::string(targetResourceName);
1958 			const size_t		rootNameEnd				= resourceName.find_first_of(".[");
1959 			const std::string	rootName				= resourceName.substr(0, rootNameEnd);
1960 			std::string			simplifiedResourceName;
1961 
1962 			if (recoveryStrategies[strategyNdx].removeTrailingArray)
1963 			{
1964 				if (de::endsWith(resourceName, "[0]"))
1965 					simplifiedResourceName = resourceName.substr(0, resourceName.length() - 3);
1966 				else
1967 					continue;
1968 			}
1969 
1970 			if (recoveryStrategies[strategyNdx].removeTrailingMember)
1971 			{
1972 				const size_t lastMember = resourceName.find_last_of('.');
1973 				if (lastMember != std::string::npos)
1974 					simplifiedResourceName = resourceName.substr(0, lastMember);
1975 				else
1976 					continue;
1977 			}
1978 
1979 			if (recoveryStrategies[strategyNdx].removeIOBlock)
1980 			{
1981 				if (deStringBeginsWith(resourceName.c_str(), "gl_PerVertex."))
1982 				{
1983 					// builtin interface bock, remove block name
1984 					simplifiedResourceName = resourceName.substr(13);
1985 				}
1986 				else if (isInterfaceBlockInterfaceName(programDefinition, interface, rootName))
1987 				{
1988 					// user-defined inteface block, remove name
1989 					const size_t accessorEnd = resourceName.find('.'); // includes potential array accessor
1990 
1991 					if (accessorEnd != std::string::npos)
1992 						simplifiedResourceName = resourceName.substr(0, accessorEnd+1);
1993 					else
1994 						continue;
1995 				}
1996 				else
1997 				{
1998 					// recovery not applicable
1999 					continue;
2000 				}
2001 			}
2002 
2003 			if (recoveryStrategies[strategyNdx].addIOBlock || recoveryStrategies[strategyNdx].addIOBlockArray)
2004 			{
2005 				const std::string arrayAccessor = (recoveryStrategies[strategyNdx].addIOBlockArray) ? ("[0]") : ("");
2006 
2007 				if (deStringBeginsWith(resourceName.c_str(), "gl_") && resourceName.find('.') == std::string::npos)
2008 				{
2009 					// free builtin variable, add block name
2010 					simplifiedResourceName = "gl_PerVertex" + arrayAccessor + "." + resourceName;
2011 				}
2012 				else
2013 				{
2014 					const std::string interafaceName = getInterfaceBlockInteraceNameByMember(programDefinition, interface, rootName);
2015 
2016 					if (!interafaceName.empty())
2017 					{
2018 						// free user variable, add block name
2019 						simplifiedResourceName = interafaceName + arrayAccessor + "." + resourceName;
2020 					}
2021 					else
2022 					{
2023 						// recovery not applicable
2024 						continue;
2025 					}
2026 				}
2027 			}
2028 
2029 			if (simplifiedResourceName.empty())
2030 				continue;
2031 
2032 			resourceNdx = gl.getProgramResourceIndex(programID, glInterface, simplifiedResourceName.c_str());
2033 			GLU_EXPECT_NO_ERROR(gl.getError(), "get resource index");
2034 
2035 			// recovery succeeded
2036 			if (resourceNdx != GL_INVALID_INDEX)
2037 			{
2038 				implementationResourceName = simplifiedResourceName;
2039 				testCtx.getLog() << tcu::TestLog::Message << "\tResource not found, continuing anyway using index obtained for resource \"" << simplifiedResourceName << "\"" << tcu::TestLog::EndMessage;
2040 				break;
2041 			}
2042 		}
2043 
2044 		if (resourceNdx == GL_INVALID_INDEX)
2045 			return;
2046 	}
2047 
2048 	gl.getProgramResourceiv(programID, glInterface, resourceNdx, (int)props.size(), &props[0], (int)propValues.size(), &written, &propValues[0]);
2049 	GLU_EXPECT_NO_ERROR(gl.getError(), "get props");
2050 
2051 	if (written != (int)props.size())
2052 	{
2053 		testCtx.getLog() << tcu::TestLog::Message << "getProgramResourceiv returned unexpected number of values, expected " << (int)props.size() << ", got " << written << tcu::TestLog::EndMessage;
2054 		testCtx.setTestResult(QP_TEST_RESULT_FAIL, "getProgramResourceiv returned unexpected number of values");
2055 		return;
2056 	}
2057 
2058 	if (propValues.back() != -2)
2059 	{
2060 		testCtx.getLog() << tcu::TestLog::Message << "getProgramResourceiv post write buffer guard value was modified, too many return values" << tcu::TestLog::EndMessage;
2061 		testCtx.setTestResult(QP_TEST_RESULT_FAIL, "getProgramResourceiv returned unexpected number of values");
2062 		return;
2063 	}
2064 	propValues.pop_back();
2065 	DE_ASSERT(validators.size() == propValues.size());
2066 
2067 	// log
2068 
2069 	{
2070 		tcu::MessageBuilder message(&testCtx.getLog());
2071 		message << "For resource index " << resourceNdx << " (\"" << targetResourceName << "\") got following properties:\n";
2072 
2073 		for (int propNdx = 0; propNdx < (int)propValues.size(); ++propNdx)
2074 			message << "\t" << glu::getProgramResourcePropertyName(props[propNdx]) << ":\t" << validators[propNdx]->getHumanReadablePropertyString(propValues[propNdx]) << "\n";
2075 
2076 		message << tcu::TestLog::EndMessage;
2077 	}
2078 
2079 	// validate
2080 
2081 	for (int propNdx = 0; propNdx < (int)propValues.size(); ++propNdx)
2082 		validators[propNdx]->validate(programDefinition, targetResourceName, propValues[propNdx], implementationResourceName);
2083 }
2084 
getAndCheckProgramDefinition(void)2085 const ProgramInterfaceDefinition::Program* ProgramInterfaceQueryTestCase::getAndCheckProgramDefinition (void)
2086 {
2087 	const ProgramInterfaceDefinition::Program* programDefinition = getProgramDefinition();
2088 	DE_ASSERT(programDefinition->isValid());
2089 
2090 	auto type = m_context.getRenderContext().getType();
2091 	if (glu::contextSupports(type, glu::ApiType::es(3, 2)) ||
2092 		glu::contextSupports(type, glu::ApiType::core(4, 5)))
2093 	{
2094 		return programDefinition;
2095 	}
2096 
2097 	if (programDefinition->hasStage(glu::SHADERTYPE_TESSELLATION_CONTROL) ||
2098 		programDefinition->hasStage(glu::SHADERTYPE_TESSELLATION_EVALUATION))
2099 	{
2100 		if (!m_context.getContextInfo().isExtensionSupported("GL_EXT_tessellation_shader"))
2101 			throw tcu::NotSupportedError("Test requires GL_EXT_tessellation_shader extension");
2102 	}
2103 
2104 	// Testing IS_PER_PATCH as a part of a larger set is ok, since the extension is checked
2105 	// before query. However, we don't want IS_PER_PATCH-specific tests to become noop and pass.
2106 	if (m_queryTarget.propFlags == PROGRAMRESOURCEPROP_IS_PER_PATCH)
2107 	{
2108 		if (!m_context.getContextInfo().isExtensionSupported("GL_EXT_tessellation_shader"))
2109 			throw tcu::NotSupportedError("Test requires GL_EXT_tessellation_shader extension");
2110 	}
2111 
2112 	if (programDefinition->hasStage(glu::SHADERTYPE_GEOMETRY))
2113 	{
2114 		if (!m_context.getContextInfo().isExtensionSupported("GL_EXT_geometry_shader"))
2115 			throw tcu::NotSupportedError("Test requires GL_EXT_geometry_shader extension");
2116 	}
2117 
2118 	if (programContainsIOBlocks(programDefinition))
2119 	{
2120 		if (!m_context.getContextInfo().isExtensionSupported("GL_EXT_shader_io_blocks"))
2121 			throw tcu::NotSupportedError("Test requires GL_EXT_shader_io_blocks extension");
2122 	}
2123 
2124 	return programDefinition;
2125 }
2126 
getMaxPatchVertices(void)2127 int ProgramInterfaceQueryTestCase::getMaxPatchVertices (void)
2128 {
2129 	const glw::Functions&	gl					= m_context.getRenderContext().getFunctions();
2130 	glw::GLint				maxPatchVertices	= 0;
2131 
2132 	gl.getIntegerv(GL_MAX_PATCH_VERTICES, &maxPatchVertices);
2133 	GLU_EXPECT_NO_ERROR(gl.getError(), "getIntegerv(GL_MAX_PATCH_VERTICES)");
2134 	return maxPatchVertices;
2135 }
2136 
iterate(void)2137 ProgramInterfaceQueryTestCase::IterateResult ProgramInterfaceQueryTestCase::iterate (void)
2138 {
2139 	struct TestProperty
2140 	{
2141 		glw::GLenum				prop;
2142 		const PropValidator*	validator;
2143 	};
2144 
2145 	const ProgramInterfaceDefinition::Program*	programDefinition	= getAndCheckProgramDefinition();
2146 	const std::vector<std::string>				targetResources		= getQueryTargetResources();
2147 	glu::ShaderProgram							program				(m_context.getRenderContext(), generateProgramInterfaceProgramSources(programDefinition));
2148 
2149 	m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
2150 
2151 	// Log program
2152 	{
2153 		const tcu::ScopedLogSection section(m_testCtx.getLog(), "Program", "Program");
2154 
2155 		// Feedback varyings
2156 		if (!programDefinition->getTransformFeedbackVaryings().empty())
2157 		{
2158 			tcu::MessageBuilder builder(&m_testCtx.getLog());
2159 			builder << "Transform feedback varyings: {";
2160 			for (int ndx = 0; ndx < (int)programDefinition->getTransformFeedbackVaryings().size(); ++ndx)
2161 			{
2162 				if (ndx)
2163 					builder << ", ";
2164 				builder << "\"" << programDefinition->getTransformFeedbackVaryings()[ndx] << "\"";
2165 			}
2166 			builder << "}" << tcu::TestLog::EndMessage;
2167 		}
2168 
2169 		m_testCtx.getLog() << program;
2170 		if (!program.isOk())
2171 		{
2172 			m_testCtx.getLog() << tcu::TestLog::Message << "Program build failed, checking if program exceeded implementation limits" << tcu::TestLog::EndMessage;
2173 			checkProgramResourceUsage(programDefinition, m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
2174 
2175 			// within limits
2176 			throw tcu::TestError("could not build program");
2177 		}
2178 	}
2179 
2180 	// Check interface props
2181 
2182 	switch (m_queryTarget.interface)
2183 	{
2184 		case PROGRAMINTERFACE_UNIFORM:
2185 		{
2186 			const VariableSearchFilter					uniformFilter						= VariableSearchFilter::createStorageFilter(glu::STORAGE_UNIFORM);
2187 
2188 			const TypeValidator							typeValidator						(m_context, program.getProgram(),						uniformFilter);
2189 			const ArraySizeValidator					arraySizeValidator					(m_context, program.getProgram(),						-1,					uniformFilter);
2190 			const ArrayStrideValidator					arrayStrideValidator				(m_context, program.getProgram(),						uniformFilter);
2191 			const BlockIndexValidator					blockIndexValidator					(m_context, program.getProgram(),						uniformFilter);
2192 			const IsRowMajorValidator					isRowMajorValidator					(m_context, program.getProgram(),						uniformFilter);
2193 			const MatrixStrideValidator					matrixStrideValidator				(m_context, program.getProgram(),						uniformFilter);
2194 			const AtomicCounterBufferIndexVerifier		atomicCounterBufferIndexVerifier	(m_context, program.getProgram(),						uniformFilter);
2195 			const LocationValidator						locationValidator					(m_context, program.getProgram(),						uniformFilter);
2196 			const VariableNameLengthValidator			nameLengthValidator					(m_context, program.getProgram(),						uniformFilter);
2197 			const OffsetValidator						offsetVerifier						(m_context, program.getProgram(),						uniformFilter);
2198 			const VariableReferencedByShaderValidator	referencedByVertexVerifier			(m_context, glu::SHADERTYPE_VERTEX,						uniformFilter);
2199 			const VariableReferencedByShaderValidator	referencedByFragmentVerifier		(m_context, glu::SHADERTYPE_FRAGMENT,					uniformFilter);
2200 			const VariableReferencedByShaderValidator	referencedByComputeVerifier			(m_context, glu::SHADERTYPE_COMPUTE,					uniformFilter);
2201 			const VariableReferencedByShaderValidator	referencedByGeometryVerifier		(m_context, glu::SHADERTYPE_GEOMETRY,					uniformFilter);
2202 			const VariableReferencedByShaderValidator	referencedByTessControlVerifier		(m_context, glu::SHADERTYPE_TESSELLATION_CONTROL,		uniformFilter);
2203 			const VariableReferencedByShaderValidator	referencedByTessEvaluationVerifier	(m_context, glu::SHADERTYPE_TESSELLATION_EVALUATION,	uniformFilter);
2204 
2205 			const TestProperty allProperties[] =
2206 			{
2207 				{ GL_ARRAY_SIZE,							&arraySizeValidator					},
2208 				{ GL_ARRAY_STRIDE,							&arrayStrideValidator				},
2209 				{ GL_ATOMIC_COUNTER_BUFFER_INDEX,			&atomicCounterBufferIndexVerifier	},
2210 				{ GL_BLOCK_INDEX,							&blockIndexValidator				},
2211 				{ GL_IS_ROW_MAJOR,							&isRowMajorValidator				},
2212 				{ GL_LOCATION,								&locationValidator					},
2213 				{ GL_MATRIX_STRIDE,							&matrixStrideValidator				},
2214 				{ GL_NAME_LENGTH,							&nameLengthValidator				},
2215 				{ GL_OFFSET,								&offsetVerifier						},
2216 				{ GL_REFERENCED_BY_VERTEX_SHADER,			&referencedByVertexVerifier			},
2217 				{ GL_REFERENCED_BY_FRAGMENT_SHADER,			&referencedByFragmentVerifier		},
2218 				{ GL_REFERENCED_BY_COMPUTE_SHADER,			&referencedByComputeVerifier		},
2219 				{ GL_REFERENCED_BY_GEOMETRY_SHADER,			&referencedByGeometryVerifier		},
2220 				{ GL_REFERENCED_BY_TESS_CONTROL_SHADER,		&referencedByTessControlVerifier	},
2221 				{ GL_REFERENCED_BY_TESS_EVALUATION_SHADER,	&referencedByTessEvaluationVerifier	},
2222 				{ GL_TYPE,									&typeValidator						},
2223 			};
2224 
2225 			for (int targetResourceNdx = 0; targetResourceNdx < (int)targetResources.size(); ++targetResourceNdx)
2226 			{
2227 				const tcu::ScopedLogSection			section			(m_testCtx.getLog(), "UniformResource", "Uniform resource \"" +  targetResources[targetResourceNdx] + "\"");
2228 				const glw::Functions&				gl				= m_context.getRenderContext().getFunctions();
2229 				std::vector<glw::GLenum>			props;
2230 				std::vector<const PropValidator*>	validators;
2231 
2232 				for (int propNdx = 0; propNdx < DE_LENGTH_OF_ARRAY(allProperties); ++propNdx)
2233 				{
2234 					if (allProperties[propNdx].validator->isSelected(m_queryTarget.propFlags) &&
2235 						allProperties[propNdx].validator->isSupported())
2236 					{
2237 						props.push_back(allProperties[propNdx].prop);
2238 						validators.push_back(allProperties[propNdx].validator);
2239 					}
2240 				}
2241 
2242 				DE_ASSERT(!props.empty());
2243 
2244 				queryAndValidateProps(m_testCtx, gl, program.getProgram(), m_queryTarget.interface, targetResources[targetResourceNdx].c_str(), programDefinition, props, validators);
2245 			}
2246 
2247 			break;
2248 		}
2249 
2250 		case PROGRAMINTERFACE_UNIFORM_BLOCK:
2251 		case PROGRAMINTERFACE_SHADER_STORAGE_BLOCK:
2252 		{
2253 			const glu::Storage						storage								= (m_queryTarget.interface == PROGRAMINTERFACE_UNIFORM_BLOCK) ? (glu::STORAGE_UNIFORM) : (glu::STORAGE_BUFFER);
2254 			const VariableSearchFilter				blockFilter							= VariableSearchFilter::createStorageFilter(storage);
2255 
2256 			const BlockNameLengthValidator			nameLengthValidator					(m_context, program.getProgram(),						blockFilter);
2257 			const BlockReferencedByShaderValidator	referencedByVertexVerifier			(m_context, glu::SHADERTYPE_VERTEX,						blockFilter);
2258 			const BlockReferencedByShaderValidator	referencedByFragmentVerifier		(m_context, glu::SHADERTYPE_FRAGMENT,					blockFilter);
2259 			const BlockReferencedByShaderValidator	referencedByComputeVerifier			(m_context, glu::SHADERTYPE_COMPUTE,					blockFilter);
2260 			const BlockReferencedByShaderValidator	referencedByGeometryVerifier		(m_context, glu::SHADERTYPE_GEOMETRY,					blockFilter);
2261 			const BlockReferencedByShaderValidator	referencedByTessControlVerifier		(m_context, glu::SHADERTYPE_TESSELLATION_CONTROL,		blockFilter);
2262 			const BlockReferencedByShaderValidator	referencedByTessEvaluationVerifier	(m_context, glu::SHADERTYPE_TESSELLATION_EVALUATION,	blockFilter);
2263 			const BufferBindingValidator			bufferBindingValidator				(m_context, program.getProgram(),						blockFilter);
2264 
2265 			const TestProperty allProperties[] =
2266 			{
2267 				{ GL_NAME_LENGTH,							&nameLengthValidator				},
2268 				{ GL_REFERENCED_BY_VERTEX_SHADER,			&referencedByVertexVerifier			},
2269 				{ GL_REFERENCED_BY_FRAGMENT_SHADER,			&referencedByFragmentVerifier		},
2270 				{ GL_REFERENCED_BY_COMPUTE_SHADER,			&referencedByComputeVerifier		},
2271 				{ GL_REFERENCED_BY_GEOMETRY_SHADER,			&referencedByGeometryVerifier		},
2272 				{ GL_REFERENCED_BY_TESS_CONTROL_SHADER,		&referencedByTessControlVerifier	},
2273 				{ GL_REFERENCED_BY_TESS_EVALUATION_SHADER,	&referencedByTessEvaluationVerifier	},
2274 				{ GL_BUFFER_BINDING,						&bufferBindingValidator				},
2275 			};
2276 
2277 			for (int targetResourceNdx = 0; targetResourceNdx < (int)targetResources.size(); ++targetResourceNdx)
2278 			{
2279 				const tcu::ScopedLogSection			section			(m_testCtx.getLog(), "BlockResource", "Interface block \"" +  targetResources[targetResourceNdx] + "\"");
2280 				const glw::Functions&				gl				= m_context.getRenderContext().getFunctions();
2281 				std::vector<glw::GLenum>			props;
2282 				std::vector<const PropValidator*>	validators;
2283 
2284 				for (int propNdx = 0; propNdx < DE_LENGTH_OF_ARRAY(allProperties); ++propNdx)
2285 				{
2286 					if (allProperties[propNdx].validator->isSelected(m_queryTarget.propFlags) &&
2287 						allProperties[propNdx].validator->isSupported())
2288 					{
2289 						props.push_back(allProperties[propNdx].prop);
2290 						validators.push_back(allProperties[propNdx].validator);
2291 					}
2292 				}
2293 
2294 				DE_ASSERT(!props.empty());
2295 
2296 				queryAndValidateProps(m_testCtx, gl, program.getProgram(), m_queryTarget.interface, targetResources[targetResourceNdx].c_str(), programDefinition, props, validators);
2297 			}
2298 
2299 			break;
2300 		}
2301 
2302 		case PROGRAMINTERFACE_PROGRAM_INPUT:
2303 		case PROGRAMINTERFACE_PROGRAM_OUTPUT:
2304 		{
2305 			const bool									isInputCase							= (m_queryTarget.interface == PROGRAMINTERFACE_PROGRAM_INPUT);
2306 			const glu::Storage							varyingStorage						= (isInputCase) ? (glu::STORAGE_IN) : (glu::STORAGE_OUT);
2307 			const glu::Storage							patchStorage						= (isInputCase) ? (glu::STORAGE_PATCH_IN) : (glu::STORAGE_PATCH_OUT);
2308 			const glu::ShaderType						shaderType							= (isInputCase) ? (programDefinition->getFirstStage()) : (programDefinition->getLastStage());
2309 			const int									unsizedArraySize					= (isInputCase && shaderType == glu::SHADERTYPE_GEOMETRY)					? (1)															// input points
2310 																							: (isInputCase && shaderType == glu::SHADERTYPE_TESSELLATION_CONTROL)		? (getMaxPatchVertices())										// input batch size
2311 																							: (!isInputCase && shaderType == glu::SHADERTYPE_TESSELLATION_CONTROL)		? (programDefinition->getTessellationNumOutputPatchVertices())	// output batch size
2312 																							: (isInputCase && shaderType == glu::SHADERTYPE_TESSELLATION_EVALUATION)	? (getMaxPatchVertices())										// input batch size
2313 																							: (-1);
2314 			const VariableSearchFilter					variableFilter						= VariableSearchFilter::logicalAnd(VariableSearchFilter::createShaderTypeFilter(shaderType),
2315 																															   VariableSearchFilter::logicalOr(VariableSearchFilter::createStorageFilter(varyingStorage),
2316 																																							   VariableSearchFilter::createStorageFilter(patchStorage)));
2317 
2318 			const TypeValidator							typeValidator						(m_context, program.getProgram(),						variableFilter);
2319 			const ArraySizeValidator					arraySizeValidator					(m_context, program.getProgram(),						unsizedArraySize,		variableFilter);
2320 			const LocationValidator						locationValidator					(m_context, program.getProgram(),						variableFilter);
2321 			const VariableNameLengthValidator			nameLengthValidator					(m_context, program.getProgram(),						variableFilter);
2322 			const VariableReferencedByShaderValidator	referencedByVertexVerifier			(m_context, glu::SHADERTYPE_VERTEX,						variableFilter);
2323 			const VariableReferencedByShaderValidator	referencedByFragmentVerifier		(m_context, glu::SHADERTYPE_FRAGMENT,					variableFilter);
2324 			const VariableReferencedByShaderValidator	referencedByComputeVerifier			(m_context, glu::SHADERTYPE_COMPUTE,					variableFilter);
2325 			const VariableReferencedByShaderValidator	referencedByGeometryVerifier		(m_context, glu::SHADERTYPE_GEOMETRY,					variableFilter);
2326 			const VariableReferencedByShaderValidator	referencedByTessControlVerifier		(m_context, glu::SHADERTYPE_TESSELLATION_CONTROL,		variableFilter);
2327 			const VariableReferencedByShaderValidator	referencedByTessEvaluationVerifier	(m_context, glu::SHADERTYPE_TESSELLATION_EVALUATION,	variableFilter);
2328 			const PerPatchValidator						perPatchValidator					(m_context, program.getProgram(),						variableFilter);
2329 
2330 			const TestProperty allProperties[] =
2331 			{
2332 				{ GL_ARRAY_SIZE,							&arraySizeValidator					},
2333 				{ GL_LOCATION,								&locationValidator					},
2334 				{ GL_NAME_LENGTH,							&nameLengthValidator				},
2335 				{ GL_REFERENCED_BY_VERTEX_SHADER,			&referencedByVertexVerifier			},
2336 				{ GL_REFERENCED_BY_FRAGMENT_SHADER,			&referencedByFragmentVerifier		},
2337 				{ GL_REFERENCED_BY_COMPUTE_SHADER,			&referencedByComputeVerifier		},
2338 				{ GL_REFERENCED_BY_GEOMETRY_SHADER,			&referencedByGeometryVerifier		},
2339 				{ GL_REFERENCED_BY_TESS_CONTROL_SHADER,		&referencedByTessControlVerifier	},
2340 				{ GL_REFERENCED_BY_TESS_EVALUATION_SHADER,	&referencedByTessEvaluationVerifier	},
2341 				{ GL_TYPE,									&typeValidator						},
2342 				{ GL_IS_PER_PATCH,							&perPatchValidator					},
2343 			};
2344 
2345 			for (int targetResourceNdx = 0; targetResourceNdx < (int)targetResources.size(); ++targetResourceNdx)
2346 			{
2347 				const std::string					resourceInterfaceName	= (m_queryTarget.interface == PROGRAMINTERFACE_PROGRAM_INPUT) ? ("Input") : ("Output");
2348 				const tcu::ScopedLogSection			section					(m_testCtx.getLog(), "BlockResource", resourceInterfaceName + " resource \"" +  targetResources[targetResourceNdx] + "\"");
2349 				const glw::Functions&				gl						= m_context.getRenderContext().getFunctions();
2350 				std::vector<glw::GLenum>			props;
2351 				std::vector<const PropValidator*>	validators;
2352 
2353 				for (int propNdx = 0; propNdx < DE_LENGTH_OF_ARRAY(allProperties); ++propNdx)
2354 				{
2355 					if (allProperties[propNdx].validator->isSelected(m_queryTarget.propFlags) &&
2356 						allProperties[propNdx].validator->isSupported())
2357 					{
2358 						props.push_back(allProperties[propNdx].prop);
2359 						validators.push_back(allProperties[propNdx].validator);
2360 					}
2361 				}
2362 
2363 				DE_ASSERT(!props.empty());
2364 
2365 				queryAndValidateProps(m_testCtx, gl, program.getProgram(), m_queryTarget.interface, targetResources[targetResourceNdx].c_str(), programDefinition, props, validators);
2366 			}
2367 
2368 			break;
2369 		}
2370 
2371 		case PROGRAMINTERFACE_BUFFER_VARIABLE:
2372 		{
2373 			const VariableSearchFilter					variableFilter						= VariableSearchFilter::createStorageFilter(glu::STORAGE_BUFFER);
2374 
2375 			const TypeValidator							typeValidator						(m_context, program.getProgram(),						variableFilter);
2376 			const ArraySizeValidator					arraySizeValidator					(m_context, program.getProgram(),						0,					variableFilter);
2377 			const ArrayStrideValidator					arrayStrideValidator				(m_context, program.getProgram(),						variableFilter);
2378 			const BlockIndexValidator					blockIndexValidator					(m_context, program.getProgram(),						variableFilter);
2379 			const IsRowMajorValidator					isRowMajorValidator					(m_context, program.getProgram(),						variableFilter);
2380 			const MatrixStrideValidator					matrixStrideValidator				(m_context, program.getProgram(),						variableFilter);
2381 			const OffsetValidator						offsetValidator						(m_context, program.getProgram(),						variableFilter);
2382 			const VariableNameLengthValidator			nameLengthValidator					(m_context, program.getProgram(),						variableFilter);
2383 			const VariableReferencedByShaderValidator	referencedByVertexVerifier			(m_context, glu::SHADERTYPE_VERTEX,						variableFilter);
2384 			const VariableReferencedByShaderValidator	referencedByFragmentVerifier		(m_context, glu::SHADERTYPE_FRAGMENT,					variableFilter);
2385 			const VariableReferencedByShaderValidator	referencedByComputeVerifier			(m_context, glu::SHADERTYPE_COMPUTE,					variableFilter);
2386 			const VariableReferencedByShaderValidator	referencedByGeometryVerifier		(m_context, glu::SHADERTYPE_GEOMETRY,					variableFilter);
2387 			const VariableReferencedByShaderValidator	referencedByTessControlVerifier		(m_context, glu::SHADERTYPE_TESSELLATION_CONTROL,		variableFilter);
2388 			const VariableReferencedByShaderValidator	referencedByTessEvaluationVerifier	(m_context, glu::SHADERTYPE_TESSELLATION_EVALUATION,	variableFilter);
2389 			const TopLevelArraySizeValidator			topLevelArraySizeValidator			(m_context, program.getProgram(),						variableFilter);
2390 			const TopLevelArrayStrideValidator			topLevelArrayStrideValidator		(m_context, program.getProgram(),						variableFilter);
2391 
2392 			const TestProperty allProperties[] =
2393 			{
2394 				{ GL_ARRAY_SIZE,							&arraySizeValidator					},
2395 				{ GL_ARRAY_STRIDE,							&arrayStrideValidator				},
2396 				{ GL_BLOCK_INDEX,							&blockIndexValidator				},
2397 				{ GL_IS_ROW_MAJOR,							&isRowMajorValidator				},
2398 				{ GL_MATRIX_STRIDE,							&matrixStrideValidator				},
2399 				{ GL_NAME_LENGTH,							&nameLengthValidator				},
2400 				{ GL_OFFSET,								&offsetValidator					},
2401 				{ GL_REFERENCED_BY_VERTEX_SHADER,			&referencedByVertexVerifier			},
2402 				{ GL_REFERENCED_BY_FRAGMENT_SHADER,			&referencedByFragmentVerifier		},
2403 				{ GL_REFERENCED_BY_COMPUTE_SHADER,			&referencedByComputeVerifier		},
2404 				{ GL_REFERENCED_BY_GEOMETRY_SHADER,			&referencedByGeometryVerifier		},
2405 				{ GL_REFERENCED_BY_TESS_CONTROL_SHADER,		&referencedByTessControlVerifier	},
2406 				{ GL_REFERENCED_BY_TESS_EVALUATION_SHADER,	&referencedByTessEvaluationVerifier	},
2407 				{ GL_TOP_LEVEL_ARRAY_SIZE,					&topLevelArraySizeValidator			},
2408 				{ GL_TOP_LEVEL_ARRAY_STRIDE,				&topLevelArrayStrideValidator		},
2409 				{ GL_TYPE,									&typeValidator						},
2410 			};
2411 
2412 			for (int targetResourceNdx = 0; targetResourceNdx < (int)targetResources.size(); ++targetResourceNdx)
2413 			{
2414 				const tcu::ScopedLogSection			section			(m_testCtx.getLog(), "BufferVariableResource", "Buffer variable \"" +  targetResources[targetResourceNdx] + "\"");
2415 				const glw::Functions&				gl				= m_context.getRenderContext().getFunctions();
2416 				std::vector<glw::GLenum>			props;
2417 				std::vector<const PropValidator*>	validators;
2418 
2419 				for (int propNdx = 0; propNdx < DE_LENGTH_OF_ARRAY(allProperties); ++propNdx)
2420 				{
2421 					if (allProperties[propNdx].validator->isSelected(m_queryTarget.propFlags) &&
2422 						allProperties[propNdx].validator->isSupported())
2423 					{
2424 						props.push_back(allProperties[propNdx].prop);
2425 						validators.push_back(allProperties[propNdx].validator);
2426 					}
2427 				}
2428 
2429 				DE_ASSERT(!props.empty());
2430 
2431 				queryAndValidateProps(m_testCtx, gl, program.getProgram(), m_queryTarget.interface, targetResources[targetResourceNdx].c_str(), programDefinition, props, validators);
2432 			}
2433 
2434 			break;
2435 		}
2436 
2437 		case PROGRAMINTERFACE_TRANSFORM_FEEDBACK_VARYING:
2438 		{
2439 			const TransformFeedbackTypeValidator		typeValidator			(m_context);
2440 			const TransformFeedbackArraySizeValidator	arraySizeValidator		(m_context);
2441 			const TransformFeedbackNameLengthValidator	nameLengthValidator		(m_context);
2442 
2443 			const TestProperty allProperties[] =
2444 			{
2445 				{ GL_ARRAY_SIZE,					&arraySizeValidator				},
2446 				{ GL_NAME_LENGTH,					&nameLengthValidator			},
2447 				{ GL_TYPE,							&typeValidator					},
2448 			};
2449 
2450 			for (int targetResourceNdx = 0; targetResourceNdx < (int)targetResources.size(); ++targetResourceNdx)
2451 			{
2452 				const tcu::ScopedLogSection			section			(m_testCtx.getLog(), "XFBVariableResource", "Transform feedback varying \"" +  targetResources[targetResourceNdx] + "\"");
2453 				const glw::Functions&				gl				= m_context.getRenderContext().getFunctions();
2454 				std::vector<glw::GLenum>			props;
2455 				std::vector<const PropValidator*>	validators;
2456 
2457 				for (int propNdx = 0; propNdx < DE_LENGTH_OF_ARRAY(allProperties); ++propNdx)
2458 				{
2459 					if (allProperties[propNdx].validator->isSelected(m_queryTarget.propFlags) &&
2460 						allProperties[propNdx].validator->isSupported())
2461 					{
2462 						props.push_back(allProperties[propNdx].prop);
2463 						validators.push_back(allProperties[propNdx].validator);
2464 					}
2465 				}
2466 
2467 				DE_ASSERT(!props.empty());
2468 
2469 				queryAndValidateProps(m_testCtx, gl, program.getProgram(), m_queryTarget.interface, targetResources[targetResourceNdx].c_str(), programDefinition, props, validators);
2470 			}
2471 
2472 			break;
2473 		}
2474 
2475 		default:
2476 			DE_ASSERT(false);
2477 	}
2478 
2479 	return STOP;
2480 }
2481 
checkLimit(glw::GLenum pname,int usage,const glw::Functions & gl,tcu::TestLog & log)2482 static bool checkLimit (glw::GLenum pname, int usage, const glw::Functions& gl, tcu::TestLog& log)
2483 {
2484 	if (usage > 0)
2485 	{
2486 		glw::GLint limit = 0;
2487 		gl.getIntegerv(pname, &limit);
2488 		GLU_EXPECT_NO_ERROR(gl.getError(), "query limits");
2489 
2490 		log << tcu::TestLog::Message << "\t" << glu::getGettableStateStr(pname) << " = " << limit << ", test requires " << usage << tcu::TestLog::EndMessage;
2491 
2492 		if (limit < usage)
2493 		{
2494 			log << tcu::TestLog::Message << "\t\tLimit exceeded" << tcu::TestLog::EndMessage;
2495 			return false;
2496 		}
2497 	}
2498 
2499 	return true;
2500 }
2501 
checkShaderResourceUsage(const ProgramInterfaceDefinition::Program * program,const ProgramInterfaceDefinition::Shader * shader,const glw::Functions & gl,tcu::TestLog & log)2502 static bool checkShaderResourceUsage (const ProgramInterfaceDefinition::Program* program, const ProgramInterfaceDefinition::Shader* shader, const glw::Functions& gl, tcu::TestLog& log)
2503 {
2504 	const ProgramInterfaceDefinition::ShaderResourceUsage usage = getShaderResourceUsage(program, shader);
2505 
2506 	switch (shader->getType())
2507 	{
2508 		case glu::SHADERTYPE_VERTEX:
2509 		{
2510 			const struct
2511 			{
2512 				glw::GLenum	pname;
2513 				int			usage;
2514 			} restrictions[] =
2515 			{
2516 				{ GL_MAX_VERTEX_ATTRIBS,						usage.numInputVectors					},
2517 				{ GL_MAX_VERTEX_UNIFORM_COMPONENTS,				usage.numDefaultBlockUniformComponents	},
2518 				{ GL_MAX_VERTEX_UNIFORM_VECTORS,				usage.numUniformVectors					},
2519 				{ GL_MAX_VERTEX_UNIFORM_BLOCKS,					usage.numUniformBlocks					},
2520 				{ GL_MAX_VERTEX_OUTPUT_COMPONENTS,				usage.numOutputComponents				},
2521 				{ GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS,			usage.numSamplers						},
2522 				{ GL_MAX_VERTEX_ATOMIC_COUNTER_BUFFERS,			usage.numAtomicCounterBuffers			},
2523 				{ GL_MAX_VERTEX_ATOMIC_COUNTERS,				usage.numAtomicCounters					},
2524 				{ GL_MAX_VERTEX_IMAGE_UNIFORMS,					usage.numImages							},
2525 				{ GL_MAX_COMBINED_VERTEX_UNIFORM_COMPONENTS,	usage.numCombinedUniformComponents		},
2526 				{ GL_MAX_VERTEX_SHADER_STORAGE_BLOCKS,			usage.numShaderStorageBlocks			},
2527 			};
2528 
2529 			bool allOk = true;
2530 
2531 			log << tcu::TestLog::Message << "Vertex shader:" << tcu::TestLog::EndMessage;
2532 			for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(restrictions); ++ndx)
2533 				allOk &= checkLimit(restrictions[ndx].pname, restrictions[ndx].usage, gl, log);
2534 
2535 			return allOk;
2536 		}
2537 
2538 		case glu::SHADERTYPE_FRAGMENT:
2539 		{
2540 			const struct
2541 			{
2542 				glw::GLenum	pname;
2543 				int			usage;
2544 			} restrictions[] =
2545 			{
2546 				{ GL_MAX_FRAGMENT_UNIFORM_COMPONENTS,			usage.numDefaultBlockUniformComponents		},
2547 				{ GL_MAX_FRAGMENT_UNIFORM_VECTORS,				usage.numUniformVectors						},
2548 				{ GL_MAX_FRAGMENT_UNIFORM_BLOCKS,				usage.numUniformBlocks						},
2549 				{ GL_MAX_FRAGMENT_INPUT_COMPONENTS,				usage.numInputComponents					},
2550 				{ GL_MAX_TEXTURE_IMAGE_UNITS,					usage.numSamplers							},
2551 				{ GL_MAX_FRAGMENT_ATOMIC_COUNTER_BUFFERS,		usage.numAtomicCounterBuffers				},
2552 				{ GL_MAX_FRAGMENT_ATOMIC_COUNTERS,				usage.numAtomicCounters						},
2553 				{ GL_MAX_FRAGMENT_IMAGE_UNIFORMS,				usage.numImages								},
2554 				{ GL_MAX_COMBINED_FRAGMENT_UNIFORM_COMPONENTS,	usage.numCombinedUniformComponents			},
2555 				{ GL_MAX_FRAGMENT_SHADER_STORAGE_BLOCKS,		usage.numShaderStorageBlocks				},
2556 			};
2557 
2558 			bool allOk = true;
2559 
2560 			log << tcu::TestLog::Message << "Fragment shader:" << tcu::TestLog::EndMessage;
2561 			for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(restrictions); ++ndx)
2562 				allOk &= checkLimit(restrictions[ndx].pname, restrictions[ndx].usage, gl, log);
2563 
2564 			return allOk;
2565 		}
2566 
2567 		case glu::SHADERTYPE_COMPUTE:
2568 		{
2569 			const struct
2570 			{
2571 				glw::GLenum	pname;
2572 				int			usage;
2573 			} restrictions[] =
2574 			{
2575 				{ GL_MAX_COMPUTE_UNIFORM_BLOCKS,				usage.numUniformBlocks					},
2576 				{ GL_MAX_COMPUTE_TEXTURE_IMAGE_UNITS,			usage.numSamplers						},
2577 				{ GL_MAX_COMPUTE_UNIFORM_COMPONENTS,			usage.numDefaultBlockUniformComponents	},
2578 				{ GL_MAX_COMPUTE_ATOMIC_COUNTER_BUFFERS,		usage.numAtomicCounterBuffers			},
2579 				{ GL_MAX_COMPUTE_ATOMIC_COUNTERS,				usage.numAtomicCounters					},
2580 				{ GL_MAX_COMPUTE_IMAGE_UNIFORMS,				usage.numImages							},
2581 				{ GL_MAX_COMBINED_COMPUTE_UNIFORM_COMPONENTS,	usage.numCombinedUniformComponents		},
2582 				{ GL_MAX_COMPUTE_SHADER_STORAGE_BLOCKS,			usage.numShaderStorageBlocks			},
2583 			};
2584 
2585 			bool allOk = true;
2586 
2587 			log << tcu::TestLog::Message << "Compute shader:" << tcu::TestLog::EndMessage;
2588 			for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(restrictions); ++ndx)
2589 				allOk &= checkLimit(restrictions[ndx].pname, restrictions[ndx].usage, gl, log);
2590 
2591 			return allOk;
2592 		}
2593 
2594 		case glu::SHADERTYPE_GEOMETRY:
2595 		{
2596 			const int totalOutputComponents = program->getGeometryNumOutputVertices() * usage.numOutputComponents;
2597 			const struct
2598 			{
2599 				glw::GLenum	pname;
2600 				int			usage;
2601 			} restrictions[] =
2602 			{
2603 				{ GL_MAX_GEOMETRY_UNIFORM_COMPONENTS,				usage.numDefaultBlockUniformComponents			},
2604 				{ GL_MAX_GEOMETRY_UNIFORM_BLOCKS,					usage.numUniformBlocks							},
2605 				{ GL_MAX_GEOMETRY_INPUT_COMPONENTS,					usage.numInputComponents						},
2606 				{ GL_MAX_GEOMETRY_OUTPUT_COMPONENTS,				usage.numOutputComponents						},
2607 				{ GL_MAX_GEOMETRY_OUTPUT_VERTICES,					(int)program->getGeometryNumOutputVertices()	},
2608 				{ GL_MAX_GEOMETRY_TOTAL_OUTPUT_COMPONENTS,			totalOutputComponents							},
2609 				{ GL_MAX_GEOMETRY_TEXTURE_IMAGE_UNITS,				usage.numSamplers								},
2610 				{ GL_MAX_GEOMETRY_ATOMIC_COUNTER_BUFFERS,			usage.numAtomicCounterBuffers					},
2611 				{ GL_MAX_GEOMETRY_ATOMIC_COUNTERS,					usage.numAtomicCounters							},
2612 				{ GL_MAX_GEOMETRY_IMAGE_UNIFORMS,					usage.numImages									},
2613 				{ GL_MAX_GEOMETRY_SHADER_STORAGE_BLOCKS,			usage.numShaderStorageBlocks					},
2614 			};
2615 
2616 			bool allOk = true;
2617 
2618 			log << tcu::TestLog::Message << "Geometry shader:" << tcu::TestLog::EndMessage;
2619 			for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(restrictions); ++ndx)
2620 				allOk &= checkLimit(restrictions[ndx].pname, restrictions[ndx].usage, gl, log);
2621 
2622 			return allOk;
2623 		}
2624 
2625 		case glu::SHADERTYPE_TESSELLATION_CONTROL:
2626 		{
2627 			const int totalOutputComponents = program->getTessellationNumOutputPatchVertices() * usage.numOutputComponents + usage.numPatchOutputComponents;
2628 			const struct
2629 			{
2630 				glw::GLenum	pname;
2631 				int			usage;
2632 			} restrictions[] =
2633 			{
2634 				{ GL_MAX_PATCH_VERTICES,								(int)program->getTessellationNumOutputPatchVertices()	},
2635 				{ GL_MAX_TESS_PATCH_COMPONENTS,							usage.numPatchOutputComponents							},
2636 				{ GL_MAX_TESS_CONTROL_UNIFORM_COMPONENTS,				usage.numDefaultBlockUniformComponents					},
2637 				{ GL_MAX_TESS_CONTROL_UNIFORM_BLOCKS,					usage.numUniformBlocks									},
2638 				{ GL_MAX_TESS_CONTROL_INPUT_COMPONENTS,					usage.numInputComponents								},
2639 				{ GL_MAX_TESS_CONTROL_OUTPUT_COMPONENTS,				usage.numOutputComponents								},
2640 				{ GL_MAX_TESS_CONTROL_TOTAL_OUTPUT_COMPONENTS,			totalOutputComponents									},
2641 				{ GL_MAX_TESS_CONTROL_TEXTURE_IMAGE_UNITS,				usage.numSamplers										},
2642 				{ GL_MAX_TESS_CONTROL_ATOMIC_COUNTER_BUFFERS,			usage.numAtomicCounterBuffers							},
2643 				{ GL_MAX_TESS_CONTROL_ATOMIC_COUNTERS,					usage.numAtomicCounters									},
2644 				{ GL_MAX_TESS_CONTROL_IMAGE_UNIFORMS,					usage.numImages											},
2645 				{ GL_MAX_TESS_CONTROL_SHADER_STORAGE_BLOCKS,			usage.numShaderStorageBlocks							},
2646 			};
2647 
2648 			bool allOk = true;
2649 
2650 			log << tcu::TestLog::Message << "Tessellation control shader:" << tcu::TestLog::EndMessage;
2651 			for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(restrictions); ++ndx)
2652 				allOk &= checkLimit(restrictions[ndx].pname, restrictions[ndx].usage, gl, log);
2653 
2654 			return allOk;
2655 		}
2656 
2657 		case glu::SHADERTYPE_TESSELLATION_EVALUATION:
2658 		{
2659 			const struct
2660 			{
2661 				glw::GLenum	pname;
2662 				int			usage;
2663 			} restrictions[] =
2664 			{
2665 				{ GL_MAX_PATCH_VERTICES,								(int)program->getTessellationNumOutputPatchVertices()	},
2666 				{ GL_MAX_TESS_PATCH_COMPONENTS,							usage.numPatchInputComponents							},
2667 				{ GL_MAX_TESS_EVALUATION_UNIFORM_COMPONENTS,			usage.numDefaultBlockUniformComponents					},
2668 				{ GL_MAX_TESS_EVALUATION_UNIFORM_BLOCKS,				usage.numUniformBlocks									},
2669 				{ GL_MAX_TESS_EVALUATION_INPUT_COMPONENTS,				usage.numInputComponents								},
2670 				{ GL_MAX_TESS_EVALUATION_OUTPUT_COMPONENTS,				usage.numOutputComponents								},
2671 				{ GL_MAX_TESS_EVALUATION_TEXTURE_IMAGE_UNITS,			usage.numSamplers										},
2672 				{ GL_MAX_TESS_EVALUATION_ATOMIC_COUNTER_BUFFERS,		usage.numAtomicCounterBuffers							},
2673 				{ GL_MAX_TESS_EVALUATION_ATOMIC_COUNTERS,				usage.numAtomicCounters									},
2674 				{ GL_MAX_TESS_EVALUATION_IMAGE_UNIFORMS,				usage.numImages											},
2675 				{ GL_MAX_TESS_EVALUATION_SHADER_STORAGE_BLOCKS,			usage.numShaderStorageBlocks							},
2676 			};
2677 
2678 			bool allOk = true;
2679 
2680 			log << tcu::TestLog::Message << "Tessellation evaluation shader:" << tcu::TestLog::EndMessage;
2681 			for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(restrictions); ++ndx)
2682 				allOk &= checkLimit(restrictions[ndx].pname, restrictions[ndx].usage, gl, log);
2683 
2684 			return allOk;
2685 		}
2686 
2687 		default:
2688 			DE_ASSERT(false);
2689 			return false;
2690 	}
2691 }
2692 
checkProgramCombinedResourceUsage(const ProgramInterfaceDefinition::Program * program,const glw::Functions & gl,tcu::TestLog & log)2693 static bool checkProgramCombinedResourceUsage (const ProgramInterfaceDefinition::Program* program, const glw::Functions& gl, tcu::TestLog& log)
2694 {
2695 	const ProgramInterfaceDefinition::ProgramResourceUsage usage = getCombinedProgramResourceUsage(program);
2696 
2697 	const struct
2698 	{
2699 		glw::GLenum	pname;
2700 		int			usage;
2701 	} restrictions[] =
2702 	{
2703 		{ GL_MAX_UNIFORM_BUFFER_BINDINGS,						usage.uniformBufferMaxBinding+1					},
2704 		{ GL_MAX_UNIFORM_BLOCK_SIZE,							usage.uniformBufferMaxSize						},
2705 		{ GL_MAX_COMBINED_UNIFORM_BLOCKS,						usage.numUniformBlocks							},
2706 		{ GL_MAX_COMBINED_VERTEX_UNIFORM_COMPONENTS,			usage.numCombinedVertexUniformComponents		},
2707 		{ GL_MAX_COMBINED_FRAGMENT_UNIFORM_COMPONENTS,			usage.numCombinedFragmentUniformComponents		},
2708 		{ GL_MAX_COMBINED_GEOMETRY_UNIFORM_COMPONENTS,			usage.numCombinedGeometryUniformComponents		},
2709 		{ GL_MAX_COMBINED_TESS_CONTROL_UNIFORM_COMPONENTS,		usage.numCombinedTessControlUniformComponents	},
2710 		{ GL_MAX_COMBINED_TESS_EVALUATION_UNIFORM_COMPONENTS,	usage.numCombinedTessEvalUniformComponents		},
2711 		{ GL_MAX_VARYING_COMPONENTS,							usage.numVaryingComponents						},
2712 		{ GL_MAX_VARYING_VECTORS,								usage.numVaryingVectors							},
2713 		{ GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS,					usage.numCombinedSamplers						},
2714 		{ GL_MAX_COMBINED_SHADER_OUTPUT_RESOURCES,				usage.numCombinedOutputResources				},
2715 		{ GL_MAX_ATOMIC_COUNTER_BUFFER_BINDINGS,				usage.atomicCounterBufferMaxBinding+1			},
2716 		{ GL_MAX_ATOMIC_COUNTER_BUFFER_SIZE,					usage.atomicCounterBufferMaxSize				},
2717 		{ GL_MAX_COMBINED_ATOMIC_COUNTER_BUFFERS,				usage.numAtomicCounterBuffers					},
2718 		{ GL_MAX_COMBINED_ATOMIC_COUNTERS,						usage.numAtomicCounters							},
2719 		{ GL_MAX_IMAGE_UNITS,									usage.maxImageBinding+1							},
2720 		{ GL_MAX_COMBINED_IMAGE_UNIFORMS,						usage.numCombinedImages							},
2721 		{ GL_MAX_SHADER_STORAGE_BUFFER_BINDINGS,				usage.shaderStorageBufferMaxBinding+1			},
2722 		{ GL_MAX_SHADER_STORAGE_BLOCK_SIZE,						usage.shaderStorageBufferMaxSize				},
2723 		{ GL_MAX_COMBINED_SHADER_STORAGE_BLOCKS,				usage.numShaderStorageBlocks					},
2724 		{ GL_MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS,		usage.numXFBInterleavedComponents				},
2725 		{ GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS,			usage.numXFBSeparateAttribs						},
2726 		{ GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS,		usage.numXFBSeparateComponents					},
2727 		{ GL_MAX_DRAW_BUFFERS,									usage.fragmentOutputMaxBinding+1				},
2728 	};
2729 
2730 	bool allOk = true;
2731 
2732 	log << tcu::TestLog::Message << "Program combined:" << tcu::TestLog::EndMessage;
2733 	for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(restrictions); ++ndx)
2734 		allOk &= checkLimit(restrictions[ndx].pname, restrictions[ndx].usage, gl, log);
2735 
2736 	return allOk;
2737 }
2738 
checkProgramResourceUsage(const ProgramInterfaceDefinition::Program * program,const glw::Functions & gl,tcu::TestLog & log)2739 void checkProgramResourceUsage (const ProgramInterfaceDefinition::Program* program, const glw::Functions& gl, tcu::TestLog& log)
2740 {
2741 	bool limitExceeded = false;
2742 
2743 	for (int shaderNdx = 0; shaderNdx < (int)program->getShaders().size(); ++shaderNdx)
2744 		limitExceeded |= !checkShaderResourceUsage(program, program->getShaders()[shaderNdx], gl, log);
2745 
2746 	limitExceeded |= !checkProgramCombinedResourceUsage(program, gl, log);
2747 
2748 	if (limitExceeded)
2749 	{
2750 		log << tcu::TestLog::Message << "One or more resource limits exceeded" << tcu::TestLog::EndMessage;
2751 		throw tcu::NotSupportedError("one or more resource limits exceeded");
2752 	}
2753 }
2754 
2755 } // Functional
2756 } // gles31
2757 } // deqp
2758