1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program OpenGL ES 2.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 Texture upload performance tests.
22  *
23  * \todo [2012-10-01 pyry]
24  *  - Test different pixel unpack alignments
25  *  - Use multiple textures
26  *  - Trash cache prior to uploading from data ptr
27  *//*--------------------------------------------------------------------*/
28 
29 #include "es2pTextureUploadTests.hpp"
30 #include "tcuTexture.hpp"
31 #include "tcuTextureUtil.hpp"
32 #include "tcuTestLog.hpp"
33 #include "tcuSurface.hpp"
34 #include "gluTextureUtil.hpp"
35 #include "gluShaderProgram.hpp"
36 #include "gluPixelTransfer.hpp"
37 #include "deStringUtil.hpp"
38 #include "deRandom.hpp"
39 #include "deClock.h"
40 #include "deString.h"
41 
42 #include "glsCalibration.hpp"
43 
44 #include "glwEnums.hpp"
45 #include "glwFunctions.hpp"
46 
47 #include <algorithm>
48 #include <vector>
49 
50 namespace deqp
51 {
52 namespace gles2
53 {
54 namespace Performance
55 {
56 
57 using tcu::Vec2;
58 using tcu::Vec3;
59 using tcu::Vec4;
60 using tcu::IVec4;
61 using std::string;
62 using std::vector;
63 using tcu::TestLog;
64 using tcu::TextureFormat;
65 using namespace glw; // GL types
66 
67 static const int	VIEWPORT_SIZE	= 64;
68 static const float	quadCoords[] =
69 {
70 	-1.0f, -1.0f,
71 	 1.0f, -1.0f,
72 	-1.0f,  1.0f,
73 	 1.0f,  1.0f
74 };
75 
76 class TextureUploadCase : public TestCase
77 {
78 public:
79 								TextureUploadCase	(Context& context, const char* name, const char* description, UploadFunction uploadFunction, deUint32 format, deUint32 type, int texSize);
80 								~TextureUploadCase	(void);
81 
82 	virtual void				init				(void);
83 	void						deinit				(void);
84 
85 	virtual IterateResult		iterate				(void) = 0;
86 	void						logResults			(void);
87 
88 protected:
89 	UploadFunction				m_uploadFunction;
90 	deUint32					m_format;
91 	deUint32					m_type;
92 	int							m_texSize;
93 	int							m_alignment;
94 
95 	gls::TheilSenCalibrator		m_calibrator;
96 	glu::ShaderProgram*			m_program;
97 	deUint32					m_texture;
98 	de::Random					m_rnd;
99 	TestLog&					m_log;
100 
101 	vector<deUint8>				m_texData;
102 };
103 
TextureUploadCase(Context & context,const char * name,const char * description,UploadFunction uploadFunction,deUint32 format,deUint32 type,int texSize)104 TextureUploadCase::TextureUploadCase (Context& context, const char* name, const char* description, UploadFunction uploadFunction, deUint32 format, deUint32 type, int texSize)
105 	: TestCase			(context, tcu::NODETYPE_PERFORMANCE, name, description)
106 	, m_uploadFunction	(uploadFunction)
107 	, m_format			(format)
108 	, m_type			(type)
109 	, m_texSize			(texSize)
110 	, m_alignment		(4)
111 	, m_calibrator		()
112 	, m_program			(DE_NULL)
113 	, m_texture			(0)
114 	, m_rnd				(deStringHash(name))
115 	, m_log				(context.getTestContext().getLog())
116 {
117 }
118 
~TextureUploadCase(void)119 TextureUploadCase::~TextureUploadCase (void)
120 {
121 	TextureUploadCase::deinit();
122 }
123 
deinit(void)124 void TextureUploadCase::deinit (void)
125 {
126 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
127 
128 	if (m_program)
129 	{
130 		delete m_program;
131 		m_program = DE_NULL;
132 	}
133 
134 	gl.deleteTextures(1, &m_texture);
135 	m_texture = 0;
136 
137 	m_texData.clear();
138 }
139 
init(void)140 void TextureUploadCase::init (void)
141 {
142 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
143 	int maxTextureSize;
144 	gl.getIntegerv(GL_MAX_TEXTURE_SIZE, &maxTextureSize);
145 
146 	if (m_texSize > maxTextureSize)
147 	{
148 		m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Unsupported texture size");
149 		return;
150 	}
151 
152 	// Create program
153 
154 	string vertexShaderSource = "";
155 	string fragmentShaderSource = "";
156 
157 	vertexShaderSource.append(	"precision mediump	float;\n"
158 								"attribute vec2		a_pos;\n"
159 								"varying   vec2		v_texCoord;\n"
160 								"\n"
161 								"void main (void)\n"
162 								"{\n"
163 								"	v_texCoord	= a_pos;\n"
164 								"	gl_Position = vec4(a_pos, 0.5, 1.0);\n"
165 								"}\n");
166 
167 	fragmentShaderSource.append("precision	mediump	float;\n"
168 								"uniform	lowp sampler2D	u_sampler;\n"
169 								"varying	vec2			v_texCoord;\n"
170 								"\n"
171 								"void main (void)\n"
172 								"{\n"
173 								"	gl_FragColor = texture2D(u_sampler, v_texCoord.xy);\n"
174 								"}\n");
175 
176 	DE_ASSERT(!m_program);
177 	m_program = new glu::ShaderProgram(m_context.getRenderContext(), glu::makeVtxFragSources(vertexShaderSource, fragmentShaderSource));
178 
179 	if (!m_program->isOk())
180 	{
181 		m_log << *m_program;
182 		TCU_FAIL("Failed to create shader program (m_programRender)");
183 	}
184 
185 	gl.useProgram (m_program->getProgram());
186 
187 	// Init GL state
188 
189 	gl.viewport		(0, 0, VIEWPORT_SIZE, VIEWPORT_SIZE);
190 	gl.disable		(GL_DEPTH_TEST);
191 	gl.disable		(GL_CULL_FACE);
192 	gl.enable		(GL_BLEND);
193 	gl.blendFunc	(GL_ONE, GL_ONE);
194 	gl.clearColor	(0.0f, 0.0f, 0.0f, 1.0f);
195 	gl.clear		(GL_COLOR_BUFFER_BIT);
196 
197 	deUint32 uSampler	= gl.getUniformLocation(m_program->getProgram(), "u_sampler");
198 	deUint32 aPos		= gl.getAttribLocation (m_program->getProgram(), "a_pos");
199 	gl.enableVertexAttribArray	(aPos);
200 	gl.vertexAttribPointer		(aPos,	2, GL_FLOAT, GL_FALSE, 0, &quadCoords[0]);
201 	gl.uniform1i				(uSampler, 0);
202 
203 	// Create texture
204 
205 	gl.activeTexture	(GL_TEXTURE0);
206 	gl.genTextures		(1, &m_texture);
207 	gl.bindTexture		(GL_TEXTURE_2D, m_texture);
208 	gl.pixelStorei		(GL_UNPACK_ALIGNMENT, m_alignment);
209 	gl.texParameteri	(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
210 	gl.texParameteri	(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
211 	gl.texParameteri	(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
212 	gl.texParameteri	(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
213 
214 	// Prepare texture data
215 
216 	{
217 		const tcu::TextureFormat&	texFmt		= glu::mapGLTransferFormat(m_format, m_type);
218 		int							pixelSize	= texFmt.getPixelSize();
219 		int							stride		= deAlign32(pixelSize*m_texSize, m_alignment);
220 
221 		m_texData.resize(stride*m_texSize);
222 
223 		tcu::PixelBufferAccess		access		(texFmt, m_texSize, m_texSize, 1, stride, 0, &m_texData[0]);
224 
225 		tcu::fillWithComponentGradients(access, tcu::Vec4(0.0f, 0.0f, 0.0f, 0.0f), tcu::Vec4(1.0f, 1.0f, 1.0f, 1.0f));
226 	}
227 
228 	// Do a dry-run to ensure the pipes are hot
229 
230 	gl.texImage2D	(GL_TEXTURE_2D, 0, m_format, m_texSize, m_texSize, 0, m_format, m_type, &m_texData[0]);
231 	gl.drawArrays	(GL_TRIANGLE_STRIP, 0, 4);
232 	gl.finish		();
233 }
234 
logResults(void)235 void TextureUploadCase::logResults (void)
236 {
237 	const gls::MeasureState& measureState = m_calibrator.getMeasureState();
238 
239 	// Log measurement details
240 
241 	m_log << TestLog::Section("Measurement details", "Measurement details");
242 	m_log << TestLog::Message << "Uploading texture with " << (m_uploadFunction == UPLOAD_TEXIMAGE2D ? "glTexImage2D" : "glTexSubImage2D") << "." << TestLog::EndMessage; // \todo [arttu] Change enum to struct with name included
243 	m_log << TestLog::Message << "Texture size = "	<< m_texSize	 << "x" << m_texSize	 << "." << TestLog::EndMessage;
244 	m_log << TestLog::Message << "Viewport size = " << VIEWPORT_SIZE << "x" << VIEWPORT_SIZE << "." << TestLog::EndMessage;
245 	m_log << TestLog::Message << measureState.numDrawCalls << " upload calls / iteration" << TestLog::EndMessage;
246 	m_log << TestLog::EndSection;
247 
248 	// Log results
249 
250 	TestLog& log = m_testCtx.getLog();
251 	log << TestLog::Section("Results", "Results");
252 
253 	// Log individual frame durations
254 	//for (int i = 0; i < m_calibrator.measureState.numFrames; i++)
255 	//	m_log << TestLog::Message	<< "Frame "	<< i+1 << " duration: \t" << m_calibrator.measureState.frameTimes[i] << " us."<< TestLog::EndMessage;
256 
257 	std::vector<deUint64> sortedFrameTimes(measureState.frameTimes.begin(), measureState.frameTimes.end());
258 	std::sort(sortedFrameTimes.begin(), sortedFrameTimes.end());
259 	vector<deUint64>::const_iterator first	= sortedFrameTimes.begin();
260 	vector<deUint64>::const_iterator last	= sortedFrameTimes.end();
261 	vector<deUint64>::const_iterator middle	= first + (last - first) / 2;
262 
263 	deUint64 medianFrameTime			=  *middle;
264 	double medianMTexelsPerSeconds		= (double)(m_texSize*m_texSize*measureState.numDrawCalls) / medianFrameTime;
265 	double medianTexelDrawDurationNs	= (double)medianFrameTime * 1000.0 / (double)(m_texSize*m_texSize*measureState.numDrawCalls);
266 
267 	deUint64	totalTime			= measureState.getTotalTime();
268 	int			numFrames			= (int)measureState.frameTimes.size();
269 	deInt64		numTexturesDrawn	= measureState.numDrawCalls * numFrames;
270 	deInt64		numPixels			= (deInt64)m_texSize * (deInt64)m_texSize * numTexturesDrawn;
271 
272 	double		framesPerSecond			= (double)numFrames / ((double)totalTime / 1000000.0);
273 	double		avgFrameTime			= (double)totalTime / (double)numFrames;
274 	double		avgMTexelsPerSeconds	= (double)numPixels / (double)totalTime;
275 	double		avgTexelDrawDurationNs	= (double)totalTime * 1000.0 / (double)numPixels;
276 
277 	log << TestLog::Float("FramesPerSecond",	"Frames per second in measurement\t\t",		"Frames/s",		QP_KEY_TAG_PERFORMANCE,	(float)framesPerSecond);
278 	log << TestLog::Float("AverageFrameTime",	"Average frame duration in measurement\t",	"us",			QP_KEY_TAG_PERFORMANCE,	(float)avgFrameTime);
279 	log << TestLog::Float("AverageTexelPerf",	"Average texel upload performance\t\t",		"MTex/s",		QP_KEY_TAG_PERFORMANCE,	(float)avgMTexelsPerSeconds);
280 	log << TestLog::Float("AverageTexelTime",	"Average texel upload duration\t\t",		"ns",			QP_KEY_TAG_PERFORMANCE,	(float)avgTexelDrawDurationNs);
281 	log << TestLog::Float("MedianTexelPerf",	"Median texel upload performance\t\t",		"MTex/s",		QP_KEY_TAG_PERFORMANCE,	(float)medianMTexelsPerSeconds);
282 	log << TestLog::Float("MedianTexelTime",	"Median texel upload duration\t\t",			"ns",			QP_KEY_TAG_PERFORMANCE,	(float)medianTexelDrawDurationNs);
283 
284 	log << TestLog::EndSection;
285 
286 	gls::logCalibrationInfo(log, m_calibrator);	// Log calibration details
287 	m_testCtx.setTestResult(QP_TEST_RESULT_PASS, de::floatToString((float)avgMTexelsPerSeconds, 2).c_str());
288 }
289 
290 // Texture upload call case
291 
292 class TextureUploadCallCase : public TextureUploadCase
293 {
294 public:
295 							TextureUploadCallCase	(Context& context, const char* name, const char* description, UploadFunction uploadFunction, deUint32 format, deUint32 type, int texSize);
296 							~TextureUploadCallCase	(void);
297 
298 	IterateResult			iterate					(void);
299 	void					render					(void);
300 };
301 
TextureUploadCallCase(Context & context,const char * name,const char * description,UploadFunction uploadFunction,deUint32 format,deUint32 type,int texSize)302 TextureUploadCallCase::TextureUploadCallCase (Context& context, const char* name, const char* description, UploadFunction uploadFunction, deUint32 format, deUint32 type, int texSize)
303 	: TextureUploadCase (context, name, description, uploadFunction, format, type, texSize)
304 {
305 }
306 
~TextureUploadCallCase(void)307 TextureUploadCallCase::~TextureUploadCallCase (void)
308 {
309 	TextureUploadCase::deinit();
310 }
311 
render(void)312 void TextureUploadCallCase::render (void)
313 {
314 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
315 
316 	// Draw multiple quads to ensure enough workload
317 
318 	switch (m_uploadFunction)
319 	{
320 		case UPLOAD_TEXIMAGE2D:
321 			gl.texImage2D(GL_TEXTURE_2D, 0, m_format, m_texSize, m_texSize, 0, m_format, m_type, &m_texData[0]);
322 			break;
323 		case UPLOAD_TEXSUBIMAGE2D:
324 			gl.texSubImage2D(GL_TEXTURE_2D, 0, 0, 0, m_texSize, m_texSize, m_format, m_type, &m_texData[0]);
325 			break;
326 		default:
327 			DE_ASSERT(false);
328 	}
329 }
330 
iterate(void)331 tcu::TestNode::IterateResult TextureUploadCallCase::iterate (void)
332 {
333 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
334 
335 	if (m_testCtx.getTestResult() == QP_TEST_RESULT_NOT_SUPPORTED)
336 		return STOP;
337 
338 	for (;;)
339 	{
340 		gls::TheilSenCalibrator::State state = m_calibrator.getState();
341 
342 		if (state == gls::TheilSenCalibrator::STATE_MEASURE)
343 		{
344 			int			numCalls	= m_calibrator.getCallCount();
345 			deUint64	startTime	= deGetMicroseconds();
346 
347 			for (int i = 0; i < numCalls; i++)
348 				render();
349 
350 			gl.finish();
351 
352 			deUint64	endTime		= deGetMicroseconds();
353 			deUint64	duration	= endTime-startTime;
354 
355 			m_calibrator.recordIteration(duration);
356 		}
357 		else if (state == gls::TheilSenCalibrator::STATE_RECOMPUTE_PARAMS)
358 		{
359 			m_calibrator.recomputeParameters();
360 		}
361 		else
362 		{
363 			DE_ASSERT(state == gls::TheilSenCalibrator::STATE_FINISHED);
364 			break;
365 		}
366 
367 		// Touch watchdog between iterations to avoid timeout.
368 		{
369 			qpWatchDog* dog = m_testCtx.getWatchDog();
370 			if (dog)
371 				qpWatchDog_touch(dog);
372 		}
373 	}
374 
375 	GLU_EXPECT_NO_ERROR(gl.getError(), "iterate");
376 	logResults();
377 	return STOP;
378 }
379 
380 // Texture upload and draw case
381 
382 class TextureUploadAndDrawCase : public TextureUploadCase
383 {
384 public:
385 								TextureUploadAndDrawCase	(Context& context, const char* name, const char* description, UploadFunction uploadFunction, deUint32 format, deUint32 type, int texSize);
386 								~TextureUploadAndDrawCase	(void);
387 
388 	IterateResult				iterate						(void);
389 	void						render						(void);
390 
391 private:
392 	bool						m_lastIterationRender;
393 	deUint64					m_renderStart;
394 };
395 
TextureUploadAndDrawCase(Context & context,const char * name,const char * description,UploadFunction uploadFunction,deUint32 format,deUint32 type,int texSize)396 TextureUploadAndDrawCase::TextureUploadAndDrawCase (Context& context, const char* name, const char* description, UploadFunction uploadFunction, deUint32 format, deUint32 type, int texSize)
397 	: TextureUploadCase		(context, name, description, uploadFunction, format, type, texSize)
398 	, m_lastIterationRender	(false)
399 	, m_renderStart			(0)
400 {
401 }
402 
~TextureUploadAndDrawCase(void)403 TextureUploadAndDrawCase::~TextureUploadAndDrawCase (void)
404 {
405 	TextureUploadCase::deinit();
406 }
407 
render(void)408 void TextureUploadAndDrawCase::render (void)
409 {
410 	const glw::Functions& gl = m_context.getRenderContext().getFunctions();
411 
412 	// Draw multiple quads to ensure enough workload
413 
414 	switch (m_uploadFunction)
415 	{
416 		case UPLOAD_TEXIMAGE2D:
417 			gl.texImage2D(GL_TEXTURE_2D, 0, m_format, m_texSize, m_texSize, 0, m_format, m_type, &m_texData[0]);
418 			break;
419 		case UPLOAD_TEXSUBIMAGE2D:
420 			gl.texSubImage2D(GL_TEXTURE_2D, 0, 0, 0, m_texSize, m_texSize, m_format, m_type, &m_texData[0]);
421 			break;
422 		default:
423 			DE_ASSERT(false);
424 	}
425 
426 	gl.drawArrays(GL_TRIANGLE_STRIP, 0, 4);
427 }
428 
iterate(void)429 tcu::TestNode::IterateResult TextureUploadAndDrawCase::iterate (void)
430 {
431 	if (m_testCtx.getTestResult() == QP_TEST_RESULT_NOT_SUPPORTED)
432 		return STOP;
433 
434 	if (m_lastIterationRender && (m_calibrator.getState() == gls::TheilSenCalibrator::STATE_MEASURE))
435 	{
436 		deUint64 curTime = deGetMicroseconds();
437 		m_calibrator.recordIteration(curTime - m_renderStart);
438 	}
439 
440 	gls::TheilSenCalibrator::State state = m_calibrator.getState();
441 
442 	if (state == gls::TheilSenCalibrator::STATE_MEASURE)
443 	{
444 		// Render
445 		int	numCalls = m_calibrator.getCallCount();
446 
447 		m_renderStart			= deGetMicroseconds();
448 		m_lastIterationRender	= true;
449 
450 		for (int i = 0; i < numCalls; i++)
451 			render();
452 
453 		return CONTINUE;
454 	}
455 	else if (state == gls::TheilSenCalibrator::STATE_RECOMPUTE_PARAMS)
456 	{
457 		m_calibrator.recomputeParameters();
458 		m_lastIterationRender = false;
459 		return CONTINUE;
460 	}
461 	else
462 	{
463 		DE_ASSERT(state == gls::TheilSenCalibrator::STATE_FINISHED);
464 		GLU_EXPECT_NO_ERROR(m_context.getRenderContext().getFunctions().getError(), "finish");
465 		logResults();
466 		return STOP;
467 	}
468 }
469 
470 // Texture upload tests
471 
TextureUploadTests(Context & context)472 TextureUploadTests::TextureUploadTests (Context& context)
473 	: TestCaseGroup(context, "upload", "Texture upload tests")
474 {
475 }
476 
~TextureUploadTests(void)477 TextureUploadTests::~TextureUploadTests (void)
478 {
479 	TextureUploadTests::deinit();
480 }
481 
deinit(void)482 void TextureUploadTests::deinit (void)
483 {
484 }
485 
init(void)486 void TextureUploadTests::init (void)
487 {
488 	TestCaseGroup* uploadCall		= new TestCaseGroup(m_context, "upload",			"Texture upload");
489 	TestCaseGroup* uploadAndDraw	= new TestCaseGroup(m_context, "upload_draw_swap",	"Texture upload, draw & buffer swap");
490 
491 	addChild(uploadCall);
492 	addChild(uploadAndDraw);
493 
494 	static const struct
495 	{
496 		const char*		name;
497 		const char*		nameLower;
498 		UploadFunction	func;
499 	} uploadFunctions[] =
500 	{
501 		{ "texImage2D",		"teximage2d",		UPLOAD_TEXIMAGE2D },
502 		{ "texSubImage2D",	"texsubimage2d",	UPLOAD_TEXSUBIMAGE2D }
503 	};
504 
505 	static const struct
506 	{
507 		const char*	name;
508 		deUint32	format;
509 		deUint32	type;
510 	} textureCombinations[] =
511 	{
512 		{ "rgb_ubyte",				GL_RGB,				GL_UNSIGNED_BYTE },
513 		{ "rgba_ubyte",				GL_RGBA,			GL_UNSIGNED_BYTE },
514 		{ "alpha_ubyte",			GL_ALPHA,			GL_UNSIGNED_BYTE },
515 		{ "luminance_ubyte",		GL_LUMINANCE,		GL_UNSIGNED_BYTE },
516 		{ "luminance-alpha_ubyte",	GL_LUMINANCE_ALPHA,	GL_UNSIGNED_BYTE },
517 		{ "rgb_ushort565",			GL_RGB,				GL_UNSIGNED_SHORT_5_6_5 },
518 		{ "rgba_ushort4444",		GL_RGBA,			GL_UNSIGNED_SHORT_4_4_4_4 },
519 		{ "rgba_ushort5551",		GL_RGBA,			GL_UNSIGNED_SHORT_5_5_5_1 },
520 	};
521 
522 	static const struct
523 	{
524 		int				size;
525 		TestCaseGroup*	uploadCallGroup;
526 		TestCaseGroup*	uploadAndDrawGroup;
527 	} textureSizes[] =
528 	{
529 		{ 16,	new TestCaseGroup(m_context, "16x16",		"Texture size 16x16"),		new TestCaseGroup(m_context, "16x16",		"Texture size 16x16") },
530 		{ 256,	new TestCaseGroup(m_context, "256x256",		"Texture size 256x256"),	new TestCaseGroup(m_context, "256x256",		"Texture size 256x256") },
531 		{ 257,	new TestCaseGroup(m_context, "257x257",		"Texture size 257x257"),	new TestCaseGroup(m_context, "257x257",		"Texture size 257x257") },
532 		{ 1024,	new TestCaseGroup(m_context, "1024x1024",	"Texture size 1024x1024"),	new TestCaseGroup(m_context, "1024x1024",	"Texture size 1024x1024") },
533 		{ 2048,	new TestCaseGroup(m_context, "2048x2048",	"Texture size 2048x2048"),	new TestCaseGroup(m_context, "2048x2048",	"Texture size 2048x2048") },
534 	};
535 
536 #define FOR_EACH(ITERATOR, ARRAY, BODY)	\
537 	for (int ITERATOR = 0; ITERATOR < DE_LENGTH_OF_ARRAY(ARRAY); ITERATOR++)	\
538 		BODY
539 
540 	FOR_EACH(uploadFunc,	 uploadFunctions,
541 	FOR_EACH(texSize,		 textureSizes,
542 	FOR_EACH(texCombination, textureCombinations,
543 	{
544 		string			caseName	= string("") + uploadFunctions[uploadFunc].nameLower + "_" + textureCombinations[texCombination].name;
545 		UploadFunction	function	= uploadFunctions[uploadFunc].func;
546 		deUint32		format		= textureCombinations[texCombination].format;
547 		deUint32		type		= textureCombinations[texCombination].type;
548 		int				size		= textureSizes[texSize].size;
549 
550 		textureSizes[texSize].uploadCallGroup->addChild		(new TextureUploadCallCase		(m_context, caseName.c_str(), "", function, format, type, size));
551 		textureSizes[texSize].uploadAndDrawGroup->addChild	(new TextureUploadAndDrawCase	(m_context, caseName.c_str(), "", function, format, type, size));
552 	})));
553 
554 	for (int i = 0; i < DE_LENGTH_OF_ARRAY(textureSizes); i++)
555 	{
556 		uploadCall->addChild	(textureSizes[i].uploadCallGroup);
557 		uploadAndDraw->addChild	(textureSizes[i].uploadAndDrawGroup);
558 	}
559 }
560 
561 
562 } // Performance
563 } // gles2
564 } // deqp
565