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 Drawing tests.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es2fDrawTests.hpp"
25 #include "glsDrawTest.hpp"
26 #include "tcuRenderTarget.hpp"
27 #include "deRandom.hpp"
28 #include "deStringUtil.hpp"
29 #include "deUniquePtr.hpp"
30 #include "deSTLUtil.hpp"
31 
32 #include "glwEnums.hpp"
33 
34 #include <set>
35 
36 namespace deqp
37 {
38 namespace gles2
39 {
40 namespace Functional
41 {
42 namespace
43 {
44 
45 enum TestIterationType
46 {
47 	TYPE_DRAW_COUNT,		// !< test with 1, 5, and 25 primitives
48 
49 	TYPE_LAST
50 };
51 
addTestIterations(gls::DrawTest * test,const gls::DrawTestSpec & baseSpec,TestIterationType type)52 static void addTestIterations (gls::DrawTest* test, const gls::DrawTestSpec& baseSpec, TestIterationType type)
53 {
54 	gls::DrawTestSpec spec(baseSpec);
55 
56 	if (type == TYPE_DRAW_COUNT)
57 	{
58 		spec.primitiveCount = 1;
59 		test->addIteration(spec, "draw count = 1");
60 
61 		spec.primitiveCount = 5;
62 		test->addIteration(spec, "draw count = 5");
63 
64 		spec.primitiveCount = 25;
65 		test->addIteration(spec, "draw count = 25");
66 	}
67 	else
68 		DE_ASSERT(false);
69 }
70 
genBasicSpec(gls::DrawTestSpec & spec,gls::DrawTestSpec::DrawMethod method)71 static void genBasicSpec (gls::DrawTestSpec& spec, gls::DrawTestSpec::DrawMethod method)
72 {
73 	spec.apiType							= glu::ApiType::es(2,0);
74 	spec.primitive							= gls::DrawTestSpec::PRIMITIVE_TRIANGLES;
75 	spec.primitiveCount						= 5;
76 	spec.drawMethod							= method;
77 	spec.indexType							= gls::DrawTestSpec::INDEXTYPE_LAST;
78 	spec.indexPointerOffset					= 0;
79 	spec.indexStorage						= gls::DrawTestSpec::STORAGE_LAST;
80 	spec.first								= 0;
81 	spec.indexMin							= 0;
82 	spec.indexMax							= 0;
83 	spec.instanceCount						= 1;
84 
85 	spec.attribs.resize(2);
86 
87 	spec.attribs[0].inputType				= gls::DrawTestSpec::INPUTTYPE_FLOAT;
88 	spec.attribs[0].outputType				= gls::DrawTestSpec::OUTPUTTYPE_VEC2;
89 	spec.attribs[0].storage					= gls::DrawTestSpec::STORAGE_BUFFER;
90 	spec.attribs[0].usage					= gls::DrawTestSpec::USAGE_STATIC_DRAW;
91 	spec.attribs[0].componentCount			= 4;
92 	spec.attribs[0].offset					= 0;
93 	spec.attribs[0].stride					= 0;
94 	spec.attribs[0].normalize				= false;
95 	spec.attribs[0].instanceDivisor			= 0;
96 	spec.attribs[0].useDefaultAttribute		= false;
97 
98 	spec.attribs[1].inputType				= gls::DrawTestSpec::INPUTTYPE_FLOAT;
99 	spec.attribs[1].outputType				= gls::DrawTestSpec::OUTPUTTYPE_VEC2;
100 	spec.attribs[1].storage					= gls::DrawTestSpec::STORAGE_BUFFER;
101 	spec.attribs[1].usage					= gls::DrawTestSpec::USAGE_STATIC_DRAW;
102 	spec.attribs[1].componentCount			= 2;
103 	spec.attribs[1].offset					= 0;
104 	spec.attribs[1].stride					= 0;
105 	spec.attribs[1].normalize				= false;
106 	spec.attribs[1].instanceDivisor			= 0;
107 	spec.attribs[1].useDefaultAttribute		= false;
108 }
109 
110 
111 class AttributeGroup : public TestCaseGroup
112 {
113 public:
114 									AttributeGroup	(Context& context, const char* name, const char* descr, gls::DrawTestSpec::DrawMethod drawMethod, gls::DrawTestSpec::Primitive primitive, gls::DrawTestSpec::IndexType indexType, gls::DrawTestSpec::Storage indexStorage);
115 									~AttributeGroup	(void);
116 
117 	void							init			(void);
118 
119 private:
120 	gls::DrawTestSpec::DrawMethod	m_method;
121 	gls::DrawTestSpec::Primitive	m_primitive;
122 	gls::DrawTestSpec::IndexType	m_indexType;
123 	gls::DrawTestSpec::Storage		m_indexStorage;
124 };
125 
AttributeGroup(Context & context,const char * name,const char * descr,gls::DrawTestSpec::DrawMethod drawMethod,gls::DrawTestSpec::Primitive primitive,gls::DrawTestSpec::IndexType indexType,gls::DrawTestSpec::Storage indexStorage)126 AttributeGroup::AttributeGroup (Context& context, const char* name, const char* descr, gls::DrawTestSpec::DrawMethod drawMethod, gls::DrawTestSpec::Primitive primitive, gls::DrawTestSpec::IndexType indexType, gls::DrawTestSpec::Storage indexStorage)
127 	: TestCaseGroup		(context, name, descr)
128 	, m_method			(drawMethod)
129 	, m_primitive		(primitive)
130 	, m_indexType		(indexType)
131 	, m_indexStorage	(indexStorage)
132 {
133 }
134 
~AttributeGroup(void)135 AttributeGroup::~AttributeGroup (void)
136 {
137 }
138 
init(void)139 void AttributeGroup::init (void)
140 {
141 	// Single attribute
142 	{
143 		gls::DrawTest*		test				= new gls::DrawTest(m_testCtx, m_context.getRenderContext(), "single_attribute", "Single attribute array.");
144 		gls::DrawTestSpec	spec;
145 
146 		spec.apiType							= glu::ApiType::es(2,0);
147 		spec.primitive							= m_primitive;
148 		spec.primitiveCount						= 0;
149 		spec.drawMethod							= m_method;
150 		spec.indexType							= m_indexType;
151 		spec.indexPointerOffset					= 0;
152 		spec.indexStorage						= m_indexStorage;
153 		spec.first								= 0;
154 		spec.indexMin							= 0;
155 		spec.indexMax							= 0;
156 		spec.instanceCount						= 1;
157 
158 		spec.attribs.resize(1);
159 
160 		spec.attribs[0].inputType				= gls::DrawTestSpec::INPUTTYPE_FLOAT;
161 		spec.attribs[0].outputType				= gls::DrawTestSpec::OUTPUTTYPE_VEC2;
162 		spec.attribs[0].storage					= gls::DrawTestSpec::STORAGE_BUFFER;
163 		spec.attribs[0].usage					= gls::DrawTestSpec::USAGE_STATIC_DRAW;
164 		spec.attribs[0].componentCount			= 2;
165 		spec.attribs[0].offset					= 0;
166 		spec.attribs[0].stride					= 0;
167 		spec.attribs[0].normalize				= false;
168 		spec.attribs[0].instanceDivisor			= 0;
169 		spec.attribs[0].useDefaultAttribute		= false;
170 
171 		addTestIterations(test, spec, TYPE_DRAW_COUNT);
172 
173 		this->addChild(test);
174 	}
175 
176 	// Multiple attribute
177 	{
178 		gls::DrawTest*		test				= new gls::DrawTest(m_testCtx, m_context.getRenderContext(), "multiple_attributes", "Multiple attribute arrays");
179 		gls::DrawTestSpec	spec;
180 
181 		spec.apiType							= glu::ApiType::es(2,0);
182 		spec.primitive							= m_primitive;
183 		spec.primitiveCount						= 0;
184 		spec.drawMethod							= m_method;
185 		spec.indexType							= m_indexType;
186 		spec.indexPointerOffset					= 0;
187 		spec.indexStorage						= m_indexStorage;
188 		spec.first								= 0;
189 		spec.indexMin							= 0;
190 		spec.indexMax							= 0;
191 		spec.instanceCount						= 1;
192 
193 		spec.attribs.resize(2);
194 
195 		spec.attribs[0].inputType				= gls::DrawTestSpec::INPUTTYPE_FLOAT;
196 		spec.attribs[0].outputType				= gls::DrawTestSpec::OUTPUTTYPE_VEC2;
197 		spec.attribs[0].storage					= gls::DrawTestSpec::STORAGE_BUFFER;
198 		spec.attribs[0].usage					= gls::DrawTestSpec::USAGE_STATIC_DRAW;
199 		spec.attribs[0].componentCount			= 4;
200 		spec.attribs[0].offset					= 0;
201 		spec.attribs[0].stride					= 0;
202 		spec.attribs[0].normalize				= false;
203 		spec.attribs[0].instanceDivisor			= 0;
204 		spec.attribs[0].useDefaultAttribute		= false;
205 
206 		spec.attribs[1].inputType				= gls::DrawTestSpec::INPUTTYPE_FLOAT;
207 		spec.attribs[1].outputType				= gls::DrawTestSpec::OUTPUTTYPE_VEC2;
208 		spec.attribs[1].storage					= gls::DrawTestSpec::STORAGE_BUFFER;
209 		spec.attribs[1].usage					= gls::DrawTestSpec::USAGE_STATIC_DRAW;
210 		spec.attribs[1].componentCount			= 2;
211 		spec.attribs[1].offset					= 0;
212 		spec.attribs[1].stride					= 0;
213 		spec.attribs[1].normalize				= false;
214 		spec.attribs[1].instanceDivisor			= 0;
215 		spec.attribs[1].useDefaultAttribute		= false;
216 
217 		addTestIterations(test, spec, TYPE_DRAW_COUNT);
218 
219 		this->addChild(test);
220 	}
221 
222 	// Multiple attribute, second one default.
223 	{
224 		gls::DrawTest*		test				= new gls::DrawTest(m_testCtx, m_context.getRenderContext(), "default_attribute", "Attribute specified with glVertexAttrib*.");
225 		gls::DrawTestSpec	spec;
226 
227 		spec.apiType							= glu::ApiType::es(2,0);
228 		spec.primitive							= m_primitive;
229 		spec.primitiveCount						= 5;
230 		spec.drawMethod							= m_method;
231 		spec.indexType							= m_indexType;
232 		spec.indexPointerOffset					= 0;
233 		spec.indexStorage						= m_indexStorage;
234 		spec.first								= 0;
235 		spec.indexMin							= 0;
236 		spec.indexMax							= 0;
237 		spec.instanceCount						= 1;
238 
239 		spec.attribs.resize(2);
240 
241 		spec.attribs[0].inputType				= gls::DrawTestSpec::INPUTTYPE_FLOAT;
242 		spec.attribs[0].outputType				= gls::DrawTestSpec::OUTPUTTYPE_VEC2;
243 		spec.attribs[0].storage					= gls::DrawTestSpec::STORAGE_BUFFER;
244 		spec.attribs[0].usage					= gls::DrawTestSpec::USAGE_STATIC_DRAW;
245 		spec.attribs[0].componentCount			= 2;
246 		spec.attribs[0].offset					= 0;
247 		spec.attribs[0].stride					= 0;
248 		spec.attribs[0].normalize				= false;
249 		spec.attribs[0].instanceDivisor			= 0;
250 		spec.attribs[0].useDefaultAttribute		= false;
251 
252 		struct IOPair
253 		{
254 			gls::DrawTestSpec::InputType  input;
255 			gls::DrawTestSpec::OutputType output;
256 			int							  componentCount;
257 		} iopairs[] =
258 		{
259 			{ gls::DrawTestSpec::INPUTTYPE_FLOAT,        gls::DrawTestSpec::OUTPUTTYPE_VEC2,  4 },
260 			{ gls::DrawTestSpec::INPUTTYPE_FLOAT,        gls::DrawTestSpec::OUTPUTTYPE_VEC4,  2 },
261 		};
262 
263 		for (int ioNdx = 0; ioNdx < DE_LENGTH_OF_ARRAY(iopairs); ++ioNdx)
264 		{
265 			const std::string desc = gls::DrawTestSpec::inputTypeToString(iopairs[ioNdx].input) + de::toString(iopairs[ioNdx].componentCount) + " to " + gls::DrawTestSpec::outputTypeToString(iopairs[ioNdx].output);
266 
267 			spec.attribs[1].inputType			= iopairs[ioNdx].input;
268 			spec.attribs[1].outputType			= iopairs[ioNdx].output;
269 			spec.attribs[1].storage				= gls::DrawTestSpec::STORAGE_BUFFER;
270 			spec.attribs[1].usage				= gls::DrawTestSpec::USAGE_STATIC_DRAW;
271 			spec.attribs[1].componentCount		= iopairs[ioNdx].componentCount;
272 			spec.attribs[1].offset				= 0;
273 			spec.attribs[1].stride				= 0;
274 			spec.attribs[1].normalize			= false;
275 			spec.attribs[1].instanceDivisor		= 0;
276 			spec.attribs[1].useDefaultAttribute	= true;
277 
278 			test->addIteration(spec, desc.c_str());
279 		}
280 
281 		this->addChild(test);
282 	}
283 }
284 
285 class IndexGroup : public TestCaseGroup
286 {
287 public:
288 									IndexGroup		(Context& context, const char* name, const char* descr, gls::DrawTestSpec::DrawMethod drawMethod);
289 									~IndexGroup		(void);
290 
291 	void							init			(void);
292 
293 private:
294 	gls::DrawTestSpec::DrawMethod	m_method;
295 };
296 
IndexGroup(Context & context,const char * name,const char * descr,gls::DrawTestSpec::DrawMethod drawMethod)297 IndexGroup::IndexGroup (Context& context, const char* name, const char* descr, gls::DrawTestSpec::DrawMethod drawMethod)
298 	: TestCaseGroup		(context, name, descr)
299 	, m_method			(drawMethod)
300 {
301 }
302 
~IndexGroup(void)303 IndexGroup::~IndexGroup (void)
304 {
305 }
306 
init(void)307 void IndexGroup::init (void)
308 {
309 	struct IndexTest
310 	{
311 		gls::DrawTestSpec::Storage		storage;
312 		gls::DrawTestSpec::IndexType	type;
313 		bool							aligned;
314 		int								offsets[3];
315 	};
316 
317 	const IndexTest tests[] =
318 	{
319 		{ gls::DrawTestSpec::STORAGE_USER,		gls::DrawTestSpec::INDEXTYPE_BYTE,	true,	{ 0, 1, -1 } },
320 		{ gls::DrawTestSpec::STORAGE_USER,		gls::DrawTestSpec::INDEXTYPE_SHORT,	true,	{ 0, 2, -1 } },
321 
322 		{ gls::DrawTestSpec::STORAGE_USER,		gls::DrawTestSpec::INDEXTYPE_SHORT,	false,	{ 1, 3, -1 } },
323 
324 		{ gls::DrawTestSpec::STORAGE_BUFFER,	gls::DrawTestSpec::INDEXTYPE_BYTE,	true,	{ 0, 1, -1 } },
325 		{ gls::DrawTestSpec::STORAGE_BUFFER,	gls::DrawTestSpec::INDEXTYPE_SHORT,	true,	{ 0, 2, -1 } },
326 	};
327 
328 	gls::DrawTestSpec spec;
329 
330 	tcu::TestCaseGroup* userPtrGroup			= new tcu::TestCaseGroup(m_testCtx, "user_ptr", "user pointer");
331 	tcu::TestCaseGroup* unalignedUserPtrGroup	= new tcu::TestCaseGroup(m_testCtx, "unaligned_user_ptr", "unaligned user pointer");
332 	tcu::TestCaseGroup* bufferGroup				= new tcu::TestCaseGroup(m_testCtx, "buffer", "buffer");
333 
334 	genBasicSpec(spec, m_method);
335 
336 	this->addChild(userPtrGroup);
337 	this->addChild(unalignedUserPtrGroup);
338 	this->addChild(bufferGroup);
339 
340 	for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(tests); ++testNdx)
341 	{
342 		const IndexTest&				indexTest	= tests[testNdx];
343 		tcu::TestCaseGroup*				group		= (indexTest.storage == gls::DrawTestSpec::STORAGE_USER)
344 													? ((indexTest.aligned) ? (userPtrGroup) : (unalignedUserPtrGroup))
345 													: ((indexTest.aligned) ? (bufferGroup) : (DE_NULL));
346 
347 		const std::string				name		= std::string("index_") + gls::DrawTestSpec::indexTypeToString(indexTest.type);
348 		const std::string				desc		= std::string("index ") + gls::DrawTestSpec::indexTypeToString(indexTest.type) + " in " + gls::DrawTestSpec::storageToString(indexTest.storage);
349 		de::MovePtr<gls::DrawTest>		test		(new gls::DrawTest(m_testCtx, m_context.getRenderContext(), name.c_str(), desc.c_str()));
350 
351 		spec.indexType			= indexTest.type;
352 		spec.indexStorage		= indexTest.storage;
353 
354 		for (int iterationNdx = 0; iterationNdx < DE_LENGTH_OF_ARRAY(indexTest.offsets) && indexTest.offsets[iterationNdx] != -1; ++iterationNdx)
355 		{
356 			const std::string iterationDesc = std::string("offset ") + de::toString(indexTest.offsets[iterationNdx]);
357 			spec.indexPointerOffset	= indexTest.offsets[iterationNdx];
358 			test->addIteration(spec, iterationDesc.c_str());
359 		}
360 
361 		DE_ASSERT(spec.isCompatibilityTest() != gls::DrawTestSpec::COMPATIBILITY_UNALIGNED_OFFSET);
362 		DE_ASSERT(spec.isCompatibilityTest() != gls::DrawTestSpec::COMPATIBILITY_UNALIGNED_STRIDE);
363 		group->addChild(test.release());
364 	}
365 }
366 
367 
368 class FirstGroup : public TestCaseGroup
369 {
370 public:
371 									FirstGroup		(Context& context, const char* name, const char* descr, gls::DrawTestSpec::DrawMethod drawMethod);
372 									~FirstGroup		(void);
373 
374 	void							init			(void);
375 
376 private:
377 	gls::DrawTestSpec::DrawMethod	m_method;
378 };
379 
FirstGroup(Context & context,const char * name,const char * descr,gls::DrawTestSpec::DrawMethod drawMethod)380 FirstGroup::FirstGroup (Context& context, const char* name, const char* descr, gls::DrawTestSpec::DrawMethod drawMethod)
381 	: TestCaseGroup		(context, name, descr)
382 	, m_method			(drawMethod)
383 {
384 }
385 
~FirstGroup(void)386 FirstGroup::~FirstGroup (void)
387 {
388 }
389 
init(void)390 void FirstGroup::init (void)
391 {
392 	const int firsts[] =
393 	{
394 		0, 1, 17
395 	};
396 
397 	gls::DrawTestSpec spec;
398 	genBasicSpec(spec, m_method);
399 
400 	for (int firstNdx = 0; firstNdx < DE_LENGTH_OF_ARRAY(firsts); ++firstNdx)
401 	{
402 		const std::string	name = std::string("first_") + de::toString(firsts[firstNdx]);
403 		const std::string	desc = std::string("first ") + de::toString(firsts[firstNdx]);
404 		gls::DrawTest*		test = new gls::DrawTest(m_testCtx, m_context.getRenderContext(), name.c_str(), desc.c_str());
405 
406 		spec.first = firsts[firstNdx];
407 
408 		addTestIterations(test, spec, TYPE_DRAW_COUNT);
409 
410 		this->addChild(test);
411 	}
412 }
413 
414 class MethodGroup : public TestCaseGroup
415 {
416 public:
417 									MethodGroup			(Context& context, const char* name, const char* descr, gls::DrawTestSpec::DrawMethod drawMethod);
418 									~MethodGroup		(void);
419 
420 	void							init				(void);
421 
422 private:
423 	gls::DrawTestSpec::DrawMethod	m_method;
424 };
425 
MethodGroup(Context & context,const char * name,const char * descr,gls::DrawTestSpec::DrawMethod drawMethod)426 MethodGroup::MethodGroup (Context& context, const char* name, const char* descr, gls::DrawTestSpec::DrawMethod drawMethod)
427 	: TestCaseGroup		(context, name, descr)
428 	, m_method			(drawMethod)
429 {
430 }
431 
~MethodGroup(void)432 MethodGroup::~MethodGroup (void)
433 {
434 }
435 
init(void)436 void MethodGroup::init (void)
437 {
438 	const bool indexed		= (m_method == gls::DrawTestSpec::DRAWMETHOD_DRAWELEMENTS) || (m_method == gls::DrawTestSpec::DRAWMETHOD_DRAWELEMENTS_INSTANCED) || (m_method == gls::DrawTestSpec::DRAWMETHOD_DRAWELEMENTS_RANGED);
439 	const bool hasFirst		= (m_method == gls::DrawTestSpec::DRAWMETHOD_DRAWARRAYS) || (m_method == gls::DrawTestSpec::DRAWMETHOD_DRAWARRAYS_INSTANCED);
440 
441 	const gls::DrawTestSpec::Primitive primitive[] =
442 	{
443 		gls::DrawTestSpec::PRIMITIVE_POINTS,
444 		gls::DrawTestSpec::PRIMITIVE_TRIANGLES,
445 		gls::DrawTestSpec::PRIMITIVE_TRIANGLE_FAN,
446 		gls::DrawTestSpec::PRIMITIVE_TRIANGLE_STRIP,
447 		gls::DrawTestSpec::PRIMITIVE_LINES,
448 		gls::DrawTestSpec::PRIMITIVE_LINE_STRIP,
449 		gls::DrawTestSpec::PRIMITIVE_LINE_LOOP
450 	};
451 
452 	if (hasFirst)
453 	{
454 		// First-tests
455 		this->addChild(new FirstGroup(m_context, "first", "First tests", m_method));
456 	}
457 
458 	if (indexed)
459 	{
460 		// Index-tests
461 		if (m_method != gls::DrawTestSpec::DRAWMETHOD_DRAWELEMENTS_RANGED)
462 			this->addChild(new IndexGroup(m_context, "indices", "Index tests", m_method));
463 	}
464 
465 	for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(primitive); ++ndx)
466 	{
467 		const std::string name = gls::DrawTestSpec::primitiveToString(primitive[ndx]);
468 		const std::string desc = gls::DrawTestSpec::primitiveToString(primitive[ndx]);
469 
470 		this->addChild(new AttributeGroup(m_context, name.c_str(), desc.c_str(), m_method, primitive[ndx], gls::DrawTestSpec::INDEXTYPE_SHORT, gls::DrawTestSpec::STORAGE_BUFFER));
471 	}
472 }
473 
474 class RandomGroup : public TestCaseGroup
475 {
476 public:
477 									RandomGroup		(Context& context, const char* name, const char* descr);
478 									~RandomGroup	(void);
479 
480 	void							init			(void);
481 };
482 
483 template <int SIZE>
484 struct UniformWeightArray
485 {
486 	float weights[SIZE];
487 
UniformWeightArraydeqp::gles2::Functional::__anone6c623820111::UniformWeightArray488 	UniformWeightArray (void)
489 	{
490 		for (int i=0; i<SIZE; ++i)
491 			weights[i] = 1.0f;
492 	}
493 };
494 
RandomGroup(Context & context,const char * name,const char * descr)495 RandomGroup::RandomGroup (Context& context, const char* name, const char* descr)
496 	: TestCaseGroup	(context, name, descr)
497 {
498 }
499 
~RandomGroup(void)500 RandomGroup::~RandomGroup (void)
501 {
502 }
503 
init(void)504 void RandomGroup::init (void)
505 {
506 	const int			numAttempts				= 100;
507 
508 	static const int	attribCounts[]			= { 1, 2, 5 };
509 	static const float	attribWeights[]			= { 30, 10, 1 };
510 	static const int	primitiveCounts[]		= { 1, 5, 64 };
511 	static const float	primitiveCountWeights[]	= { 20, 10, 1 };
512 	static const int	indexOffsets[]			= { 0, 7, 13 };
513 	static const float	indexOffsetWeights[]	= { 20, 20, 1 };
514 	static const int	firsts[]				= { 0, 7, 13 };
515 	static const float	firstWeights[]			= { 20, 20, 1 };
516 	static const int	offsets[]				= { 0, 1, 5, 12 };
517 	static const float	offsetWeights[]			= { 50, 10, 10, 10 };
518 	static const int	strides[]				= { 0, 7, 16, 17 };
519 	static const float	strideWeights[]			= { 50, 10, 10, 10 };
520 
521 	static const gls::DrawTestSpec::Primitive primitives[] =
522 	{
523 		gls::DrawTestSpec::PRIMITIVE_POINTS,
524 		gls::DrawTestSpec::PRIMITIVE_TRIANGLES,
525 		gls::DrawTestSpec::PRIMITIVE_TRIANGLE_FAN,
526 		gls::DrawTestSpec::PRIMITIVE_TRIANGLE_STRIP,
527 		gls::DrawTestSpec::PRIMITIVE_LINES,
528 		gls::DrawTestSpec::PRIMITIVE_LINE_STRIP,
529 		gls::DrawTestSpec::PRIMITIVE_LINE_LOOP
530 	};
531 	const UniformWeightArray<DE_LENGTH_OF_ARRAY(primitives)> primitiveWeights;
532 
533 	static const gls::DrawTestSpec::DrawMethod drawMethods[] =
534 	{
535 		gls::DrawTestSpec::DRAWMETHOD_DRAWARRAYS,
536 		gls::DrawTestSpec::DRAWMETHOD_DRAWELEMENTS,
537 	};
538 	const UniformWeightArray<DE_LENGTH_OF_ARRAY(drawMethods)> drawMethodWeights;
539 
540 	static const gls::DrawTestSpec::IndexType indexTypes[] =
541 	{
542 		gls::DrawTestSpec::INDEXTYPE_BYTE,
543 		gls::DrawTestSpec::INDEXTYPE_SHORT,
544 	};
545 	const UniformWeightArray<DE_LENGTH_OF_ARRAY(indexTypes)> indexTypeWeights;
546 
547 	static const gls::DrawTestSpec::Storage storages[] =
548 	{
549 		gls::DrawTestSpec::STORAGE_USER,
550 		gls::DrawTestSpec::STORAGE_BUFFER,
551 	};
552 	const UniformWeightArray<DE_LENGTH_OF_ARRAY(storages)> storageWeights;
553 
554 	static const gls::DrawTestSpec::InputType inputTypes[] =
555 	{
556 		gls::DrawTestSpec::INPUTTYPE_FLOAT,
557 		gls::DrawTestSpec::INPUTTYPE_FIXED,
558 		gls::DrawTestSpec::INPUTTYPE_BYTE,
559 		gls::DrawTestSpec::INPUTTYPE_SHORT,
560 		gls::DrawTestSpec::INPUTTYPE_UNSIGNED_BYTE,
561 		gls::DrawTestSpec::INPUTTYPE_UNSIGNED_SHORT
562 	};
563 	const UniformWeightArray<DE_LENGTH_OF_ARRAY(inputTypes)> inputTypeWeights;
564 
565 	static const gls::DrawTestSpec::OutputType outputTypes[] =
566 	{
567 		gls::DrawTestSpec::OUTPUTTYPE_FLOAT,
568 		gls::DrawTestSpec::OUTPUTTYPE_VEC2,
569 		gls::DrawTestSpec::OUTPUTTYPE_VEC3,
570 		gls::DrawTestSpec::OUTPUTTYPE_VEC4,
571 	};
572 	const UniformWeightArray<DE_LENGTH_OF_ARRAY(outputTypes)> outputTypeWeights;
573 
574 	static const gls::DrawTestSpec::Usage usages[] =
575 	{
576 		gls::DrawTestSpec::USAGE_STATIC_DRAW,
577 		gls::DrawTestSpec::USAGE_DYNAMIC_DRAW,
578 		gls::DrawTestSpec::USAGE_STREAM_DRAW,
579 	};
580 	const UniformWeightArray<DE_LENGTH_OF_ARRAY(usages)> usageWeights;
581 
582 	static const deUint32 blacklistedCases[]=
583 	{
584 		3153,	//!< extremely narrow triangle, results depend on sample positions
585 	};
586 
587 	std::set<deUint32>	insertedHashes;
588 	size_t				insertedCount = 0;
589 
590 	for (int ndx = 0; ndx < numAttempts; ++ndx)
591 	{
592 		de::Random random(0xc551393 + ndx); // random does not depend on previous cases
593 
594 		int					attributeCount = random.chooseWeighted<int, const int*, const float*>(DE_ARRAY_BEGIN(attribCounts), DE_ARRAY_END(attribCounts), attribWeights);
595 		gls::DrawTestSpec	spec;
596 
597 		spec.apiType				= glu::ApiType::es(2,0);
598 		spec.primitive				= random.chooseWeighted<gls::DrawTestSpec::Primitive>	(DE_ARRAY_BEGIN(primitives),		DE_ARRAY_END(primitives),		primitiveWeights.weights);
599 		spec.primitiveCount			= random.chooseWeighted<int, const int*, const float*>	(DE_ARRAY_BEGIN(primitiveCounts),	DE_ARRAY_END(primitiveCounts),	primitiveCountWeights);
600 		spec.drawMethod				= random.chooseWeighted<gls::DrawTestSpec::DrawMethod>	(DE_ARRAY_BEGIN(drawMethods),		DE_ARRAY_END(drawMethods),		drawMethodWeights.weights);
601 		spec.indexType				= random.chooseWeighted<gls::DrawTestSpec::IndexType>	(DE_ARRAY_BEGIN(indexTypes),		DE_ARRAY_END(indexTypes),		indexTypeWeights.weights);
602 		spec.indexPointerOffset		= random.chooseWeighted<int, const int*, const float*>	(DE_ARRAY_BEGIN(indexOffsets),		DE_ARRAY_END(indexOffsets),		indexOffsetWeights);
603 		spec.indexStorage			= random.chooseWeighted<gls::DrawTestSpec::Storage>		(DE_ARRAY_BEGIN(storages),			DE_ARRAY_END(storages),			storageWeights.weights);
604 		spec.first					= random.chooseWeighted<int, const int*, const float*>	(DE_ARRAY_BEGIN(firsts),			DE_ARRAY_END(firsts),			firstWeights);
605 		spec.indexMin				= 0;
606 		spec.indexMax				= 0;
607 		spec.instanceCount			= 0;
608 
609 		// check spec is legal
610 		if (!spec.valid())
611 			continue;
612 
613 		for (int attrNdx = 0; attrNdx < attributeCount;)
614 		{
615 			bool valid;
616 			gls::DrawTestSpec::AttributeSpec attribSpec;
617 
618 			attribSpec.inputType			= random.chooseWeighted<gls::DrawTestSpec::InputType>	(DE_ARRAY_BEGIN(inputTypes),		DE_ARRAY_END(inputTypes),		inputTypeWeights.weights);
619 			attribSpec.outputType			= random.chooseWeighted<gls::DrawTestSpec::OutputType>	(DE_ARRAY_BEGIN(outputTypes),		DE_ARRAY_END(outputTypes),		outputTypeWeights.weights);
620 			attribSpec.storage				= random.chooseWeighted<gls::DrawTestSpec::Storage>		(DE_ARRAY_BEGIN(storages),			DE_ARRAY_END(storages),			storageWeights.weights);
621 			attribSpec.usage				= random.chooseWeighted<gls::DrawTestSpec::Usage>		(DE_ARRAY_BEGIN(usages),			DE_ARRAY_END(usages),			usageWeights.weights);
622 			attribSpec.componentCount		= random.getInt(1, 4);
623 			attribSpec.offset				= random.chooseWeighted<int, const int*, const float*>(DE_ARRAY_BEGIN(offsets), DE_ARRAY_END(offsets), offsetWeights);
624 			attribSpec.stride				= random.chooseWeighted<int, const int*, const float*>(DE_ARRAY_BEGIN(strides), DE_ARRAY_END(strides), strideWeights);
625 			attribSpec.normalize			= random.getBool();
626 			attribSpec.instanceDivisor		= 0;
627 			attribSpec.useDefaultAttribute	= random.getBool();
628 
629 			// check spec is legal
630 			valid = attribSpec.valid(spec.apiType);
631 
632 			// we do not want interleaved elements. (Might result in some weird floating point values)
633 			if (attribSpec.stride && attribSpec.componentCount * gls::DrawTestSpec::inputTypeSize(attribSpec.inputType) > attribSpec.stride)
634 				valid = false;
635 
636 			// try again if not valid
637 			if (valid)
638 			{
639 				spec.attribs.push_back(attribSpec);
640 				++attrNdx;
641 			}
642 		}
643 
644 		// Do not collapse all vertex positions to a single positions
645 		if (spec.primitive != gls::DrawTestSpec::PRIMITIVE_POINTS)
646 			spec.attribs[0].instanceDivisor = 0;
647 
648 		// Is render result meaningful?
649 		{
650 			// Only one vertex
651 			if (spec.drawMethod == gls::DrawTestSpec::DRAWMETHOD_DRAWELEMENTS_RANGED && spec.indexMin == spec.indexMax && spec.primitive != gls::DrawTestSpec::PRIMITIVE_POINTS)
652 				continue;
653 			if (spec.attribs[0].useDefaultAttribute && spec.primitive != gls::DrawTestSpec::PRIMITIVE_POINTS)
654 				continue;
655 
656 			// Triangle only on one axis
657 			if (spec.primitive == gls::DrawTestSpec::PRIMITIVE_TRIANGLES || spec.primitive == gls::DrawTestSpec::PRIMITIVE_TRIANGLE_FAN || spec.primitive == gls::DrawTestSpec::PRIMITIVE_TRIANGLE_STRIP)
658 			{
659 				if (spec.attribs[0].componentCount == 1)
660 					continue;
661 				if (spec.attribs[0].outputType == gls::DrawTestSpec::OUTPUTTYPE_FLOAT || spec.attribs[0].outputType == gls::DrawTestSpec::OUTPUTTYPE_INT || spec.attribs[0].outputType == gls::DrawTestSpec::OUTPUTTYPE_UINT)
662 					continue;
663 				if (spec.drawMethod == gls::DrawTestSpec::DRAWMETHOD_DRAWELEMENTS_RANGED && (spec.indexMax - spec.indexMin) < 2)
664 					continue;
665 			}
666 		}
667 
668 		// Add case
669 		{
670 			deUint32 hash = spec.hash();
671 			for (int attrNdx = 0; attrNdx < attributeCount; ++attrNdx)
672 				hash = (hash << 2) ^ (deUint32)spec.attribs[attrNdx].hash();
673 
674 			if (insertedHashes.find(hash) == insertedHashes.end() &&
675 				!de::contains(DE_ARRAY_BEGIN(blacklistedCases), DE_ARRAY_END(blacklistedCases), hash))
676 			{
677 				// Only aligned cases
678 				if (spec.isCompatibilityTest() != gls::DrawTestSpec::COMPATIBILITY_UNALIGNED_OFFSET &&
679 					spec.isCompatibilityTest() != gls::DrawTestSpec::COMPATIBILITY_UNALIGNED_STRIDE)
680 					this->addChild(new gls::DrawTest(m_testCtx, m_context.getRenderContext(), spec, de::toString(insertedCount).c_str(), spec.getDesc().c_str()));
681 				insertedHashes.insert(hash);
682 
683 				++insertedCount;
684 			}
685 		}
686 	}
687 }
688 
689 } // anonymous
690 
DrawTests(Context & context)691 DrawTests::DrawTests (Context& context)
692 	: TestCaseGroup(context, "draw", "Drawing tests")
693 {
694 }
695 
~DrawTests(void)696 DrawTests::~DrawTests (void)
697 {
698 }
699 
init(void)700 void DrawTests::init (void)
701 {
702 	// Basic
703 	{
704 		const gls::DrawTestSpec::DrawMethod basicMethods[] =
705 		{
706 			gls::DrawTestSpec::DRAWMETHOD_DRAWARRAYS,
707 			gls::DrawTestSpec::DRAWMETHOD_DRAWELEMENTS,
708 		};
709 
710 		for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(basicMethods); ++ndx)
711 		{
712 			std::string name = gls::DrawTestSpec::drawMethodToString(basicMethods[ndx]);
713 			std::string desc = gls::DrawTestSpec::drawMethodToString(basicMethods[ndx]);
714 
715 			this->addChild(new MethodGroup(m_context, name.c_str(), desc.c_str(), basicMethods[ndx]));
716 		}
717 	}
718 
719 	// Random
720 
721 	this->addChild(new RandomGroup(m_context, "random", "random draw commands."));
722 }
723 
724 } // Functional
725 } // gles2
726 } // deqp
727