1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program OpenGL ES 3.0 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 Shader compilation performance tests.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es3pShaderCompilationCases.hpp"
25 #include "tcuTestLog.hpp"
26 #include "tcuVector.hpp"
27 #include "tcuMatrix.hpp"
28 #include "tcuTextureUtil.hpp"
29 #include "tcuPlatform.hpp"
30 #include "tcuCommandLine.hpp"
31 #include "tcuRenderTarget.hpp"
32 #include "tcuCPUWarmup.hpp"
33 #include "tcuStringTemplate.hpp"
34 #include "gluTexture.hpp"
35 #include "gluPixelTransfer.hpp"
36 #include "gluRenderContext.hpp"
37 #include "deStringUtil.hpp"
38 #include "deRandom.hpp"
39 #include "deClock.h"
40 #include "deMath.h"
41 
42 #include "glwEnums.hpp"
43 #include "glwFunctions.hpp"
44 
45 #include <map>
46 #include <algorithm>
47 #include <limits>
48 #include <iomanip>
49 
50 using tcu::TestLog;
51 using tcu::Vec3;
52 using tcu::Vec4;
53 using tcu::Mat3;
54 using tcu::Mat4;
55 using std::string;
56 using std::vector;
57 using namespace glw; // GL types
58 
59 namespace deqp
60 {
61 
62 namespace gles3
63 {
64 
65 namespace Performance
66 {
67 
68 static const bool	WARMUP_CPU_AT_BEGINNING_OF_CASE					= false;
69 static const bool	WARMUP_CPU_BEFORE_EACH_MEASUREMENT				= true;
70 
71 static const int	MAX_VIEWPORT_WIDTH								= 64;
72 static const int	MAX_VIEWPORT_HEIGHT								= 64;
73 
74 static const int	DEFAULT_MINIMUM_MEASUREMENT_COUNT				= 15;
75 static const float	RELATIVE_MEDIAN_ABSOLUTE_DEVIATION_THRESHOLD	= 0.05f;
76 
77 // Texture size for the light shader and texture lookup shader cases.
78 static const int	TEXTURE_WIDTH									= 64;
79 static const int	TEXTURE_HEIGHT									= 64;
80 
81 template <typename T>
toStringWithPadding(T value,int minLength)82 inline string toStringWithPadding (T value, int minLength)
83 {
84 	std::ostringstream s;
85 	s << std::setfill('0') << std::setw(minLength) << value;
86 	return s.str();
87 }
88 
89 // Add some whitespace and comments to str. They should depend on uniqueNumber.
strWithWhiteSpaceAndComments(const string & str,deUint32 uniqueNumber)90 static string strWithWhiteSpaceAndComments (const string& str, deUint32 uniqueNumber)
91 {
92 	string res("");
93 
94 	// Find the first newline.
95 	int firstLineEndNdx = 0;
96 	while (firstLineEndNdx < (int)str.size() && str[firstLineEndNdx] != '\n')
97 	{
98 		res += str[firstLineEndNdx];
99 		firstLineEndNdx++;
100 	}
101 	res += '\n';
102 	DE_ASSERT(firstLineEndNdx < (int)str.size());
103 
104 	// Add the whitespaces and comments just after the first line.
105 
106 	de::Random		rnd		(uniqueNumber);
107 	int				numWS	= rnd.getInt(10, 20);
108 
109 	for (int i = 0; i < numWS; i++)
110 		res += " \t\n"[rnd.getInt(0, 2)];
111 
112 	res += "/* unique comment " + de::toString(uniqueNumber) + " */\n";
113 	res += "// unique comment " + de::toString(uniqueNumber) + "\n";
114 
115 	for (int i = 0; i < numWS; i++)
116 		res += " \t\n"[rnd.getInt(0, 2)];
117 
118 	// Add the rest of the string.
119 	res.append(&str.c_str()[firstLineEndNdx + 1]);
120 
121 	return res;
122 }
123 
124 //! Helper for computing relative magnitudes while avoiding division by zero.
hackySafeRelativeResult(float x,float y)125 static float hackySafeRelativeResult (float x, float y)
126 {
127 	// \note A possible case is that x is standard deviation, and y is average
128 	//		 (or similarly for median or some such). So, if y is 0, that
129 	//		 probably means that x is also 0(ish) (because in practice we're
130 	//		 dealing with non-negative values, in which case an average of 0
131 	//		 implies that the samples are all 0 - note that the same isn't
132 	//		 strictly true for things like median) so a relative result of 0
133 	//		 wouldn't be that far from the truth.
134 	return y == 0.0f ? 0.0f : x/y;
135 }
136 
137 template <typename T>
vectorFloatAverage(const vector<T> & v)138 static float vectorFloatAverage (const vector<T>& v)
139 {
140 	DE_ASSERT(!v.empty());
141 	float result = 0.0f;
142 	for (int i = 0; i < (int)v.size(); i++)
143 		result += (float)v[i];
144 	return result / (float)v.size();
145 }
146 
147 template <typename T>
vectorFloatMedian(const vector<T> & v)148 static float vectorFloatMedian (const vector<T>& v)
149 {
150 	DE_ASSERT(!v.empty());
151 	vector<T> temp = v;
152 	std::sort(temp.begin(), temp.end());
153 	return temp.size() % 2 == 0
154 		   ? 0.5f * ((float)temp[temp.size()/2-1] + (float)temp[temp.size()/2])
155 		   : (float)temp[temp.size()/2];
156 }
157 
158 template <typename T>
vectorFloatMinimum(const vector<T> & v)159 static float vectorFloatMinimum (const vector<T>& v)
160 {
161 	DE_ASSERT(!v.empty());
162 	return (float)*std::min_element(v.begin(), v.end());
163 }
164 
165 template <typename T>
vectorFloatMaximum(const vector<T> & v)166 static float vectorFloatMaximum (const vector<T>& v)
167 {
168 	DE_ASSERT(!v.empty());
169 	return (float)*std::max_element(v.begin(), v.end());
170 }
171 
172 template <typename T>
vectorFloatStandardDeviation(const vector<T> & v)173 static float vectorFloatStandardDeviation (const vector<T>& v)
174 {
175 	float average	= vectorFloatAverage(v);
176 	float result	= 0.0f;
177 	for (int i = 0; i < (int)v.size(); i++)
178 	{
179 		float d = (float)v[i] - average;
180 		result += d*d;
181 	}
182 	return deFloatSqrt(result/(float)v.size());
183 }
184 
185 template <typename T>
vectorFloatRelativeStandardDeviation(const vector<T> & v)186 static float vectorFloatRelativeStandardDeviation (const vector<T>& v)
187 {
188 	return hackySafeRelativeResult(vectorFloatStandardDeviation(v), vectorFloatAverage(v));
189 }
190 
191 template <typename T>
vectorFloatMedianAbsoluteDeviation(const vector<T> & v)192 static float vectorFloatMedianAbsoluteDeviation (const vector<T>& v)
193 {
194 	float			median				= vectorFloatMedian(v);
195 	vector<float>	absoluteDeviations	(v.size());
196 
197 	for (int i = 0; i < (int)v.size(); i++)
198 		absoluteDeviations[i] = deFloatAbs((float)v[i] - median);
199 
200 	return vectorFloatMedian(absoluteDeviations);
201 }
202 
203 template <typename T>
vectorFloatRelativeMedianAbsoluteDeviation(const vector<T> & v)204 static float vectorFloatRelativeMedianAbsoluteDeviation (const vector<T>& v)
205 {
206 	return hackySafeRelativeResult(vectorFloatMedianAbsoluteDeviation(v), vectorFloatMedian(v));
207 }
208 
209 template <typename T>
vectorFloatMaximumMinusMinimum(const vector<T> & v)210 static float vectorFloatMaximumMinusMinimum (const vector<T>& v)
211 {
212 	return vectorFloatMaximum(v) - vectorFloatMinimum(v);
213 }
214 
215 template <typename T>
vectorFloatRelativeMaximumMinusMinimum(const vector<T> & v)216 static float vectorFloatRelativeMaximumMinusMinimum (const vector<T>& v)
217 {
218 	return hackySafeRelativeResult(vectorFloatMaximumMinusMinimum(v), vectorFloatMaximum(v));
219 }
220 
221 template <typename T>
vectorLowestPercentage(const vector<T> & v,float factor)222 static vector<T> vectorLowestPercentage (const vector<T>& v, float factor)
223 {
224 	DE_ASSERT(0.0f < factor && factor <= 1.0f);
225 
226 	int			targetSize	= (int)(deFloatCeil(factor*(float)v.size()));
227 	vector<T>	temp		= v;
228 	std::sort(temp.begin(), temp.end());
229 
230 	while ((int)temp.size() > targetSize)
231 		temp.pop_back();
232 
233 	return temp;
234 }
235 
236 template <typename T>
vectorFloatFirstQuartile(const vector<T> & v)237 static float vectorFloatFirstQuartile (const vector<T>& v)
238 {
239 	return vectorFloatMedian(vectorLowestPercentage(v, 0.5f));
240 }
241 
242 // Helper function for combining 4 tcu::Vec4's into one tcu::Vector<float, 16>.
combineVec4ToVec16(const Vec4 & a0,const Vec4 & a1,const Vec4 & a2,const Vec4 & a3)243 static tcu::Vector<float, 16> combineVec4ToVec16 (const Vec4& a0, const Vec4& a1, const Vec4& a2, const Vec4& a3)
244 {
245 	tcu::Vector<float, 16> result;
246 
247 	for (int vecNdx = 0; vecNdx < 4; vecNdx++)
248 	{
249 		const Vec4& srcVec = vecNdx == 0 ? a0 : vecNdx == 1 ? a1 : vecNdx == 2 ? a2 : a3;
250 		for (int i = 0; i < 4; i++)
251 			result[vecNdx*4 + i] = srcVec[i];
252 	}
253 
254 	return result;
255 }
256 
257 // Helper function for extending an n-sized (n <= 16) vector to a 16-sized vector (padded with zeros).
258 template <int Size>
vecTo16(const tcu::Vector<float,Size> & vec)259 static tcu::Vector<float, 16> vecTo16 (const tcu::Vector<float, Size>& vec)
260 {
261 	DE_STATIC_ASSERT(Size <= 16);
262 
263 	tcu::Vector<float, 16> res(0.0f);
264 
265 	for (int i = 0; i < Size; i++)
266 		res[i] = vec[i];
267 
268 	return res;
269 }
270 
271 // Helper function for extending an n-sized (n <= 16) array to a 16-sized vector (padded with zeros).
272 template <int Size>
arrTo16(const tcu::Array<float,Size> & arr)273 static tcu::Vector<float, 16> arrTo16 (const tcu::Array<float, Size>& arr)
274 {
275 	DE_STATIC_ASSERT(Size <= 16);
276 
277 	tcu::Vector<float, 16> res(0.0f);
278 
279 	for(int i = 0; i < Size; i++)
280 		res[i] = arr[i];
281 
282 	return res;
283 }
284 
getShaderInfoLog(const glw::Functions & gl,deUint32 shader)285 static string getShaderInfoLog (const glw::Functions& gl, deUint32 shader)
286 {
287 	string			result;
288 	int				infoLogLen;
289 	vector<char>	infoLogBuf;
290 
291 	gl.getShaderiv(shader, GL_INFO_LOG_LENGTH, &infoLogLen);
292 	infoLogBuf.resize(infoLogLen + 1);
293 	gl.getShaderInfoLog(shader, infoLogLen + 1, DE_NULL, &infoLogBuf[0]);
294 	result = &infoLogBuf[0];
295 
296 	return result;
297 }
298 
getProgramInfoLog(const glw::Functions & gl,deUint32 program)299 static string getProgramInfoLog (const glw::Functions& gl, deUint32 program)
300 {
301 	string			result;
302 	int				infoLogLen;
303 	vector<char>	infoLogBuf;
304 
305 	gl.getProgramiv(program, GL_INFO_LOG_LENGTH, &infoLogLen);
306 	infoLogBuf.resize(infoLogLen + 1);
307 	gl.getProgramInfoLog(program, infoLogLen + 1, DE_NULL, &infoLogBuf[0]);
308 	result = &infoLogBuf[0];
309 
310 	return result;
311 }
312 
313 enum LightType
314 {
315 	LIGHT_DIRECTIONAL = 0,
316 	LIGHT_POINT,
317 
318 	LIGHT_LAST,
319 };
320 
321 enum LoopType
322 {
323 	LOOP_TYPE_STATIC = 0,
324 	LOOP_TYPE_UNIFORM,
325 	LOOP_TYPE_DYNAMIC,
326 
327 	LOOP_LAST
328 };
329 
330 // For texture lookup cases: which texture lookups are inside a conditional statement.
331 enum ConditionalUsage
332 {
333 	CONDITIONAL_USAGE_NONE = 0,		// No conditional statements.
334 	CONDITIONAL_USAGE_FIRST_HALF,	// First numLookUps/2 lookups are inside a conditional statement.
335 	CONDITIONAL_USAGE_EVERY_OTHER,	// First, third etc. lookups are inside conditional statements.
336 
337 	CONDITIONAL_USAGE_LAST
338 };
339 
340 enum ConditionalType
341 {
342 	CONDITIONAL_TYPE_STATIC = 0,
343 	CONDITIONAL_TYPE_UNIFORM,
344 	CONDITIONAL_TYPE_DYNAMIC,
345 
346 	CONDITIONAL_TYPE_LAST
347 };
348 
349 // For the invalid shader compilation tests; what kind of invalidity a shader shall contain.
350 enum ShaderValidity
351 {
352 	SHADER_VALIDITY_VALID = 0,
353 	SHADER_VALIDITY_INVALID_CHAR,
354 	SHADER_VALIDITY_SEMANTIC_ERROR,
355 
356 	SHADER_VALIDITY_LAST
357 };
358 
359 class ShaderCompilerCase : public TestCase
360 {
361 public:
362 	struct AttribSpec
363 	{
364 		string					name;
365 		tcu::Vector<float, 16>	value;
366 
AttribSpecdeqp::gles3::Performance::ShaderCompilerCase::AttribSpec367 		AttribSpec (const string& n, const tcu::Vector<float, 16>& v) : name(n), value(v) {}
368 	};
369 
370 	struct UniformSpec
371 	{
372 		enum Type
373 		{
374 			TYPE_FLOAT = 0,
375 			TYPE_VEC2,
376 			TYPE_VEC3,
377 			TYPE_VEC4,
378 
379 			TYPE_MAT3,
380 			TYPE_MAT4,
381 
382 			TYPE_TEXTURE_UNIT,
383 
384 			TYPE_LAST
385 		};
386 
387 		string					name;
388 		Type					type;
389 		tcu::Vector<float, 16>	value;
390 
UniformSpecdeqp::gles3::Performance::ShaderCompilerCase::UniformSpec391 		UniformSpec (const string& n, Type t, float v)							: name(n), type(t), value(v) {}
UniformSpecdeqp::gles3::Performance::ShaderCompilerCase::UniformSpec392 		UniformSpec (const string& n, Type t, const tcu::Vector<float, 16>& v)	: name(n), type(t), value(v) {}
393 	};
394 
395 								ShaderCompilerCase		(Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments);
396 								~ShaderCompilerCase		(void);
397 
398 	void						init					(void);
399 
400 	IterateResult				iterate					(void);
401 
402 protected:
403 	struct ProgramContext
404 	{
405 		string					vertShaderSource;
406 		string					fragShaderSource;
407 		vector<AttribSpec>		vertexAttributes;
408 		vector<UniformSpec>		uniforms;
409 	};
410 
411 	deUint32					getSpecializationID		(int measurementNdx) const;		// Return an ID that depends on the case ID, current measurement index and time; used to specialize attribute names etc. (avoid shader caching).
412 	virtual ProgramContext		generateShaderData		(int measurementNdx) const = 0;	// Generate shader sources and inputs. Attribute etc. names depend on above name specialization.
413 
414 private:
415 	struct Measurement
416 	{
417 		// \note All times in microseconds. 32-bit integers would probably suffice (would need over an hour of test case runtime to overflow), but better safe than sorry.
418 		deInt64 sourceSetTime;
419 		deInt64 vertexCompileTime;
420 		deInt64 fragmentCompileTime;
421 		deInt64 programLinkTime;
422 		deInt64 firstInputSetTime;
423 		deInt64 firstDrawTime;
424 
425 		deInt64 secondInputSetTime;
426 		deInt64 secondDrawTime;
427 
firstPhasedeqp::gles3::Performance::ShaderCompilerCase::Measurement428 		deInt64 firstPhase				(void) const { return sourceSetTime + vertexCompileTime + fragmentCompileTime + programLinkTime + firstInputSetTime + firstDrawTime; }
secondPhasedeqp::gles3::Performance::ShaderCompilerCase::Measurement429 		deInt64 secondPhase				(void) const { return secondInputSetTime + secondDrawTime; }
430 
totalTimeWithoutDrawdeqp::gles3::Performance::ShaderCompilerCase::Measurement431 		deInt64 totalTimeWithoutDraw	(void) const { return firstPhase() - de::min(secondPhase(), firstInputSetTime + firstDrawTime); }
432 
Measurementdeqp::gles3::Performance::ShaderCompilerCase::Measurement433 		Measurement (deInt64 sourceSetTime_,
434 					 deInt64 vertexCompileTime_,
435 					 deInt64 fragmentCompileTime_,
436 					 deInt64 programLinkTime_,
437 					 deInt64 firstInputSetTime_,
438 					 deInt64 firstDrawTime_,
439 					 deInt64 secondInputSetTime_,
440 					 deInt64 secondDrawTime_)
441 			: sourceSetTime			(sourceSetTime_)
442 			, vertexCompileTime		(vertexCompileTime_)
443 			, fragmentCompileTime	(fragmentCompileTime_)
444 			, programLinkTime		(programLinkTime_)
445 			, firstInputSetTime		(firstInputSetTime_)
446 			, firstDrawTime			(firstDrawTime_)
447 			, secondInputSetTime	(secondInputSetTime_)
448 			, secondDrawTime		(secondDrawTime_)
449 		{
450 		}
451 	};
452 
453 	struct ShadersAndProgram
454 	{
455 		deUint32 vertShader;
456 		deUint32 fragShader;
457 		deUint32 program;
458 	};
459 
460 	struct Logs
461 	{
462 		string vert;
463 		string frag;
464 		string link;
465 	};
466 
467 	struct BuildInfo
468 	{
469 		bool vertCompileSuccess;
470 		bool fragCompileSuccess;
471 		bool linkSuccess;
472 
473 		Logs logs;
474 	};
475 
476 	ShadersAndProgram			createShadersAndProgram		(void) const;
477 	void						setShaderSources			(deUint32 vertShader, deUint32 fragShader, const ProgramContext&) const;
478 	bool						compileShader				(deUint32 shader) const;
479 	bool						linkAndUseProgram			(deUint32 program) const;
480 	void						setShaderInputs				(deUint32 program, const ProgramContext&) const;							// Set attribute pointers and uniforms.
481 	void						draw						(void) const;																// Clear, draw and finish.
482 	void						cleanup						(const ShadersAndProgram&, const ProgramContext&, bool linkSuccess) const;	// Do GL deinitializations.
483 
484 	Logs						getLogs						(const ShadersAndProgram&) const;
485 	void						logProgramData				(const BuildInfo&, const ProgramContext&) const;
486 	bool						goodEnoughMeasurements		(const vector<Measurement>& measurements) const;
487 
488 	int							m_viewportWidth;
489 	int							m_viewportHeight;
490 
491 	bool						m_avoidCache;				// If true, avoid caching between measurements as well (and not only between test cases).
492 	bool						m_addWhitespaceAndComments;	// If true, add random whitespace and comments to the source (good caching should ignore those).
493 	deUint32					m_startHash;				// A hash from case id and time, at the time of construction.
494 
495 	int							m_minimumMeasurementCount;
496 	int							m_maximumMeasurementCount;
497 };
498 
499 class ShaderCompilerLightCase : public ShaderCompilerCase
500 {
501 public:
502 							ShaderCompilerLightCase		(Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, bool isVertexCase, int numLights, LightType lightType);
503 							~ShaderCompilerLightCase	(void);
504 
505 	void					init						(void);
506 	void					deinit						(void);
507 
508 protected:
509 	ProgramContext			generateShaderData			(int measurementNdx) const;
510 
511 private:
512 	int						m_numLights;
513 	bool					m_isVertexCase;
514 	LightType				m_lightType;
515 	glu::Texture2D*			m_texture;
516 };
517 
518 class ShaderCompilerTextureCase : public ShaderCompilerCase
519 {
520 public:
521 									ShaderCompilerTextureCase	(Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, int numLookups, ConditionalUsage conditionalUsage, ConditionalType conditionalType);
522 									~ShaderCompilerTextureCase	(void);
523 
524 	void							init						(void);
525 	void							deinit						(void);
526 
527 protected:
528 	ProgramContext					generateShaderData			(int measurementNdx) const;
529 
530 private:
531 	int								m_numLookups;
532 	vector<glu::Texture2D*>			m_textures;
533 	ConditionalUsage				m_conditionalUsage;
534 	ConditionalType					m_conditionalType;
535 };
536 
537 class ShaderCompilerLoopCase : public ShaderCompilerCase
538 {
539 public:
540 						ShaderCompilerLoopCase	(Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, bool isVertexCase, LoopType type, int numLoopIterations, int nestingDepth);
541 						~ShaderCompilerLoopCase	(void);
542 
543 protected:
544 	ProgramContext		generateShaderData		(int measurementNdx) const;
545 
546 private:
547 	int					m_numLoopIterations;
548 	int					m_nestingDepth;
549 	bool				m_isVertexCase;
550 	LoopType			m_type;
551 };
552 
553 class ShaderCompilerOperCase : public ShaderCompilerCase
554 {
555 public:
556 						ShaderCompilerOperCase	(Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, bool isVertexCase, const char* oper, int numOperations);
557 						~ShaderCompilerOperCase	(void);
558 
559 protected:
560 	ProgramContext		generateShaderData		(int measurementNdx) const;
561 
562 private:
563 	string				m_oper;
564 	int					m_numOperations;
565 	bool				m_isVertexCase;
566 };
567 
568 class ShaderCompilerMandelbrotCase : public ShaderCompilerCase
569 {
570 public:
571 						ShaderCompilerMandelbrotCase	(Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, int numFractalIterations);
572 						~ShaderCompilerMandelbrotCase	(void);
573 
574 protected:
575 	ProgramContext		generateShaderData				(int measurementNdx) const;
576 
577 private:
578 	int					m_numFractalIterations;
579 };
580 
581 class InvalidShaderCompilerCase : public TestCase
582 {
583 public:
584 	// \note Similar to the ShaderValidity enum, but doesn't have a VALID type.
585 	enum InvalidityType
586 	{
587 		INVALIDITY_INVALID_CHAR = 0,
588 		INVALIDITY_SEMANTIC_ERROR,
589 
590 		INVALIDITY_LAST
591 	};
592 
593 						InvalidShaderCompilerCase	(Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType);
594 						~InvalidShaderCompilerCase	(void);
595 
596 	IterateResult		iterate						(void);
597 
598 protected:
599 	struct ProgramContext
600 	{
601 		string vertShaderSource;
602 		string fragShaderSource;
603 	};
604 
605 	deUint32				getSpecializationID		(int measurementNdx) const;			// Return an ID that depends on the case ID, current measurement index and time; used to specialize attribute names etc. (avoid shader caching).
606 	virtual ProgramContext	generateShaderSources	(int measurementNdx) const = 0;		// Generate shader sources. Attribute etc. names depend on above name specialization.
607 
608 	InvalidityType			m_invalidityType;
609 
610 private:
611 	struct Measurement
612 	{
613 		// \note All times in microseconds. 32-bit integers would probably suffice (would need over an hour of test case runtime to overflow), but better safe than sorry.
614 		deInt64 sourceSetTime;
615 		deInt64 vertexCompileTime;
616 		deInt64 fragmentCompileTime;
617 
totalTimedeqp::gles3::Performance::InvalidShaderCompilerCase::Measurement618 		deInt64 totalTime (void) const { return sourceSetTime + vertexCompileTime + fragmentCompileTime; }
619 
Measurementdeqp::gles3::Performance::InvalidShaderCompilerCase::Measurement620 		Measurement (deInt64 sourceSetTime_,
621 					 deInt64 vertexCompileTime_,
622 					 deInt64 fragmentCompileTime_)
623 			: sourceSetTime			(sourceSetTime_)
624 			, vertexCompileTime		(vertexCompileTime_)
625 			, fragmentCompileTime	(fragmentCompileTime_)
626 		{
627 		}
628 	};
629 
630 	struct Shaders
631 	{
632 		deUint32 vertShader;
633 		deUint32 fragShader;
634 	};
635 
636 	struct Logs
637 	{
638 		string vert;
639 		string frag;
640 	};
641 
642 	struct BuildInfo
643 	{
644 		bool vertCompileSuccess;
645 		bool fragCompileSuccess;
646 
647 		Logs logs;
648 	};
649 
650 	Shaders						createShaders			(void) const;
651 	void						setShaderSources		(const Shaders&, const ProgramContext&) const;
652 	bool						compileShader			(deUint32 shader) const;
653 	void						cleanup					(const Shaders&) const;
654 
655 	Logs						getLogs					(const Shaders&) const;
656 	void						logProgramData			(const BuildInfo&, const ProgramContext&) const;
657 	bool						goodEnoughMeasurements	(const vector<Measurement>& measurements) const;
658 
659 	deUint32					m_startHash; // A hash from case id and time, at the time of construction.
660 
661 	int							m_minimumMeasurementCount;
662 	int							m_maximumMeasurementCount;
663 };
664 
665 class InvalidShaderCompilerLightCase : public InvalidShaderCompilerCase
666 {
667 public:
668 							InvalidShaderCompilerLightCase	(Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, bool isVertexCase, int numLights, LightType lightType);
669 							~InvalidShaderCompilerLightCase	(void);
670 
671 protected:
672 	ProgramContext			generateShaderSources			(int measurementNdx) const;
673 
674 private:
675 	bool					m_isVertexCase;
676 	int						m_numLights;
677 	LightType				m_lightType;
678 };
679 
680 class InvalidShaderCompilerTextureCase : public InvalidShaderCompilerCase
681 {
682 public:
683 							InvalidShaderCompilerTextureCase	(Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, int numLookups, ConditionalUsage conditionalUsage, ConditionalType conditionalType);
684 							~InvalidShaderCompilerTextureCase	(void);
685 
686 protected:
687 	ProgramContext			generateShaderSources				(int measurementNdx) const;
688 
689 private:
690 	int						m_numLookups;
691 	ConditionalUsage		m_conditionalUsage;
692 	ConditionalType			m_conditionalType;
693 };
694 
695 class InvalidShaderCompilerLoopCase : public InvalidShaderCompilerCase
696 {
697 public:
698 						InvalidShaderCompilerLoopCase	(Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, bool , LoopType type, int numLoopIterations, int nestingDepth);
699 						~InvalidShaderCompilerLoopCase	(void);
700 
701 protected:
702 	ProgramContext		generateShaderSources			(int measurementNdx) const;
703 
704 private:
705 	bool				m_isVertexCase;
706 	int					m_numLoopIterations;
707 	int					m_nestingDepth;
708 	LoopType			m_type;
709 };
710 
711 class InvalidShaderCompilerOperCase : public InvalidShaderCompilerCase
712 {
713 public:
714 						InvalidShaderCompilerOperCase	(Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, bool isVertexCase, const char* oper, int numOperations);
715 						~InvalidShaderCompilerOperCase	(void);
716 
717 protected:
718 	ProgramContext		generateShaderSources			(int measurementNdx) const;
719 
720 private:
721 	bool				m_isVertexCase;
722 	string				m_oper;
723 	int					m_numOperations;
724 };
725 
726 class InvalidShaderCompilerMandelbrotCase : public InvalidShaderCompilerCase
727 {
728 public:
729 						InvalidShaderCompilerMandelbrotCase		(Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, int numFractalIterations);
730 						~InvalidShaderCompilerMandelbrotCase	(void);
731 
732 protected:
733 	ProgramContext		generateShaderSources					(int measurementNdx) const;
734 
735 private:
736 	int					m_numFractalIterations;
737 };
738 
getNameSpecialization(deUint32 id)739 static string getNameSpecialization (deUint32 id)
740 {
741 	return "_" + toStringWithPadding(id, 10);
742 }
743 
744 // Substitute StringTemplate parameters for attribute/uniform/varying name and constant expression specialization as well as possible shader compilation error causes.
specializeShaderSource(const string & shaderSourceTemplate,deUint32 cacheAvoidanceID,ShaderValidity validity)745 static string specializeShaderSource (const string& shaderSourceTemplate, deUint32 cacheAvoidanceID, ShaderValidity validity)
746 {
747 	std::map<string, string> params;
748 	params["NAME_SPEC"]			= getNameSpecialization(cacheAvoidanceID);
749 	params["FLOAT01"]			= de::floatToString((float)cacheAvoidanceID / (float)(std::numeric_limits<deUint32>::max()), 6);
750 	params["SEMANTIC_ERROR"]	= validity != SHADER_VALIDITY_SEMANTIC_ERROR	? "" : "\tfloat invalid = sin(1.0, 2.0);\n";
751 	params["INVALID_CHAR"]		= validity != SHADER_VALIDITY_INVALID_CHAR		? "" : "@\n"; // \note Some implementations crash when the invalid character is the last character in the source, so use newline.
752 
753 	return tcu::StringTemplate(shaderSourceTemplate).specialize(params);
754 }
755 
756 // Function for generating the vertex shader of a (directional or point) light case.
lightVertexTemplate(int numLights,bool isVertexCase,LightType lightType)757 static string lightVertexTemplate (int numLights, bool isVertexCase, LightType lightType)
758 {
759 	string resultTemplate;
760 
761 	resultTemplate +=
762 		"#version 300 es\n"
763 		"in highp vec4 a_position${NAME_SPEC};\n"
764 		"in mediump vec3 a_normal${NAME_SPEC};\n"
765 		"in mediump vec4 a_texCoord0${NAME_SPEC};\n"
766 		"uniform mediump vec3 u_material_ambientColor${NAME_SPEC};\n"
767 		"uniform mediump vec4 u_material_diffuseColor${NAME_SPEC};\n"
768 		"uniform mediump vec3 u_material_emissiveColor${NAME_SPEC};\n"
769 		"uniform mediump vec3 u_material_specularColor${NAME_SPEC};\n"
770 		"uniform mediump float u_material_shininess${NAME_SPEC};\n";
771 
772 	for (int lightNdx = 0; lightNdx < numLights; lightNdx++)
773 	{
774 		string ndxStr = de::toString(lightNdx);
775 
776 		resultTemplate +=
777 			"uniform mediump vec3 u_light" + ndxStr + "_color${NAME_SPEC};\n"
778 			"uniform mediump vec3 u_light" + ndxStr + "_direction${NAME_SPEC};\n";
779 
780 		if (lightType == LIGHT_POINT)
781 			resultTemplate +=
782 				"uniform mediump vec4 u_light" + ndxStr + "_position${NAME_SPEC};\n"
783 				"uniform mediump float u_light" + ndxStr + "_constantAttenuation${NAME_SPEC};\n"
784 				"uniform mediump float u_light" + ndxStr + "_linearAttenuation${NAME_SPEC};\n"
785 				"uniform mediump float u_light" + ndxStr + "_quadraticAttenuation${NAME_SPEC};\n";
786 	}
787 
788 	resultTemplate +=
789 		"uniform highp mat4 u_mvpMatrix${NAME_SPEC};\n"
790 		"uniform highp mat4 u_modelViewMatrix${NAME_SPEC};\n"
791 		"uniform mediump mat3 u_normalMatrix${NAME_SPEC};\n"
792 		"uniform mediump mat4 u_texCoordMatrix0${NAME_SPEC};\n"
793 		"out mediump vec4 v_color${NAME_SPEC};\n"
794 		"out mediump vec2 v_texCoord0${NAME_SPEC};\n";
795 
796 	if (!isVertexCase)
797 	{
798 		resultTemplate += "out mediump vec3 v_eyeNormal${NAME_SPEC};\n";
799 
800 		if (lightType == LIGHT_POINT)
801 			resultTemplate +=
802 				"out mediump vec3 v_directionToLight${NAME_SPEC}[" + de::toString(numLights) + "];\n"
803 				"out mediump float v_distanceToLight${NAME_SPEC}[" + de::toString(numLights) + "];\n";
804 	}
805 
806 	resultTemplate +=
807 		"mediump vec3 direction (mediump vec4 from, mediump vec4 to)\n"
808 		"{\n"
809 		"	return vec3(to.xyz * from.w - from.xyz * to.w);\n"
810 		"}\n"
811 		"\n"
812 		"mediump vec3 computeLighting (\n"
813 		"	mediump vec3 directionToLight,\n"
814 		"	mediump vec3 halfVector,\n"
815 		"	mediump vec3 normal,\n"
816 		"	mediump vec3 lightColor,\n"
817 		"	mediump vec3 diffuseColor,\n"
818 		"	mediump vec3 specularColor,\n"
819 		"	mediump float shininess)\n"
820 		"{\n"
821 		"	mediump float normalDotDirection  = max(dot(normal, directionToLight), 0.0);\n"
822 		"	mediump vec3  color               = normalDotDirection * diffuseColor * lightColor;\n"
823 		"\n"
824 		"	if (normalDotDirection != 0.0)\n"
825 		"		color += pow(max(dot(normal, halfVector), 0.0), shininess) * specularColor * lightColor;\n"
826 		"\n"
827 		"	return color;\n"
828 		"}\n"
829 		"\n";
830 
831 	if (lightType == LIGHT_POINT)
832 		resultTemplate +=
833 			"mediump float computeDistanceAttenuation (mediump float distToLight, mediump float constAtt, mediump float linearAtt, mediump float quadraticAtt)\n"
834 			"{\n"
835 			"	return 1.0 / (constAtt + linearAtt * distToLight + quadraticAtt * distToLight * distToLight);\n"
836 			"}\n"
837 			"\n";
838 
839 	resultTemplate +=
840 		"void main (void)\n"
841 		"{\n"
842 		"	highp vec4 position = a_position${NAME_SPEC};\n"
843 		"	highp vec3 normal = a_normal${NAME_SPEC};\n"
844 		"	gl_Position = u_mvpMatrix${NAME_SPEC} * position * (0.95 + 0.05*${FLOAT01});\n"
845 		"	v_texCoord0${NAME_SPEC} = (u_texCoordMatrix0${NAME_SPEC} * a_texCoord0${NAME_SPEC}).xy;\n"
846 		"	mediump vec4 color = vec4(u_material_emissiveColor${NAME_SPEC}, u_material_diffuseColor${NAME_SPEC}.a);\n"
847 		"\n"
848 		"	highp vec4 eyePosition = u_modelViewMatrix${NAME_SPEC} * position;\n"
849 		"	mediump vec3 eyeNormal = normalize(u_normalMatrix${NAME_SPEC} * normal);\n";
850 
851 	if (!isVertexCase)
852 		resultTemplate += "\tv_eyeNormal${NAME_SPEC} = eyeNormal;\n";
853 
854 	resultTemplate += "\n";
855 
856 	for (int lightNdx = 0; lightNdx < numLights; lightNdx++)
857 	{
858 		string ndxStr = de::toString(lightNdx);
859 
860 		resultTemplate +=
861 			"	/* Light " + ndxStr + " */\n";
862 
863 		if (lightType == LIGHT_POINT)
864 		{
865 			resultTemplate +=
866 				"	mediump float distanceToLight" + ndxStr + " = distance(eyePosition, u_light" + ndxStr + "_position${NAME_SPEC});\n"
867 				"	mediump vec3 directionToLight" + ndxStr + " = normalize(direction(eyePosition, u_light" + ndxStr + "_position${NAME_SPEC}));\n";
868 
869 			if (isVertexCase)
870 				resultTemplate +=
871 					"	mediump vec3 halfVector" + ndxStr + " = normalize(directionToLight" + ndxStr + " + vec3(0.0, 0.0, 1.0));\n"
872 					"	color.rgb += computeLighting(directionToLight" + ndxStr + ", halfVector" + ndxStr + ", eyeNormal, u_light" + ndxStr + "_color${NAME_SPEC}, u_material_diffuseColor${NAME_SPEC}.rgb, "
873 					"u_material_specularColor${NAME_SPEC}, u_material_shininess${NAME_SPEC}) * computeDistanceAttenuation(distanceToLight" + ndxStr + ", u_light" + ndxStr + "_constantAttenuation${NAME_SPEC}, "
874 					"u_light" + ndxStr + "_linearAttenuation${NAME_SPEC}, u_light" + ndxStr + "_quadraticAttenuation${NAME_SPEC});\n";
875 			else
876 				resultTemplate +=
877 					"	v_directionToLight${NAME_SPEC}[" + ndxStr + "] = directionToLight" + ndxStr + ";\n"
878 					"	v_distanceToLight${NAME_SPEC}[" + ndxStr + "]  = distanceToLight" + ndxStr + ";\n";
879 		}
880 		else if (lightType == LIGHT_DIRECTIONAL)
881 		{
882 			if (isVertexCase)
883 				resultTemplate +=
884 					"	mediump vec3 directionToLight" + ndxStr + " = -u_light" + ndxStr + "_direction${NAME_SPEC};\n"
885 					"	mediump vec3 halfVector" + ndxStr + " = normalize(directionToLight" + ndxStr + " + vec3(0.0, 0.0, 1.0));\n"
886 					"	color.rgb += computeLighting(directionToLight" + ndxStr + ", halfVector" + ndxStr + ", eyeNormal, u_light" + ndxStr + "_color${NAME_SPEC}, u_material_diffuseColor${NAME_SPEC}.rgb, u_material_specularColor${NAME_SPEC}, u_material_shininess${NAME_SPEC});\n";
887 		}
888 		else
889 			DE_ASSERT(DE_FALSE);
890 
891 		resultTemplate += "\n";
892 	}
893 
894 	resultTemplate +=
895 		"	v_color${NAME_SPEC} = color;\n"
896 		"${SEMANTIC_ERROR}"
897 		"}\n"
898 		"${INVALID_CHAR}";
899 
900 	return resultTemplate;
901 }
902 
903 // Function for generating the fragment shader of a (directional or point) light case.
lightFragmentTemplate(int numLights,bool isVertexCase,LightType lightType)904 static string lightFragmentTemplate (int numLights, bool isVertexCase, LightType lightType)
905 {
906 	string resultTemplate;
907 
908 	resultTemplate +=
909 		"#version 300 es\n"
910 		"layout(location = 0) out mediump vec4 o_color;\n";
911 
912 	if (!isVertexCase)
913 	{
914 		resultTemplate +=
915 			"uniform mediump vec3 u_material_ambientColor${NAME_SPEC};\n"
916 			"uniform mediump vec4 u_material_diffuseColor${NAME_SPEC};\n"
917 			"uniform mediump vec3 u_material_emissiveColor${NAME_SPEC};\n"
918 			"uniform mediump vec3 u_material_specularColor${NAME_SPEC};\n"
919 			"uniform mediump float u_material_shininess${NAME_SPEC};\n";
920 
921 		for (int lightNdx = 0; lightNdx < numLights; lightNdx++)
922 		{
923 			string ndxStr = de::toString(lightNdx);
924 
925 			resultTemplate +=
926 				"uniform mediump vec3 u_light" + ndxStr + "_color${NAME_SPEC};\n"
927 				"uniform mediump vec3 u_light" + ndxStr + "_direction${NAME_SPEC};\n";
928 
929 			if (lightType == LIGHT_POINT)
930 				resultTemplate +=
931 					"uniform mediump vec4 u_light" + ndxStr + "_position${NAME_SPEC};\n"
932 					"uniform mediump float u_light" + ndxStr + "_constantAttenuation${NAME_SPEC};\n"
933 					"uniform mediump float u_light" + ndxStr + "_linearAttenuation${NAME_SPEC};\n"
934 					"uniform mediump float u_light" + ndxStr + "_quadraticAttenuation${NAME_SPEC};\n";
935 		}
936 	}
937 
938 	resultTemplate +=
939 		"uniform sampler2D u_sampler0${NAME_SPEC};\n"
940 		"in mediump vec4 v_color${NAME_SPEC};\n"
941 		"in mediump vec2 v_texCoord0${NAME_SPEC};\n";
942 
943 	if (!isVertexCase)
944 	{
945 		resultTemplate +=
946 			"in mediump vec3 v_eyeNormal${NAME_SPEC};\n";
947 
948 		if (lightType == LIGHT_POINT)
949 			resultTemplate +=
950 				"in mediump vec3 v_directionToLight${NAME_SPEC}[" + de::toString(numLights) + "];\n"
951 				"in mediump float v_distanceToLight${NAME_SPEC}[" + de::toString(numLights) + "];\n";
952 
953 		resultTemplate +=
954 			"mediump vec3 direction (mediump vec4 from, mediump vec4 to)\n"
955 			"{\n"
956 			"	return vec3(to.xyz * from.w - from.xyz * to.w);\n"
957 			"}\n"
958 			"\n";
959 
960 		resultTemplate +=
961 			"mediump vec3 computeLighting (\n"
962 			"	mediump vec3 directionToLight,\n"
963 			"	mediump vec3 halfVector,\n"
964 			"	mediump vec3 normal,\n"
965 			"	mediump vec3 lightColor,\n"
966 			"	mediump vec3 diffuseColor,\n"
967 			"	mediump vec3 specularColor,\n"
968 			"	mediump float shininess)\n"
969 			"{\n"
970 			"	mediump float normalDotDirection  = max(dot(normal, directionToLight), 0.0);\n"
971 			"	mediump vec3  color               = normalDotDirection * diffuseColor * lightColor;\n"
972 			"\n"
973 			"	if (normalDotDirection != 0.0)\n"
974 			"		color += pow(max(dot(normal, halfVector), 0.0), shininess) * specularColor * lightColor;\n"
975 			"\n"
976 			"	return color;\n"
977 			"}\n"
978 			"\n";
979 
980 		if (lightType == LIGHT_POINT)
981 			resultTemplate +=
982 				"mediump float computeDistanceAttenuation (mediump float distToLight, mediump float constAtt, mediump float linearAtt, mediump float quadraticAtt)\n"
983 				"{\n"
984 				"	return 1.0 / (constAtt + linearAtt * distToLight + quadraticAtt * distToLight * distToLight);\n"
985 				"}\n"
986 				"\n";
987 	}
988 
989 	resultTemplate +=
990 		"void main (void)\n"
991 		"{\n"
992 		"	mediump vec2 texCoord0 = v_texCoord0${NAME_SPEC}.xy;\n"
993 		"	mediump vec4 color = v_color${NAME_SPEC};\n";
994 
995 	if (!isVertexCase)
996 	{
997 		resultTemplate +=
998 			"	mediump vec3 eyeNormal = normalize(v_eyeNormal${NAME_SPEC});\n"
999 			"\n";
1000 
1001 		for (int lightNdx = 0; lightNdx < numLights; lightNdx++)
1002 		{
1003 			string ndxStr = de::toString(lightNdx);
1004 
1005 			resultTemplate +=
1006 				"	/* Light " + ndxStr + " */\n";
1007 
1008 			if (lightType == LIGHT_POINT)
1009 				resultTemplate +=
1010 					"	mediump vec3 directionToLight" + ndxStr + " = normalize(v_directionToLight${NAME_SPEC}[" + ndxStr + "]);\n"
1011 					"	mediump float distanceToLight" + ndxStr + " = v_distanceToLight${NAME_SPEC}[" + ndxStr + "];\n"
1012 					"	mediump vec3 halfVector" + ndxStr + " = normalize(directionToLight" + ndxStr + " + vec3(0.0, 0.0, 1.0));\n"
1013 					"	color.rgb += computeLighting(directionToLight" + ndxStr + ", halfVector" + ndxStr + ", eyeNormal, u_light" + ndxStr + "_color${NAME_SPEC}, u_material_diffuseColor${NAME_SPEC}.rgb, "
1014 					"u_material_specularColor${NAME_SPEC}, u_material_shininess${NAME_SPEC}) * computeDistanceAttenuation(distanceToLight" + ndxStr + ", u_light" + ndxStr + "_constantAttenuation${NAME_SPEC}, "
1015 					"u_light" + ndxStr + "_linearAttenuation${NAME_SPEC}, u_light" + ndxStr + "_quadraticAttenuation${NAME_SPEC});\n"
1016 					"\n";
1017 			else if (lightType == LIGHT_DIRECTIONAL)
1018 				resultTemplate +=
1019 					"	mediump vec3 directionToLight" + ndxStr + " = -u_light" + ndxStr + "_direction${NAME_SPEC};\n"
1020 					"	mediump vec3 halfVector" + ndxStr + " = normalize(directionToLight" + ndxStr + " + vec3(0.0, 0.0, 1.0));\n"
1021 					"	color.rgb += computeLighting(directionToLight" + ndxStr + ", halfVector" + ndxStr + ", eyeNormal, u_light" + ndxStr + "_color${NAME_SPEC}, u_material_diffuseColor${NAME_SPEC}.rgb, u_material_specularColor${NAME_SPEC}, u_material_shininess${NAME_SPEC});\n"
1022 					"\n";
1023 			else
1024 				DE_ASSERT(DE_FALSE);
1025 		}
1026 	}
1027 
1028 	resultTemplate +=
1029 		"	color *= texture(u_sampler0${NAME_SPEC}, texCoord0);\n"
1030 		"	o_color = color + ${FLOAT01};\n"
1031 		"${SEMANTIC_ERROR}"
1032 		"}\n"
1033 		"${INVALID_CHAR}";
1034 
1035 	return resultTemplate;
1036 }
1037 
1038 // Function for generating the shader attributes of a (directional or point) light case.
lightShaderAttributes(const string & nameSpecialization)1039 static vector<ShaderCompilerCase::AttribSpec> lightShaderAttributes (const string& nameSpecialization)
1040 {
1041 	vector<ShaderCompilerCase::AttribSpec> result;
1042 
1043 	result.push_back(ShaderCompilerCase::AttribSpec("a_position" + nameSpecialization,
1044 													combineVec4ToVec16(Vec4(-1.0f, -1.0f,  0.0f,  1.0f),
1045 																	   Vec4(-1.0f,  1.0f,  0.0f,  1.0f),
1046 																	   Vec4( 1.0f, -1.0f,  0.0f,  1.0f),
1047 																	   Vec4( 1.0f,  1.0f,  0.0f,  1.0f))));
1048 
1049 	result.push_back(ShaderCompilerCase::AttribSpec("a_normal" + nameSpecialization,
1050 													combineVec4ToVec16(Vec4(0.0f, 0.0f, -1.0f, 0.0f),
1051 																	   Vec4(0.0f, 0.0f, -1.0f, 0.0f),
1052 																	   Vec4(0.0f, 0.0f, -1.0f, 0.0f),
1053 																	   Vec4(0.0f, 0.0f, -1.0f, 0.0f))));
1054 
1055 	result.push_back(ShaderCompilerCase::AttribSpec("a_texCoord0" + nameSpecialization,
1056 													combineVec4ToVec16(Vec4(0.0f, 0.0f, 0.0f, 0.0f),
1057 																	   Vec4(1.0f, 0.0f, 0.0f, 0.0f),
1058 																	   Vec4(0.0f, 1.0f, 0.0f, 0.0f),
1059 																	   Vec4(1.0f, 1.0f, 0.0f, 0.0f))));
1060 
1061 	return result;
1062 }
1063 
1064 // Function for generating the shader uniforms of a (directional or point) light case.
lightShaderUniforms(const string & nameSpecialization,int numLights,LightType lightType)1065 static vector<ShaderCompilerCase::UniformSpec> lightShaderUniforms (const string& nameSpecialization, int numLights, LightType lightType)
1066 {
1067 	vector<ShaderCompilerCase::UniformSpec> result;
1068 
1069 	result.push_back(ShaderCompilerCase::UniformSpec("u_material_ambientColor" + nameSpecialization,
1070 													 ShaderCompilerCase::UniformSpec::TYPE_VEC3,
1071 													 vecTo16(Vec3(0.5f, 0.7f, 0.9f))));
1072 
1073 	result.push_back(ShaderCompilerCase::UniformSpec("u_material_diffuseColor" + nameSpecialization,
1074 													 ShaderCompilerCase:: UniformSpec::TYPE_VEC4,
1075 													 vecTo16(Vec4(0.3f, 0.4f, 0.5f, 1.0f))));
1076 
1077 	result.push_back(ShaderCompilerCase::UniformSpec("u_material_emissiveColor" + nameSpecialization,
1078 													 ShaderCompilerCase::UniformSpec::TYPE_VEC3,
1079 													 vecTo16(Vec3(0.7f, 0.2f, 0.2f))));
1080 
1081 	result.push_back(ShaderCompilerCase::UniformSpec("u_material_specularColor" + nameSpecialization,
1082 													 ShaderCompilerCase::UniformSpec::TYPE_VEC3,
1083 													 vecTo16(Vec3(0.2f, 0.6f, 1.0f))));
1084 
1085 	result.push_back(ShaderCompilerCase::UniformSpec("u_material_shininess" + nameSpecialization,
1086 													 ShaderCompilerCase::UniformSpec::TYPE_FLOAT,
1087 													 0.8f));
1088 
1089 	for (int lightNdx = 0; lightNdx < numLights; lightNdx++)
1090 	{
1091 		string ndxStr = de::toString(lightNdx);
1092 
1093 		result.push_back(ShaderCompilerCase::UniformSpec("u_light" + ndxStr + "_color" + nameSpecialization,
1094 														 ShaderCompilerCase::UniformSpec::TYPE_VEC3,
1095 														 vecTo16(Vec3(0.8f, 0.6f, 0.3f))));
1096 
1097 		result.push_back(ShaderCompilerCase::UniformSpec("u_light" + ndxStr + "_direction" + nameSpecialization,
1098 														 ShaderCompilerCase::UniformSpec::TYPE_VEC3,
1099 														 vecTo16(Vec3(0.2f, 0.3f, 0.4f))));
1100 
1101 		if (lightType == LIGHT_POINT)
1102 		{
1103 			result.push_back(ShaderCompilerCase::UniformSpec("u_light" + ndxStr + "_position" + nameSpecialization,
1104 															 ShaderCompilerCase::UniformSpec::TYPE_VEC4,
1105 															 vecTo16(Vec4(1.0f, 0.6f, 0.3f, 0.2f))));
1106 
1107 			result.push_back(ShaderCompilerCase::UniformSpec("u_light" + ndxStr + "_constantAttenuation" + nameSpecialization,
1108 															 ShaderCompilerCase::UniformSpec::TYPE_FLOAT,
1109 															 0.6f));
1110 
1111 			result.push_back(ShaderCompilerCase::UniformSpec("u_light" + ndxStr + "_linearAttenuation" + nameSpecialization,
1112 															 ShaderCompilerCase::UniformSpec::TYPE_FLOAT,
1113 															 0.5f));
1114 
1115 			result.push_back(ShaderCompilerCase::UniformSpec("u_light" + ndxStr + "_quadraticAttenuation" + nameSpecialization,
1116 															 ShaderCompilerCase::UniformSpec::TYPE_FLOAT,
1117 															 0.4f));
1118 		}
1119 	}
1120 
1121 	result.push_back(ShaderCompilerCase::UniformSpec("u_mvpMatrix" + nameSpecialization,
1122 													 ShaderCompilerCase::UniformSpec::TYPE_MAT4,
1123 													 arrTo16(Mat4(1.0f).getColumnMajorData())));
1124 
1125 	result.push_back(ShaderCompilerCase::UniformSpec("u_modelViewMatrix" + nameSpecialization,
1126 													 ShaderCompilerCase::UniformSpec::TYPE_MAT4,
1127 													 arrTo16(Mat4(1.0f).getColumnMajorData())));
1128 
1129 	result.push_back(ShaderCompilerCase::UniformSpec("u_normalMatrix" + nameSpecialization,
1130 													 ShaderCompilerCase::UniformSpec::TYPE_MAT3,
1131 													 arrTo16(Mat3(1.0f).getColumnMajorData())));
1132 
1133 	result.push_back(ShaderCompilerCase::UniformSpec("u_texCoordMatrix0" + nameSpecialization,
1134 													 ShaderCompilerCase::UniformSpec::TYPE_MAT4,
1135 													 arrTo16(Mat4(1.0f).getColumnMajorData())));
1136 
1137 	result.push_back(ShaderCompilerCase::UniformSpec("u_sampler0" + nameSpecialization,
1138 													 ShaderCompilerCase::UniformSpec::TYPE_TEXTURE_UNIT,
1139 													 0.0f));
1140 
1141 	return result;
1142 }
1143 
1144 // Function for generating a vertex shader with a for loop.
loopVertexTemplate(LoopType type,bool isVertexCase,int numLoopIterations,int nestingDepth)1145 static string loopVertexTemplate (LoopType type, bool isVertexCase, int numLoopIterations, int nestingDepth)
1146 {
1147 	string resultTemplate;
1148 	string loopBound		= type == LOOP_TYPE_STATIC	? de::toString(numLoopIterations)
1149 							: type == LOOP_TYPE_UNIFORM	? "int(u_loopBound${NAME_SPEC})"
1150 							: type == LOOP_TYPE_DYNAMIC	? "int(a_loopBound${NAME_SPEC})"
1151 							: "";
1152 
1153 	DE_ASSERT(!loopBound.empty());
1154 
1155 	resultTemplate +=
1156 		"#version 300 es\n"
1157 		"in highp vec4 a_position${NAME_SPEC};\n";
1158 
1159 	if (type == LOOP_TYPE_DYNAMIC)
1160 		resultTemplate +=
1161 			"in mediump float a_loopBound${NAME_SPEC};\n";
1162 
1163 	resultTemplate +=
1164 		"in mediump vec4 a_value${NAME_SPEC};\n"
1165 		"out mediump vec4 v_value${NAME_SPEC};\n";
1166 
1167 	if (isVertexCase)
1168 	{
1169 		if (type == LOOP_TYPE_UNIFORM)
1170 			resultTemplate += "uniform mediump float u_loopBound${NAME_SPEC};\n";
1171 
1172 		resultTemplate +=
1173 			"\n"
1174 			"void main()\n"
1175 			"{\n"
1176 			"	gl_Position = a_position${NAME_SPEC} * (0.95 + 0.05*${FLOAT01});\n"
1177 			"	mediump vec4 value = a_value${NAME_SPEC};\n";
1178 
1179 		for (int i = 0; i < nestingDepth; i++)
1180 		{
1181 			string iterName = "i" + de::toString(i);
1182 			resultTemplate += string(i + 1, '\t') + "for (int " + iterName + " = 0; " + iterName + " < " + loopBound + "; " + iterName + "++)\n";
1183 		}
1184 
1185 		resultTemplate += string(nestingDepth + 1, '\t') + "value *= a_value${NAME_SPEC};\n";
1186 
1187 		resultTemplate +=
1188 			"	v_value${NAME_SPEC} = value;\n";
1189 	}
1190 	else
1191 	{
1192 		if (type == LOOP_TYPE_DYNAMIC)
1193 			resultTemplate +=
1194 				"out mediump float v_loopBound${NAME_SPEC};\n";
1195 
1196 		resultTemplate +=
1197 			"\n"
1198 			"void main()\n"
1199 			"{\n"
1200 			"	gl_Position = a_position${NAME_SPEC} * (0.95 + 0.05*${FLOAT01});\n"
1201 			"	v_value${NAME_SPEC} = a_value${NAME_SPEC};\n";
1202 
1203 		if (type == LOOP_TYPE_DYNAMIC)
1204 			resultTemplate +=
1205 				"	v_loopBound${NAME_SPEC} = a_loopBound${NAME_SPEC};\n";
1206 	}
1207 
1208 	resultTemplate +=
1209 		"${SEMANTIC_ERROR}"
1210 		"}\n"
1211 		"${INVALID_CHAR}";
1212 
1213 	return resultTemplate;
1214 }
1215 
1216 // Function for generating a fragment shader with a for loop.
loopFragmentTemplate(LoopType type,bool isVertexCase,int numLoopIterations,int nestingDepth)1217 static string loopFragmentTemplate (LoopType type, bool isVertexCase, int numLoopIterations, int nestingDepth)
1218 {
1219 	string resultTemplate;
1220 	string loopBound		= type == LOOP_TYPE_STATIC	? de::toString(numLoopIterations)
1221 							: type == LOOP_TYPE_UNIFORM	? "int(u_loopBound${NAME_SPEC})"
1222 							: type == LOOP_TYPE_DYNAMIC	? "int(v_loopBound${NAME_SPEC})"
1223 							: "";
1224 
1225 	DE_ASSERT(!loopBound.empty());
1226 
1227 	resultTemplate +=
1228 		"#version 300 es\n"
1229 		"layout(location = 0) out mediump vec4 o_color;\n"
1230 		"in mediump vec4 v_value${NAME_SPEC};\n";
1231 
1232 	if (!isVertexCase)
1233 	{
1234 		if (type == LOOP_TYPE_DYNAMIC)
1235 			resultTemplate +=
1236 				"in mediump float v_loopBound${NAME_SPEC};\n";
1237 		else if (type == LOOP_TYPE_UNIFORM)
1238 			resultTemplate +=
1239 				"uniform mediump float u_loopBound${NAME_SPEC};\n";
1240 
1241 		resultTemplate +=
1242 			"\n"
1243 			"void main()\n"
1244 			"{\n"
1245 			"	mediump vec4 value = v_value${NAME_SPEC};\n";
1246 
1247 		for (int i = 0; i < nestingDepth; i++)
1248 		{
1249 			string iterName = "i" + de::toString(i);
1250 			resultTemplate += string(i + 1, '\t') + "for (int " + iterName + " = 0; " + iterName + " < " + loopBound + "; " + iterName + "++)\n";
1251 		}
1252 
1253 		resultTemplate += string(nestingDepth + 1, '\t') + "value *= v_value${NAME_SPEC};\n";
1254 
1255 		resultTemplate +=
1256 			"	o_color = value + ${FLOAT01};\n";
1257 	}
1258 	else
1259 		resultTemplate +=
1260 			"\n"
1261 			"void main()\n"
1262 			"{\n"
1263 			"	o_color = v_value${NAME_SPEC} + ${FLOAT01};\n";
1264 
1265 	resultTemplate +=
1266 		"${SEMANTIC_ERROR}"
1267 		"}\n"
1268 		"${INVALID_CHAR}";
1269 
1270 	return resultTemplate;
1271 }
1272 
1273 // Function for generating the shader attributes for a loop case.
loopShaderAttributes(const string & nameSpecialization,LoopType type,int numLoopIterations)1274 static vector<ShaderCompilerCase::AttribSpec> loopShaderAttributes (const string& nameSpecialization, LoopType type, int numLoopIterations)
1275 {
1276 	vector<ShaderCompilerCase::AttribSpec> result;
1277 
1278 	result.push_back(ShaderCompilerCase::AttribSpec("a_position" + nameSpecialization,
1279 													combineVec4ToVec16(Vec4(-1.0f, -1.0f,  0.0f,  1.0f),
1280 																	   Vec4(-1.0f,  1.0f,  0.0f,  1.0f),
1281 																	   Vec4( 1.0f, -1.0f,  0.0f,  1.0f),
1282 																	   Vec4( 1.0f,  1.0f,  0.0f,  1.0f))));
1283 
1284 	result.push_back(ShaderCompilerCase::AttribSpec("a_value" + nameSpecialization,
1285 													combineVec4ToVec16(Vec4( 1.0f,  1.0f,  1.0f,  1.0f),
1286 																	   Vec4( 1.0f,  1.0f,  1.0f,  1.0f),
1287 																	   Vec4( 1.0f,  1.0f,  1.0f,  1.0f),
1288 																	   Vec4( 1.0f,  1.0f,  1.0f,  1.0f))));
1289 
1290 	if (type == LOOP_TYPE_DYNAMIC)
1291 		result.push_back(ShaderCompilerCase::AttribSpec("a_loopBound" + nameSpecialization,
1292 														combineVec4ToVec16(Vec4((float)numLoopIterations, 0.0f, 0.0f, 0.0f),
1293 																		   Vec4((float)numLoopIterations, 0.0f, 0.0f, 0.0f),
1294 																		   Vec4((float)numLoopIterations, 0.0f, 0.0f, 0.0f),
1295 																		   Vec4((float)numLoopIterations, 0.0f, 0.0f, 0.0f))));
1296 
1297 	return result;
1298 }
1299 
loopShaderUniforms(const string & nameSpecialization,LoopType type,int numLoopIterations)1300 static vector<ShaderCompilerCase::UniformSpec> loopShaderUniforms (const string& nameSpecialization, LoopType type, int numLoopIterations)
1301 {
1302 	vector<ShaderCompilerCase::UniformSpec> result;
1303 
1304 	if (type == LOOP_TYPE_UNIFORM)
1305 		result.push_back(ShaderCompilerCase::UniformSpec("u_loopBound" + nameSpecialization,
1306 														 ShaderCompilerCase::UniformSpec::TYPE_FLOAT,
1307 														 (float)numLoopIterations));
1308 
1309 	return result;
1310 }
1311 
1312 // Function for generating the shader attributes for a case with only one attribute value in addition to the position attribute.
singleValueShaderAttributes(const string & nameSpecialization)1313 static vector<ShaderCompilerCase::AttribSpec> singleValueShaderAttributes (const string& nameSpecialization)
1314 {
1315 	vector<ShaderCompilerCase::AttribSpec> result;
1316 
1317 	result.push_back(ShaderCompilerCase::AttribSpec("a_position" + nameSpecialization,
1318 													combineVec4ToVec16(Vec4(-1.0f, -1.0f,  0.0f,  1.0f),
1319 																	   Vec4(-1.0f,  1.0f,  0.0f,  1.0f),
1320 																	   Vec4( 1.0f, -1.0f,  0.0f,  1.0f),
1321 																	   Vec4( 1.0f,  1.0f,  0.0f,  1.0f))));
1322 
1323 	result.push_back(ShaderCompilerCase::AttribSpec("a_value" + nameSpecialization,
1324 													combineVec4ToVec16(Vec4( 1.0f,  1.0f,  1.0f,  1.0f),
1325 																	   Vec4( 1.0f,  1.0f,  1.0f,  1.0f),
1326 																	   Vec4( 1.0f,  1.0f,  1.0f,  1.0f),
1327 																	   Vec4( 1.0f,  1.0f,  1.0f,  1.0f))));
1328 
1329 	return result;
1330 }
1331 
1332 // Function for generating a vertex shader with a binary operation chain.
binaryOpVertexTemplate(int numOperations,const char * op)1333 static string binaryOpVertexTemplate (int numOperations, const char* op)
1334 {
1335 	string resultTemplate;
1336 
1337 	resultTemplate +=
1338 		"#version 300 es\n"
1339 		"in highp vec4 a_position${NAME_SPEC};\n"
1340 		"in mediump vec4 a_value${NAME_SPEC};\n"
1341 		"out mediump vec4 v_value${NAME_SPEC};\n"
1342 		"\n"
1343 		"void main()\n"
1344 		"{\n"
1345 		"	gl_Position = a_position${NAME_SPEC} * (0.95 + 0.05*${FLOAT01});\n"
1346 		"	mediump vec4 value = ";
1347 
1348 	for (int i = 0; i < numOperations; i++)
1349 		resultTemplate += string(i > 0 ? op : "") + "a_value${NAME_SPEC}";
1350 
1351 	resultTemplate +=
1352 		";\n"
1353 		"	v_value${NAME_SPEC} = value;\n"
1354 		"${SEMANTIC_ERROR}"
1355 		"}\n"
1356 		"${INVALID_CHAR}";
1357 
1358 	return resultTemplate;
1359 }
1360 
1361 // Function for generating a fragment shader with a binary operation chain.
binaryOpFragmentTemplate(int numOperations,const char * op)1362 static string binaryOpFragmentTemplate (int numOperations, const char* op)
1363 {
1364 	string resultTemplate;
1365 
1366 	resultTemplate +=
1367 		"#version 300 es\n"
1368 		"layout(location = 0) out mediump vec4 o_color;\n"
1369 		"in mediump vec4 v_value${NAME_SPEC};\n"
1370 		"\n"
1371 		"void main()\n"
1372 		"{\n"
1373 		"	mediump vec4 value = ";
1374 
1375 	for (int i = 0; i < numOperations; i++)
1376 		resultTemplate += string(i > 0 ? op : "") + "v_value${NAME_SPEC}";
1377 
1378 	resultTemplate +=
1379 		";\n"
1380 		"	o_color = value + ${FLOAT01};\n"
1381 		"${SEMANTIC_ERROR}"
1382 		"}\n"
1383 		"${INVALID_CHAR}";
1384 
1385 	return resultTemplate;
1386 }
1387 
1388 // Function for generating a vertex that takes one attribute in addition to position and just passes it to the fragment shader as a varying.
singleVaryingVertexTemplate(void)1389 static string singleVaryingVertexTemplate (void)
1390 {
1391 	const char* resultTemplate =
1392 		"#version 300 es\n"
1393 		"in highp vec4 a_position${NAME_SPEC};\n"
1394 		"in mediump vec4 a_value${NAME_SPEC};\n"
1395 		"out mediump vec4 v_value${NAME_SPEC};\n"
1396 		"\n"
1397 		"void main()\n"
1398 		"{\n"
1399 		"	gl_Position = a_position${NAME_SPEC} * (0.95 + 0.05*${FLOAT01});\n"
1400 		"	v_value${NAME_SPEC} = a_value${NAME_SPEC};\n"
1401 		"${SEMANTIC_ERROR}"
1402 		"}\n"
1403 		"${INVALID_CHAR}";
1404 
1405 	return resultTemplate;
1406 }
1407 
1408 // Function for generating a fragment shader that takes a single varying and uses it as the color.
singleVaryingFragmentTemplate(void)1409 static string singleVaryingFragmentTemplate (void)
1410 {
1411 	const char* resultTemplate =
1412 		"#version 300 es\n"
1413 		"layout(location = 0) out mediump vec4 o_color;\n"
1414 		"in mediump vec4 v_value${NAME_SPEC};\n"
1415 		"\n"
1416 		"void main()\n"
1417 		"{\n"
1418 		"	o_color = v_value${NAME_SPEC} + ${FLOAT01};\n"
1419 		"${SEMANTIC_ERROR}"
1420 		"}\n"
1421 		"${INVALID_CHAR}";
1422 
1423 	return resultTemplate;
1424 }
1425 
1426 // Function for generating the vertex shader of a texture lookup case.
textureLookupVertexTemplate(ConditionalUsage conditionalUsage,ConditionalType conditionalType)1427 static string textureLookupVertexTemplate (ConditionalUsage conditionalUsage, ConditionalType conditionalType)
1428 {
1429 	string	resultTemplate;
1430 	bool	conditionVaryingNeeded = conditionalUsage != CONDITIONAL_USAGE_NONE && conditionalType == CONDITIONAL_TYPE_DYNAMIC;
1431 
1432 	resultTemplate +=
1433 		"#version 300 es\n"
1434 		"in highp vec4 a_position${NAME_SPEC};\n"
1435 		"in mediump vec2 a_coords${NAME_SPEC};\n"
1436 		"out mediump vec2 v_coords${NAME_SPEC};\n";
1437 
1438 	if (conditionVaryingNeeded)
1439 		resultTemplate +=
1440 			"in mediump float a_condition${NAME_SPEC};\n"
1441 			"out mediump float v_condition${NAME_SPEC};\n";
1442 
1443 	resultTemplate +=
1444 		"\n"
1445 		"void main()\n"
1446 		"{\n"
1447 		"	gl_Position = a_position${NAME_SPEC} * (0.95 + 0.05*${FLOAT01});\n"
1448 		"	v_coords${NAME_SPEC} = a_coords${NAME_SPEC};\n";
1449 
1450 	if (conditionVaryingNeeded)
1451 		resultTemplate +=
1452 			"	v_condition${NAME_SPEC} = a_condition${NAME_SPEC};\n";
1453 
1454 	resultTemplate +=
1455 		"${SEMANTIC_ERROR}"
1456 		"}\n"
1457 		"${INVALID_CHAR}";
1458 
1459 	return resultTemplate;
1460 }
1461 
1462 // Function for generating the fragment shader of a texture lookup case.
textureLookupFragmentTemplate(int numLookups,ConditionalUsage conditionalUsage,ConditionalType conditionalType)1463 static string textureLookupFragmentTemplate (int numLookups, ConditionalUsage conditionalUsage, ConditionalType conditionalType)
1464 {
1465 	string resultTemplate;
1466 
1467 	resultTemplate +=
1468 		"#version 300 es\n"
1469 		"layout(location = 0) out mediump vec4 o_color;\n"
1470 		"in mediump vec2 v_coords${NAME_SPEC};\n";
1471 
1472 	if (conditionalUsage != CONDITIONAL_USAGE_NONE && conditionalType == CONDITIONAL_TYPE_DYNAMIC)
1473 		resultTemplate +=
1474 			"in mediump float v_condition${NAME_SPEC};\n";
1475 
1476 	for (int i = 0; i < numLookups; i++)
1477 		resultTemplate +=
1478 			"uniform sampler2D u_sampler" + de::toString(i) + "${NAME_SPEC};\n";
1479 
1480 	if (conditionalUsage != CONDITIONAL_USAGE_NONE && conditionalType == CONDITIONAL_TYPE_UNIFORM)
1481 		resultTemplate +=
1482 			"uniform mediump float u_condition${NAME_SPEC};\n";
1483 
1484 	resultTemplate +=
1485 		"\n"
1486 		"void main()\n"
1487 		"{\n"
1488 		"	mediump vec4 color = vec4(0.0);\n";
1489 
1490 	const char* conditionalTerm = conditionalType == CONDITIONAL_TYPE_STATIC	? "1.0 > 0.0"
1491 								: conditionalType == CONDITIONAL_TYPE_UNIFORM	? "u_condition${NAME_SPEC} > 0.0"
1492 								: conditionalType == CONDITIONAL_TYPE_DYNAMIC	? "v_condition${NAME_SPEC} > 0.0"
1493 								: DE_NULL;
1494 
1495 	DE_ASSERT(conditionalTerm != DE_NULL);
1496 
1497 	if (conditionalUsage == CONDITIONAL_USAGE_FIRST_HALF)
1498 		resultTemplate += string("") +
1499 			"	if (" + conditionalTerm + ")\n"
1500 			"	{\n";
1501 
1502 	for (int i = 0; i < numLookups; i++)
1503 	{
1504 		if (conditionalUsage == CONDITIONAL_USAGE_FIRST_HALF)
1505 		{
1506 			if (i < (numLookups + 1) / 2)
1507 				resultTemplate += "\t";
1508 		}
1509 		else if (conditionalUsage == CONDITIONAL_USAGE_EVERY_OTHER)
1510 		{
1511 			if (i % 2 == 0)
1512 				resultTemplate += string("") +
1513 					"	if (" + conditionalTerm + ")\n"
1514 					"\t";
1515 		}
1516 
1517 		resultTemplate +=
1518 			"	color += texture(u_sampler" + de::toString(i) + "${NAME_SPEC}, v_coords${NAME_SPEC});\n";
1519 
1520 		if (conditionalUsage == CONDITIONAL_USAGE_FIRST_HALF && i == (numLookups - 1) / 2)
1521 			resultTemplate += "\t}\n";
1522 	}
1523 
1524 	resultTemplate +=
1525 		"	o_color = color/" + de::toString(numLookups) + ".0 + ${FLOAT01};\n" +
1526 		"${SEMANTIC_ERROR}"
1527 		"}\n"
1528 		"${INVALID_CHAR}";
1529 
1530 	return resultTemplate;
1531 }
1532 
1533 // Function for generating the shader attributes of a texture lookup case.
textureLookupShaderAttributes(const string & nameSpecialization,ConditionalUsage conditionalUsage,ConditionalType conditionalType)1534 static vector<ShaderCompilerCase::AttribSpec> textureLookupShaderAttributes (const string& nameSpecialization, ConditionalUsage conditionalUsage, ConditionalType conditionalType)
1535 {
1536 	vector<ShaderCompilerCase::AttribSpec> result;
1537 
1538 	result.push_back(ShaderCompilerCase::AttribSpec("a_position" + nameSpecialization,
1539 													combineVec4ToVec16(Vec4(-1.0f, -1.0f,  0.0f,  1.0f),
1540 																	   Vec4(-1.0f,  1.0f,  0.0f,  1.0f),
1541 																	   Vec4( 1.0f, -1.0f,  0.0f,  1.0f),
1542 																	   Vec4( 1.0f,  1.0f,  0.0f,  1.0f))));
1543 
1544 	result.push_back(ShaderCompilerCase::AttribSpec("a_coords" + nameSpecialization,
1545 													combineVec4ToVec16(Vec4(0.0f, 0.0f, 0.0f, 0.0f),
1546 																	   Vec4(0.0f, 1.0f, 0.0f, 0.0f),
1547 																	   Vec4(1.0f, 0.0f, 0.0f, 0.0f),
1548 																	   Vec4(1.0f, 1.0f, 0.0f, 0.0f))));
1549 
1550 	if (conditionalUsage != CONDITIONAL_USAGE_NONE && conditionalType == CONDITIONAL_TYPE_DYNAMIC)
1551 		result.push_back(ShaderCompilerCase::AttribSpec("a_condition" + nameSpecialization,
1552 														combineVec4ToVec16(Vec4(1.0f), Vec4(1.0f), Vec4(1.0f), Vec4(1.0f))));
1553 
1554 	return result;
1555 }
1556 
1557 // Function for generating the shader uniforms of a texture lookup case.
textureLookupShaderUniforms(const string & nameSpecialization,int numLookups,ConditionalUsage conditionalUsage,ConditionalType conditionalType)1558 static vector<ShaderCompilerCase::UniformSpec> textureLookupShaderUniforms (const string& nameSpecialization, int numLookups, ConditionalUsage conditionalUsage, ConditionalType conditionalType)
1559 {
1560 	vector<ShaderCompilerCase::UniformSpec> result;
1561 
1562 	for (int i = 0; i < numLookups; i++)
1563 		result.push_back(ShaderCompilerCase::UniformSpec("u_sampler" + de::toString(i) + nameSpecialization,
1564 														 ShaderCompilerCase::UniformSpec::TYPE_TEXTURE_UNIT,
1565 														 (float)i));
1566 
1567 	if (conditionalUsage != CONDITIONAL_USAGE_NONE && conditionalType == CONDITIONAL_TYPE_UNIFORM)
1568 		result.push_back(ShaderCompilerCase::UniformSpec("u_condition" + nameSpecialization,
1569 														 ShaderCompilerCase::UniformSpec::TYPE_FLOAT,
1570 														 1.0f));
1571 
1572 	return result;
1573 }
1574 
mandelbrotVertexTemplate(void)1575 static string mandelbrotVertexTemplate (void)
1576 {
1577 	const char* resultTemplate =
1578 		"#version 300 es\n"
1579 		"uniform highp mat4 u_mvp${NAME_SPEC};\n"
1580 		"\n"
1581 		"in highp vec4 a_vertex${NAME_SPEC};\n"
1582 		"in highp vec4 a_coord${NAME_SPEC};\n"
1583 		"\n"
1584 		"out mediump vec2 v_coord${NAME_SPEC};\n"
1585 		"\n"
1586 		"void main(void)\n"
1587 		"{\n"
1588 		"	gl_Position = u_mvp${NAME_SPEC} * a_vertex${NAME_SPEC} * (0.95 + 0.05*${FLOAT01});\n"
1589 		"\n"
1590 		"	float xMin = -2.0;\n"
1591 		"	float xMax = +0.5;\n"
1592 		"	float yMin = -1.5;\n"
1593 		"	float yMax = +1.5;\n"
1594 		"\n"
1595 		"	v_coord${NAME_SPEC}.x = a_coord${NAME_SPEC}.x * (xMax - xMin) + xMin;\n"
1596 		"	v_coord${NAME_SPEC}.y = a_coord${NAME_SPEC}.y * (yMax - yMin) + yMin;\n"
1597 		"${SEMANTIC_ERROR}"
1598 		"}\n"
1599 		"${INVALID_CHAR}";
1600 
1601 	return resultTemplate;
1602 }
1603 
mandelbrotFragmentTemplate(int numFractalIterations)1604 static string mandelbrotFragmentTemplate (int numFractalIterations)
1605 {
1606 	string resultTemplate =
1607 		"#version 300 es\n"
1608 		"layout(location = 0) out mediump vec4 o_color;\n"
1609 		"in mediump vec2 v_coord${NAME_SPEC};\n"
1610 		"\n"
1611 		"precision mediump float;\n"
1612 		"\n"
1613 		"#define NUM_ITERS " + de::toString(numFractalIterations) + "\n"
1614 		"\n"
1615 		"void main (void)\n"
1616 		"{\n"
1617 		"	vec2 coords = v_coord${NAME_SPEC};\n"
1618 		"	float u_limit = 2.0 * 2.0;\n"
1619 		"	vec2 tmp = vec2(0, 0);\n"
1620 		"	int iter;\n"
1621 		"\n"
1622 		"	for (iter = 0; iter < NUM_ITERS; iter++)\n"
1623 		"	{\n"
1624 		"		tmp = vec2((tmp.x + tmp.y) * (tmp.x - tmp.y), 2.0 * (tmp.x * tmp.y)) + coords;\n"
1625 		"\n"
1626 		"		if (dot(tmp, tmp) > u_limit)\n"
1627 		"			break;\n"
1628 		"	}\n"
1629 		"\n"
1630 		"	vec3 color = vec3(float(iter) * (1.0 / float(NUM_ITERS)));\n"
1631 		"\n"
1632 		"	o_color = vec4(color, 1.0) + ${FLOAT01};\n"
1633 		"${SEMANTIC_ERROR}"
1634 		"}\n"
1635 		"${INVALID_CHAR}";
1636 
1637 	return resultTemplate;
1638 }
1639 
mandelbrotShaderAttributes(const string & nameSpecialization)1640 static vector<ShaderCompilerCase::AttribSpec> mandelbrotShaderAttributes (const string& nameSpecialization)
1641 {
1642 	vector<ShaderCompilerCase::AttribSpec> result;
1643 
1644 	result.push_back(ShaderCompilerCase::AttribSpec("a_vertex" + nameSpecialization,
1645 													combineVec4ToVec16(Vec4(-1.0f, -1.0f,  0.0f,  1.0f),
1646 																	   Vec4(-1.0f,  1.0f,  0.0f,  1.0f),
1647 																	   Vec4( 1.0f, -1.0f,  0.0f,  1.0f),
1648 																	   Vec4( 1.0f,  1.0f,  0.0f,  1.0f))));
1649 
1650 	result.push_back(ShaderCompilerCase::AttribSpec("a_coord" + nameSpecialization,
1651 													combineVec4ToVec16(Vec4(0.0f, 0.0f, 0.0f, 1.0f),
1652 																	   Vec4(0.0f, 1.0f, 0.0f, 1.0f),
1653 																	   Vec4(1.0f, 0.0f, 0.0f, 1.0f),
1654 																	   Vec4(1.0f, 1.0f, 0.0f, 1.0f))));
1655 
1656 	return result;
1657 }
1658 
mandelbrotShaderUniforms(const string & nameSpecialization)1659 static vector<ShaderCompilerCase::UniformSpec> mandelbrotShaderUniforms (const string& nameSpecialization)
1660 {
1661 	vector<ShaderCompilerCase::UniformSpec> result;
1662 
1663 	result.push_back(ShaderCompilerCase::UniformSpec("u_mvp" + nameSpecialization,
1664 													 ShaderCompilerCase::UniformSpec::TYPE_MAT4,
1665 													 arrTo16(Mat4(1.0f).getColumnMajorData())));
1666 
1667 	return result;
1668 }
1669 
ShaderCompilerCase(Context & context,const char * name,const char * description,int caseID,bool avoidCache,bool addWhitespaceAndComments)1670 ShaderCompilerCase::ShaderCompilerCase (Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments)
1671 	: TestCase								(context, tcu::NODETYPE_PERFORMANCE, name, description)
1672 	, m_viewportWidth						(0)
1673 	, m_viewportHeight						(0)
1674 	, m_avoidCache							(avoidCache)
1675 	, m_addWhitespaceAndComments			(addWhitespaceAndComments)
1676 	, m_startHash							((deUint32)(deUint64Hash(deGetTime()) ^ deUint64Hash(deGetMicroseconds()) ^ deInt32Hash(caseID)))
1677 {
1678 	int cmdLineIterCount = context.getTestContext().getCommandLine().getTestIterationCount();
1679 	m_minimumMeasurementCount = cmdLineIterCount > 0 ? cmdLineIterCount : DEFAULT_MINIMUM_MEASUREMENT_COUNT;
1680 	m_maximumMeasurementCount = m_minimumMeasurementCount*3;
1681 }
1682 
~ShaderCompilerCase(void)1683 ShaderCompilerCase::~ShaderCompilerCase (void)
1684 {
1685 }
1686 
getSpecializationID(int measurementNdx) const1687 deUint32 ShaderCompilerCase::getSpecializationID (int measurementNdx) const
1688 {
1689 	if (m_avoidCache)
1690 		return m_startHash ^ (deUint32)deInt32Hash((deInt32)measurementNdx);
1691 	else
1692 		return m_startHash;
1693 }
1694 
init(void)1695 void ShaderCompilerCase::init (void)
1696 {
1697 	const glw::Functions&		gl				= m_context.getRenderContext().getFunctions();
1698 	const tcu::RenderTarget&	renderTarget	= m_context.getRenderContext().getRenderTarget();
1699 
1700 	m_viewportWidth		= deMin32(MAX_VIEWPORT_WIDTH, renderTarget.getWidth());
1701 	m_viewportHeight	= deMin32(MAX_VIEWPORT_HEIGHT, renderTarget.getHeight());
1702 
1703 	gl.viewport(0, 0, m_viewportWidth, m_viewportHeight);
1704 }
1705 
createShadersAndProgram(void) const1706 ShaderCompilerCase::ShadersAndProgram ShaderCompilerCase::createShadersAndProgram (void) const
1707 {
1708 	const glw::Functions&	gl = m_context.getRenderContext().getFunctions();
1709 	ShadersAndProgram		result;
1710 
1711 	result.vertShader	= gl.createShader(GL_VERTEX_SHADER);
1712 	result.fragShader	= gl.createShader(GL_FRAGMENT_SHADER);
1713 	result.program		= gl.createProgram();
1714 
1715 	gl.attachShader(result.program, result.vertShader);
1716 	gl.attachShader(result.program, result.fragShader);
1717 
1718 	return result;
1719 }
1720 
setShaderSources(deUint32 vertShader,deUint32 fragShader,const ProgramContext & progCtx) const1721 void ShaderCompilerCase::setShaderSources (deUint32 vertShader, deUint32 fragShader, const ProgramContext& progCtx) const
1722 {
1723 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1724 	const char* vertShaderSourceCStr = progCtx.vertShaderSource.c_str();
1725 	const char* fragShaderSourceCStr = progCtx.fragShaderSource.c_str();
1726 	gl.shaderSource(vertShader, 1, &vertShaderSourceCStr, DE_NULL);
1727 	gl.shaderSource(fragShader, 1, &fragShaderSourceCStr, DE_NULL);
1728 }
1729 
compileShader(deUint32 shader) const1730 bool ShaderCompilerCase::compileShader (deUint32 shader) const
1731 {
1732 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1733 	GLint status;
1734 	gl.compileShader(shader);
1735 	gl.getShaderiv(shader, GL_COMPILE_STATUS, &status);
1736 	return status != 0;
1737 }
1738 
linkAndUseProgram(deUint32 program) const1739 bool ShaderCompilerCase::linkAndUseProgram (deUint32 program) const
1740 {
1741 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1742 	GLint linkStatus;
1743 
1744 	gl.linkProgram(program);
1745 	gl.getProgramiv(program, GL_LINK_STATUS, &linkStatus);
1746 
1747 	if (linkStatus != 0)
1748 		gl.useProgram(program);
1749 
1750 	return linkStatus != 0;
1751 }
1752 
setShaderInputs(deUint32 program,const ProgramContext & progCtx) const1753 void ShaderCompilerCase::setShaderInputs (deUint32 program, const ProgramContext& progCtx) const
1754 {
1755 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1756 
1757 	// Setup attributes.
1758 
1759 	for (int attribNdx = 0; attribNdx < (int)progCtx.vertexAttributes.size(); attribNdx++)
1760 	{
1761 		int location = gl.getAttribLocation(program, progCtx.vertexAttributes[attribNdx].name.c_str());
1762 		if (location >= 0)
1763 		{
1764 			gl.enableVertexAttribArray(location);
1765 			gl.vertexAttribPointer(location, 4, GL_FLOAT, GL_FALSE, 0, progCtx.vertexAttributes[attribNdx].value.getPtr());
1766 		}
1767 	}
1768 
1769 	// Setup uniforms.
1770 
1771 	for (int uniformNdx = 0; uniformNdx < (int)progCtx.uniforms.size(); uniformNdx++)
1772 	{
1773 		int location = gl.getUniformLocation(program, progCtx.uniforms[uniformNdx].name.c_str());
1774 		if (location >= 0)
1775 		{
1776 			const float* floatPtr = progCtx.uniforms[uniformNdx].value.getPtr();
1777 
1778 			switch (progCtx.uniforms[uniformNdx].type)
1779 			{
1780 				case UniformSpec::TYPE_FLOAT:			gl.uniform1fv(location, 1, floatPtr);								break;
1781 				case UniformSpec::TYPE_VEC2:			gl.uniform2fv(location, 1, floatPtr);								break;
1782 				case UniformSpec::TYPE_VEC3:			gl.uniform3fv(location, 1, floatPtr);								break;
1783 				case UniformSpec::TYPE_VEC4:			gl.uniform4fv(location, 1, floatPtr);								break;
1784 				case UniformSpec::TYPE_MAT3:			gl.uniformMatrix3fv(location, 1, GL_FALSE, floatPtr);				break;
1785 				case UniformSpec::TYPE_MAT4:			gl.uniformMatrix4fv(location, 1, GL_FALSE, floatPtr);				break;
1786 				case UniformSpec::TYPE_TEXTURE_UNIT:	gl.uniform1i(location, (GLint)deRoundFloatToInt32(*floatPtr));		break;
1787 				default:
1788 					DE_ASSERT(DE_FALSE);
1789 			}
1790 		}
1791 	}
1792 }
1793 
draw(void) const1794 void ShaderCompilerCase::draw (void) const
1795 {
1796 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1797 
1798 	static const deUint8 indices[] =
1799 	{
1800 		0, 1, 2,
1801 		2, 1, 3
1802 	};
1803 
1804 	gl.clear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
1805 	gl.drawElements(GL_TRIANGLES, DE_LENGTH_OF_ARRAY(indices), GL_UNSIGNED_BYTE, indices);
1806 
1807 	// \note Read one pixel to force compilation.
1808 	deUint32 pixel;
1809 	gl.readPixels(0, 0, 1, 1, GL_RGBA, GL_UNSIGNED_BYTE, &pixel);
1810 }
1811 
cleanup(const ShadersAndProgram & shadersAndProgram,const ProgramContext & progCtx,bool linkSuccess) const1812 void ShaderCompilerCase::cleanup (const ShadersAndProgram& shadersAndProgram, const ProgramContext& progCtx, bool linkSuccess) const
1813 {
1814 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1815 
1816 	if (linkSuccess)
1817 	{
1818 		for (int attribNdx = 0; attribNdx < (int)progCtx.vertexAttributes.size(); attribNdx++)
1819 		{
1820 			int location = gl.getAttribLocation(shadersAndProgram.program, progCtx.vertexAttributes[attribNdx].name.c_str());
1821 			if (location >= 0)
1822 				gl.disableVertexAttribArray(location);
1823 		}
1824 	}
1825 
1826 	gl.useProgram(0);
1827 	gl.detachShader(shadersAndProgram.program, shadersAndProgram.vertShader);
1828 	gl.detachShader(shadersAndProgram.program, shadersAndProgram.fragShader);
1829 	gl.deleteShader(shadersAndProgram.vertShader);
1830 	gl.deleteShader(shadersAndProgram.fragShader);
1831 	gl.deleteProgram(shadersAndProgram.program);
1832 }
1833 
logProgramData(const BuildInfo & buildInfo,const ProgramContext & progCtx) const1834 void ShaderCompilerCase::logProgramData (const BuildInfo& buildInfo, const ProgramContext& progCtx) const
1835 {
1836 	m_testCtx.getLog() << TestLog::ShaderProgram(buildInfo.linkSuccess, buildInfo.logs.link)
1837 					   << TestLog::Shader(QP_SHADER_TYPE_VERTEX,	progCtx.vertShaderSource, buildInfo.vertCompileSuccess, buildInfo.logs.vert)
1838 					   << TestLog::Shader(QP_SHADER_TYPE_FRAGMENT,	progCtx.fragShaderSource, buildInfo.fragCompileSuccess, buildInfo.logs.frag)
1839 					   << TestLog::EndShaderProgram;
1840 }
1841 
getLogs(const ShadersAndProgram & shadersAndProgram) const1842 ShaderCompilerCase::Logs ShaderCompilerCase::getLogs (const ShadersAndProgram& shadersAndProgram) const
1843 {
1844 	const glw::Functions&	gl = m_context.getRenderContext().getFunctions();
1845 	Logs					result;
1846 
1847 	result.vert = getShaderInfoLog(gl, shadersAndProgram.vertShader);
1848 	result.frag = getShaderInfoLog(gl, shadersAndProgram.fragShader);
1849 	result.link = getProgramInfoLog(gl, shadersAndProgram.program);
1850 
1851 	return result;
1852 }
1853 
goodEnoughMeasurements(const vector<Measurement> & measurements) const1854 bool ShaderCompilerCase::goodEnoughMeasurements (const vector<Measurement>& measurements) const
1855 {
1856 	if ((int)measurements.size() < m_minimumMeasurementCount)
1857 		return false;
1858 	else
1859 	{
1860 		if ((int)measurements.size() >= m_maximumMeasurementCount)
1861 			return true;
1862 		else
1863 		{
1864 			vector<deInt64> totalTimesWithoutDraw;
1865 			for (int i = 0; i < (int)measurements.size(); i++)
1866 				totalTimesWithoutDraw.push_back(measurements[i].totalTimeWithoutDraw());
1867 			return vectorFloatRelativeMedianAbsoluteDeviation(vectorLowestPercentage(totalTimesWithoutDraw, 0.5f)) < RELATIVE_MEDIAN_ABSOLUTE_DEVIATION_THRESHOLD;
1868 		}
1869 	}
1870 }
1871 
iterate(void)1872 ShaderCompilerCase::IterateResult ShaderCompilerCase::iterate (void)
1873 {
1874 	// Before actual measurements, compile and draw with a dummy shader to avoid possible initial slowdowns in the actual test.
1875 	{
1876 		deUint32		specID = getSpecializationID(0);
1877 		ProgramContext	progCtx;
1878 		progCtx.vertShaderSource = specializeShaderSource(singleVaryingVertexTemplate(), specID, SHADER_VALIDITY_VALID);
1879 		progCtx.fragShaderSource = specializeShaderSource(singleVaryingFragmentTemplate(), specID, SHADER_VALIDITY_VALID);
1880 		progCtx.vertexAttributes = singleValueShaderAttributes(getNameSpecialization(specID));
1881 
1882 		ShadersAndProgram shadersAndProgram = createShadersAndProgram();
1883 		setShaderSources(shadersAndProgram.vertShader, shadersAndProgram.fragShader, progCtx);
1884 
1885 		BuildInfo buildInfo;
1886 		buildInfo.vertCompileSuccess	= compileShader(shadersAndProgram.vertShader);
1887 		buildInfo.fragCompileSuccess	= compileShader(shadersAndProgram.fragShader);
1888 		buildInfo.linkSuccess			= linkAndUseProgram(shadersAndProgram.program);
1889 		if (!(buildInfo.vertCompileSuccess && buildInfo.fragCompileSuccess && buildInfo.linkSuccess))
1890 		{
1891 			buildInfo.logs = getLogs(shadersAndProgram);
1892 			logProgramData(buildInfo, progCtx);
1893 			cleanup(shadersAndProgram, progCtx, buildInfo.linkSuccess);
1894 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Compilation failed");
1895 			return STOP;
1896 		}
1897 		setShaderInputs(shadersAndProgram.program, progCtx);
1898 		draw();
1899 		cleanup(shadersAndProgram, progCtx, buildInfo.linkSuccess);
1900 	}
1901 
1902 	vector<Measurement>		measurements;
1903 	// \note These are logged after measurements are done.
1904 	ProgramContext			latestProgramContext;
1905 	BuildInfo				latestBuildInfo;
1906 
1907 	if (WARMUP_CPU_AT_BEGINNING_OF_CASE)
1908 		tcu::warmupCPU();
1909 
1910 	// Actual test measurements.
1911 	while (!goodEnoughMeasurements(measurements))
1912 	{
1913 		// Create shaders, compile & link, set shader inputs and draw. Time measurement is done at relevant points.
1914 		// \note Setting inputs and drawing are done twice in order to find out the time for actual compiling.
1915 
1916 		// \note Shader data (sources and inputs) are generated and GL shader and program objects are created before any time measurements.
1917 		ProgramContext		progCtx				= generateShaderData((int)measurements.size());
1918 		ShadersAndProgram	shadersAndProgram	= createShadersAndProgram();
1919 		BuildInfo			buildInfo;
1920 
1921 		if (m_addWhitespaceAndComments)
1922 		{
1923 			const deUint32 hash = m_startHash ^ (deUint32)deInt32Hash((deInt32)measurements.size());
1924 			progCtx.vertShaderSource = strWithWhiteSpaceAndComments(progCtx.vertShaderSource, hash);
1925 			progCtx.fragShaderSource = strWithWhiteSpaceAndComments(progCtx.fragShaderSource, hash);
1926 		}
1927 
1928 		if (WARMUP_CPU_BEFORE_EACH_MEASUREMENT)
1929 			tcu::warmupCPU();
1930 
1931 		// \note Do NOT do anything too hefty between the first and last deGetMicroseconds() here (other than the gl calls); it would disturb the measurement.
1932 
1933 		deUint64 startTime = deGetMicroseconds();
1934 
1935 		setShaderSources(shadersAndProgram.vertShader, shadersAndProgram.fragShader, progCtx);
1936 		deUint64 shaderSourceSetEndTime = deGetMicroseconds();
1937 
1938 		buildInfo.vertCompileSuccess = compileShader(shadersAndProgram.vertShader);
1939 		deUint64 vertexShaderCompileEndTime = deGetMicroseconds();
1940 
1941 		buildInfo.fragCompileSuccess = compileShader(shadersAndProgram.fragShader);
1942 		deUint64 fragmentShaderCompileEndTime = deGetMicroseconds();
1943 
1944 		buildInfo.linkSuccess = linkAndUseProgram(shadersAndProgram.program);
1945 		deUint64 programLinkEndTime = deGetMicroseconds();
1946 
1947 		// Check compilation and linking status here, after all compilation and linking gl calls are made.
1948 		if (!(buildInfo.vertCompileSuccess && buildInfo.fragCompileSuccess && buildInfo.linkSuccess))
1949 		{
1950 			buildInfo.logs = getLogs(shadersAndProgram);
1951 			logProgramData(buildInfo, progCtx);
1952 			cleanup(shadersAndProgram, progCtx, buildInfo.linkSuccess);
1953 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Compilation failed");
1954 			return STOP;
1955 		}
1956 
1957 		setShaderInputs(shadersAndProgram.program, progCtx);
1958 		deUint64 firstShaderInputSetEndTime = deGetMicroseconds();
1959 
1960 		// Draw for the first time.
1961 		draw();
1962 		deUint64 firstDrawEndTime = deGetMicroseconds();
1963 
1964 		// Set inputs and draw again.
1965 
1966 		setShaderInputs(shadersAndProgram.program, progCtx);
1967 		deUint64 secondShaderInputSetEndTime = deGetMicroseconds();
1968 
1969 		draw();
1970 		deUint64 secondDrawEndTime = deGetMicroseconds();
1971 
1972 		// De-initializations (detach shaders etc.).
1973 
1974 		buildInfo.logs = getLogs(shadersAndProgram);
1975 		cleanup(shadersAndProgram, progCtx, buildInfo.linkSuccess);
1976 
1977 		// Output measurement log later (after last measurement).
1978 
1979 		measurements.push_back(Measurement((deInt64)(shaderSourceSetEndTime			- startTime),
1980 										   (deInt64)(vertexShaderCompileEndTime		- shaderSourceSetEndTime),
1981 										   (deInt64)(fragmentShaderCompileEndTime	- vertexShaderCompileEndTime),
1982 										   (deInt64)(programLinkEndTime				- fragmentShaderCompileEndTime),
1983 										   (deInt64)(firstShaderInputSetEndTime		- programLinkEndTime),
1984 										   (deInt64)(firstDrawEndTime				- firstShaderInputSetEndTime),
1985 										   (deInt64)(secondShaderInputSetEndTime	- firstDrawEndTime),
1986 										   (deInt64)(secondDrawEndTime				- secondShaderInputSetEndTime)));
1987 
1988 		latestBuildInfo			= buildInfo;
1989 		latestProgramContext	= progCtx;
1990 
1991 		m_testCtx.touchWatchdog(); // \note Measurements may take a while in a bad case.
1992 	}
1993 
1994 	// End of test case, log information about measurements.
1995 	{
1996 		TestLog& log = m_testCtx.getLog();
1997 
1998 		vector<deInt64> sourceSetTimes;
1999 		vector<deInt64> vertexCompileTimes;
2000 		vector<deInt64> fragmentCompileTimes;
2001 		vector<deInt64> programLinkTimes;
2002 		vector<deInt64> firstInputSetTimes;
2003 		vector<deInt64> firstDrawTimes;
2004 		vector<deInt64> secondInputTimes;
2005 		vector<deInt64> secondDrawTimes;
2006 		vector<deInt64> firstPhaseTimes;
2007 		vector<deInt64> secondPhaseTimes;
2008 		vector<deInt64> totalTimesWithoutDraw;
2009 		vector<deInt64> specializationTimes;
2010 
2011 		if (!m_avoidCache)
2012 			log << TestLog::Message << "Note: Testing cache hits, so the medians and averages exclude the first iteration." << TestLog::EndMessage;
2013 
2014 		log << TestLog::Message << "Note: \"Specialization time\" means first draw time minus second draw time." << TestLog::EndMessage
2015 			<< TestLog::Message << "Note: \"Compilation time\" means the time up to (and including) linking, plus specialization time." << TestLog::EndMessage;
2016 
2017 		log << TestLog::Section("IterationMeasurements", "Iteration measurements of compilation and linking times");
2018 
2019 		DE_ASSERT((int)measurements.size() > (m_avoidCache ? 0 : 1));
2020 
2021 		for (int ndx = 0; ndx < (int)measurements.size(); ndx++)
2022 		{
2023 			const Measurement& curMeas = measurements[ndx];
2024 
2025 			// Subtract time of second phase (second input setup and draw) from first (from start to end of first draw).
2026 			// \note Cap if second phase seems unreasonably high (higher than first input set and draw).
2027 			deInt64 timeWithoutDraw		= curMeas.totalTimeWithoutDraw();
2028 
2029 			// Specialization time = first draw - second draw time. Again, cap at 0 if second draw was longer than first draw.
2030 			deInt64 specializationTime	= de::max<deInt64>(0, curMeas.firstDrawTime - curMeas.secondDrawTime);
2031 
2032 			if (ndx > 0 || m_avoidCache) // \note When allowing cache hits, don't account for the first measurement when calculating median or average.
2033 			{
2034 				sourceSetTimes.push_back		(curMeas.sourceSetTime);
2035 				vertexCompileTimes.push_back	(curMeas.vertexCompileTime);
2036 				fragmentCompileTimes.push_back	(curMeas.fragmentCompileTime);
2037 				programLinkTimes.push_back		(curMeas.programLinkTime);
2038 				firstInputSetTimes.push_back	(curMeas.firstInputSetTime);
2039 				firstDrawTimes.push_back		(curMeas.firstDrawTime);
2040 				firstPhaseTimes.push_back		(curMeas.firstPhase());
2041 				secondDrawTimes.push_back		(curMeas.secondDrawTime);
2042 				secondInputTimes.push_back		(curMeas.secondInputSetTime);
2043 				secondPhaseTimes.push_back		(curMeas.secondPhase());
2044 				totalTimesWithoutDraw.push_back	(timeWithoutDraw);
2045 				specializationTimes.push_back	(specializationTime);
2046 			}
2047 
2048 			// Log this measurement.
2049 			log << TestLog::Float("Measurement" + de::toString(ndx) + "CompilationTime",
2050 								  "Measurement " + de::toString(ndx) + " compilation time",
2051 								  "ms", QP_KEY_TAG_TIME, timeWithoutDraw / 1000.0f)
2052 				<< TestLog::Float("Measurement" + de::toString(ndx) + "SpecializationTime",
2053 								  "Measurement " + de::toString(ndx) + " specialization time",
2054 								  "ms", QP_KEY_TAG_TIME, specializationTime / 1000.0f);
2055 		}
2056 
2057 		// Log some statistics.
2058 
2059 		for (int entireRangeOrLowestHalf = 0; entireRangeOrLowestHalf < 2; entireRangeOrLowestHalf++)
2060 		{
2061 			bool				isEntireRange				= entireRangeOrLowestHalf == 0;
2062 			string				statNamePrefix				= isEntireRange ? "" : "LowestHalf";
2063 			vector<deInt64>		rangeTotalTimes				= isEntireRange ? totalTimesWithoutDraw	: vectorLowestPercentage(totalTimesWithoutDraw,	0.5f);
2064 			vector<deInt64>		rangeSpecializationTimes	= isEntireRange ? specializationTimes	: vectorLowestPercentage(specializationTimes,	0.5f);
2065 
2066 #define LOG_COMPILE_SPECIALIZE_TIME_STAT(NAME, DESC, FUNC)																													\
2067 	log << TestLog::Float(statNamePrefix + "CompilationTime" + (NAME), (DESC) + string(" of compilation time"), "ms", QP_KEY_TAG_TIME, (FUNC)(rangeTotalTimes)/1000.0f)		\
2068 		<< TestLog::Float(statNamePrefix + "SpecializationTime" + (NAME), (DESC) + string(" of specialization time"), "ms", QP_KEY_TAG_TIME, (FUNC)(rangeSpecializationTimes)/1000.0f)
2069 
2070 #define LOG_COMPILE_SPECIALIZE_RELATIVE_STAT(NAME, DESC, FUNC)																										\
2071 	log << TestLog::Float(statNamePrefix + "CompilationTime" + (NAME), (DESC) + string(" of compilation time"), "", QP_KEY_TAG_NONE, (FUNC)(rangeTotalTimes))		\
2072 		<< TestLog::Float(statNamePrefix + "SpecializationTime" + (NAME), (DESC) + string(" of specialization time"), "", QP_KEY_TAG_NONE, (FUNC)(rangeSpecializationTimes))
2073 
2074 			log << TestLog::Message << "\nStatistics computed from "
2075 									<< (isEntireRange ? "all" : "only the lowest 50%")
2076 									<< " of the above measurements:"
2077 									<< TestLog::EndMessage;
2078 
2079 			LOG_COMPILE_SPECIALIZE_TIME_STAT		("Median",							"Median",								vectorFloatMedian);
2080 			LOG_COMPILE_SPECIALIZE_TIME_STAT		("Average",							"Average",								vectorFloatAverage);
2081 			LOG_COMPILE_SPECIALIZE_TIME_STAT		("Minimum",							"Minimum",								vectorFloatMinimum);
2082 			LOG_COMPILE_SPECIALIZE_TIME_STAT		("Maximum",							"Maximum",								vectorFloatMaximum);
2083 			LOG_COMPILE_SPECIALIZE_TIME_STAT		("MedianAbsoluteDeviation",			"Median absolute deviation",			vectorFloatMedianAbsoluteDeviation);
2084 			LOG_COMPILE_SPECIALIZE_RELATIVE_STAT	("RelativeMedianAbsoluteDeviation",	"Relative median absolute deviation",	vectorFloatRelativeMedianAbsoluteDeviation);
2085 			LOG_COMPILE_SPECIALIZE_TIME_STAT		("StandardDeviation",				"Standard deviation",					vectorFloatStandardDeviation);
2086 			LOG_COMPILE_SPECIALIZE_RELATIVE_STAT	("RelativeStandardDeviation",		"Relative standard deviation",			vectorFloatRelativeStandardDeviation);
2087 			LOG_COMPILE_SPECIALIZE_TIME_STAT		("MaxMinusMin",						"Max-min",								vectorFloatMaximumMinusMinimum);
2088 			LOG_COMPILE_SPECIALIZE_RELATIVE_STAT	("RelativeMaxMinusMin",				"Relative max-min",						vectorFloatRelativeMaximumMinusMinimum);
2089 
2090 #undef LOG_COMPILE_SPECIALIZE_RELATIVE_STAT
2091 #undef LOG_COMPILE_SPECIALIZE_TIME_STAT
2092 
2093 			if (!isEntireRange && vectorFloatRelativeMedianAbsoluteDeviation(rangeTotalTimes) > RELATIVE_MEDIAN_ABSOLUTE_DEVIATION_THRESHOLD)
2094 				log << TestLog::Message << "\nWARNING: couldn't achieve relative median absolute deviation under threshold value "
2095 										<< RELATIVE_MEDIAN_ABSOLUTE_DEVIATION_THRESHOLD
2096 										<< " for compilation time of the lowest 50% of measurements" << TestLog::EndMessage;
2097 		}
2098 
2099 		log << TestLog::EndSection; // End section IterationMeasurements
2100 
2101 		for (int medianOrAverage = 0; medianOrAverage < 2; medianOrAverage++)
2102 		{
2103 			typedef float (*VecFunc)(const vector<deInt64>&);
2104 
2105 			bool	isMedian						= medianOrAverage == 0;
2106 			string	singular						= isMedian ? "Median" : "Average";
2107 			string	plural							= singular + "s";
2108 			VecFunc func							= isMedian ? (VecFunc) vectorFloatMedian<deInt64> : (VecFunc) vectorFloatAverage<deInt64>;
2109 
2110 			log << TestLog::Section(plural + "PerPhase", plural + " per phase");
2111 
2112 			for (int entireRangeOrLowestHalf = 0; entireRangeOrLowestHalf < 2; entireRangeOrLowestHalf++)
2113 			{
2114 				bool	isEntireRange	= entireRangeOrLowestHalf == 0;
2115 				string	statNamePrefix	= isEntireRange ? "" : "LowestHalf";
2116 				float	rangeSizeRatio	= isEntireRange ? 1.0f : 0.5f;
2117 
2118 #define LOG_TIME(NAME, DESC, DATA) log << TestLog::Float(statNamePrefix + (NAME) + singular, singular + " of " + (DESC), "ms", QP_KEY_TAG_TIME, func(vectorLowestPercentage((DATA), rangeSizeRatio))/1000.0f);
2119 
2120 				log << TestLog::Message << (isEntireRange ? "For all measurements:" : "\nFor only the lowest 50% of the measurements:") << TestLog::EndMessage;
2121 				LOG_TIME("ShaderSourceSetTime",			"shader source set time",			sourceSetTimes);
2122 				LOG_TIME("VertexShaderCompileTime",		"vertex shader compile time",		vertexCompileTimes);
2123 				LOG_TIME("FragmentShaderCompileTime",	"fragment shader compile time",		fragmentCompileTimes);
2124 				LOG_TIME("ProgramLinkTime",				"program link time",				programLinkTimes);
2125 				LOG_TIME("FirstShaderInputSetTime",		"first shader input set time",		firstInputSetTimes);
2126 				LOG_TIME("FirstDrawTime",				"first draw time",					firstDrawTimes);
2127 				LOG_TIME("SecondShaderInputSetTime",	"second shader input set time",		secondInputTimes);
2128 				LOG_TIME("SecondDrawTime",				"second draw time",					secondDrawTimes);
2129 
2130 #undef LOG_TIME
2131 			}
2132 
2133 			log << TestLog::EndSection;
2134 		}
2135 
2136 		// Set result.
2137 
2138 		{
2139 			log << TestLog::Message << "Note: test result is the first quartile (i.e. median of the lowest half of measurements) of compilation times" << TestLog::EndMessage;
2140 			float result = vectorFloatFirstQuartile(totalTimesWithoutDraw) / 1000.0f;
2141 			m_testCtx.setTestResult(QP_TEST_RESULT_PASS, de::floatToString(result, 2).c_str());
2142 		}
2143 
2144 		// Log shaders.
2145 
2146 		if (m_avoidCache || m_addWhitespaceAndComments)
2147 		{
2148 			string msg = "Note: the following shaders are the ones from the last iteration; ";
2149 
2150 			if (m_avoidCache)
2151 				msg += "variables' names and some constant expressions";
2152 			if (m_addWhitespaceAndComments)
2153 				msg += string(m_avoidCache ? " as well as " : "") + "whitespace and comments";
2154 
2155 			msg += " differ between iterations.";
2156 
2157 			log << TestLog::Message << msg.c_str() << TestLog::EndMessage;
2158 		}
2159 
2160 		logProgramData(latestBuildInfo, latestProgramContext);
2161 
2162 		return STOP;
2163 	}
2164 }
2165 
ShaderCompilerLightCase(Context & context,const char * name,const char * description,int caseID,bool avoidCache,bool addWhitespaceAndComments,bool isVertexCase,int numLights,LightType lightType)2166 ShaderCompilerLightCase::ShaderCompilerLightCase (Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, bool isVertexCase, int numLights, LightType lightType)
2167 	: ShaderCompilerCase	(context, name, description, caseID, avoidCache, addWhitespaceAndComments)
2168 	, m_numLights			(numLights)
2169 	, m_isVertexCase		(isVertexCase)
2170 	, m_lightType			(lightType)
2171 	, m_texture				(DE_NULL)
2172 {
2173 }
2174 
~ShaderCompilerLightCase(void)2175 ShaderCompilerLightCase::~ShaderCompilerLightCase (void)
2176 {
2177 	ShaderCompilerLightCase::deinit();
2178 }
2179 
deinit(void)2180 void ShaderCompilerLightCase::deinit (void)
2181 {
2182 	delete m_texture;
2183 	m_texture = DE_NULL;
2184 }
2185 
init(void)2186 void ShaderCompilerLightCase::init (void)
2187 {
2188 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2189 
2190 	// Setup texture.
2191 
2192 	DE_ASSERT(m_texture == DE_NULL);
2193 
2194 	m_texture = new glu::Texture2D(m_context.getRenderContext(), GL_RGB, GL_UNSIGNED_BYTE, TEXTURE_WIDTH, TEXTURE_HEIGHT);
2195 
2196 	tcu::TextureFormatInfo fmtInfo = tcu::getTextureFormatInfo(m_texture->getRefTexture().getFormat());
2197 
2198 	m_texture->getRefTexture().allocLevel(0);
2199 	tcu::fillWithComponentGradients(m_texture->getRefTexture().getLevel(0), fmtInfo.valueMin, fmtInfo.valueMax);
2200 
2201 	gl.activeTexture(GL_TEXTURE0);
2202 	gl.bindTexture(GL_TEXTURE_2D, m_texture->getGLTexture());
2203 	gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
2204 	gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
2205 	gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
2206 	gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
2207 	m_texture->upload();
2208 
2209 	ShaderCompilerCase::init();
2210 }
2211 
generateShaderData(int measurementNdx) const2212 ShaderCompilerCase::ProgramContext ShaderCompilerLightCase::generateShaderData (int measurementNdx) const
2213 {
2214 	deUint32		specID		= getSpecializationID(measurementNdx);
2215 	string			nameSpec	= getNameSpecialization(specID);
2216 	ProgramContext	result;
2217 
2218 	result.vertShaderSource		= specializeShaderSource(lightVertexTemplate(m_numLights, m_isVertexCase, m_lightType), specID, SHADER_VALIDITY_VALID);
2219 	result.fragShaderSource		= specializeShaderSource(lightFragmentTemplate(m_numLights, m_isVertexCase, m_lightType), specID, SHADER_VALIDITY_VALID);
2220 	result.vertexAttributes		= lightShaderAttributes(nameSpec);
2221 	result.uniforms				= lightShaderUniforms(nameSpec, m_numLights, m_lightType);
2222 
2223 	return result;
2224 }
2225 
ShaderCompilerTextureCase(Context & context,const char * name,const char * description,int caseID,bool avoidCache,bool addWhitespaceAndComments,int numLookups,ConditionalUsage conditionalUsage,ConditionalType conditionalType)2226 ShaderCompilerTextureCase::ShaderCompilerTextureCase (Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, int numLookups, ConditionalUsage conditionalUsage, ConditionalType conditionalType)
2227 	: ShaderCompilerCase	(context, name, description, caseID, avoidCache, addWhitespaceAndComments)
2228 	, m_numLookups			(numLookups)
2229 	, m_conditionalUsage	(conditionalUsage)
2230 	, m_conditionalType		(conditionalType)
2231 {
2232 }
2233 
~ShaderCompilerTextureCase(void)2234 ShaderCompilerTextureCase::~ShaderCompilerTextureCase (void)
2235 {
2236 	ShaderCompilerTextureCase::deinit();
2237 }
2238 
deinit(void)2239 void ShaderCompilerTextureCase::deinit (void)
2240 {
2241 	for (vector<glu::Texture2D*>::iterator i = m_textures.begin(); i != m_textures.end(); i++)
2242 		delete *i;
2243 	m_textures.clear();
2244 }
2245 
init(void)2246 void ShaderCompilerTextureCase::init (void)
2247 {
2248 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2249 
2250 	// Setup texture.
2251 
2252 	DE_ASSERT(m_textures.empty());
2253 
2254 	m_textures.reserve(m_numLookups);
2255 
2256 	for (int i = 0; i < m_numLookups; i++)
2257 	{
2258 		glu::Texture2D*			tex		= new glu::Texture2D(m_context.getRenderContext(), GL_RGB, GL_UNSIGNED_BYTE, TEXTURE_WIDTH, TEXTURE_HEIGHT);
2259 		tcu::TextureFormatInfo	fmtInfo	= tcu::getTextureFormatInfo(tex->getRefTexture().getFormat());
2260 
2261 		tex->getRefTexture().allocLevel(0);
2262 		tcu::fillWithComponentGradients(tex->getRefTexture().getLevel(0), fmtInfo.valueMin, fmtInfo.valueMax);
2263 
2264 		gl.activeTexture(GL_TEXTURE0 + i);
2265 		gl.bindTexture(GL_TEXTURE_2D, tex->getGLTexture());
2266 		gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
2267 		gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
2268 		gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
2269 		gl.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
2270 		tex->upload();
2271 
2272 		m_textures.push_back(tex);
2273 	}
2274 
2275 	ShaderCompilerCase::init();
2276 }
2277 
generateShaderData(int measurementNdx) const2278 ShaderCompilerCase::ProgramContext ShaderCompilerTextureCase::generateShaderData (int measurementNdx) const
2279 {
2280 	deUint32		specID		= getSpecializationID(measurementNdx);
2281 	string			nameSpec	= getNameSpecialization(specID);
2282 	ProgramContext	result;
2283 
2284 	result.vertShaderSource		= specializeShaderSource(textureLookupVertexTemplate(m_conditionalUsage, m_conditionalType), specID, SHADER_VALIDITY_VALID);
2285 	result.fragShaderSource		= specializeShaderSource(textureLookupFragmentTemplate(m_numLookups, m_conditionalUsage, m_conditionalType), specID, SHADER_VALIDITY_VALID);
2286 	result.vertexAttributes		= textureLookupShaderAttributes(nameSpec, m_conditionalUsage, m_conditionalType);
2287 	result.uniforms				= textureLookupShaderUniforms(nameSpec, m_numLookups, m_conditionalUsage, m_conditionalType);
2288 
2289 	return result;
2290 }
2291 
ShaderCompilerLoopCase(Context & context,const char * name,const char * description,int caseID,bool avoidCache,bool addWhitespaceAndComments,bool isVertexCase,LoopType type,int numLoopIterations,int nestingDepth)2292 ShaderCompilerLoopCase::ShaderCompilerLoopCase (Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, bool isVertexCase, LoopType type, int numLoopIterations, int nestingDepth)
2293 	: ShaderCompilerCase	(context, name, description, caseID, avoidCache, addWhitespaceAndComments)
2294 	, m_numLoopIterations	(numLoopIterations)
2295 	, m_nestingDepth		(nestingDepth)
2296 	, m_isVertexCase		(isVertexCase)
2297 	, m_type				(type)
2298 {
2299 }
2300 
~ShaderCompilerLoopCase(void)2301 ShaderCompilerLoopCase::~ShaderCompilerLoopCase (void)
2302 {
2303 }
2304 
generateShaderData(int measurementNdx) const2305 ShaderCompilerCase::ProgramContext ShaderCompilerLoopCase::generateShaderData (int measurementNdx) const
2306 {
2307 	deUint32		specID		= getSpecializationID(measurementNdx);
2308 	string			nameSpec	= getNameSpecialization(specID);
2309 	ProgramContext	result;
2310 
2311 	result.vertShaderSource		= specializeShaderSource(loopVertexTemplate(m_type, m_isVertexCase, m_numLoopIterations, m_nestingDepth), specID, SHADER_VALIDITY_VALID);
2312 	result.fragShaderSource		= specializeShaderSource(loopFragmentTemplate(m_type, m_isVertexCase, m_numLoopIterations, m_nestingDepth), specID, SHADER_VALIDITY_VALID);
2313 
2314 	result.vertexAttributes		= loopShaderAttributes(nameSpec, m_type, m_numLoopIterations);
2315 	result.uniforms				= loopShaderUniforms(nameSpec, m_type, m_numLoopIterations);
2316 
2317 	return result;
2318 }
2319 
ShaderCompilerOperCase(Context & context,const char * name,const char * description,int caseID,bool avoidCache,bool addWhitespaceAndComments,bool isVertexCase,const char * oper,int numOperations)2320 ShaderCompilerOperCase::ShaderCompilerOperCase (Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, bool isVertexCase, const char* oper, int numOperations)
2321 	: ShaderCompilerCase	(context, name, description, caseID, avoidCache, addWhitespaceAndComments)
2322 	, m_oper				(oper)
2323 	, m_numOperations		(numOperations)
2324 	, m_isVertexCase		(isVertexCase)
2325 {
2326 }
2327 
~ShaderCompilerOperCase(void)2328 ShaderCompilerOperCase::~ShaderCompilerOperCase (void)
2329 {
2330 }
2331 
generateShaderData(int measurementNdx) const2332 ShaderCompilerCase::ProgramContext ShaderCompilerOperCase::generateShaderData (int measurementNdx) const
2333 {
2334 	deUint32		specID		= getSpecializationID(measurementNdx);
2335 	string			nameSpec	= getNameSpecialization(specID);
2336 	ProgramContext	result;
2337 
2338 	if (m_isVertexCase)
2339 	{
2340 		result.vertShaderSource = specializeShaderSource(binaryOpVertexTemplate(m_numOperations, m_oper.c_str()), specID, SHADER_VALIDITY_VALID);
2341 		result.fragShaderSource = specializeShaderSource(singleVaryingFragmentTemplate(), specID, SHADER_VALIDITY_VALID);
2342 	}
2343 	else
2344 	{
2345 		result.vertShaderSource = specializeShaderSource(singleVaryingVertexTemplate(), specID, SHADER_VALIDITY_VALID);
2346 		result.fragShaderSource = specializeShaderSource(binaryOpFragmentTemplate(m_numOperations, m_oper.c_str()), specID, SHADER_VALIDITY_VALID);
2347 	}
2348 
2349 	result.vertexAttributes = singleValueShaderAttributes(nameSpec);
2350 
2351 	result.uniforms.clear(); // No uniforms used.
2352 
2353 	return result;
2354 }
2355 
ShaderCompilerMandelbrotCase(Context & context,const char * name,const char * description,int caseID,bool avoidCache,bool addWhitespaceAndComments,int numFractalIterations)2356 ShaderCompilerMandelbrotCase::ShaderCompilerMandelbrotCase (Context& context, const char* name, const char* description, int caseID, bool avoidCache, bool addWhitespaceAndComments, int numFractalIterations)
2357 	: ShaderCompilerCase		(context, name, description, caseID, avoidCache, addWhitespaceAndComments)
2358 	, m_numFractalIterations	(numFractalIterations)
2359 {
2360 }
2361 
~ShaderCompilerMandelbrotCase(void)2362 ShaderCompilerMandelbrotCase::~ShaderCompilerMandelbrotCase (void)
2363 {
2364 }
2365 
generateShaderData(int measurementNdx) const2366 ShaderCompilerCase::ProgramContext ShaderCompilerMandelbrotCase::generateShaderData (int measurementNdx) const
2367 {
2368 	deUint32		specID		= getSpecializationID(measurementNdx);
2369 	string			nameSpec	= getNameSpecialization(specID);
2370 	ProgramContext	result;
2371 
2372 	result.vertShaderSource = specializeShaderSource(mandelbrotVertexTemplate(), specID, SHADER_VALIDITY_VALID);
2373 	result.fragShaderSource = specializeShaderSource(mandelbrotFragmentTemplate(m_numFractalIterations), specID, SHADER_VALIDITY_VALID);
2374 
2375 	result.vertexAttributes = mandelbrotShaderAttributes(nameSpec);
2376 	result.uniforms = mandelbrotShaderUniforms(nameSpec);
2377 
2378 	return result;
2379 }
2380 
InvalidShaderCompilerCase(Context & context,const char * name,const char * description,int caseID,InvalidityType invalidityType)2381 InvalidShaderCompilerCase::InvalidShaderCompilerCase (Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType)
2382 	: TestCase						(context, tcu::NODETYPE_PERFORMANCE, name, description)
2383 	, m_invalidityType				(invalidityType)
2384 	, m_startHash					((deUint32)(deUint64Hash(deGetTime()) ^ deUint64Hash(deGetMicroseconds()) ^ deInt32Hash(caseID)))
2385 {
2386 	int cmdLineIterCount = context.getTestContext().getCommandLine().getTestIterationCount();
2387 	m_minimumMeasurementCount = cmdLineIterCount > 0 ? cmdLineIterCount : DEFAULT_MINIMUM_MEASUREMENT_COUNT;
2388 	m_maximumMeasurementCount = 3*m_minimumMeasurementCount;
2389 }
2390 
~InvalidShaderCompilerCase(void)2391 InvalidShaderCompilerCase::~InvalidShaderCompilerCase (void)
2392 {
2393 }
2394 
getSpecializationID(int measurementNdx) const2395 deUint32 InvalidShaderCompilerCase::getSpecializationID (int measurementNdx) const
2396 {
2397 	return m_startHash ^ (deUint32)deInt32Hash((deInt32)measurementNdx);
2398 }
2399 
createShaders(void) const2400 InvalidShaderCompilerCase::Shaders InvalidShaderCompilerCase::createShaders (void) const
2401 {
2402 	const glw::Functions&	gl = m_context.getRenderContext().getFunctions();
2403 	Shaders					result;
2404 
2405 	result.vertShader = gl.createShader(GL_VERTEX_SHADER);
2406 	result.fragShader = gl.createShader(GL_FRAGMENT_SHADER);
2407 
2408 	return result;
2409 }
2410 
setShaderSources(const Shaders & shaders,const ProgramContext & progCtx) const2411 void InvalidShaderCompilerCase::setShaderSources (const Shaders& shaders, const ProgramContext& progCtx) const
2412 {
2413 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2414 	const char* vertShaderSourceCStr = progCtx.vertShaderSource.c_str();
2415 	const char* fragShaderSourceCStr = progCtx.fragShaderSource.c_str();
2416 	gl.shaderSource(shaders.vertShader, 1, &vertShaderSourceCStr, DE_NULL);
2417 	gl.shaderSource(shaders.fragShader, 1, &fragShaderSourceCStr, DE_NULL);
2418 }
2419 
compileShader(deUint32 shader) const2420 bool InvalidShaderCompilerCase::compileShader (deUint32 shader) const
2421 {
2422 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2423 	GLint status;
2424 	gl.compileShader(shader);
2425 	gl.getShaderiv(shader, GL_COMPILE_STATUS, &status);
2426 	return status != 0;
2427 }
2428 
logProgramData(const BuildInfo & buildInfo,const ProgramContext & progCtx) const2429 void InvalidShaderCompilerCase::logProgramData (const BuildInfo& buildInfo, const ProgramContext& progCtx) const
2430 {
2431 	m_testCtx.getLog() << TestLog::ShaderProgram(false, "(No linking done)")
2432 					   << TestLog::Shader(QP_SHADER_TYPE_VERTEX,	progCtx.vertShaderSource, buildInfo.vertCompileSuccess, buildInfo.logs.vert)
2433 					   << TestLog::Shader(QP_SHADER_TYPE_FRAGMENT,	progCtx.fragShaderSource, buildInfo.fragCompileSuccess, buildInfo.logs.frag)
2434 					   << TestLog::EndShaderProgram;
2435 }
2436 
getLogs(const Shaders & shaders) const2437 InvalidShaderCompilerCase::Logs InvalidShaderCompilerCase::getLogs (const Shaders& shaders) const
2438 {
2439 	const glw::Functions&	gl = m_context.getRenderContext().getFunctions();
2440 	Logs					result;
2441 
2442 	result.vert = getShaderInfoLog(gl, shaders.vertShader);
2443 	result.frag = getShaderInfoLog(gl, shaders.fragShader);
2444 
2445 	return result;
2446 }
2447 
cleanup(const Shaders & shaders) const2448 void InvalidShaderCompilerCase::cleanup (const Shaders& shaders) const
2449 {
2450 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
2451 
2452 	gl.deleteShader(shaders.vertShader);
2453 	gl.deleteShader(shaders.fragShader);
2454 }
2455 
goodEnoughMeasurements(const vector<Measurement> & measurements) const2456 bool InvalidShaderCompilerCase::goodEnoughMeasurements (const vector<Measurement>& measurements) const
2457 {
2458 	if ((int)measurements.size() < m_minimumMeasurementCount)
2459 		return false;
2460 	else
2461 	{
2462 		if ((int)measurements.size() >= m_maximumMeasurementCount)
2463 			return true;
2464 		else
2465 		{
2466 			vector<deInt64> totalTimes;
2467 			for (int i = 0; i < (int)measurements.size(); i++)
2468 				totalTimes.push_back(measurements[i].totalTime());
2469 			return vectorFloatRelativeMedianAbsoluteDeviation(vectorLowestPercentage(totalTimes, 0.5f)) < RELATIVE_MEDIAN_ABSOLUTE_DEVIATION_THRESHOLD;
2470 		}
2471 	}
2472 }
2473 
iterate(void)2474 InvalidShaderCompilerCase::IterateResult InvalidShaderCompilerCase::iterate (void)
2475 {
2476 	ShaderValidity shaderValidity = m_invalidityType == INVALIDITY_INVALID_CHAR		? SHADER_VALIDITY_INVALID_CHAR
2477 								  : m_invalidityType == INVALIDITY_SEMANTIC_ERROR	? SHADER_VALIDITY_SEMANTIC_ERROR
2478 								  : SHADER_VALIDITY_LAST;
2479 
2480 	DE_ASSERT(shaderValidity != SHADER_VALIDITY_LAST);
2481 
2482 	// Before actual measurements, compile a dummy shader to avoid possible initial slowdowns in the actual test.
2483 	{
2484 		deUint32		specID = getSpecializationID(0);
2485 		ProgramContext	progCtx;
2486 		progCtx.vertShaderSource = specializeShaderSource(singleVaryingVertexTemplate(), specID, shaderValidity);
2487 		progCtx.fragShaderSource = specializeShaderSource(singleVaryingFragmentTemplate(), specID, shaderValidity);
2488 
2489 		Shaders shaders = createShaders();
2490 		setShaderSources(shaders, progCtx);
2491 
2492 		BuildInfo buildInfo;
2493 		buildInfo.vertCompileSuccess = compileShader(shaders.vertShader);
2494 		buildInfo.fragCompileSuccess = compileShader(shaders.fragShader);
2495 		if (buildInfo.vertCompileSuccess || buildInfo.fragCompileSuccess)
2496 		{
2497 			buildInfo.logs = getLogs(shaders);
2498 			logProgramData(buildInfo, progCtx);
2499 			cleanup(shaders);
2500 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Compilation of a shader erroneously succeeded");
2501 			return STOP;
2502 		}
2503 		cleanup(shaders);
2504 	}
2505 
2506 	vector<Measurement>		measurements;
2507 	// \note These are logged after measurements are done.
2508 	ProgramContext			latestProgramContext;
2509 	BuildInfo				latestBuildInfo;
2510 
2511 	if (WARMUP_CPU_AT_BEGINNING_OF_CASE)
2512 		tcu::warmupCPU();
2513 
2514 	// Actual test measurements.
2515 	while (!goodEnoughMeasurements(measurements))
2516 	{
2517 		// Create shader and compile. Measure time.
2518 
2519 		// \note Shader sources are generated and GL shader objects are created before any time measurements.
2520 		ProgramContext	progCtx		= generateShaderSources((int)measurements.size());
2521 		Shaders			shaders		= createShaders();
2522 		BuildInfo		buildInfo;
2523 
2524 		if (WARMUP_CPU_BEFORE_EACH_MEASUREMENT)
2525 			tcu::warmupCPU();
2526 
2527 		// \note Do NOT do anything too hefty between the first and last deGetMicroseconds() here (other than the gl calls); it would disturb the measurement.
2528 
2529 		deUint64 startTime = deGetMicroseconds();
2530 
2531 		setShaderSources(shaders, progCtx);
2532 		deUint64 shaderSourceSetEndTime = deGetMicroseconds();
2533 
2534 		buildInfo.vertCompileSuccess = compileShader(shaders.vertShader);
2535 		deUint64 vertexShaderCompileEndTime = deGetMicroseconds();
2536 
2537 		buildInfo.fragCompileSuccess = compileShader(shaders.fragShader);
2538 		deUint64 fragmentShaderCompileEndTime = deGetMicroseconds();
2539 
2540 		buildInfo.logs = getLogs(shaders);
2541 
2542 		// Both shader compilations should have failed.
2543 		if (buildInfo.vertCompileSuccess || buildInfo.fragCompileSuccess)
2544 		{
2545 			logProgramData(buildInfo, progCtx);
2546 			cleanup(shaders);
2547 			m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Compilation of a shader erroneously succeeded");
2548 			return STOP;
2549 		}
2550 
2551 		// De-initializations (delete shaders).
2552 
2553 		cleanup(shaders);
2554 
2555 		// Output measurement log later (after last measurement).
2556 
2557 		measurements.push_back(Measurement((deInt64)(shaderSourceSetEndTime			- startTime),
2558 										   (deInt64)(vertexShaderCompileEndTime		- shaderSourceSetEndTime),
2559 										   (deInt64)(fragmentShaderCompileEndTime	- vertexShaderCompileEndTime)));
2560 
2561 		latestBuildInfo			= buildInfo;
2562 		latestProgramContext	= progCtx;
2563 
2564 		m_testCtx.touchWatchdog(); // \note Measurements may take a while in a bad case.
2565 	}
2566 
2567 	// End of test case, log information about measurements.
2568 	{
2569 		TestLog& log = m_testCtx.getLog();
2570 
2571 		vector<deInt64> sourceSetTimes;
2572 		vector<deInt64> vertexCompileTimes;
2573 		vector<deInt64> fragmentCompileTimes;
2574 		vector<deInt64> totalTimes;
2575 
2576 		log << TestLog::Section("IterationMeasurements", "Iteration measurements of compilation times");
2577 
2578 		for (int ndx = 0; ndx < (int)measurements.size(); ndx++)
2579 		{
2580 			sourceSetTimes.push_back		(measurements[ndx].sourceSetTime);
2581 			vertexCompileTimes.push_back	(measurements[ndx].vertexCompileTime);
2582 			fragmentCompileTimes.push_back	(measurements[ndx].fragmentCompileTime);
2583 			totalTimes.push_back			(measurements[ndx].totalTime());
2584 
2585 			// Log this measurement.
2586 			log << TestLog::Float("Measurement" + de::toString(ndx) + "Time",
2587 								  "Measurement " + de::toString(ndx) + " time",
2588 								  "ms", QP_KEY_TAG_TIME, measurements[ndx].totalTime()/1000.0f);
2589 		}
2590 
2591 		// Log some statistics.
2592 
2593 		for (int entireRangeOrLowestHalf = 0; entireRangeOrLowestHalf < 2; entireRangeOrLowestHalf++)
2594 		{
2595 			bool				isEntireRange	= entireRangeOrLowestHalf == 0;
2596 			string				statNamePrefix	= isEntireRange ? "" : "LowestHalf";
2597 			vector<deInt64>		rangeTimes		= isEntireRange ? totalTimes : vectorLowestPercentage(totalTimes, 0.5f);
2598 
2599 			log << TestLog::Message << "\nStatistics computed from "
2600 									<< (isEntireRange ? "all" : "only the lowest 50%")
2601 									<< " of the above measurements:"
2602 									<< TestLog::EndMessage;
2603 
2604 #define LOG_TIME_STAT(NAME, DESC, FUNC)			log << TestLog::Float(statNamePrefix + "TotalTime" + (NAME), (DESC) + string(" of total time"), "ms",	QP_KEY_TAG_TIME, (FUNC)(rangeTimes)/1000.0f)
2605 #define LOG_RELATIVE_STAT(NAME, DESC, FUNC)		log << TestLog::Float(statNamePrefix + "TotalTime" + (NAME), (DESC) + string(" of total time"), "",		QP_KEY_TAG_NONE, (FUNC)(rangeTimes))
2606 
2607 			LOG_TIME_STAT		("Median",							"Median",								vectorFloatMedian);
2608 			LOG_TIME_STAT		("Average",							"Average",								vectorFloatAverage);
2609 			LOG_TIME_STAT		("Minimum",							"Minimum",								vectorFloatMinimum);
2610 			LOG_TIME_STAT		("Maximum",							"Maximum",								vectorFloatMaximum);
2611 			LOG_TIME_STAT		("MedianAbsoluteDeviation",			"Median absolute deviation",			vectorFloatMedianAbsoluteDeviation);
2612 			LOG_RELATIVE_STAT	("RelativeMedianAbsoluteDeviation",	"Relative median absolute deviation",	vectorFloatRelativeMedianAbsoluteDeviation);
2613 			LOG_TIME_STAT		("StandardDeviation",				"Standard deviation",					vectorFloatStandardDeviation);
2614 			LOG_RELATIVE_STAT	("RelativeStandardDeviation",		"Relative standard deviation",			vectorFloatRelativeStandardDeviation);
2615 			LOG_TIME_STAT		("MaxMinusMin",						"Max-min",								vectorFloatMaximumMinusMinimum);
2616 			LOG_RELATIVE_STAT	("RelativeMaxMinusMin",				"Relative max-min",						vectorFloatRelativeMaximumMinusMinimum);
2617 
2618 #undef LOG_TIME_STAT
2619 #undef LOG_RELATIVE_STAT
2620 
2621 			if (!isEntireRange && vectorFloatRelativeMedianAbsoluteDeviation(rangeTimes) > RELATIVE_MEDIAN_ABSOLUTE_DEVIATION_THRESHOLD)
2622 				log << TestLog::Message << "\nWARNING: couldn't achieve relative median absolute deviation under threshold value " << RELATIVE_MEDIAN_ABSOLUTE_DEVIATION_THRESHOLD << TestLog::EndMessage;
2623 		}
2624 
2625 		log << TestLog::EndSection; // End section IterationMeasurements
2626 
2627 		for (int medianOrAverage = 0; medianOrAverage < 2; medianOrAverage++)
2628 		{
2629 			typedef float (*VecFunc)(const vector<deInt64>&);
2630 
2631 			bool	isMedian						= medianOrAverage == 0;
2632 			string	singular						= isMedian ? "Median" : "Average";
2633 			string	plural							= singular + "s";
2634 			VecFunc func							= isMedian ? (VecFunc) vectorFloatMedian<deInt64> : (VecFunc) vectorFloatAverage<deInt64>;
2635 
2636 			log << TestLog::Section(plural + "PerPhase", plural + " per phase");
2637 
2638 			for (int entireRangeOrLowestHalf = 0; entireRangeOrLowestHalf < 2; entireRangeOrLowestHalf++)
2639 			{
2640 				bool	isEntireRange	= entireRangeOrLowestHalf == 0;
2641 				string	statNamePrefix	= isEntireRange ? "" : "LowestHalf";
2642 				float	rangeSizeRatio	= isEntireRange ? 1.0f : 0.5f;
2643 
2644 #define LOG_TIME(NAME, DESC, DATA) log << TestLog::Float(statNamePrefix + (NAME) + singular, singular + " of " + (DESC), "ms", QP_KEY_TAG_TIME, func(vectorLowestPercentage((DATA), rangeSizeRatio))/1000.0f);
2645 
2646 				log << TestLog::Message << (isEntireRange ? "For all measurements:" : "\nFor only the lowest 50% of the measurements:") << TestLog::EndMessage;
2647 				LOG_TIME("ShaderSourceSetTime",			"shader source set time",			sourceSetTimes);
2648 				LOG_TIME("VertexShaderCompileTime",		"vertex shader compile time",		vertexCompileTimes);
2649 				LOG_TIME("FragmentShaderCompileTime",	"fragment shader compile time",		fragmentCompileTimes);
2650 
2651 #undef LOG_TIME
2652 			}
2653 
2654 			log << TestLog::EndSection;
2655 		}
2656 
2657 		// Set result.
2658 
2659 		{
2660 			log << TestLog::Message << "Note: test result is the first quartile (i.e. median of the lowest half of measurements) of total times" << TestLog::EndMessage;
2661 			float result = vectorFloatFirstQuartile(totalTimes) / 1000.0f;
2662 			m_testCtx.setTestResult(QP_TEST_RESULT_PASS, de::floatToString(result, 2).c_str());
2663 		}
2664 
2665 		// Log shaders.
2666 
2667 		log << TestLog::Message << "Note: the following shaders are the ones from the last iteration; variables' names and some constant expressions differ between iterations." << TestLog::EndMessage;
2668 
2669 		logProgramData(latestBuildInfo, latestProgramContext);
2670 
2671 		return STOP;
2672 	}
2673 }
2674 
InvalidShaderCompilerLightCase(Context & context,const char * name,const char * description,int caseID,InvalidityType invalidityType,bool isVertexCase,int numLights,LightType lightType)2675 InvalidShaderCompilerLightCase::InvalidShaderCompilerLightCase (Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, bool isVertexCase, int numLights, LightType lightType)
2676 	: InvalidShaderCompilerCase	(context, name, description, caseID, invalidityType)
2677 	, m_isVertexCase			(isVertexCase)
2678 	, m_numLights				(numLights)
2679 	, m_lightType				(lightType)
2680 {
2681 }
2682 
~InvalidShaderCompilerLightCase(void)2683 InvalidShaderCompilerLightCase::~InvalidShaderCompilerLightCase (void)
2684 {
2685 }
2686 
generateShaderSources(int measurementNdx) const2687 InvalidShaderCompilerCase::ProgramContext InvalidShaderCompilerLightCase::generateShaderSources (int measurementNdx) const
2688 {
2689 	deUint32		specID			= getSpecializationID(measurementNdx);
2690 	ProgramContext	result;
2691 	ShaderValidity	shaderValidity	= m_invalidityType == INVALIDITY_INVALID_CHAR	? SHADER_VALIDITY_INVALID_CHAR
2692 									: m_invalidityType == INVALIDITY_SEMANTIC_ERROR	? SHADER_VALIDITY_SEMANTIC_ERROR
2693 									: SHADER_VALIDITY_LAST;
2694 
2695 	DE_ASSERT(shaderValidity != SHADER_VALIDITY_LAST);
2696 
2697 	result.vertShaderSource = specializeShaderSource(lightVertexTemplate(m_numLights, m_isVertexCase, m_lightType), specID, shaderValidity);
2698 	result.fragShaderSource = specializeShaderSource(lightFragmentTemplate(m_numLights, m_isVertexCase, m_lightType), specID, shaderValidity);
2699 
2700 	return result;
2701 }
2702 
InvalidShaderCompilerTextureCase(Context & context,const char * name,const char * description,int caseID,InvalidityType invalidityType,int numLookups,ConditionalUsage conditionalUsage,ConditionalType conditionalType)2703 InvalidShaderCompilerTextureCase::InvalidShaderCompilerTextureCase (Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, int numLookups, ConditionalUsage conditionalUsage, ConditionalType conditionalType)
2704 	: InvalidShaderCompilerCase	(context, name, description, caseID, invalidityType)
2705 	, m_numLookups				(numLookups)
2706 	, m_conditionalUsage		(conditionalUsage)
2707 	, m_conditionalType			(conditionalType)
2708 {
2709 }
2710 
~InvalidShaderCompilerTextureCase(void)2711 InvalidShaderCompilerTextureCase::~InvalidShaderCompilerTextureCase (void)
2712 {
2713 }
2714 
generateShaderSources(int measurementNdx) const2715 InvalidShaderCompilerCase::ProgramContext InvalidShaderCompilerTextureCase::generateShaderSources (int measurementNdx) const
2716 {
2717 	deUint32		specID			= getSpecializationID(measurementNdx);
2718 	ProgramContext	result;
2719 	ShaderValidity	shaderValidity	= m_invalidityType == INVALIDITY_INVALID_CHAR	? SHADER_VALIDITY_INVALID_CHAR
2720 									: m_invalidityType == INVALIDITY_SEMANTIC_ERROR	? SHADER_VALIDITY_SEMANTIC_ERROR
2721 									: SHADER_VALIDITY_LAST;
2722 
2723 	DE_ASSERT(shaderValidity != SHADER_VALIDITY_LAST);
2724 
2725 	result.vertShaderSource = specializeShaderSource(textureLookupVertexTemplate(m_conditionalUsage, m_conditionalType), specID, shaderValidity);
2726 	result.fragShaderSource = specializeShaderSource(textureLookupFragmentTemplate(m_numLookups, m_conditionalUsage, m_conditionalType), specID, shaderValidity);
2727 
2728 	return result;
2729 }
2730 
InvalidShaderCompilerLoopCase(Context & context,const char * name,const char * description,int caseID,InvalidityType invalidityType,bool isVertexCase,LoopType type,int numLoopIterations,int nestingDepth)2731 InvalidShaderCompilerLoopCase::InvalidShaderCompilerLoopCase (Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, bool isVertexCase, LoopType type, int numLoopIterations, int nestingDepth)
2732 	: InvalidShaderCompilerCase	(context, name, description, caseID, invalidityType)
2733 	, m_isVertexCase			(isVertexCase)
2734 	, m_numLoopIterations		(numLoopIterations)
2735 	, m_nestingDepth			(nestingDepth)
2736 	, m_type					(type)
2737 {
2738 }
2739 
~InvalidShaderCompilerLoopCase(void)2740 InvalidShaderCompilerLoopCase::~InvalidShaderCompilerLoopCase (void)
2741 {
2742 }
2743 
generateShaderSources(int measurementNdx) const2744 InvalidShaderCompilerCase::ProgramContext InvalidShaderCompilerLoopCase::generateShaderSources (int measurementNdx) const
2745 {
2746 	deUint32		specID			= getSpecializationID(measurementNdx);
2747 	ProgramContext	result;
2748 	ShaderValidity	shaderValidity	= m_invalidityType == INVALIDITY_INVALID_CHAR	? SHADER_VALIDITY_INVALID_CHAR
2749 									: m_invalidityType == INVALIDITY_SEMANTIC_ERROR	? SHADER_VALIDITY_SEMANTIC_ERROR
2750 									: SHADER_VALIDITY_LAST;
2751 
2752 	DE_ASSERT(shaderValidity != SHADER_VALIDITY_LAST);
2753 
2754 	result.vertShaderSource = specializeShaderSource(loopVertexTemplate(m_type, m_isVertexCase, m_numLoopIterations, m_nestingDepth), specID, shaderValidity);
2755 	result.fragShaderSource = specializeShaderSource(loopFragmentTemplate(m_type, m_isVertexCase, m_numLoopIterations, m_nestingDepth), specID, shaderValidity);
2756 
2757 	return result;
2758 }
2759 
InvalidShaderCompilerOperCase(Context & context,const char * name,const char * description,int caseID,InvalidityType invalidityType,bool isVertexCase,const char * oper,int numOperations)2760 InvalidShaderCompilerOperCase::InvalidShaderCompilerOperCase (Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, bool isVertexCase, const char* oper, int numOperations)
2761 	: InvalidShaderCompilerCase	(context, name, description, caseID, invalidityType)
2762 	, m_isVertexCase			(isVertexCase)
2763 	, m_oper					(oper)
2764 	, m_numOperations			(numOperations)
2765 {
2766 }
2767 
~InvalidShaderCompilerOperCase(void)2768 InvalidShaderCompilerOperCase::~InvalidShaderCompilerOperCase (void)
2769 {
2770 }
2771 
generateShaderSources(int measurementNdx) const2772 InvalidShaderCompilerCase::ProgramContext InvalidShaderCompilerOperCase::generateShaderSources (int measurementNdx) const
2773 {
2774 	deUint32		specID			= getSpecializationID(measurementNdx);
2775 	ProgramContext	result;
2776 	ShaderValidity	shaderValidity	= m_invalidityType == INVALIDITY_INVALID_CHAR	? SHADER_VALIDITY_INVALID_CHAR
2777 									: m_invalidityType == INVALIDITY_SEMANTIC_ERROR	? SHADER_VALIDITY_SEMANTIC_ERROR
2778 									: SHADER_VALIDITY_LAST;
2779 
2780 	DE_ASSERT(shaderValidity != SHADER_VALIDITY_LAST);
2781 
2782 	if (m_isVertexCase)
2783 	{
2784 		result.vertShaderSource = specializeShaderSource(binaryOpVertexTemplate(m_numOperations, m_oper.c_str()), specID, shaderValidity);
2785 		result.fragShaderSource = specializeShaderSource(singleVaryingFragmentTemplate(), specID, shaderValidity);
2786 	}
2787 	else
2788 	{
2789 		result.vertShaderSource = specializeShaderSource(singleVaryingVertexTemplate(), specID, shaderValidity);
2790 		result.fragShaderSource = specializeShaderSource(binaryOpFragmentTemplate(m_numOperations, m_oper.c_str()), specID, shaderValidity);
2791 	}
2792 
2793 	return result;
2794 }
2795 
InvalidShaderCompilerMandelbrotCase(Context & context,const char * name,const char * description,int caseID,InvalidityType invalidityType,int numFractalIterations)2796 InvalidShaderCompilerMandelbrotCase::InvalidShaderCompilerMandelbrotCase (Context& context, const char* name, const char* description, int caseID, InvalidityType invalidityType, int numFractalIterations)
2797 	: InvalidShaderCompilerCase	(context, name, description, caseID, invalidityType)
2798 	, m_numFractalIterations	(numFractalIterations)
2799 {
2800 }
2801 
~InvalidShaderCompilerMandelbrotCase(void)2802 InvalidShaderCompilerMandelbrotCase::~InvalidShaderCompilerMandelbrotCase (void)
2803 {
2804 }
2805 
generateShaderSources(int measurementNdx) const2806 InvalidShaderCompilerCase::ProgramContext InvalidShaderCompilerMandelbrotCase::generateShaderSources (int measurementNdx) const
2807 {
2808 	deUint32		specID			= getSpecializationID(measurementNdx);
2809 	ProgramContext	result;
2810 	ShaderValidity	shaderValidity	= m_invalidityType == INVALIDITY_INVALID_CHAR	? SHADER_VALIDITY_INVALID_CHAR
2811 									: m_invalidityType == INVALIDITY_SEMANTIC_ERROR	? SHADER_VALIDITY_SEMANTIC_ERROR
2812 									: SHADER_VALIDITY_LAST;
2813 
2814 	DE_ASSERT(shaderValidity != SHADER_VALIDITY_LAST);
2815 
2816 	result.vertShaderSource = specializeShaderSource(mandelbrotVertexTemplate(), specID, shaderValidity);
2817 	result.fragShaderSource = specializeShaderSource(mandelbrotFragmentTemplate(m_numFractalIterations), specID, shaderValidity);
2818 
2819 	return result;
2820 }
2821 
addShaderCompilationPerformanceCases(TestCaseGroup & parentGroup)2822 void addShaderCompilationPerformanceCases (TestCaseGroup& parentGroup)
2823 {
2824 	Context&	context		= parentGroup.getContext();
2825 	int			caseID		= 0; // Increment this after adding each case. Used for avoiding cache hits between cases.
2826 
2827 	TestCaseGroup* validGroup			= new TestCaseGroup(context, "valid_shader",	"Valid Shader Compiler Cases");
2828 	TestCaseGroup* invalidGroup			= new TestCaseGroup(context, "invalid_shader",	"Invalid Shader Compiler Cases");
2829 	TestCaseGroup* cacheGroup			= new TestCaseGroup(context, "cache",			"Allow shader caching");
2830 	parentGroup.addChild(validGroup);
2831 	parentGroup.addChild(invalidGroup);
2832 	parentGroup.addChild(cacheGroup);
2833 
2834 	TestCaseGroup* invalidCharGroup		= new TestCaseGroup(context, "invalid_char",	"Invalid Character Shader Compiler Cases");
2835 	TestCaseGroup* semanticErrorGroup	= new TestCaseGroup(context, "semantic_error",	"Semantic Error Shader Compiler Cases");
2836 	invalidGroup->addChild(invalidCharGroup);
2837 	invalidGroup->addChild(semanticErrorGroup);
2838 
2839 	// Lighting shader compilation cases.
2840 
2841 	{
2842 		static const int lightCounts[] = { 1, 2, 4, 8 };
2843 
2844 		TestCaseGroup* validLightingGroup			= new TestCaseGroup(context, "lighting", "Shader Compiler Lighting Cases");
2845 		TestCaseGroup* invalidCharLightingGroup		= new TestCaseGroup(context, "lighting", "Invalid Character Shader Compiler Lighting Cases");
2846 		TestCaseGroup* semanticErrorLightingGroup	= new TestCaseGroup(context, "lighting", "Semantic Error Shader Compiler Lighting Cases");
2847 		TestCaseGroup* cacheLightingGroup			= new TestCaseGroup(context, "lighting", "Shader Compiler Lighting Cache Cases");
2848 		validGroup->addChild(validLightingGroup);
2849 		invalidCharGroup->addChild(invalidCharLightingGroup);
2850 		semanticErrorGroup->addChild(semanticErrorLightingGroup);
2851 		cacheGroup->addChild(cacheLightingGroup);
2852 
2853 		for (int lightType = 0; lightType < (int)LIGHT_LAST; lightType++)
2854 		{
2855 			const char* lightTypeName = lightType == (int)LIGHT_DIRECTIONAL	? "directional"
2856 									  : lightType == (int)LIGHT_POINT		? "point"
2857 									  : DE_NULL;
2858 
2859 			DE_ASSERT(lightTypeName != DE_NULL);
2860 
2861 			for (int isFrag = 0; isFrag <= 1; isFrag++)
2862 			{
2863 				bool		isVertex	= isFrag == 0;
2864 				const char*	vertFragStr	= isVertex ? "vertex" : "fragment";
2865 
2866 				for (int lightCountNdx = 0; lightCountNdx < DE_LENGTH_OF_ARRAY(lightCounts); lightCountNdx++)
2867 				{
2868 					int numLights = lightCounts[lightCountNdx];
2869 
2870 					string caseName = string("") + lightTypeName + "_" + de::toString(numLights) + "_lights_" + vertFragStr;
2871 
2872 					// Valid shader case, no-cache and cache versions.
2873 
2874 					validLightingGroup->addChild(new ShaderCompilerLightCase(context, caseName.c_str(), "", caseID++, true  /* avoid cache */, false, isVertex, numLights, (LightType)lightType));
2875 					cacheLightingGroup->addChild(new ShaderCompilerLightCase(context, caseName.c_str(), "", caseID++, false /* allow cache */, false, isVertex, numLights, (LightType)lightType));
2876 
2877 					// Invalid shader cases.
2878 
2879 					for (int invalidityType = 0; invalidityType < (int)InvalidShaderCompilerCase::INVALIDITY_LAST; invalidityType++)
2880 					{
2881 						TestCaseGroup* curInvalidGroup	= invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_INVALID_CHAR		? invalidCharLightingGroup
2882 														: invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_SEMANTIC_ERROR	? semanticErrorLightingGroup
2883 														: DE_NULL;
2884 
2885 						DE_ASSERT(curInvalidGroup != DE_NULL);
2886 
2887 						curInvalidGroup->addChild(new InvalidShaderCompilerLightCase(context, caseName.c_str(), "", caseID++, (InvalidShaderCompilerCase::InvalidityType)invalidityType, isVertex, numLights, (LightType)lightType));
2888 					}
2889 				}
2890 			}
2891 		}
2892 	}
2893 
2894 	// Texture lookup shader compilation cases.
2895 
2896 	{
2897 		static const int texLookupCounts[] = { 1, 2, 4, 8 };
2898 
2899 		TestCaseGroup* validTexGroup			= new TestCaseGroup(context, "texture", "Shader Compiler Texture Lookup Cases");
2900 		TestCaseGroup* invalidCharTexGroup		= new TestCaseGroup(context, "texture", "Invalid Character Shader Compiler Texture Lookup Cases");
2901 		TestCaseGroup* semanticErrorTexGroup	= new TestCaseGroup(context, "texture", "Semantic Error Shader Compiler Texture Lookup Cases");
2902 		TestCaseGroup* cacheTexGroup			= new TestCaseGroup(context, "texture", "Shader Compiler Texture Lookup Cache Cases");
2903 		validGroup->addChild(validTexGroup);
2904 		invalidCharGroup->addChild(invalidCharTexGroup);
2905 		semanticErrorGroup->addChild(semanticErrorTexGroup);
2906 		cacheGroup->addChild(cacheTexGroup);
2907 
2908 		for (int conditionalUsage = 0; conditionalUsage < (int)CONDITIONAL_USAGE_LAST; conditionalUsage++)
2909 		{
2910 			const char* conditionalUsageName = conditionalUsage == (int)CONDITIONAL_USAGE_NONE			? "no_conditionals"
2911 											 : conditionalUsage == (int)CONDITIONAL_USAGE_FIRST_HALF	? "first_half"
2912 											 : conditionalUsage == (int)CONDITIONAL_USAGE_EVERY_OTHER	? "every_other"
2913 											 : DE_NULL;
2914 
2915 			DE_ASSERT(conditionalUsageName != DE_NULL);
2916 
2917 			int lastConditionalType = conditionalUsage == (int)CONDITIONAL_USAGE_NONE ? 1 : (int)CONDITIONAL_TYPE_LAST;
2918 
2919 			for (int conditionalType = 0; conditionalType < lastConditionalType; conditionalType++)
2920 			{
2921 				const char* conditionalTypeName = conditionalType == (int)CONDITIONAL_TYPE_STATIC	? "static_conditionals"
2922 												: conditionalType == (int)CONDITIONAL_TYPE_UNIFORM	? "uniform_conditionals"
2923 												: conditionalType == (int)CONDITIONAL_TYPE_DYNAMIC	? "dynamic_conditionals"
2924 												: DE_NULL;
2925 
2926 				DE_ASSERT(conditionalTypeName != DE_NULL);
2927 
2928 				for (int lookupCountNdx = 0; lookupCountNdx < DE_LENGTH_OF_ARRAY(texLookupCounts); lookupCountNdx++)
2929 				{
2930 					int numLookups = texLookupCounts[lookupCountNdx];
2931 
2932 					string caseName = de::toString(numLookups) + "_lookups_" + conditionalUsageName + (conditionalUsage == (int)CONDITIONAL_USAGE_NONE ? "" : string("_") + conditionalTypeName);
2933 
2934 					// Valid shader case, no-cache and cache versions.
2935 
2936 					validTexGroup->addChild(new ShaderCompilerTextureCase(context, caseName.c_str(), "", caseID++, true  /* avoid cache */, false, numLookups, (ConditionalUsage)conditionalUsage, (ConditionalType)conditionalType));
2937 					cacheTexGroup->addChild(new ShaderCompilerTextureCase(context, caseName.c_str(), "", caseID++, false /* allow cache */, false, numLookups, (ConditionalUsage)conditionalUsage, (ConditionalType)conditionalType));
2938 
2939 					// Invalid shader cases.
2940 
2941 					for (int invalidityType = 0; invalidityType < (int)InvalidShaderCompilerCase::INVALIDITY_LAST; invalidityType++)
2942 					{
2943 						TestCaseGroup* curInvalidGroup	= invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_INVALID_CHAR		? invalidCharTexGroup
2944 														: invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_SEMANTIC_ERROR	? semanticErrorTexGroup
2945 														: DE_NULL;
2946 
2947 						DE_ASSERT(curInvalidGroup != DE_NULL);
2948 
2949 						curInvalidGroup->addChild(new InvalidShaderCompilerTextureCase(context, caseName.c_str(), "", caseID++, (InvalidShaderCompilerCase::InvalidityType)invalidityType, numLookups, (ConditionalUsage)conditionalUsage, (ConditionalType)conditionalType));
2950 					}
2951 				}
2952 			}
2953 		}
2954 	}
2955 
2956 	// Loop shader compilation cases.
2957 
2958 	{
2959 		static const int loopIterCounts[]		= { 10, 100, 1000 };
2960 		static const int maxLoopNestingDepth	= 3;
2961 		static const int maxTotalLoopIterations	= 2000; // If <loop iteration count> ** <loop nesting depth> (where ** is exponentiation) exceeds this, don't generate the case.
2962 
2963 		TestCaseGroup* validLoopGroup			= new TestCaseGroup(context, "loop", "Shader Compiler Loop Cases");
2964 		TestCaseGroup* invalidCharLoopGroup		= new TestCaseGroup(context, "loop", "Invalid Character Shader Compiler Loop Cases");
2965 		TestCaseGroup* semanticErrorLoopGroup	= new TestCaseGroup(context, "loop", "Semantic Error Shader Compiler Loop Cases");
2966 		TestCaseGroup* cacheLoopGroup			= new TestCaseGroup(context, "loop", "Shader Compiler Loop Cache Cases");
2967 		validGroup->addChild(validLoopGroup);
2968 		invalidCharGroup->addChild(invalidCharLoopGroup);
2969 		semanticErrorGroup->addChild(semanticErrorLoopGroup);
2970 		cacheGroup->addChild(cacheLoopGroup);
2971 
2972 		for (int loopType = 0; loopType < (int)LOOP_LAST; loopType++)
2973 		{
2974 			const char* loopTypeName = loopType == (int)LOOP_TYPE_STATIC	? "static"
2975 									 : loopType == (int)LOOP_TYPE_UNIFORM	? "uniform"
2976 									 : loopType == (int)LOOP_TYPE_DYNAMIC	? "dynamic"
2977 									 : DE_NULL;
2978 
2979 			DE_ASSERT(loopTypeName != DE_NULL);
2980 
2981 			TestCaseGroup* validLoopTypeGroup			= new TestCaseGroup(context, loopTypeName, "");
2982 			TestCaseGroup* invalidCharLoopTypeGroup		= new TestCaseGroup(context, loopTypeName, "");
2983 			TestCaseGroup* semanticErrorLoopTypeGroup	= new TestCaseGroup(context, loopTypeName, "");
2984 			TestCaseGroup* cacheLoopTypeGroup			= new TestCaseGroup(context, loopTypeName, "");
2985 			validLoopGroup->addChild(validLoopTypeGroup);
2986 			invalidCharLoopGroup->addChild(invalidCharLoopTypeGroup);
2987 			semanticErrorLoopGroup->addChild(semanticErrorLoopTypeGroup);
2988 			cacheLoopGroup->addChild(cacheLoopTypeGroup);
2989 
2990 			for (int isFrag = 0; isFrag <= 1; isFrag++)
2991 			{
2992 				bool		isVertex	= isFrag == 0;
2993 				const char*	vertFragStr	= isVertex ? "vertex" : "fragment";
2994 
2995 				// \note Non-static loop cases with different iteration counts have identical shaders, so only make one of each.
2996 				int loopIterCountMaxNdx = loopType != (int)LOOP_TYPE_STATIC ? 1 : DE_LENGTH_OF_ARRAY(loopIterCounts);
2997 
2998 				for (int nestingDepth = 1; nestingDepth <= maxLoopNestingDepth; nestingDepth++)
2999 				{
3000 					for (int loopIterCountNdx = 0; loopIterCountNdx < loopIterCountMaxNdx; loopIterCountNdx++)
3001 					{
3002 						int numIterations = loopIterCounts[loopIterCountNdx];
3003 
3004 						if (deFloatPow((float)numIterations, (float)nestingDepth) > (float)maxTotalLoopIterations)
3005 							continue; // Don't generate too heavy tasks.
3006 
3007 						string validCaseName = de::toString(numIterations) + "_iterations_" + de::toString(nestingDepth) + "_levels_" + vertFragStr;
3008 
3009 						// Valid shader case, no-cache and cache versions.
3010 
3011 						validLoopTypeGroup->addChild(new ShaderCompilerLoopCase(context, validCaseName.c_str(), "", caseID++, true  /* avoid cache */, false, isVertex, (LoopType)loopType, numIterations, nestingDepth));
3012 						cacheLoopTypeGroup->addChild(new ShaderCompilerLoopCase(context, validCaseName.c_str(), "", caseID++, false /* allow cache */, false, isVertex, (LoopType)loopType, numIterations, nestingDepth));
3013 
3014 						// Invalid shader cases.
3015 
3016 						for (int invalidityType = 0; invalidityType < (int)InvalidShaderCompilerCase::INVALIDITY_LAST; invalidityType++)
3017 						{
3018 							TestCaseGroup* curInvalidGroup	= invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_INVALID_CHAR		? invalidCharLoopTypeGroup
3019 															: invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_SEMANTIC_ERROR	? semanticErrorLoopTypeGroup
3020 															: DE_NULL;
3021 
3022 							DE_ASSERT(curInvalidGroup != DE_NULL);
3023 
3024 							string invalidCaseName = de::toString(nestingDepth) + "_levels_" + vertFragStr;
3025 
3026 							if (loopType == (int)LOOP_TYPE_STATIC)
3027 								invalidCaseName = de::toString(numIterations) + "_iterations_" + invalidCaseName; // \note For invalid, non-static loop cases the iteration count means nothing (since no uniforms or attributes are set).
3028 
3029 							curInvalidGroup->addChild(new InvalidShaderCompilerLoopCase(context, invalidCaseName.c_str(), "", caseID++, (InvalidShaderCompilerCase::InvalidityType)invalidityType, isVertex, (LoopType)loopType, numIterations, nestingDepth));
3030 						}
3031 					}
3032 				}
3033 			}
3034 		}
3035 	}
3036 
3037 	// Multiplication shader compilation cases.
3038 
3039 	{
3040 		static const int multiplicationCounts[] = { 10, 100, 1000 };
3041 
3042 		TestCaseGroup* validMulGroup			= new TestCaseGroup(context, "multiplication", "Shader Compiler Multiplication Cases");
3043 		TestCaseGroup* invalidCharMulGroup		= new TestCaseGroup(context, "multiplication", "Invalid Character Shader Compiler Multiplication Cases");
3044 		TestCaseGroup* semanticErrorMulGroup	= new TestCaseGroup(context, "multiplication", "Semantic Error Shader Compiler Multiplication Cases");
3045 		TestCaseGroup* cacheMulGroup			= new TestCaseGroup(context, "multiplication", "Shader Compiler Multiplication Cache Cases");
3046 		validGroup->addChild(validMulGroup);
3047 		invalidCharGroup->addChild(invalidCharMulGroup);
3048 		semanticErrorGroup->addChild(semanticErrorMulGroup);
3049 		cacheGroup->addChild(cacheMulGroup);
3050 
3051 		for (int isFrag = 0; isFrag <= 1; isFrag++)
3052 		{
3053 			bool		isVertex	= isFrag == 0;
3054 			const char*	vertFragStr	= isVertex ? "vertex" : "fragment";
3055 
3056 			for (int operCountNdx = 0; operCountNdx < DE_LENGTH_OF_ARRAY(multiplicationCounts); operCountNdx++)
3057 			{
3058 				int numOpers = multiplicationCounts[operCountNdx];
3059 
3060 				string caseName = de::toString(numOpers) + "_operations_" + vertFragStr;
3061 
3062 				// Valid shader case, no-cache and cache versions.
3063 
3064 				validMulGroup->addChild(new ShaderCompilerOperCase(context, caseName.c_str(), "", caseID++, true  /* avoid cache */, false, isVertex, "*", numOpers));
3065 				cacheMulGroup->addChild(new ShaderCompilerOperCase(context, caseName.c_str(), "", caseID++, false /* allow cache */, false, isVertex, "*", numOpers));
3066 
3067 				// Invalid shader cases.
3068 
3069 				for (int invalidityType = 0; invalidityType < (int)InvalidShaderCompilerCase::INVALIDITY_LAST; invalidityType++)
3070 				{
3071 					TestCaseGroup* curInvalidGroup	= invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_INVALID_CHAR		? invalidCharMulGroup
3072 													: invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_SEMANTIC_ERROR	? semanticErrorMulGroup
3073 													: DE_NULL;
3074 
3075 					DE_ASSERT(curInvalidGroup != DE_NULL);
3076 
3077 					curInvalidGroup->addChild(new InvalidShaderCompilerOperCase(context, caseName.c_str(), "", caseID++, (InvalidShaderCompilerCase::InvalidityType)invalidityType, isVertex, "*", numOpers));
3078 				}
3079 			}
3080 		}
3081 	}
3082 
3083 	// Mandelbrot shader compilation cases.
3084 
3085 	{
3086 		static const int mandelbrotIterationCounts[] = { 32, 64, 128 };
3087 
3088 		TestCaseGroup* validMandelbrotGroup			= new TestCaseGroup(context, "mandelbrot", "Shader Compiler Mandelbrot Fractal Cases");
3089 		TestCaseGroup* invalidCharMandelbrotGroup	= new TestCaseGroup(context, "mandelbrot", "Invalid Character Shader Compiler Mandelbrot Fractal Cases");
3090 		TestCaseGroup* semanticErrorMandelbrotGroup	= new TestCaseGroup(context, "mandelbrot", "Semantic Error Shader Compiler Mandelbrot Fractal Cases");
3091 		TestCaseGroup* cacheMandelbrotGroup			= new TestCaseGroup(context, "mandelbrot", "Shader Compiler Mandelbrot Fractal Cache Cases");
3092 		validGroup->addChild(validMandelbrotGroup);
3093 		invalidCharGroup->addChild(invalidCharMandelbrotGroup);
3094 		semanticErrorGroup->addChild(semanticErrorMandelbrotGroup);
3095 		cacheGroup->addChild(cacheMandelbrotGroup);
3096 
3097 		for (int iterCountNdx = 0; iterCountNdx < DE_LENGTH_OF_ARRAY(mandelbrotIterationCounts); iterCountNdx++)
3098 		{
3099 			int		numFractalIterations	= mandelbrotIterationCounts[iterCountNdx];
3100 			string	caseName				= de::toString(numFractalIterations) + "_iterations";
3101 
3102 			// Valid shader case, no-cache and cache versions.
3103 
3104 			validMandelbrotGroup->addChild(new ShaderCompilerMandelbrotCase(context, caseName.c_str(), "", caseID++, true  /* avoid cache */, false, numFractalIterations));
3105 			cacheMandelbrotGroup->addChild(new ShaderCompilerMandelbrotCase(context, caseName.c_str(), "", caseID++, false /* allow cache */, false, numFractalIterations));
3106 
3107 			// Invalid shader cases.
3108 
3109 			for (int invalidityType = 0; invalidityType < (int)InvalidShaderCompilerCase::INVALIDITY_LAST; invalidityType++)
3110 			{
3111 				TestCaseGroup* curInvalidGroup	= invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_INVALID_CHAR		? invalidCharMandelbrotGroup
3112 												: invalidityType == (int)InvalidShaderCompilerCase::INVALIDITY_SEMANTIC_ERROR	? semanticErrorMandelbrotGroup
3113 												: DE_NULL;
3114 
3115 				DE_ASSERT(curInvalidGroup != DE_NULL);
3116 
3117 				curInvalidGroup->addChild(new InvalidShaderCompilerMandelbrotCase(context, caseName.c_str(), "", caseID++, (InvalidShaderCompilerCase::InvalidityType)invalidityType, numFractalIterations));
3118 			}
3119 		}
3120 	}
3121 
3122 	// Cases testing cache behaviour when whitespace and comments are added.
3123 
3124 	{
3125 		TestCaseGroup* whitespaceCommentCacheGroup = new TestCaseGroup(context, "cache_whitespace_comment", "Cases testing the effect of whitespace and comments on caching");
3126 		parentGroup.addChild(whitespaceCommentCacheGroup);
3127 
3128 		// \note Add just a small subset of the cases that were added above for the main performance tests.
3129 
3130 		// Cases with both vertex and fragment variants.
3131 		for (int isFrag = 0; isFrag <= 1; isFrag++)
3132 		{
3133 			bool	isVertex		= isFrag == 0;
3134 			string	vtxFragSuffix	= isVertex ? "_vertex" : "_fragment";
3135 			string	dirLightName	= "directional_2_lights" + vtxFragSuffix;
3136 			string	loopName		= "static_loop_100_iterations" + vtxFragSuffix;
3137 			string	multCase		= "multiplication_100_operations" + vtxFragSuffix;
3138 
3139 			whitespaceCommentCacheGroup->addChild(new ShaderCompilerLightCase(context, dirLightName.c_str(), "", caseID++, false, true, isVertex, 2, LIGHT_DIRECTIONAL));
3140 			whitespaceCommentCacheGroup->addChild(new ShaderCompilerLoopCase(context, loopName.c_str(), "", caseID++, false, true, isVertex, LOOP_TYPE_STATIC, 100, 1));
3141 			whitespaceCommentCacheGroup->addChild(new ShaderCompilerOperCase(context, multCase.c_str(), "", caseID++, false, true, isVertex, "*", 100));
3142 		}
3143 
3144 		// Cases that don't have vertex and fragment variants.
3145 		whitespaceCommentCacheGroup->addChild(new ShaderCompilerTextureCase(context, "texture_4_lookups", "", caseID++, false, true, 4, CONDITIONAL_USAGE_NONE, CONDITIONAL_TYPE_STATIC));
3146 		whitespaceCommentCacheGroup->addChild(new ShaderCompilerMandelbrotCase(context, "mandelbrot_32_operations", "", caseID++, false, true, 32));
3147 	}
3148 }
3149 
3150 } // Performance
3151 } // gles3
3152 } // deqp
3153