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 Transform feedback tests.
22 *//*--------------------------------------------------------------------*/
23
24 #include "es3fTransformFeedbackTests.hpp"
25 #include "tcuTestLog.hpp"
26 #include "tcuSurface.hpp"
27 #include "tcuImageCompare.hpp"
28 #include "tcuVector.hpp"
29 #include "tcuFormatUtil.hpp"
30 #include "tcuRenderTarget.hpp"
31 #include "gluShaderUtil.hpp"
32 #include "gluVarType.hpp"
33 #include "gluVarTypeUtil.hpp"
34 #include "gluPixelTransfer.hpp"
35 #include "gluRenderContext.hpp"
36 #include "gluShaderProgram.hpp"
37 #include "gluObjectWrapper.hpp"
38 #include "glwFunctions.hpp"
39 #include "glwEnums.hpp"
40 #include "deRandom.hpp"
41 #include "deStringUtil.hpp"
42 #include "deMemory.h"
43 #include "deString.h"
44
45 #include <set>
46 #include <map>
47 #include <algorithm>
48
49 using std::string;
50 using std::vector;
51 using std::set;
52
53 using std::map;
54 using std::set;
55
56 using tcu::TestLog;
57
58 namespace deqp
59 {
60 namespace gles3
61 {
62 namespace Functional
63 {
64 namespace TransformFeedback
65 {
66
67 enum
68 {
69 VIEWPORT_WIDTH = 128,
70 VIEWPORT_HEIGHT = 128,
71 BUFFER_GUARD_MULTIPLIER = 2 //!< stride*BUFFER_GUARD_MULTIPLIER bytes are added to the end of tf buffer and used to check for overruns.
72 };
73
74 enum Interpolation
75 {
76 INTERPOLATION_SMOOTH = 0,
77 INTERPOLATION_FLAT,
78 INTERPOLATION_CENTROID,
79
80 INTERPOLATION_LAST
81 };
82
getInterpolationName(Interpolation interp)83 static const char* getInterpolationName (Interpolation interp)
84 {
85 switch (interp)
86 {
87 case INTERPOLATION_SMOOTH: return "smooth";
88 case INTERPOLATION_FLAT: return "flat";
89 case INTERPOLATION_CENTROID: return "centroid";
90 default:
91 DE_ASSERT(false);
92 return DE_NULL;
93 }
94 }
95
96 struct Varying
97 {
Varyingdeqp::gles3::Functional::TransformFeedback::Varying98 Varying (const char* name_, const glu::VarType& type_, Interpolation interp_)
99 : name (name_)
100 , type (type_)
101 , interpolation (interp_)
102 {
103 }
104
105 std::string name; //!< Variable name.
106 glu::VarType type; //!< Variable type.
107 Interpolation interpolation; //!< Interpolation mode (smooth, flat, centroid).
108 };
109
110 struct VaryingNameEquals
111 {
VaryingNameEqualsdeqp::gles3::Functional::TransformFeedback::VaryingNameEquals112 VaryingNameEquals (const std::string& name_) : name(name_) {}
operator ()deqp::gles3::Functional::TransformFeedback::VaryingNameEquals113 bool operator() (const Varying& var) const { return var.name == name; }
114
115 std::string name;
116 };
117
118 struct Attribute
119 {
Attributedeqp::gles3::Functional::TransformFeedback::Attribute120 Attribute (const std::string& name_, const glu::VarType& type_, int offset_)
121 : name (name_)
122 , type (type_)
123 , offset (offset_)
124 {
125 }
126
127 std::string name;
128 glu::VarType type;
129 int offset;
130 };
131
132 struct AttributeNameEquals
133 {
AttributeNameEqualsdeqp::gles3::Functional::TransformFeedback::AttributeNameEquals134 AttributeNameEquals (const std::string& name_) : name(name_) {}
operator ()deqp::gles3::Functional::TransformFeedback::AttributeNameEquals135 bool operator() (const Attribute& attr) const { return attr.name == name; }
136
137 std::string name;
138 };
139
140 struct Output
141 {
Outputdeqp::gles3::Functional::TransformFeedback::Output142 Output (void)
143 : bufferNdx (0)
144 , offset (0)
145 {
146 }
147
148 std::string name;
149 glu::VarType type;
150 int bufferNdx;
151 int offset;
152 vector<const Attribute*> inputs;
153 };
154
155 struct DrawCall
156 {
DrawCalldeqp::gles3::Functional::TransformFeedback::DrawCall157 DrawCall (int numElements_, bool tfEnabled_)
158 : numElements (numElements_)
159 , transformFeedbackEnabled (tfEnabled_)
160 {
161 }
162
DrawCalldeqp::gles3::Functional::TransformFeedback::DrawCall163 DrawCall (void)
164 : numElements (0)
165 , transformFeedbackEnabled (false)
166 {
167 }
168
169 int numElements;
170 bool transformFeedbackEnabled;
171 };
172
operator <<(std::ostream & str,const DrawCall & call)173 std::ostream& operator<< (std::ostream& str, const DrawCall& call)
174 {
175 return str << "(" << call.numElements << ", " << (call.transformFeedbackEnabled ? "resumed" : "paused") << ")";
176 }
177
178 class ProgramSpec
179 {
180 public:
181 ProgramSpec (void);
182 ~ProgramSpec (void);
183
184 glu::StructType* createStruct (const char* name);
185 void addVarying (const char* name, const glu::VarType& type, Interpolation interp);
186 void addTransformFeedbackVarying (const char* name);
187
getStructs(void) const188 const vector<glu::StructType*>& getStructs (void) const { return m_structs; }
getVaryings(void) const189 const vector<Varying>& getVaryings (void) const { return m_varyings; }
getTransformFeedbackVaryings(void) const190 const vector<string>& getTransformFeedbackVaryings (void) const { return m_transformFeedbackVaryings; }
191 bool isPointSizeUsed (void) const;
192
193 private:
194 ProgramSpec (const ProgramSpec& other);
195 ProgramSpec& operator= (const ProgramSpec& other);
196
197 vector<glu::StructType*> m_structs;
198 vector<Varying> m_varyings;
199 vector<string> m_transformFeedbackVaryings;
200 };
201
202 // ProgramSpec
203
ProgramSpec(void)204 ProgramSpec::ProgramSpec (void)
205 {
206 }
207
~ProgramSpec(void)208 ProgramSpec::~ProgramSpec (void)
209 {
210 for (vector<glu::StructType*>::iterator i = m_structs.begin(); i != m_structs.end(); i++)
211 delete *i;
212 }
213
createStruct(const char * name)214 glu::StructType* ProgramSpec::createStruct (const char* name)
215 {
216 m_structs.reserve(m_structs.size()+1);
217 m_structs.push_back(new glu::StructType(name));
218 return m_structs.back();
219 }
220
addVarying(const char * name,const glu::VarType & type,Interpolation interp)221 void ProgramSpec::addVarying (const char* name, const glu::VarType& type, Interpolation interp)
222 {
223 m_varyings.push_back(Varying(name, type, interp));
224 }
225
addTransformFeedbackVarying(const char * name)226 void ProgramSpec::addTransformFeedbackVarying (const char* name)
227 {
228 m_transformFeedbackVaryings.push_back(name);
229 }
230
isPointSizeUsed(void) const231 bool ProgramSpec::isPointSizeUsed (void) const
232 {
233 return std::find(m_transformFeedbackVaryings.begin(), m_transformFeedbackVaryings.end(), "gl_PointSize") != m_transformFeedbackVaryings.end();
234 }
235
isProgramSupported(const glw::Functions & gl,const ProgramSpec & spec,deUint32 tfMode)236 static bool isProgramSupported (const glw::Functions& gl, const ProgramSpec& spec, deUint32 tfMode)
237 {
238 int maxVertexAttribs = 0;
239 int maxTfInterleavedComponents = 0;
240 int maxTfSeparateAttribs = 0;
241 int maxTfSeparateComponents = 0;
242
243 gl.getIntegerv(GL_MAX_VERTEX_ATTRIBS, &maxVertexAttribs);
244 gl.getIntegerv(GL_MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS, &maxTfInterleavedComponents);
245 gl.getIntegerv(GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS, &maxTfSeparateAttribs);
246 gl.getIntegerv(GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS, &maxTfSeparateComponents);
247
248 // Check vertex attribs.
249 int totalVertexAttribs = 1 /* a_position */ + (spec.isPointSizeUsed() ? 1 : 0);
250 for (vector<Varying>::const_iterator var = spec.getVaryings().begin(); var != spec.getVaryings().end(); var++)
251 {
252 for (glu::VectorTypeIterator vecIter = glu::VectorTypeIterator::begin(&var->type); vecIter != glu::VectorTypeIterator::end(&var->type); vecIter++)
253 totalVertexAttribs += 1;
254 }
255
256 if (totalVertexAttribs > maxVertexAttribs)
257 return false; // Vertex attribute count exceeded.
258
259 // Check varyings.
260 int totalTfComponents = 0;
261 int totalTfAttribs = 0;
262 for (vector<string>::const_iterator iter = spec.getTransformFeedbackVaryings().begin(); iter != spec.getTransformFeedbackVaryings().end(); iter++)
263 {
264 const string& name = *iter;
265 int numComponents = 0;
266
267 if (name == "gl_Position")
268 numComponents = 4;
269 else if (name == "gl_PointSize")
270 numComponents = 1;
271 else
272 {
273 string varName = glu::parseVariableName(name.c_str());
274 const Varying& varying = *std::find_if(spec.getVaryings().begin(), spec.getVaryings().end(), VaryingNameEquals(varName));
275 glu::TypeComponentVector varPath;
276
277 glu::parseTypePath(name.c_str(), varying.type, varPath);
278 numComponents = glu::getVarType(varying.type, varPath).getScalarSize();
279 }
280
281 if (tfMode == GL_SEPARATE_ATTRIBS && numComponents > maxTfSeparateComponents)
282 return false; // Per-attribute component count exceeded.
283
284 totalTfComponents += numComponents;
285 totalTfAttribs += 1;
286 }
287
288 if (tfMode == GL_SEPARATE_ATTRIBS && totalTfAttribs > maxTfSeparateAttribs)
289 return false;
290
291 if (tfMode == GL_INTERLEAVED_ATTRIBS && totalTfComponents > maxTfInterleavedComponents)
292 return false;
293
294 return true;
295 }
296
297 // Program
298
getAttributeName(const char * varyingName,const glu::TypeComponentVector & path)299 static std::string getAttributeName (const char* varyingName, const glu::TypeComponentVector& path)
300 {
301 std::ostringstream str;
302
303 str << "a_" << (deStringBeginsWith(varyingName, "v_") ? varyingName+2 : varyingName);
304
305 for (glu::TypeComponentVector::const_iterator iter = path.begin(); iter != path.end(); iter++)
306 {
307 const char* prefix = DE_NULL;
308
309 switch (iter->type)
310 {
311 case glu::VarTypeComponent::STRUCT_MEMBER: prefix = "_m"; break;
312 case glu::VarTypeComponent::ARRAY_ELEMENT: prefix = "_e"; break;
313 case glu::VarTypeComponent::MATRIX_COLUMN: prefix = "_c"; break;
314 case glu::VarTypeComponent::VECTOR_COMPONENT: prefix = "_s"; break;
315 default:
316 DE_ASSERT(false);
317 }
318
319 str << prefix << iter->index;
320 }
321
322 return str.str();
323 }
324
genShaderSources(const ProgramSpec & spec,std::string & vertSource,std::string & fragSource,bool pointSizeRequired)325 static void genShaderSources (const ProgramSpec& spec, std::string& vertSource, std::string& fragSource, bool pointSizeRequired)
326 {
327 std::ostringstream vtx;
328 std::ostringstream frag;
329 bool addPointSize = spec.isPointSizeUsed();
330
331 vtx << "#version 300 es\n"
332 << "in highp vec4 a_position;\n";
333 frag << "#version 300 es\n"
334 << "layout(location = 0) out mediump vec4 o_color;\n"
335 << "uniform highp vec4 u_scale;\n"
336 << "uniform highp vec4 u_bias;\n";
337
338 if (addPointSize)
339 vtx << "in highp float a_pointSize;\n";
340
341 // Declare attributes.
342 for (vector<Varying>::const_iterator var = spec.getVaryings().begin(); var != spec.getVaryings().end(); var++)
343 {
344 const char* name = var->name.c_str();
345 const glu::VarType& type = var->type;
346
347 for (glu::VectorTypeIterator vecIter = glu::VectorTypeIterator::begin(&type); vecIter != glu::VectorTypeIterator::end(&type); vecIter++)
348 {
349 glu::VarType attribType = glu::getVarType(type, vecIter.getPath());
350 string attribName = getAttributeName(name, vecIter.getPath());
351
352 vtx << "in " << glu::declare(attribType, attribName.c_str()) << ";\n";
353 }
354 }
355
356 // Declare vayrings.
357 for (int ndx = 0; ndx < 2; ndx++)
358 {
359 const char* inout = ndx ? "in" : "out";
360 std::ostringstream& str = ndx ? frag : vtx;
361
362 // Declare structs that have type name.
363 for (vector<glu::StructType*>::const_iterator structIter = spec.getStructs().begin(); structIter != spec.getStructs().end(); structIter++)
364 {
365 const glu::StructType* structPtr = *structIter;
366 if (structPtr->hasTypeName())
367 str << glu::declare(structPtr) << ";\n";
368 }
369
370 for (vector<Varying>::const_iterator var = spec.getVaryings().begin(); var != spec.getVaryings().end(); var++)
371 str << getInterpolationName(var->interpolation) << " " << inout << " " << glu::declare(var->type, var->name.c_str()) << ";\n";
372 }
373
374 vtx << "\nvoid main (void)\n{\n"
375 << "\tgl_Position = a_position;\n";
376 frag << "\nvoid main (void)\n{\n"
377 << "\thighp vec4 res = vec4(0.0);\n";
378
379 if (addPointSize)
380 vtx << "\tgl_PointSize = a_pointSize;\n";
381 else if (pointSizeRequired)
382 vtx << "\tgl_PointSize = 1.0;\n";
383
384 // Generate assignments / usage.
385 for (vector<Varying>::const_iterator var = spec.getVaryings().begin(); var != spec.getVaryings().end(); var++)
386 {
387 const char* name = var->name.c_str();
388 const glu::VarType& type = var->type;
389
390 for (glu::VectorTypeIterator vecIter = glu::VectorTypeIterator::begin(&type); vecIter != glu::VectorTypeIterator::end(&type); vecIter++)
391 {
392 glu::VarType subType = glu::getVarType(type, vecIter.getPath());
393 string attribName = getAttributeName(name, vecIter.getPath());
394
395 DE_ASSERT(subType.isBasicType() && glu::isDataTypeScalarOrVector(subType.getBasicType()));
396
397 // Vertex: assign from attribute.
398 vtx << "\t" << name << vecIter << " = " << attribName << ";\n";
399
400 // Fragment: add to res variable.
401 int scalarSize = glu::getDataTypeScalarSize(subType.getBasicType());
402
403 frag << "\tres += ";
404 if (scalarSize == 1) frag << "vec4(" << name << vecIter << ")";
405 else if (scalarSize == 2) frag << "vec2(" << name << vecIter << ").xxyy";
406 else if (scalarSize == 3) frag << "vec3(" << name << vecIter << ").xyzx";
407 else if (scalarSize == 4) frag << "vec4(" << name << vecIter << ")";
408
409 frag << ";\n";
410 }
411 }
412
413 frag << "\to_color = res * u_scale + u_bias;\n";
414
415 vtx << "}\n";
416 frag << "}\n";
417
418 vertSource = vtx.str();
419 fragSource = frag.str();
420 }
421
createVertexCaptureProgram(const glu::RenderContext & context,const ProgramSpec & spec,deUint32 bufferMode,deUint32 primitiveType)422 static glu::ShaderProgram* createVertexCaptureProgram (const glu::RenderContext& context, const ProgramSpec& spec, deUint32 bufferMode, deUint32 primitiveType)
423 {
424 std::string vertSource, fragSource;
425
426 genShaderSources(spec, vertSource, fragSource, primitiveType == GL_POINTS /* Is point size required? */);
427
428 return new glu::ShaderProgram(context, glu::ProgramSources()
429 << glu::VertexSource(vertSource)
430 << glu::FragmentSource(fragSource)
431 << glu::TransformFeedbackVaryings<vector<string>::const_iterator>(spec.getTransformFeedbackVaryings().begin(), spec.getTransformFeedbackVaryings().end())
432 << glu::TransformFeedbackMode(bufferMode));
433 }
434
435 // Helpers.
436
computeInputLayout(vector<Attribute> & attributes,int & inputStride,const vector<Varying> & varyings,bool usePointSize)437 static void computeInputLayout (vector<Attribute>& attributes, int& inputStride, const vector<Varying>& varyings, bool usePointSize)
438 {
439 inputStride = 0;
440
441 // Add position.
442 attributes.push_back(Attribute("a_position", glu::VarType(glu::TYPE_FLOAT_VEC4, glu::PRECISION_HIGHP), inputStride));
443 inputStride += 4*sizeof(deUint32);
444
445 if (usePointSize)
446 {
447 attributes.push_back(Attribute("a_pointSize", glu::VarType(glu::TYPE_FLOAT, glu::PRECISION_HIGHP), inputStride));
448 inputStride += 1*sizeof(deUint32);
449 }
450
451 // Compute attribute vector.
452 for (vector<Varying>::const_iterator var = varyings.begin(); var != varyings.end(); var++)
453 {
454 for (glu::VectorTypeIterator vecIter = glu::VectorTypeIterator::begin(&var->type); vecIter != glu::VectorTypeIterator::end(&var->type); vecIter++)
455 {
456 glu::VarType type = vecIter.getType();
457 string name = getAttributeName(var->name.c_str(), vecIter.getPath());
458
459 attributes.push_back(Attribute(name, type, inputStride));
460 inputStride += glu::getDataTypeScalarSize(type.getBasicType())*sizeof(deUint32);
461 }
462 }
463 }
464
computeTransformFeedbackOutputs(vector<Output> & transformFeedbackOutputs,const vector<Attribute> & attributes,const vector<Varying> & varyings,const vector<string> & transformFeedbackVaryings,deUint32 bufferMode)465 static void computeTransformFeedbackOutputs (vector<Output>& transformFeedbackOutputs, const vector<Attribute>& attributes, const vector<Varying>& varyings, const vector<string>& transformFeedbackVaryings, deUint32 bufferMode)
466 {
467 int accumulatedSize = 0;
468
469 transformFeedbackOutputs.resize(transformFeedbackVaryings.size());
470 for (int varNdx = 0; varNdx < (int)transformFeedbackVaryings.size(); varNdx++)
471 {
472 const string& name = transformFeedbackVaryings[varNdx];
473 int bufNdx = (bufferMode == GL_SEPARATE_ATTRIBS ? varNdx : 0);
474 int offset = (bufferMode == GL_SEPARATE_ATTRIBS ? 0 : accumulatedSize);
475 Output& output = transformFeedbackOutputs[varNdx];
476
477 output.name = name;
478 output.bufferNdx = bufNdx;
479 output.offset = offset;
480
481 if (name == "gl_Position")
482 {
483 const Attribute* posIn = &(*std::find_if(attributes.begin(), attributes.end(), AttributeNameEquals("a_position")));
484 output.type = posIn->type;
485 output.inputs.push_back(posIn);
486 }
487 else if (name == "gl_PointSize")
488 {
489 const Attribute* sizeIn = &(*std::find_if(attributes.begin(), attributes.end(), AttributeNameEquals("a_pointSize")));
490 output.type = sizeIn->type;
491 output.inputs.push_back(sizeIn);
492 }
493 else
494 {
495 string varName = glu::parseVariableName(name.c_str());
496 const Varying& varying = *std::find_if(varyings.begin(), varyings.end(), VaryingNameEquals(varName));
497 glu::TypeComponentVector varPath;
498
499 glu::parseTypePath(name.c_str(), varying.type, varPath);
500
501 output.type = glu::getVarType(varying.type, varPath);
502
503 // Add all vectorized attributes as inputs.
504 for (glu::VectorTypeIterator iter = glu::VectorTypeIterator::begin(&output.type); iter != glu::VectorTypeIterator::end(&output.type); iter++)
505 {
506 // Full path.
507 glu::TypeComponentVector fullPath(varPath.size() + iter.getPath().size());
508
509 std::copy(varPath.begin(), varPath.end(), fullPath.begin());
510 std::copy(iter.getPath().begin(), iter.getPath().end(), fullPath.begin()+varPath.size());
511
512 string attribName = getAttributeName(varName.c_str(), fullPath);
513 const Attribute* attrib = &(*std::find_if(attributes.begin(), attributes.end(), AttributeNameEquals(attribName)));
514
515 output.inputs.push_back(attrib);
516 }
517 }
518
519 accumulatedSize += output.type.getScalarSize()*sizeof(deUint32);
520 }
521 }
522
signExtend(deUint32 value,deUint32 numBits)523 static deUint32 signExtend (deUint32 value, deUint32 numBits)
524 {
525 DE_ASSERT(numBits >= 1u && numBits <= 32u);
526 if (numBits == 32u)
527 return value;
528 else if ((value & (1u << (numBits-1u))) == 0u)
529 return value;
530 else
531 return value | ~((1u << numBits) - 1u);
532 }
533
genAttributeData(const Attribute & attrib,deUint8 * basePtr,int stride,int numElements,de::Random & rnd)534 static void genAttributeData (const Attribute& attrib, deUint8* basePtr, int stride, int numElements, de::Random& rnd)
535 {
536 const int elementSize = (int)sizeof(deUint32);
537 const bool isFloat = glu::isDataTypeFloatOrVec(attrib.type.getBasicType());
538 const bool isInt = glu::isDataTypeIntOrIVec(attrib.type.getBasicType());
539 const bool isUint = glu::isDataTypeUintOrUVec(attrib.type.getBasicType());
540 const glu::Precision precision = attrib.type.getPrecision();
541 const int numComps = glu::getDataTypeScalarSize(attrib.type.getBasicType());
542
543 for (int elemNdx = 0; elemNdx < numElements; elemNdx++)
544 {
545 for (int compNdx = 0; compNdx < numComps; compNdx++)
546 {
547 int offset = attrib.offset+elemNdx*stride+compNdx*elementSize;
548 if (isFloat)
549 {
550 float* comp = (float*)(basePtr+offset);
551 switch (precision)
552 {
553 case glu::PRECISION_LOWP: *comp = 0.0f + 0.25f*(float)rnd.getInt(0, 4); break;
554 case glu::PRECISION_MEDIUMP: *comp = rnd.getFloat(-1e3f, 1e3f); break;
555 case glu::PRECISION_HIGHP: *comp = rnd.getFloat(-1e5f, 1e5f); break;
556 default:
557 DE_ASSERT(false);
558 }
559 }
560 else if (isInt)
561 {
562 int* comp = (int*)(basePtr+offset);
563 switch (precision)
564 {
565 case glu::PRECISION_LOWP: *comp = (int)signExtend(rnd.getUint32()&0xff, 8); break;
566 case glu::PRECISION_MEDIUMP: *comp = (int)signExtend(rnd.getUint32()&0xffff, 16); break;
567 case glu::PRECISION_HIGHP: *comp = (int)rnd.getUint32(); break;
568 default:
569 DE_ASSERT(false);
570 }
571 }
572 else if (isUint)
573 {
574 deUint32* comp = (deUint32*)(basePtr+offset);
575 switch (precision)
576 {
577 case glu::PRECISION_LOWP: *comp = rnd.getUint32()&0xff; break;
578 case glu::PRECISION_MEDIUMP: *comp = rnd.getUint32()&0xffff; break;
579 case glu::PRECISION_HIGHP: *comp = rnd.getUint32(); break;
580 default:
581 DE_ASSERT(false);
582 }
583 }
584 else
585 DE_ASSERT(false);
586 }
587 }
588 }
589
genInputData(const vector<Attribute> & attributes,int numInputs,int inputStride,deUint8 * inputBasePtr,de::Random & rnd)590 static void genInputData (const vector<Attribute>& attributes, int numInputs, int inputStride, deUint8* inputBasePtr, de::Random& rnd)
591 {
592 // Random positions.
593 const Attribute& position = *std::find_if(attributes.begin(), attributes.end(), AttributeNameEquals("a_position"));
594
595 for (int ndx = 0; ndx < numInputs; ndx++)
596 {
597 deUint8* ptr = inputBasePtr + position.offset + inputStride*ndx;
598 *((float*)(ptr+ 0)) = rnd.getFloat(-1.2f, 1.2f);
599 *((float*)(ptr+ 4)) = rnd.getFloat(-1.2f, 1.2f);
600 *((float*)(ptr+ 8)) = rnd.getFloat(-1.2f, 1.2f);
601 *((float*)(ptr+12)) = rnd.getFloat(0.1f, 2.0f);
602 }
603
604 // Point size.
605 vector<Attribute>::const_iterator pointSizePos = std::find_if(attributes.begin(), attributes.end(), AttributeNameEquals("a_pointSize"));
606 if (pointSizePos != attributes.end())
607 {
608 for (int ndx = 0; ndx < numInputs; ndx++)
609 {
610 deUint8* ptr = inputBasePtr + pointSizePos->offset + inputStride*ndx;
611 *((float*)ptr) = rnd.getFloat(1.0f, 8.0f);
612 }
613 }
614
615 // Random data for rest of components.
616 for (vector<Attribute>::const_iterator attrib = attributes.begin(); attrib != attributes.end(); attrib++)
617 {
618 if (attrib->name == "a_position" || attrib->name == "a_pointSize")
619 continue;
620
621 genAttributeData(*attrib, inputBasePtr, inputStride, numInputs, rnd);
622 }
623 }
624
getTransformFeedbackOutputCount(deUint32 primitiveType,int numElements)625 static deUint32 getTransformFeedbackOutputCount (deUint32 primitiveType, int numElements)
626 {
627 switch (primitiveType)
628 {
629 case GL_TRIANGLES: return numElements - numElements%3;
630 case GL_TRIANGLE_STRIP: return de::max(0, numElements-2)*3;
631 case GL_TRIANGLE_FAN: return de::max(0, numElements-2)*3;
632 case GL_LINES: return numElements - numElements%2;
633 case GL_LINE_STRIP: return de::max(0, numElements-1)*2;
634 case GL_LINE_LOOP: return numElements > 1 ? numElements*2 : 0;
635 case GL_POINTS: return numElements;
636
637 default:
638 DE_ASSERT(false);
639 return 0;
640 }
641 }
642
getTransformFeedbackPrimitiveCount(deUint32 primitiveType,int numElements)643 static deUint32 getTransformFeedbackPrimitiveCount (deUint32 primitiveType, int numElements)
644 {
645 switch (primitiveType)
646 {
647 case GL_TRIANGLES: return numElements/3;
648 case GL_TRIANGLE_STRIP: return de::max(0, numElements-2);
649 case GL_TRIANGLE_FAN: return de::max(0, numElements-2);
650 case GL_LINES: return numElements/2;
651 case GL_LINE_STRIP: return de::max(0, numElements-1);
652 case GL_LINE_LOOP: return numElements > 1 ? numElements : 0;
653 case GL_POINTS: return numElements;
654
655 default:
656 DE_ASSERT(false);
657 return 0;
658 }
659 }
660
getTransformFeedbackPrimitiveMode(deUint32 primitiveType)661 static deUint32 getTransformFeedbackPrimitiveMode (deUint32 primitiveType)
662 {
663 switch (primitiveType)
664 {
665 case GL_TRIANGLES:
666 case GL_TRIANGLE_STRIP:
667 case GL_TRIANGLE_FAN:
668 return GL_TRIANGLES;
669
670 case GL_LINES:
671 case GL_LINE_LOOP:
672 case GL_LINE_STRIP:
673 return GL_LINES;
674
675 case GL_POINTS:
676 return GL_POINTS;
677
678 default:
679 DE_ASSERT(false);
680 return 0;
681 }
682 }
683
getAttributeIndex(deUint32 primitiveType,int numInputs,int outNdx)684 static int getAttributeIndex (deUint32 primitiveType, int numInputs, int outNdx)
685 {
686 switch (primitiveType)
687 {
688 case GL_TRIANGLES: return outNdx;
689 case GL_LINES: return outNdx;
690 case GL_POINTS: return outNdx;
691
692 case GL_TRIANGLE_STRIP:
693 {
694 int triNdx = outNdx/3;
695 int vtxNdx = outNdx%3;
696 return (triNdx%2 != 0 && vtxNdx < 2) ? (triNdx+1-vtxNdx) : (triNdx+vtxNdx);
697 }
698
699 case GL_TRIANGLE_FAN:
700 return (outNdx%3 != 0) ? (outNdx/3 + outNdx%3) : 0;
701
702 case GL_LINE_STRIP:
703 return outNdx/2 + outNdx%2;
704
705 case GL_LINE_LOOP:
706 {
707 int inNdx = outNdx/2 + outNdx%2;
708 return inNdx < numInputs ? inNdx : 0;
709 }
710
711 default:
712 DE_ASSERT(false);
713 return 0;
714 }
715 }
716
compareTransformFeedbackOutput(tcu::TestLog & log,deUint32 primitiveType,const Output & output,int numInputs,const deUint8 * inBasePtr,int inStride,const deUint8 * outBasePtr,int outStride)717 static bool compareTransformFeedbackOutput (tcu::TestLog& log, deUint32 primitiveType, const Output& output, int numInputs, const deUint8* inBasePtr, int inStride, const deUint8* outBasePtr, int outStride)
718 {
719 bool isOk = true;
720 int outOffset = output.offset;
721
722 for (int attrNdx = 0; attrNdx < (int)output.inputs.size(); attrNdx++)
723 {
724 const Attribute& attribute = *output.inputs[attrNdx];
725 glu::DataType type = attribute.type.getBasicType();
726 int numComponents = glu::getDataTypeScalarSize(type);
727 glu::Precision precision = attribute.type.getPrecision();
728 glu::DataType scalarType = glu::getDataTypeScalarType(type);
729 int numOutputs = getTransformFeedbackOutputCount(primitiveType, numInputs);
730
731 for (int outNdx = 0; outNdx < numOutputs; outNdx++)
732 {
733 int inNdx = getAttributeIndex(primitiveType, numInputs, outNdx);
734
735 for (int compNdx = 0; compNdx < numComponents; compNdx++)
736 {
737 const deUint8* inPtr = inBasePtr + inStride*inNdx + attribute.offset + compNdx*sizeof(deUint32);
738 const deUint8* outPtr = outBasePtr + outStride*outNdx + outOffset + compNdx*sizeof(deUint32);
739 deUint32 inVal = *(const deUint32*)inPtr;
740 deUint32 outVal = *(const deUint32*)outPtr;
741 bool isEqual = false;
742
743 if (scalarType == glu::TYPE_FLOAT)
744 {
745 // ULP comparison is used for highp and mediump. Lowp uses threshold-comparison.
746 switch (precision)
747 {
748 case glu::PRECISION_HIGHP: isEqual = de::abs((int)inVal - (int)outVal) < 2; break;
749 case glu::PRECISION_MEDIUMP: isEqual = de::abs((int)inVal - (int)outVal) < 2+(1<<13); break;
750 case glu::PRECISION_LOWP:
751 {
752 float inF = *(const float*)inPtr;
753 float outF = *(const float*)outPtr;
754 isEqual = de::abs(inF - outF) < 0.1f;
755 break;
756 }
757 default:
758 DE_ASSERT(false);
759 }
760 }
761 else
762 isEqual = (inVal == outVal); // Bit-exact match required for integer types.
763
764 if (!isEqual)
765 {
766 log << TestLog::Message << "Mismatch in " << output.name << " (" << attribute.name << "), output = " << outNdx << ", input = " << inNdx << ", component = " << compNdx << TestLog::EndMessage;
767 isOk = false;
768 break;
769 }
770 }
771
772 if (!isOk)
773 break;
774 }
775
776 if (!isOk)
777 break;
778
779 outOffset += numComponents*sizeof(deUint32);
780 }
781
782 return isOk;
783 }
784
computeTransformFeedbackPrimitiveCount(deUint32 primitiveType,const DrawCall * first,const DrawCall * end)785 static int computeTransformFeedbackPrimitiveCount (deUint32 primitiveType, const DrawCall* first, const DrawCall* end)
786 {
787 int primCount = 0;
788
789 for (const DrawCall* call = first; call != end; ++call)
790 {
791 if (call->transformFeedbackEnabled)
792 primCount += getTransformFeedbackPrimitiveCount(primitiveType, call->numElements);
793 }
794
795 return primCount;
796 }
797
writeBufferGuard(const glw::Functions & gl,deUint32 target,int bufferSize,int guardSize)798 static void writeBufferGuard (const glw::Functions& gl, deUint32 target, int bufferSize, int guardSize)
799 {
800 deUint8* ptr = (deUint8*)gl.mapBufferRange(target, bufferSize, guardSize, GL_MAP_WRITE_BIT);
801 if (ptr)
802 deMemset(ptr, 0xcd, guardSize);
803 gl.unmapBuffer(target);
804 GLU_EXPECT_NO_ERROR(gl.getError(), "guardband write");
805 }
806
verifyGuard(const deUint8 * ptr,int guardSize)807 static bool verifyGuard (const deUint8* ptr, int guardSize)
808 {
809 for (int ndx = 0; ndx < guardSize; ndx++)
810 {
811 if (ptr[ndx] != 0xcd)
812 return false;
813 }
814 return true;
815 }
816
logTransformFeedbackVaryings(TestLog & log,const glw::Functions & gl,deUint32 program)817 static void logTransformFeedbackVaryings (TestLog& log, const glw::Functions& gl, deUint32 program)
818 {
819 int numTfVaryings = 0;
820 int maxNameLen = 0;
821
822 gl.getProgramiv(program, GL_TRANSFORM_FEEDBACK_VARYINGS, &numTfVaryings);
823 gl.getProgramiv(program, GL_TRANSFORM_FEEDBACK_VARYING_MAX_LENGTH, &maxNameLen);
824 GLU_EXPECT_NO_ERROR(gl.getError(), "Query TF varyings");
825
826 log << TestLog::Message << "GL_TRANSFORM_FEEDBACK_VARYINGS = " << numTfVaryings << TestLog::EndMessage;
827
828 vector<char> nameBuf(maxNameLen+1);
829
830 for (int ndx = 0; ndx < numTfVaryings; ndx++)
831 {
832 glw::GLsizei size = 0;
833 glw::GLenum type = 0;
834
835 gl.getTransformFeedbackVarying(program, ndx, (glw::GLsizei)nameBuf.size(), DE_NULL, &size, &type, &nameBuf[0]);
836 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetTransformFeedbackVarying()");
837
838 const glu::DataType dataType = glu::getDataTypeFromGLType(type);
839 const std::string typeName = dataType != glu::TYPE_LAST ? std::string(glu::getDataTypeName(dataType))
840 : (std::string("unknown(") + tcu::toHex(type).toString() + ")");
841
842 log << TestLog::Message << (const char*)&nameBuf[0] << ": " << typeName << "[" << size << "]" << TestLog::EndMessage;
843 }
844 }
845
846 class TransformFeedbackCase : public TestCase
847 {
848 public:
849 TransformFeedbackCase (Context& context, const char* name, const char* desc, deUint32 bufferMode, deUint32 primitiveType);
850 ~TransformFeedbackCase (void);
851
852 void init (void);
853 void deinit (void);
854 IterateResult iterate (void);
855
856 protected:
857 ProgramSpec m_progSpec;
858 deUint32 m_bufferMode;
859 deUint32 m_primitiveType;
860
861 private:
862 TransformFeedbackCase (const TransformFeedbackCase& other);
863 TransformFeedbackCase& operator= (const TransformFeedbackCase& other);
864
865 bool runTest (const DrawCall* first, const DrawCall* end, deUint32 seed);
866
867 // Derived from ProgramSpec in init()
868 int m_inputStride;
869 vector<Attribute> m_attributes;
870 vector<Output> m_transformFeedbackOutputs;
871 vector<int> m_bufferStrides;
872
873 // GL state.
874 glu::ShaderProgram* m_program;
875 glu::TransformFeedback* m_transformFeedback;
876 vector<deUint32> m_outputBuffers;
877
878 int m_iterNdx;
879 };
880
TransformFeedbackCase(Context & context,const char * name,const char * desc,deUint32 bufferMode,deUint32 primitiveType)881 TransformFeedbackCase::TransformFeedbackCase (Context& context, const char* name, const char* desc, deUint32 bufferMode, deUint32 primitiveType)
882 : TestCase (context, name, desc)
883 , m_bufferMode (bufferMode)
884 , m_primitiveType (primitiveType)
885 , m_inputStride (0)
886 , m_program (DE_NULL)
887 , m_transformFeedback (DE_NULL)
888 , m_iterNdx (0)
889 {
890 }
891
~TransformFeedbackCase(void)892 TransformFeedbackCase::~TransformFeedbackCase (void)
893 {
894 TransformFeedbackCase::deinit();
895 }
896
hasArraysInTFVaryings(const ProgramSpec & spec)897 static bool hasArraysInTFVaryings (const ProgramSpec& spec)
898 {
899 for (vector<string>::const_iterator tfVar = spec.getTransformFeedbackVaryings().begin(); tfVar != spec.getTransformFeedbackVaryings().end(); ++tfVar)
900 {
901 string varName = glu::parseVariableName(tfVar->c_str());
902 vector<Varying>::const_iterator varIter = std::find_if(spec.getVaryings().begin(), spec.getVaryings().end(), VaryingNameEquals(varName));
903
904 if (varName == "gl_Position" || varName == "gl_PointSize")
905 continue;
906
907 DE_ASSERT(varIter != spec.getVaryings().end());
908
909 if (varIter->type.isArrayType())
910 return true;
911 }
912
913 return false;
914 }
915
init(void)916 void TransformFeedbackCase::init (void)
917 {
918 TestLog& log = m_testCtx.getLog();
919 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
920
921 DE_ASSERT(!m_program);
922 m_program = createVertexCaptureProgram(m_context.getRenderContext(), m_progSpec, m_bufferMode, m_primitiveType);
923
924 log << *m_program;
925 if (!m_program->isOk())
926 {
927 const bool linkFail = m_program->getShaderInfo(glu::SHADERTYPE_VERTEX).compileOk &&
928 m_program->getShaderInfo(glu::SHADERTYPE_FRAGMENT).compileOk &&
929 !m_program->getProgramInfo().linkOk;
930
931 if (linkFail)
932 {
933 if (!isProgramSupported(gl, m_progSpec, m_bufferMode))
934 throw tcu::NotSupportedError("Implementation limits execeeded", "", __FILE__, __LINE__);
935 else if (hasArraysInTFVaryings(m_progSpec))
936 throw tcu::NotSupportedError("Capturing arrays is not supported (undefined in specification)", "", __FILE__, __LINE__);
937 else
938 throw tcu::TestError("Link failed", "", __FILE__, __LINE__);
939 }
940 else
941 throw tcu::TestError("Compile failed", "", __FILE__, __LINE__);
942 }
943
944 log << TestLog::Message << "Transform feedback varyings: " << tcu::formatArray(m_progSpec.getTransformFeedbackVaryings().begin(), m_progSpec.getTransformFeedbackVaryings().end()) << TestLog::EndMessage;
945
946 // Print out transform feedback points reported by GL.
947 log << TestLog::Message << "Transform feedback varyings reported by compiler:" << TestLog::EndMessage;
948 logTransformFeedbackVaryings(log, gl, m_program->getProgram());
949
950 // Compute input specification.
951 computeInputLayout(m_attributes, m_inputStride, m_progSpec.getVaryings(), m_progSpec.isPointSizeUsed());
952
953 // Build list of varyings used in transform feedback.
954 computeTransformFeedbackOutputs(m_transformFeedbackOutputs, m_attributes, m_progSpec.getVaryings(), m_progSpec.getTransformFeedbackVaryings(), m_bufferMode);
955 DE_ASSERT(!m_transformFeedbackOutputs.empty());
956
957 // Buffer strides.
958 DE_ASSERT(m_bufferStrides.empty());
959 if (m_bufferMode == GL_SEPARATE_ATTRIBS)
960 {
961 for (vector<Output>::const_iterator outIter = m_transformFeedbackOutputs.begin(); outIter != m_transformFeedbackOutputs.end(); outIter++)
962 m_bufferStrides.push_back(outIter->type.getScalarSize()*sizeof(deUint32));
963 }
964 else
965 {
966 int totalSize = 0;
967 for (vector<Output>::const_iterator outIter = m_transformFeedbackOutputs.begin(); outIter != m_transformFeedbackOutputs.end(); outIter++)
968 totalSize += outIter->type.getScalarSize()*sizeof(deUint32);
969
970 m_bufferStrides.push_back(totalSize);
971 }
972
973 // \note Actual storage is allocated in iterate().
974 m_outputBuffers.resize(m_bufferStrides.size());
975 gl.genBuffers((glw::GLsizei)m_outputBuffers.size(), &m_outputBuffers[0]);
976
977 DE_ASSERT(!m_transformFeedback);
978 m_transformFeedback = new glu::TransformFeedback(m_context.getRenderContext());
979
980 GLU_EXPECT_NO_ERROR(gl.getError(), "init");
981
982 m_iterNdx = 0;
983 m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
984 }
985
deinit(void)986 void TransformFeedbackCase::deinit (void)
987 {
988 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
989
990 if (!m_outputBuffers.empty())
991 {
992 gl.deleteBuffers((glw::GLsizei)m_outputBuffers.size(), &m_outputBuffers[0]);
993 m_outputBuffers.clear();
994 }
995
996 delete m_transformFeedback;
997 m_transformFeedback = DE_NULL;
998
999 delete m_program;
1000 m_program = DE_NULL;
1001
1002 // Clean up state.
1003 m_attributes.clear();
1004 m_transformFeedbackOutputs.clear();
1005 m_bufferStrides.clear();
1006 m_inputStride = 0;
1007 }
1008
iterate(void)1009 TransformFeedbackCase::IterateResult TransformFeedbackCase::iterate (void)
1010 {
1011 // Test cases.
1012 static const DrawCall s_elemCount1[] = { DrawCall(1, true) };
1013 static const DrawCall s_elemCount2[] = { DrawCall(2, true) };
1014 static const DrawCall s_elemCount3[] = { DrawCall(3, true) };
1015 static const DrawCall s_elemCount4[] = { DrawCall(4, true) };
1016 static const DrawCall s_elemCount123[] = { DrawCall(123, true) };
1017 static const DrawCall s_basicPause1[] = { DrawCall(64, true), DrawCall(64, false), DrawCall(64, true) };
1018 static const DrawCall s_basicPause2[] = { DrawCall(13, true), DrawCall(5, true), DrawCall(17, false), DrawCall(3, true), DrawCall(7, false) };
1019 static const DrawCall s_startPaused[] = { DrawCall(123, false), DrawCall(123, true) };
1020 static const DrawCall s_random1[] = { DrawCall(65, true), DrawCall(135, false), DrawCall(74, true), DrawCall(16, false), DrawCall(226, false), DrawCall(9, true), DrawCall(174, false) };
1021 static const DrawCall s_random2[] = { DrawCall(217, true), DrawCall(171, true), DrawCall(147, true), DrawCall(152, false), DrawCall(55, true) };
1022
1023 static const struct
1024 {
1025 const DrawCall* calls;
1026 int numCalls;
1027 } s_iterations[] =
1028 {
1029 #define ITER(ARR) { ARR, DE_LENGTH_OF_ARRAY(ARR) }
1030 ITER(s_elemCount1),
1031 ITER(s_elemCount2),
1032 ITER(s_elemCount3),
1033 ITER(s_elemCount4),
1034 ITER(s_elemCount123),
1035 ITER(s_basicPause1),
1036 ITER(s_basicPause2),
1037 ITER(s_startPaused),
1038 ITER(s_random1),
1039 ITER(s_random2)
1040 #undef ITER
1041 };
1042
1043 TestLog& log = m_testCtx.getLog();
1044 bool isOk = true;
1045 deUint32 seed = deStringHash(getName()) ^ deInt32Hash(m_iterNdx);
1046 int numIterations = DE_LENGTH_OF_ARRAY(s_iterations);
1047 const DrawCall* first = s_iterations[m_iterNdx].calls;
1048 const DrawCall* end = s_iterations[m_iterNdx].calls + s_iterations[m_iterNdx].numCalls;
1049
1050 std::string sectionName = std::string("Iteration") + de::toString(m_iterNdx+1);
1051 std::string sectionDesc = std::string("Iteration ") + de::toString(m_iterNdx+1) + " / " + de::toString(numIterations);
1052 tcu::ScopedLogSection section (log, sectionName, sectionDesc);
1053
1054 log << TestLog::Message << "Testing " << s_iterations[m_iterNdx].numCalls << " draw calls, (element count, TF state): " << tcu::formatArray(first, end) << TestLog::EndMessage;
1055
1056 isOk = runTest(first, end, seed);
1057
1058 if (!isOk)
1059 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Result comparison failed");
1060
1061 m_iterNdx += 1;
1062 return (isOk && m_iterNdx < numIterations) ? CONTINUE : STOP;
1063 }
1064
runTest(const DrawCall * first,const DrawCall * end,deUint32 seed)1065 bool TransformFeedbackCase::runTest (const DrawCall* first, const DrawCall* end, deUint32 seed)
1066 {
1067 TestLog& log = m_testCtx.getLog();
1068 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
1069 de::Random rnd (seed);
1070 int numInputs = 0; //!< Sum of element counts in calls.
1071 int numOutputs = 0; //!< Sum of output counts for calls that have transform feedback enabled.
1072 int width = m_context.getRenderContext().getRenderTarget().getWidth();
1073 int height = m_context.getRenderContext().getRenderTarget().getHeight();
1074 int viewportW = de::min((int)VIEWPORT_WIDTH, width);
1075 int viewportH = de::min((int)VIEWPORT_HEIGHT, height);
1076 int viewportX = rnd.getInt(0, width-viewportW);
1077 int viewportY = rnd.getInt(0, height-viewportH);
1078 tcu::Surface frameWithTf (viewportW, viewportH);
1079 tcu::Surface frameWithoutTf (viewportW, viewportH);
1080 glu::Query primitiveQuery (m_context.getRenderContext());
1081 bool outputsOk = true;
1082 bool imagesOk = true;
1083 bool queryOk = true;
1084
1085 // Compute totals.
1086 for (const DrawCall* call = first; call != end; call++)
1087 {
1088 numInputs += call->numElements;
1089 numOutputs += call->transformFeedbackEnabled ? getTransformFeedbackOutputCount(m_primitiveType, call->numElements) : 0;
1090 }
1091
1092 // Input data.
1093 vector<deUint8> inputData(m_inputStride*numInputs);
1094 genInputData(m_attributes, numInputs, m_inputStride, &inputData[0], rnd);
1095
1096 gl.bindTransformFeedback(GL_TRANSFORM_FEEDBACK, m_transformFeedback->get());
1097 GLU_EXPECT_NO_ERROR(gl.getError(), "glBindTransformFeedback()");
1098
1099 // Allocate storage for transform feedback output buffers and bind to targets.
1100 for (int bufNdx = 0; bufNdx < (int)m_outputBuffers.size(); bufNdx++)
1101 {
1102 deUint32 buffer = m_outputBuffers[bufNdx];
1103 int stride = m_bufferStrides[bufNdx];
1104 int target = bufNdx;
1105 int size = stride*numOutputs;
1106 int guardSize = stride*BUFFER_GUARD_MULTIPLIER;
1107 const deUint32 usage = GL_DYNAMIC_READ;
1108
1109 gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, buffer);
1110 gl.bufferData(GL_TRANSFORM_FEEDBACK_BUFFER, size+guardSize, DE_NULL, usage);
1111 writeBufferGuard(gl, GL_TRANSFORM_FEEDBACK_BUFFER, size, guardSize);
1112
1113 // \todo [2012-07-30 pyry] glBindBufferRange()?
1114 gl.bindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, target, buffer);
1115
1116 GLU_EXPECT_NO_ERROR(gl.getError(), "transform feedback buffer setup");
1117 }
1118
1119 // Setup attributes.
1120 for (vector<Attribute>::const_iterator attrib = m_attributes.begin(); attrib != m_attributes.end(); attrib++)
1121 {
1122 int loc = gl.getAttribLocation(m_program->getProgram(), attrib->name.c_str());
1123 glu::DataType scalarType = glu::getDataTypeScalarType(attrib->type.getBasicType());
1124 int numComponents = glu::getDataTypeScalarSize(attrib->type.getBasicType());
1125 const void* ptr = &inputData[0] + attrib->offset;
1126
1127 if (loc >= 0)
1128 {
1129 gl.enableVertexAttribArray(loc);
1130
1131 if (scalarType == glu::TYPE_FLOAT) gl.vertexAttribPointer (loc, numComponents, GL_FLOAT, GL_FALSE, m_inputStride, ptr);
1132 else if (scalarType == glu::TYPE_INT) gl.vertexAttribIPointer (loc, numComponents, GL_INT, m_inputStride, ptr);
1133 else if (scalarType == glu::TYPE_UINT) gl.vertexAttribIPointer (loc, numComponents, GL_UNSIGNED_INT, m_inputStride, ptr);
1134 }
1135 }
1136
1137 // Setup viewport.
1138 gl.viewport(viewportX, viewportY, viewportW, viewportH);
1139
1140 // Setup program.
1141 gl.useProgram(m_program->getProgram());
1142
1143 gl.uniform4fv(gl.getUniformLocation(m_program->getProgram(), "u_scale"), 1, tcu::Vec4(0.01f).getPtr());
1144 gl.uniform4fv(gl.getUniformLocation(m_program->getProgram(), "u_bias"), 1, tcu::Vec4(0.5f).getPtr());
1145
1146 // Enable query.
1147 gl.beginQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN, *primitiveQuery);
1148 GLU_EXPECT_NO_ERROR(gl.getError(), "glBeginQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN)");
1149
1150 // Draw.
1151 {
1152 int offset = 0;
1153 bool tfEnabled = true;
1154
1155 gl.clear(GL_COLOR_BUFFER_BIT);
1156
1157 gl.beginTransformFeedback(getTransformFeedbackPrimitiveMode(m_primitiveType));
1158
1159 for (const DrawCall* call = first; call != end; call++)
1160 {
1161 // Pause or resume transform feedback if necessary.
1162 if (call->transformFeedbackEnabled != tfEnabled)
1163 {
1164 if (call->transformFeedbackEnabled)
1165 gl.resumeTransformFeedback();
1166 else
1167 gl.pauseTransformFeedback();
1168 tfEnabled = call->transformFeedbackEnabled;
1169 }
1170
1171 gl.drawArrays(m_primitiveType, offset, call->numElements);
1172 offset += call->numElements;
1173 }
1174
1175 // Resume feedback before finishing it.
1176 if (!tfEnabled)
1177 gl.resumeTransformFeedback();
1178
1179 gl.endTransformFeedback();
1180 GLU_EXPECT_NO_ERROR(gl.getError(), "render");
1181 }
1182
1183 gl.endQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN);
1184 GLU_EXPECT_NO_ERROR(gl.getError(), "glEndQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN)");
1185
1186 // Check and log query status right after submit
1187 {
1188 deUint32 available = GL_FALSE;
1189 gl.getQueryObjectuiv(*primitiveQuery, GL_QUERY_RESULT_AVAILABLE, &available);
1190 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetQueryObjectuiv()");
1191
1192 log << TestLog::Message << "GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN status after submit: " << (available != GL_FALSE ? "GL_TRUE" : "GL_FALSE") << TestLog::EndMessage;
1193 }
1194
1195 // Compare result buffers.
1196 for (int bufferNdx = 0; bufferNdx < (int)m_outputBuffers.size(); bufferNdx++)
1197 {
1198 deUint32 buffer = m_outputBuffers[bufferNdx];
1199 int stride = m_bufferStrides[bufferNdx];
1200 int size = stride*numOutputs;
1201 int guardSize = stride*BUFFER_GUARD_MULTIPLIER;
1202 const void* bufPtr = DE_NULL;
1203
1204 // Bind buffer for reading.
1205 gl.bindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, buffer);
1206 bufPtr = gl.mapBufferRange(GL_TRANSFORM_FEEDBACK_BUFFER, 0, size+guardSize, GL_MAP_READ_BIT);
1207 GLU_EXPECT_NO_ERROR(gl.getError(), "mapping buffer");
1208
1209 // Verify all output variables that are written to this buffer.
1210 for (vector<Output>::const_iterator out = m_transformFeedbackOutputs.begin(); out != m_transformFeedbackOutputs.end(); out++)
1211 {
1212 if (out->bufferNdx != bufferNdx)
1213 continue;
1214
1215 int inputOffset = 0;
1216 int outputOffset = 0;
1217
1218 // Process all draw calls and check ones with transform feedback enabled.
1219 for (const DrawCall* call = first; call != end; call++)
1220 {
1221 if (call->transformFeedbackEnabled)
1222 {
1223 const deUint8* inputPtr = &inputData[0] + inputOffset*m_inputStride;
1224 const deUint8* outputPtr = (const deUint8*)bufPtr + outputOffset*stride;
1225
1226 if (!compareTransformFeedbackOutput(log, m_primitiveType, *out, call->numElements, inputPtr, m_inputStride, outputPtr, stride))
1227 {
1228 outputsOk = false;
1229 break;
1230 }
1231 }
1232
1233 inputOffset += call->numElements;
1234 outputOffset += call->transformFeedbackEnabled ? getTransformFeedbackOutputCount(m_primitiveType, call->numElements) : 0;
1235 }
1236 }
1237
1238 // Verify guardband.
1239 if (!verifyGuard((const deUint8*)bufPtr + size, guardSize))
1240 {
1241 log << TestLog::Message << "Error: Transform feedback buffer overrun detected" << TestLog::EndMessage;
1242 outputsOk = false;
1243 }
1244
1245 gl.unmapBuffer(GL_TRANSFORM_FEEDBACK_BUFFER);
1246 }
1247
1248 // Check status after mapping buffers.
1249 {
1250 const bool mustBeReady = !m_outputBuffers.empty(); // Mapping buffer forces synchronization.
1251 const int expectedCount = computeTransformFeedbackPrimitiveCount(m_primitiveType, first, end);
1252 deUint32 available = GL_FALSE;
1253 deUint32 numPrimitives = 0;
1254
1255 gl.getQueryObjectuiv(*primitiveQuery, GL_QUERY_RESULT_AVAILABLE, &available);
1256 gl.getQueryObjectuiv(*primitiveQuery, GL_QUERY_RESULT, &numPrimitives);
1257 GLU_EXPECT_NO_ERROR(gl.getError(), "glGetQueryObjectuiv()");
1258
1259 if (!mustBeReady && available == GL_FALSE)
1260 {
1261 log << TestLog::Message << "ERROR: GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN result not available after mapping buffers!" << TestLog::EndMessage;
1262 queryOk = false;
1263 }
1264
1265 log << TestLog::Message << "GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN = " << numPrimitives << TestLog::EndMessage;
1266
1267 if ((int)numPrimitives != expectedCount)
1268 {
1269 log << TestLog::Message << "ERROR: Expected " << expectedCount << " primitives!" << TestLog::EndMessage;
1270 queryOk = false;
1271 }
1272 }
1273
1274 // Clear transform feedback state.
1275 gl.bindTransformFeedback(GL_TRANSFORM_FEEDBACK, 0);
1276 for (int bufNdx = 0; bufNdx < (int)m_outputBuffers.size(); bufNdx++)
1277 {
1278 gl.bindBuffer (GL_TRANSFORM_FEEDBACK_BUFFER, 0);
1279 gl.bindBufferBase (GL_TRANSFORM_FEEDBACK_BUFFER, bufNdx, 0);
1280 }
1281
1282 // Read back rendered image.
1283 glu::readPixels(m_context.getRenderContext(), viewportX, viewportY, frameWithTf.getAccess());
1284
1285 // Render without transform feedback.
1286 {
1287 int offset = 0;
1288
1289 gl.clear(GL_COLOR_BUFFER_BIT);
1290
1291 for (const DrawCall* call = first; call != end; call++)
1292 {
1293 gl.drawArrays(m_primitiveType, offset, call->numElements);
1294 offset += call->numElements;
1295 }
1296
1297 GLU_EXPECT_NO_ERROR(gl.getError(), "render");
1298 glu::readPixels(m_context.getRenderContext(), viewportX, viewportY, frameWithoutTf.getAccess());
1299 }
1300
1301 // Compare images with and without transform feedback.
1302 imagesOk = tcu::pixelThresholdCompare(log, "Result", "Image comparison result", frameWithoutTf, frameWithTf, tcu::RGBA(1, 1, 1, 1), tcu::COMPARE_LOG_ON_ERROR);
1303
1304 if (imagesOk)
1305 m_testCtx.getLog() << TestLog::Message << "Rendering result comparison between TF enabled and TF disabled passed." << TestLog::EndMessage;
1306 else
1307 m_testCtx.getLog() << TestLog::Message << "ERROR: Rendering result comparison between TF enabled and TF disabled failed!" << TestLog::EndMessage;
1308
1309 return outputsOk && imagesOk && queryOk;
1310 }
1311
1312 // Test cases.
1313
1314 class PositionCase : public TransformFeedbackCase
1315 {
1316 public:
PositionCase(Context & context,const char * name,const char * desc,deUint32 bufferType,deUint32 primitiveType)1317 PositionCase (Context& context, const char* name, const char* desc, deUint32 bufferType, deUint32 primitiveType)
1318 : TransformFeedbackCase(context, name, desc, bufferType, primitiveType)
1319 {
1320 m_progSpec.addTransformFeedbackVarying("gl_Position");
1321 }
1322 };
1323
1324 class PointSizeCase : public TransformFeedbackCase
1325 {
1326 public:
PointSizeCase(Context & context,const char * name,const char * desc,deUint32 bufferType,deUint32 primitiveType)1327 PointSizeCase (Context& context, const char* name, const char* desc, deUint32 bufferType, deUint32 primitiveType)
1328 : TransformFeedbackCase(context, name, desc, bufferType, primitiveType)
1329 {
1330 m_progSpec.addTransformFeedbackVarying("gl_PointSize");
1331 }
1332 };
1333
1334 class BasicTypeCase : public TransformFeedbackCase
1335 {
1336 public:
BasicTypeCase(Context & context,const char * name,const char * desc,deUint32 bufferType,deUint32 primitiveType,glu::DataType type,glu::Precision precision,Interpolation interpolation)1337 BasicTypeCase (Context& context, const char* name, const char* desc, deUint32 bufferType, deUint32 primitiveType, glu::DataType type, glu::Precision precision, Interpolation interpolation)
1338 : TransformFeedbackCase(context, name, desc, bufferType, primitiveType)
1339 {
1340 m_progSpec.addVarying("v_varA", glu::VarType(type, precision), interpolation);
1341 m_progSpec.addVarying("v_varB", glu::VarType(type, precision), interpolation);
1342
1343 m_progSpec.addTransformFeedbackVarying("v_varA");
1344 m_progSpec.addTransformFeedbackVarying("v_varB");
1345 }
1346 };
1347
1348 class BasicArrayCase : public TransformFeedbackCase
1349 {
1350 public:
BasicArrayCase(Context & context,const char * name,const char * desc,deUint32 bufferType,deUint32 primitiveType,glu::DataType type,glu::Precision precision,Interpolation interpolation)1351 BasicArrayCase (Context& context, const char* name, const char* desc, deUint32 bufferType, deUint32 primitiveType, glu::DataType type, glu::Precision precision, Interpolation interpolation)
1352 : TransformFeedbackCase(context, name, desc, bufferType, primitiveType)
1353 {
1354 if (glu::isDataTypeMatrix(type) || m_bufferMode == GL_SEPARATE_ATTRIBS)
1355 {
1356 // \note For matrix types we need to use reduced array sizes or otherwise we will exceed maximum attribute (16)
1357 // or transform feedback component count (64).
1358 // On separate attribs mode maximum component count per varying is 4.
1359 m_progSpec.addVarying("v_varA", glu::VarType(glu::VarType(type, precision), 1), interpolation);
1360 m_progSpec.addVarying("v_varB", glu::VarType(glu::VarType(type, precision), 2), interpolation);
1361 }
1362 else
1363 {
1364 m_progSpec.addVarying("v_varA", glu::VarType(glu::VarType(type, precision), 3), interpolation);
1365 m_progSpec.addVarying("v_varB", glu::VarType(glu::VarType(type, precision), 4), interpolation);
1366 }
1367
1368 m_progSpec.addTransformFeedbackVarying("v_varA");
1369 m_progSpec.addTransformFeedbackVarying("v_varB");
1370 }
1371 };
1372
1373 class ArrayElementCase : public TransformFeedbackCase
1374 {
1375 public:
ArrayElementCase(Context & context,const char * name,const char * desc,deUint32 bufferType,deUint32 primitiveType,glu::DataType type,glu::Precision precision,Interpolation interpolation)1376 ArrayElementCase (Context& context, const char* name, const char* desc, deUint32 bufferType, deUint32 primitiveType, glu::DataType type, glu::Precision precision, Interpolation interpolation)
1377 : TransformFeedbackCase(context, name, desc, bufferType, primitiveType)
1378 {
1379 m_progSpec.addVarying("v_varA", glu::VarType(glu::VarType(type, precision), 3), interpolation);
1380 m_progSpec.addVarying("v_varB", glu::VarType(glu::VarType(type, precision), 4), interpolation);
1381
1382 m_progSpec.addTransformFeedbackVarying("v_varA[1]");
1383 m_progSpec.addTransformFeedbackVarying("v_varB[0]");
1384 m_progSpec.addTransformFeedbackVarying("v_varB[3]");
1385 }
1386 };
1387
1388 class RandomCase : public TransformFeedbackCase
1389 {
1390 public:
RandomCase(Context & context,const char * name,const char * desc,deUint32 bufferType,deUint32 primitiveType,deUint32 seed)1391 RandomCase (Context& context, const char* name, const char* desc, deUint32 bufferType, deUint32 primitiveType, deUint32 seed)
1392 : TransformFeedbackCase (context, name, desc, bufferType, primitiveType)
1393 , m_seed (seed)
1394 {
1395 }
1396
init(void)1397 void init (void)
1398 {
1399 // \note Hard-coded indices and hackery are used when indexing this, beware.
1400 static const glu::DataType typeCandidates[] =
1401 {
1402 glu::TYPE_FLOAT,
1403 glu::TYPE_FLOAT_VEC2,
1404 glu::TYPE_FLOAT_VEC3,
1405 glu::TYPE_FLOAT_VEC4,
1406 glu::TYPE_INT,
1407 glu::TYPE_INT_VEC2,
1408 glu::TYPE_INT_VEC3,
1409 glu::TYPE_INT_VEC4,
1410 glu::TYPE_UINT,
1411 glu::TYPE_UINT_VEC2,
1412 glu::TYPE_UINT_VEC3,
1413 glu::TYPE_UINT_VEC4,
1414
1415 glu::TYPE_FLOAT_MAT2,
1416 glu::TYPE_FLOAT_MAT2X3,
1417 glu::TYPE_FLOAT_MAT2X4,
1418
1419 glu::TYPE_FLOAT_MAT3X2,
1420 glu::TYPE_FLOAT_MAT3,
1421 glu::TYPE_FLOAT_MAT3X4,
1422
1423 glu::TYPE_FLOAT_MAT4X2,
1424 glu::TYPE_FLOAT_MAT4X3,
1425 glu::TYPE_FLOAT_MAT4
1426 };
1427
1428 static const glu::Precision precisions[] =
1429 {
1430 glu::PRECISION_LOWP,
1431 glu::PRECISION_MEDIUMP,
1432 glu::PRECISION_HIGHP
1433 };
1434
1435 static const Interpolation interpModes[] =
1436 {
1437 INTERPOLATION_FLAT,
1438 INTERPOLATION_SMOOTH,
1439 INTERPOLATION_CENTROID
1440 };
1441
1442 const int maxAttributeVectors = 16;
1443 // const int maxTransformFeedbackComponents = 64; // \note It is enough to limit attribute set size.
1444 bool isSeparateMode = m_bufferMode == GL_SEPARATE_ATTRIBS;
1445 int maxTransformFeedbackVars = isSeparateMode ? 4 : maxAttributeVectors;
1446 const float arrayWeight = 0.3f;
1447 const float positionWeight = 0.7f;
1448 const float pointSizeWeight = 0.1f;
1449 const float captureFullArrayWeight = 0.5f;
1450
1451 de::Random rnd (m_seed);
1452 bool usePosition = rnd.getFloat() < positionWeight;
1453 bool usePointSize = rnd.getFloat() < pointSizeWeight;
1454 int numAttribVectorsToUse = rnd.getInt(1, maxAttributeVectors - 1/*position*/ - (usePointSize ? 1 : 0));
1455
1456 int numAttributeVectors = 0;
1457 int varNdx = 0;
1458
1459 // Generate varyings.
1460 while (numAttributeVectors < numAttribVectorsToUse)
1461 {
1462 int maxVecs = isSeparateMode ? de::min(2 /*at most 2*mat2*/, numAttribVectorsToUse-numAttributeVectors) : numAttribVectorsToUse-numAttributeVectors;
1463 const glu::DataType* begin = &typeCandidates[0];
1464 const glu::DataType* end = begin + (maxVecs >= 4 ? 21 :
1465 maxVecs >= 3 ? 18 :
1466 maxVecs >= 2 ? (isSeparateMode ? 13 : 15) : 12);
1467
1468 glu::DataType type = rnd.choose<glu::DataType>(begin, end);
1469 glu::Precision precision = rnd.choose<glu::Precision>(&precisions[0], &precisions[0]+DE_LENGTH_OF_ARRAY(precisions));
1470 Interpolation interp = glu::getDataTypeScalarType(type) == glu::TYPE_FLOAT
1471 ? rnd.choose<Interpolation>(&interpModes[0], &interpModes[0]+DE_LENGTH_OF_ARRAY(interpModes))
1472 : INTERPOLATION_FLAT;
1473 int numVecs = glu::isDataTypeMatrix(type) ? glu::getDataTypeMatrixNumColumns(type) : 1;
1474 int numComps = glu::getDataTypeScalarSize(type);
1475 int maxArrayLen = de::max(1, isSeparateMode ? 4/numComps : maxVecs/numVecs);
1476 bool useArray = rnd.getFloat() < arrayWeight;
1477 int arrayLen = useArray ? rnd.getInt(1, maxArrayLen) : 1;
1478 std::string name = "v_var" + de::toString(varNdx);
1479
1480 if (useArray)
1481 m_progSpec.addVarying(name.c_str(), glu::VarType(glu::VarType(type, precision), arrayLen), interp);
1482 else
1483 m_progSpec.addVarying(name.c_str(), glu::VarType(type, precision), interp);
1484
1485 numAttributeVectors += arrayLen*numVecs;
1486 varNdx += 1;
1487 }
1488
1489 // Generate transform feedback candidate set.
1490 vector<string> tfCandidates;
1491
1492 if (usePosition) tfCandidates.push_back("gl_Position");
1493 if (usePointSize) tfCandidates.push_back("gl_PointSize");
1494
1495 for (int ndx = 0; ndx < varNdx /* num varyings */; ndx++)
1496 {
1497 const Varying& var = m_progSpec.getVaryings()[ndx];
1498
1499 if (var.type.isArrayType())
1500 {
1501 const bool captureFull = rnd.getFloat() < captureFullArrayWeight;
1502
1503 if (captureFull)
1504 tfCandidates.push_back(var.name);
1505 else
1506 {
1507 const int numElem = var.type.getArraySize();
1508 for (int elemNdx = 0; elemNdx < numElem; elemNdx++)
1509 tfCandidates.push_back(var.name + "[" + de::toString(elemNdx) + "]");
1510 }
1511 }
1512 else
1513 tfCandidates.push_back(var.name);
1514 }
1515
1516 // Pick random selection.
1517 vector<string> tfVaryings(de::min((int)tfCandidates.size(), maxTransformFeedbackVars));
1518 rnd.choose(tfCandidates.begin(), tfCandidates.end(), tfVaryings.begin(), (int)tfVaryings.size());
1519 rnd.shuffle(tfVaryings.begin(), tfVaryings.end());
1520
1521 for (vector<string>::const_iterator var = tfVaryings.begin(); var != tfVaryings.end(); var++)
1522 m_progSpec.addTransformFeedbackVarying(var->c_str());
1523
1524 TransformFeedbackCase::init();
1525 }
1526
1527 private:
1528 deUint32 m_seed;
1529 };
1530
1531 } // TransformFeedback
1532
1533 using namespace TransformFeedback;
1534
TransformFeedbackTests(Context & context)1535 TransformFeedbackTests::TransformFeedbackTests (Context& context)
1536 : TestCaseGroup(context, "transform_feedback", "Transform feedback tests")
1537 {
1538 }
1539
~TransformFeedbackTests(void)1540 TransformFeedbackTests::~TransformFeedbackTests (void)
1541 {
1542 }
1543
init(void)1544 void TransformFeedbackTests::init (void)
1545 {
1546 static const struct
1547 {
1548 const char* name;
1549 deUint32 mode;
1550 } bufferModes[] =
1551 {
1552 { "separate", GL_SEPARATE_ATTRIBS },
1553 { "interleaved", GL_INTERLEAVED_ATTRIBS }
1554 };
1555
1556 static const struct
1557 {
1558 const char* name;
1559 deUint32 type;
1560 } primitiveTypes[] =
1561 {
1562 { "points", GL_POINTS },
1563 { "lines", GL_LINES },
1564 { "triangles", GL_TRIANGLES }
1565
1566 // Not supported by GLES3.
1567 // { "line_strip", GL_LINE_STRIP },
1568 // { "line_loop", GL_LINE_LOOP },
1569 // { "triangle_fan", GL_TRIANGLE_FAN },
1570 // { "triangle_strip", GL_TRIANGLE_STRIP }
1571 };
1572
1573 static const glu::DataType basicTypes[] =
1574 {
1575 glu::TYPE_FLOAT,
1576 glu::TYPE_FLOAT_VEC2,
1577 glu::TYPE_FLOAT_VEC3,
1578 glu::TYPE_FLOAT_VEC4,
1579 glu::TYPE_FLOAT_MAT2,
1580 glu::TYPE_FLOAT_MAT2X3,
1581 glu::TYPE_FLOAT_MAT2X4,
1582 glu::TYPE_FLOAT_MAT3X2,
1583 glu::TYPE_FLOAT_MAT3,
1584 glu::TYPE_FLOAT_MAT3X4,
1585 glu::TYPE_FLOAT_MAT4X2,
1586 glu::TYPE_FLOAT_MAT4X3,
1587 glu::TYPE_FLOAT_MAT4,
1588 glu::TYPE_INT,
1589 glu::TYPE_INT_VEC2,
1590 glu::TYPE_INT_VEC3,
1591 glu::TYPE_INT_VEC4,
1592 glu::TYPE_UINT,
1593 glu::TYPE_UINT_VEC2,
1594 glu::TYPE_UINT_VEC3,
1595 glu::TYPE_UINT_VEC4
1596 };
1597
1598 static const glu::Precision precisions[] =
1599 {
1600 glu::PRECISION_LOWP,
1601 glu::PRECISION_MEDIUMP,
1602 glu::PRECISION_HIGHP
1603 };
1604
1605 static const struct
1606 {
1607 const char* name;
1608 Interpolation interp;
1609 } interpModes[] =
1610 {
1611 { "smooth", INTERPOLATION_SMOOTH },
1612 { "flat", INTERPOLATION_FLAT },
1613 { "centroid", INTERPOLATION_CENTROID }
1614 };
1615
1616 // .position
1617 {
1618 tcu::TestCaseGroup* positionGroup = new tcu::TestCaseGroup(m_testCtx, "position", "gl_Position capture using transform feedback");
1619 addChild(positionGroup);
1620
1621 for (int primitiveType = 0; primitiveType < DE_LENGTH_OF_ARRAY(primitiveTypes); primitiveType++)
1622 {
1623 for (int bufferMode = 0; bufferMode < DE_LENGTH_OF_ARRAY(bufferModes); bufferMode++)
1624 {
1625 string name = string(primitiveTypes[primitiveType].name) + "_" + bufferModes[bufferMode].name;
1626 positionGroup->addChild(new PositionCase(m_context, name.c_str(), "", bufferModes[bufferMode].mode, primitiveTypes[primitiveType].type));
1627 }
1628 }
1629 }
1630
1631 // .point_size
1632 {
1633 tcu::TestCaseGroup* pointSizeGroup = new tcu::TestCaseGroup(m_testCtx, "point_size", "gl_PointSize capture using transform feedback");
1634 addChild(pointSizeGroup);
1635
1636 for (int primitiveType = 0; primitiveType < DE_LENGTH_OF_ARRAY(primitiveTypes); primitiveType++)
1637 {
1638 for (int bufferMode = 0; bufferMode < DE_LENGTH_OF_ARRAY(bufferModes); bufferMode++)
1639 {
1640 string name = string(primitiveTypes[primitiveType].name) + "_" + bufferModes[bufferMode].name;
1641 pointSizeGroup->addChild(new PointSizeCase(m_context, name.c_str(), "", bufferModes[bufferMode].mode, primitiveTypes[primitiveType].type));
1642 }
1643 }
1644 }
1645
1646 // .basic_type
1647 {
1648 tcu::TestCaseGroup* basicTypeGroup = new tcu::TestCaseGroup(m_testCtx, "basic_types", "Basic types in transform feedback");
1649 addChild(basicTypeGroup);
1650
1651 for (int bufferModeNdx = 0; bufferModeNdx < DE_LENGTH_OF_ARRAY(bufferModes); bufferModeNdx++)
1652 {
1653 tcu::TestCaseGroup* modeGroup = new tcu::TestCaseGroup(m_testCtx, bufferModes[bufferModeNdx].name, "");
1654 deUint32 bufferMode = bufferModes[bufferModeNdx].mode;
1655 basicTypeGroup->addChild(modeGroup);
1656
1657 for (int primitiveTypeNdx = 0; primitiveTypeNdx < DE_LENGTH_OF_ARRAY(primitiveTypes); primitiveTypeNdx++)
1658 {
1659 tcu::TestCaseGroup* primitiveGroup = new tcu::TestCaseGroup(m_testCtx, primitiveTypes[primitiveTypeNdx].name, "");
1660 deUint32 primitiveType = primitiveTypes[primitiveTypeNdx].type;
1661 modeGroup->addChild(primitiveGroup);
1662
1663 for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(basicTypes); typeNdx++)
1664 {
1665 glu::DataType type = basicTypes[typeNdx];
1666 bool isFloat = glu::getDataTypeScalarType(type) == glu::TYPE_FLOAT;
1667
1668 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1669 {
1670 glu::Precision precision = precisions[precNdx];
1671
1672 string name = string(glu::getPrecisionName(precision)) + "_" + glu::getDataTypeName(type);
1673 primitiveGroup->addChild(new BasicTypeCase(m_context, name.c_str(), "", bufferMode, primitiveType, type, precision, isFloat ? INTERPOLATION_SMOOTH : INTERPOLATION_FLAT));
1674 }
1675 }
1676 }
1677 }
1678 }
1679
1680 // .array
1681 {
1682 tcu::TestCaseGroup* arrayGroup = new tcu::TestCaseGroup(m_testCtx, "array", "Capturing whole array in TF");
1683 addChild(arrayGroup);
1684
1685 for (int bufferModeNdx = 0; bufferModeNdx < DE_LENGTH_OF_ARRAY(bufferModes); bufferModeNdx++)
1686 {
1687 tcu::TestCaseGroup* modeGroup = new tcu::TestCaseGroup(m_testCtx, bufferModes[bufferModeNdx].name, "");
1688 deUint32 bufferMode = bufferModes[bufferModeNdx].mode;
1689 arrayGroup->addChild(modeGroup);
1690
1691 for (int primitiveTypeNdx = 0; primitiveTypeNdx < DE_LENGTH_OF_ARRAY(primitiveTypes); primitiveTypeNdx++)
1692 {
1693 tcu::TestCaseGroup* primitiveGroup = new tcu::TestCaseGroup(m_testCtx, primitiveTypes[primitiveTypeNdx].name, "");
1694 deUint32 primitiveType = primitiveTypes[primitiveTypeNdx].type;
1695 modeGroup->addChild(primitiveGroup);
1696
1697 for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(basicTypes); typeNdx++)
1698 {
1699 glu::DataType type = basicTypes[typeNdx];
1700 bool isFloat = glu::getDataTypeScalarType(type) == glu::TYPE_FLOAT;
1701
1702 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1703 {
1704 glu::Precision precision = precisions[precNdx];
1705
1706 string name = string(glu::getPrecisionName(precision)) + "_" + glu::getDataTypeName(type);
1707 primitiveGroup->addChild(new BasicArrayCase(m_context, name.c_str(), "", bufferMode, primitiveType, type, precision, isFloat ? INTERPOLATION_SMOOTH : INTERPOLATION_FLAT));
1708 }
1709 }
1710 }
1711 }
1712 }
1713
1714 // .array_element
1715 {
1716 tcu::TestCaseGroup* arrayElemGroup = new tcu::TestCaseGroup(m_testCtx, "array_element", "Capturing single array element in TF");
1717 addChild(arrayElemGroup);
1718
1719 for (int bufferModeNdx = 0; bufferModeNdx < DE_LENGTH_OF_ARRAY(bufferModes); bufferModeNdx++)
1720 {
1721 tcu::TestCaseGroup* modeGroup = new tcu::TestCaseGroup(m_testCtx, bufferModes[bufferModeNdx].name, "");
1722 deUint32 bufferMode = bufferModes[bufferModeNdx].mode;
1723 arrayElemGroup->addChild(modeGroup);
1724
1725 for (int primitiveTypeNdx = 0; primitiveTypeNdx < DE_LENGTH_OF_ARRAY(primitiveTypes); primitiveTypeNdx++)
1726 {
1727 tcu::TestCaseGroup* primitiveGroup = new tcu::TestCaseGroup(m_testCtx, primitiveTypes[primitiveTypeNdx].name, "");
1728 deUint32 primitiveType = primitiveTypes[primitiveTypeNdx].type;
1729 modeGroup->addChild(primitiveGroup);
1730
1731 for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(basicTypes); typeNdx++)
1732 {
1733 glu::DataType type = basicTypes[typeNdx];
1734 bool isFloat = glu::getDataTypeScalarType(type) == glu::TYPE_FLOAT;
1735
1736 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1737 {
1738 glu::Precision precision = precisions[precNdx];
1739
1740 string name = string(glu::getPrecisionName(precision)) + "_" + glu::getDataTypeName(type);
1741 primitiveGroup->addChild(new ArrayElementCase(m_context, name.c_str(), "", bufferMode, primitiveType, type, precision, isFloat ? INTERPOLATION_SMOOTH : INTERPOLATION_FLAT));
1742 }
1743 }
1744 }
1745 }
1746 }
1747
1748 // .interpolation
1749 {
1750 tcu::TestCaseGroup* interpolationGroup = new tcu::TestCaseGroup(m_testCtx, "interpolation", "Different interpolation modes in transform feedback varyings");
1751 addChild(interpolationGroup);
1752
1753 for (int modeNdx = 0; modeNdx < DE_LENGTH_OF_ARRAY(interpModes); modeNdx++)
1754 {
1755 Interpolation interp = interpModes[modeNdx].interp;
1756 tcu::TestCaseGroup* modeGroup = new tcu::TestCaseGroup(m_testCtx, interpModes[modeNdx].name, "");
1757
1758 interpolationGroup->addChild(modeGroup);
1759
1760 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1761 {
1762 glu::Precision precision = precisions[precNdx];
1763
1764 for (int primitiveType = 0; primitiveType < DE_LENGTH_OF_ARRAY(primitiveTypes); primitiveType++)
1765 {
1766 for (int bufferMode = 0; bufferMode < DE_LENGTH_OF_ARRAY(bufferModes); bufferMode++)
1767 {
1768 string name = string(glu::getPrecisionName(precision)) + "_vec4_" + primitiveTypes[primitiveType].name + "_" + bufferModes[bufferMode].name;
1769 modeGroup->addChild(new BasicTypeCase(m_context, name.c_str(), "", bufferModes[bufferMode].mode, primitiveTypes[primitiveType].type, glu::TYPE_FLOAT_VEC4, precision, interp));
1770 }
1771 }
1772 }
1773 }
1774 }
1775
1776 // .random
1777 {
1778 tcu::TestCaseGroup* randomGroup = new tcu::TestCaseGroup(m_testCtx, "random", "Randomized transform feedback cases");
1779 addChild(randomGroup);
1780
1781 for (int bufferModeNdx = 0; bufferModeNdx < DE_LENGTH_OF_ARRAY(bufferModes); bufferModeNdx++)
1782 {
1783 tcu::TestCaseGroup* modeGroup = new tcu::TestCaseGroup(m_testCtx, bufferModes[bufferModeNdx].name, "");
1784 deUint32 bufferMode = bufferModes[bufferModeNdx].mode;
1785 randomGroup->addChild(modeGroup);
1786
1787 for (int primitiveTypeNdx = 0; primitiveTypeNdx < DE_LENGTH_OF_ARRAY(primitiveTypes); primitiveTypeNdx++)
1788 {
1789 tcu::TestCaseGroup* primitiveGroup = new tcu::TestCaseGroup(m_testCtx, primitiveTypes[primitiveTypeNdx].name, "");
1790 deUint32 primitiveType = primitiveTypes[primitiveTypeNdx].type;
1791 modeGroup->addChild(primitiveGroup);
1792
1793 for (int ndx = 0; ndx < 10; ndx++)
1794 {
1795 deUint32 seed = deInt32Hash(bufferMode) ^ deInt32Hash(primitiveType) ^ deInt32Hash(ndx);
1796 primitiveGroup->addChild(new RandomCase(m_context, de::toString(ndx+1).c_str(), "", bufferMode, primitiveType, seed));
1797 }
1798 }
1799 }
1800 }
1801 }
1802
1803 } // Functional
1804 } // gles3
1805 } // deqp
1806