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 Fragment shader output tests.
22 *
23 * \todo [2012-04-10 pyry] Missing:
24 * + non-contiguous attachments in framebuffer
25 *//*--------------------------------------------------------------------*/
26
27 #include "es3fFragmentOutputTests.hpp"
28 #include "gluShaderUtil.hpp"
29 #include "gluShaderProgram.hpp"
30 #include "gluTextureUtil.hpp"
31 #include "gluStrUtil.hpp"
32 #include "tcuTestLog.hpp"
33 #include "tcuTexture.hpp"
34 #include "tcuTextureUtil.hpp"
35 #include "tcuVector.hpp"
36 #include "tcuVectorUtil.hpp"
37 #include "tcuImageCompare.hpp"
38 #include "deRandom.hpp"
39 #include "deStringUtil.hpp"
40 #include "deMath.h"
41
42 // For getFormatName() \todo [pyry] Move to glu?
43 #include "es3fFboTestUtil.hpp"
44
45 #include "glwEnums.hpp"
46 #include "glwFunctions.hpp"
47
48 namespace deqp
49 {
50 namespace gles3
51 {
52 namespace Functional
53 {
54
55 using std::vector;
56 using std::string;
57 using tcu::IVec2;
58 using tcu::IVec4;
59 using tcu::UVec2;
60 using tcu::UVec4;
61 using tcu::Vec2;
62 using tcu::Vec3;
63 using tcu::Vec4;
64 using tcu::BVec4;
65 using tcu::TestLog;
66 using FboTestUtil::getFormatName;
67 using FboTestUtil::getFramebufferReadFormat;
68
69 struct BufferSpec
70 {
BufferSpecdeqp::gles3::Functional::BufferSpec71 BufferSpec (void)
72 : format (GL_NONE)
73 , width (0)
74 , height (0)
75 , samples (0)
76 {
77 }
78
BufferSpecdeqp::gles3::Functional::BufferSpec79 BufferSpec (deUint32 format_, int width_, int height_, int samples_)
80 : format (format_)
81 , width (width_)
82 , height (height_)
83 , samples (samples_)
84 {
85 }
86
87 deUint32 format;
88 int width;
89 int height;
90 int samples;
91 };
92
93 struct FragmentOutput
94 {
FragmentOutputdeqp::gles3::Functional::FragmentOutput95 FragmentOutput (void)
96 : type (glu::TYPE_LAST)
97 , precision (glu::PRECISION_LAST)
98 , location (0)
99 , arrayLength (0)
100 {
101 }
102
FragmentOutputdeqp::gles3::Functional::FragmentOutput103 FragmentOutput (glu::DataType type_, glu::Precision precision_, int location_, int arrayLength_ = 0)
104 : type (type_)
105 , precision (precision_)
106 , location (location_)
107 , arrayLength (arrayLength_)
108 {
109 }
110
111 glu::DataType type;
112 glu::Precision precision;
113 int location;
114 int arrayLength; //!< 0 if not an array.
115 };
116
117 struct OutputVec
118 {
119 vector<FragmentOutput> outputs;
120
operator <<deqp::gles3::Functional::OutputVec121 OutputVec& operator<< (const FragmentOutput& output)
122 {
123 outputs.push_back(output);
124 return *this;
125 }
126
toVecdeqp::gles3::Functional::OutputVec127 vector<FragmentOutput> toVec (void) const
128 {
129 return outputs;
130 }
131 };
132
133 class FragmentOutputCase : public TestCase
134 {
135 public:
136 FragmentOutputCase (Context& context, const char* name, const char* desc, const vector<BufferSpec>& fboSpec, const vector<FragmentOutput>& outputs);
137 ~FragmentOutputCase (void);
138
139 void init (void);
140 void deinit (void);
141 IterateResult iterate (void);
142
143 private:
144 FragmentOutputCase (const FragmentOutputCase& other);
145 FragmentOutputCase& operator= (const FragmentOutputCase& other);
146
147 vector<BufferSpec> m_fboSpec;
148 vector<FragmentOutput> m_outputs;
149
150 glu::ShaderProgram* m_program;
151 deUint32 m_framebuffer;
152 vector<deUint32> m_renderbuffers;
153 };
154
FragmentOutputCase(Context & context,const char * name,const char * desc,const vector<BufferSpec> & fboSpec,const vector<FragmentOutput> & outputs)155 FragmentOutputCase::FragmentOutputCase (Context& context, const char* name, const char* desc, const vector<BufferSpec>& fboSpec, const vector<FragmentOutput>& outputs)
156 : TestCase (context, name, desc)
157 , m_fboSpec (fboSpec)
158 , m_outputs (outputs)
159 , m_program (DE_NULL)
160 , m_framebuffer (0)
161 {
162 }
163
~FragmentOutputCase(void)164 FragmentOutputCase::~FragmentOutputCase (void)
165 {
166 deinit();
167 }
168
createProgram(const glu::RenderContext & context,const vector<FragmentOutput> & outputs)169 static glu::ShaderProgram* createProgram (const glu::RenderContext& context, const vector<FragmentOutput>& outputs)
170 {
171 std::ostringstream vtx;
172 std::ostringstream frag;
173
174 vtx << "#version 300 es\n"
175 << "in highp vec4 a_position;\n";
176 frag << "#version 300 es\n";
177
178 // Input-output declarations.
179 for (int outNdx = 0; outNdx < (int)outputs.size(); outNdx++)
180 {
181 const FragmentOutput& output = outputs[outNdx];
182 bool isArray = output.arrayLength > 0;
183 const char* typeName = glu::getDataTypeName(output.type);
184 const char* outputPrec = glu::getPrecisionName(output.precision);
185 bool isFloat = glu::isDataTypeFloatOrVec(output.type);
186 const char* interp = isFloat ? "smooth" : "flat";
187 const char* interpPrec = isFloat ? "highp" : outputPrec;
188
189 if (isArray)
190 {
191 for (int elemNdx = 0; elemNdx < output.arrayLength; elemNdx++)
192 {
193 vtx << "in " << interpPrec << " " << typeName << " in" << outNdx << "_" << elemNdx << ";\n"
194 << interp << " out " << interpPrec << " " << typeName << " var" << outNdx << "_" << elemNdx << ";\n";
195 frag << interp << " in " << interpPrec << " " << typeName << " var" << outNdx << "_" << elemNdx << ";\n";
196 }
197 frag << "layout(location = " << output.location << ") out " << outputPrec << " " << typeName << " out" << outNdx << "[" << output.arrayLength << "];\n";
198 }
199 else
200 {
201 vtx << "in " << interpPrec << " " << typeName << " in" << outNdx << ";\n"
202 << interp << " out " << interpPrec << " " << typeName << " var" << outNdx << ";\n";
203 frag << interp << " in " << interpPrec << " " << typeName << " var" << outNdx << ";\n"
204 << "layout(location = " << output.location << ") out " << outputPrec << " " << typeName << " out" << outNdx << ";\n";
205 }
206 }
207
208 vtx << "\nvoid main()\n{\n";
209 frag << "\nvoid main()\n{\n";
210
211 vtx << " gl_Position = a_position;\n";
212
213 // Copy body
214 for (int outNdx = 0; outNdx < (int)outputs.size(); outNdx++)
215 {
216 const FragmentOutput& output = outputs[outNdx];
217 bool isArray = output.arrayLength > 0;
218
219 if (isArray)
220 {
221 for (int elemNdx = 0; elemNdx < output.arrayLength; elemNdx++)
222 {
223 vtx << "\tvar" << outNdx << "_" << elemNdx << " = in" << outNdx << "_" << elemNdx << ";\n";
224 frag << "\tout" << outNdx << "[" << elemNdx << "] = var" << outNdx << "_" << elemNdx << ";\n";
225 }
226 }
227 else
228 {
229 vtx << "\tvar" << outNdx << " = in" << outNdx << ";\n";
230 frag << "\tout" << outNdx << " = var" << outNdx << ";\n";
231 }
232 }
233
234 vtx << "}\n";
235 frag << "}\n";
236
237 return new glu::ShaderProgram(context, glu::makeVtxFragSources(vtx.str(), frag.str()));
238 }
239
init(void)240 void FragmentOutputCase::init (void)
241 {
242 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
243 TestLog& log = m_testCtx.getLog();
244
245 // Check that all attachments are supported
246 for (std::vector<BufferSpec>::const_iterator bufIter = m_fboSpec.begin(); bufIter != m_fboSpec.end(); ++bufIter)
247 {
248 if (!glu::isSizedFormatColorRenderable(m_context.getRenderContext(), m_context.getContextInfo(), bufIter->format))
249 throw tcu::NotSupportedError("Unsupported attachment format");
250 }
251
252 DE_ASSERT(!m_program);
253 m_program = createProgram(m_context.getRenderContext(), m_outputs);
254
255 log << *m_program;
256 if (!m_program->isOk())
257 TCU_FAIL("Compile failed");
258
259 // Print render target info to log.
260 log << TestLog::Section("Framebuffer", "Framebuffer configuration");
261
262 for (int ndx = 0; ndx < (int)m_fboSpec.size(); ndx++)
263 log << TestLog::Message << "COLOR_ATTACHMENT" << ndx << ": "
264 << glu::getTextureFormatStr(m_fboSpec[ndx].format) << ", "
265 << m_fboSpec[ndx].width << "x" << m_fboSpec[ndx].height << ", "
266 << m_fboSpec[ndx].samples << " samples"
267 << TestLog::EndMessage;
268
269 log << TestLog::EndSection;
270
271 // Create framebuffer.
272 m_renderbuffers.resize(m_fboSpec.size(), 0);
273 gl.genFramebuffers(1, &m_framebuffer);
274 gl.genRenderbuffers((int)m_renderbuffers.size(), &m_renderbuffers[0]);
275
276 gl.bindFramebuffer(GL_FRAMEBUFFER, m_framebuffer);
277
278 for (int bufNdx = 0; bufNdx < (int)m_renderbuffers.size(); bufNdx++)
279 {
280 deUint32 rbo = m_renderbuffers[bufNdx];
281 const BufferSpec& bufSpec = m_fboSpec[bufNdx];
282 deUint32 attachment = GL_COLOR_ATTACHMENT0+bufNdx;
283
284 gl.bindRenderbuffer(GL_RENDERBUFFER, rbo);
285 gl.renderbufferStorageMultisample(GL_RENDERBUFFER, bufSpec.samples, bufSpec.format, bufSpec.width, bufSpec.height);
286 gl.framebufferRenderbuffer(GL_FRAMEBUFFER, attachment, GL_RENDERBUFFER, rbo);
287 }
288 GLU_EXPECT_NO_ERROR(gl.getError(), "After framebuffer setup");
289
290 deUint32 fboStatus = gl.checkFramebufferStatus(GL_FRAMEBUFFER);
291 if (fboStatus == GL_FRAMEBUFFER_UNSUPPORTED)
292 throw tcu::NotSupportedError("Framebuffer not supported", "", __FILE__, __LINE__);
293 else if (fboStatus != GL_FRAMEBUFFER_COMPLETE)
294 throw tcu::TestError((string("Incomplete framebuffer: ") + glu::getFramebufferStatusStr(fboStatus).toString()).c_str(), "", __FILE__, __LINE__);
295
296 gl.bindFramebuffer(GL_FRAMEBUFFER, 0);
297 GLU_EXPECT_NO_ERROR(gl.getError(), "After init");
298 }
299
deinit(void)300 void FragmentOutputCase::deinit (void)
301 {
302 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
303
304 if (m_framebuffer)
305 {
306 gl.deleteFramebuffers(1, &m_framebuffer);
307 m_framebuffer = 0;
308 }
309
310 if (!m_renderbuffers.empty())
311 {
312 gl.deleteRenderbuffers((int)m_renderbuffers.size(), &m_renderbuffers[0]);
313 m_renderbuffers.clear();
314 }
315
316 delete m_program;
317 m_program = DE_NULL;
318 }
319
getMinSize(const vector<BufferSpec> & fboSpec)320 static IVec2 getMinSize (const vector<BufferSpec>& fboSpec)
321 {
322 IVec2 minSize(0x7fffffff, 0x7fffffff);
323 for (vector<BufferSpec>::const_iterator i = fboSpec.begin(); i != fboSpec.end(); i++)
324 {
325 minSize.x() = de::min(minSize.x(), i->width);
326 minSize.y() = de::min(minSize.y(), i->height);
327 }
328 return minSize;
329 }
330
getNumInputVectors(const vector<FragmentOutput> & outputs)331 static int getNumInputVectors (const vector<FragmentOutput>& outputs)
332 {
333 int numVecs = 0;
334 for (vector<FragmentOutput>::const_iterator i = outputs.begin(); i != outputs.end(); i++)
335 numVecs += (i->arrayLength > 0 ? i->arrayLength : 1);
336 return numVecs;
337 }
338
getFloatRange(glu::Precision precision)339 static Vec2 getFloatRange (glu::Precision precision)
340 {
341 // \todo [2012-04-09 pyry] Not quite the full ranges.
342 static const Vec2 ranges[] =
343 {
344 Vec2(-2.0f, 2.0f),
345 Vec2(-16000.0f, 16000.0f),
346 Vec2(-1e35f, 1e35f)
347 };
348 DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(ranges) == glu::PRECISION_LAST);
349 DE_ASSERT(de::inBounds<int>(precision, 0, DE_LENGTH_OF_ARRAY(ranges)));
350 return ranges[precision];
351 }
352
getIntRange(glu::Precision precision)353 static IVec2 getIntRange (glu::Precision precision)
354 {
355 static const IVec2 ranges[] =
356 {
357 IVec2(-(1<< 7), (1<< 7)-1),
358 IVec2(-(1<<15), (1<<15)-1),
359 IVec2(0x80000000, 0x7fffffff)
360 };
361 DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(ranges) == glu::PRECISION_LAST);
362 DE_ASSERT(de::inBounds<int>(precision, 0, DE_LENGTH_OF_ARRAY(ranges)));
363 return ranges[precision];
364 }
365
getUintRange(glu::Precision precision)366 static UVec2 getUintRange (glu::Precision precision)
367 {
368 static const UVec2 ranges[] =
369 {
370 UVec2(0, (1<< 8)-1),
371 UVec2(0, (1<<16)-1),
372 UVec2(0, 0xffffffffu)
373 };
374 DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(ranges) == glu::PRECISION_LAST);
375 DE_ASSERT(de::inBounds<int>(precision, 0, DE_LENGTH_OF_ARRAY(ranges)));
376 return ranges[precision];
377 }
378
readVec4(const float * ptr,int numComponents)379 static inline Vec4 readVec4 (const float* ptr, int numComponents)
380 {
381 DE_ASSERT(numComponents >= 1);
382 return Vec4(ptr[0],
383 numComponents >= 2 ? ptr[1] : 0.0f,
384 numComponents >= 3 ? ptr[2] : 0.0f,
385 numComponents >= 4 ? ptr[3] : 0.0f);
386 }
387
readIVec4(const int * ptr,int numComponents)388 static inline IVec4 readIVec4 (const int* ptr, int numComponents)
389 {
390 DE_ASSERT(numComponents >= 1);
391 return IVec4(ptr[0],
392 numComponents >= 2 ? ptr[1] : 0,
393 numComponents >= 3 ? ptr[2] : 0,
394 numComponents >= 4 ? ptr[3] : 0);
395 }
396
renderFloatReference(const tcu::PixelBufferAccess & dst,int gridWidth,int gridHeight,int numComponents,const float * vertices)397 static void renderFloatReference (const tcu::PixelBufferAccess& dst, int gridWidth, int gridHeight, int numComponents, const float* vertices)
398 {
399 const bool isSRGB = tcu::isSRGB(dst.getFormat());
400 const float cellW = (float)dst.getWidth() / (float)(gridWidth-1);
401 const float cellH = (float)dst.getHeight() / (float)(gridHeight-1);
402
403 for (int y = 0; y < dst.getHeight(); y++)
404 {
405 for (int x = 0; x < dst.getWidth(); x++)
406 {
407 const int cellX = de::clamp(deFloorFloatToInt32((float)x / cellW), 0, gridWidth-2);
408 const int cellY = de::clamp(deFloorFloatToInt32((float)y / cellH), 0, gridHeight-2);
409 const float xf = ((float)x - (float)cellX*cellW + 0.5f) / cellW;
410 const float yf = ((float)y - (float)cellY*cellH + 0.5f) / cellH;
411 const Vec4 v00 = readVec4(vertices + ((cellY+0)*gridWidth + cellX+0)*numComponents, numComponents);
412 const Vec4 v01 = readVec4(vertices + ((cellY+1)*gridWidth + cellX+0)*numComponents, numComponents);
413 const Vec4 v10 = readVec4(vertices + ((cellY+0)*gridWidth + cellX+1)*numComponents, numComponents);
414 const Vec4 v11 = readVec4(vertices + ((cellY+1)*gridWidth + cellX+1)*numComponents, numComponents);
415 const bool tri = xf + yf >= 1.0f;
416 const Vec4& v0 = tri ? v11 : v00;
417 const Vec4& v1 = tri ? v01 : v10;
418 const Vec4& v2 = tri ? v10 : v01;
419 const float s = tri ? 1.0f-xf : xf;
420 const float t = tri ? 1.0f-yf : yf;
421 const Vec4 color = v0 + (v1-v0)*s + (v2-v0)*t;
422
423 dst.setPixel(isSRGB ? tcu::linearToSRGB(color) : color, x, y);
424 }
425 }
426 }
427
renderIntReference(const tcu::PixelBufferAccess & dst,int gridWidth,int gridHeight,int numComponents,const int * vertices)428 static void renderIntReference (const tcu::PixelBufferAccess& dst, int gridWidth, int gridHeight, int numComponents, const int* vertices)
429 {
430 float cellW = (float)dst.getWidth() / (float)(gridWidth-1);
431 float cellH = (float)dst.getHeight() / (float)(gridHeight-1);
432
433 for (int y = 0; y < dst.getHeight(); y++)
434 {
435 for (int x = 0; x < dst.getWidth(); x++)
436 {
437 int cellX = de::clamp(deFloorFloatToInt32((float)x / cellW), 0, gridWidth-2);
438 int cellY = de::clamp(deFloorFloatToInt32((float)y / cellH), 0, gridHeight-2);
439 IVec4 c = readIVec4(vertices + (cellY*gridWidth + cellX+1)*numComponents, numComponents);
440
441 dst.setPixel(c, x, y);
442 }
443 }
444 }
445
446 static const IVec4 s_swizzles[] =
447 {
448 IVec4(0,1,2,3),
449 IVec4(1,2,3,0),
450 IVec4(2,3,0,1),
451 IVec4(3,0,1,2),
452 IVec4(3,2,1,0),
453 IVec4(2,1,0,3),
454 IVec4(1,0,3,2),
455 IVec4(0,3,2,1)
456 };
457
458 template <typename T>
swizzleVec(const tcu::Vector<T,4> & vec,int swzNdx)459 inline tcu::Vector<T, 4> swizzleVec (const tcu::Vector<T, 4>& vec, int swzNdx)
460 {
461 const IVec4& swz = s_swizzles[swzNdx % DE_LENGTH_OF_ARRAY(s_swizzles)];
462 return vec.swizzle(swz[0], swz[1], swz[2], swz[3]);
463 }
464
465 namespace
466 {
467
468 struct AttachmentData
469 {
470 tcu::TextureFormat format; //!< Actual format of attachment.
471 tcu::TextureFormat referenceFormat; //!< Used for reference rendering.
472 tcu::TextureFormat readFormat;
473 int numWrittenChannels;
474 glu::Precision outPrecision;
475 vector<deUint8> renderedData;
476 vector<deUint8> referenceData;
477 };
478
479 template<typename Type>
valueRangeToString(int numValidChannels,const tcu::Vector<Type,4> & minValue,const tcu::Vector<Type,4> & maxValue)480 string valueRangeToString (int numValidChannels, const tcu::Vector<Type, 4>& minValue, const tcu::Vector<Type, 4>& maxValue)
481 {
482 std::ostringstream stream;
483
484 stream << "(";
485
486 for (int i = 0; i < 4; i++)
487 {
488 if (i != 0)
489 stream << ", ";
490
491 if (i < numValidChannels)
492 stream << minValue[i] << " -> " << maxValue[i];
493 else
494 stream << "Undef";
495 }
496
497 stream << ")";
498
499 return stream.str();
500 }
501
clearUndefined(const tcu::PixelBufferAccess & access,int numValidChannels)502 void clearUndefined (const tcu::PixelBufferAccess& access, int numValidChannels)
503 {
504 for (int y = 0; y < access.getHeight(); y++)
505 for (int x = 0; x < access.getWidth(); x++)
506 {
507 switch (tcu::getTextureChannelClass(access.getFormat().type))
508 {
509 case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
510 {
511 const Vec4 srcPixel = access.getPixel(x, y);
512 Vec4 dstPixel (0.0f, 0.0f, 0.0f, 1.0f);
513
514 for (int channelNdx = 0; channelNdx < numValidChannels; channelNdx++)
515 dstPixel[channelNdx] = srcPixel[channelNdx];
516
517 access.setPixel(dstPixel, x, y);
518 break;
519 }
520
521 case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
522 case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
523 case tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT:
524 case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
525 {
526 const IVec4 bitDepth = tcu::getTextureFormatBitDepth(access.getFormat());
527 const IVec4 srcPixel = access.getPixelInt(x, y);
528 IVec4 dstPixel (0, 0, 0, (int)(((deInt64)0x1u << bitDepth.w()) - 1));
529
530 for (int channelNdx = 0; channelNdx < numValidChannels; channelNdx++)
531 dstPixel[channelNdx] = srcPixel[channelNdx];
532
533 access.setPixel(dstPixel, x, y);
534 break;
535 }
536
537 default:
538 DE_ASSERT(false);
539 }
540 }
541 }
542
543 } // anonymous
544
iterate(void)545 FragmentOutputCase::IterateResult FragmentOutputCase::iterate (void)
546 {
547 TestLog& log = m_testCtx.getLog();
548 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
549
550 // Compute grid size & index list.
551 const int minCellSize = 8;
552 const IVec2 minBufSize = getMinSize(m_fboSpec);
553 const int gridWidth = de::clamp(minBufSize.x()/minCellSize, 1, 255)+1;
554 const int gridHeight = de::clamp(minBufSize.y()/minCellSize, 1, 255)+1;
555 const int numVertices = gridWidth*gridHeight;
556 const int numQuads = (gridWidth-1)*(gridHeight-1);
557 const int numIndices = numQuads*6;
558
559 const int numInputVecs = getNumInputVectors(m_outputs);
560 vector<vector<deUint32> > inputs (numInputVecs);
561 vector<float> positions (numVertices*4);
562 vector<deUint16> indices (numIndices);
563
564 const int readAlignment = 4;
565 const int viewportW = minBufSize.x();
566 const int viewportH = minBufSize.y();
567 const int numAttachments = (int)m_fboSpec.size();
568
569 vector<deUint32> drawBuffers (numAttachments);
570 vector<AttachmentData> attachments (numAttachments);
571
572 // Initialize attachment data.
573 for (int ndx = 0; ndx < numAttachments; ndx++)
574 {
575 const tcu::TextureFormat texFmt = glu::mapGLInternalFormat(m_fboSpec[ndx].format);
576 const tcu::TextureChannelClass chnClass = tcu::getTextureChannelClass(texFmt.type);
577 const bool isFixedPoint = chnClass == tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT ||
578 chnClass == tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT;
579
580 // \note Fixed-point formats use float reference to enable more accurate result verification.
581 const tcu::TextureFormat refFmt = isFixedPoint ? tcu::TextureFormat(texFmt.order, tcu::TextureFormat::FLOAT) : texFmt;
582 const tcu::TextureFormat readFmt = getFramebufferReadFormat(texFmt);
583 const int attachmentW = m_fboSpec[ndx].width;
584 const int attachmentH = m_fboSpec[ndx].height;
585
586 drawBuffers[ndx] = GL_COLOR_ATTACHMENT0+ndx;
587 attachments[ndx].format = texFmt;
588 attachments[ndx].readFormat = readFmt;
589 attachments[ndx].referenceFormat = refFmt;
590 attachments[ndx].renderedData.resize(readFmt.getPixelSize()*attachmentW*attachmentH);
591 attachments[ndx].referenceData.resize(refFmt.getPixelSize()*attachmentW*attachmentH);
592 }
593
594 // Initialize indices.
595 for (int quadNdx = 0; quadNdx < numQuads; quadNdx++)
596 {
597 int quadY = quadNdx / (gridWidth-1);
598 int quadX = quadNdx - quadY*(gridWidth-1);
599
600 indices[quadNdx*6+0] = deUint16(quadX + quadY*gridWidth);
601 indices[quadNdx*6+1] = deUint16(quadX + (quadY+1)*gridWidth);
602 indices[quadNdx*6+2] = deUint16(quadX + quadY*gridWidth + 1);
603 indices[quadNdx*6+3] = indices[quadNdx*6+1];
604 indices[quadNdx*6+4] = deUint16(quadX + (quadY+1)*gridWidth + 1);
605 indices[quadNdx*6+5] = indices[quadNdx*6+2];
606 }
607
608 for (int y = 0; y < gridHeight; y++)
609 {
610 for (int x = 0; x < gridWidth; x++)
611 {
612 float xf = (float)x / (float)(gridWidth-1);
613 float yf = (float)y / (float)(gridHeight-1);
614
615 positions[(y*gridWidth + x)*4 + 0] = 2.0f*xf - 1.0f;
616 positions[(y*gridWidth + x)*4 + 1] = 2.0f*yf - 1.0f;
617 positions[(y*gridWidth + x)*4 + 2] = 0.0f;
618 positions[(y*gridWidth + x)*4 + 3] = 1.0f;
619 }
620 }
621
622 // Initialize input vectors.
623 {
624 int curInVec = 0;
625 for (int outputNdx = 0; outputNdx < (int)m_outputs.size(); outputNdx++)
626 {
627 const FragmentOutput& output = m_outputs[outputNdx];
628 bool isFloat = glu::isDataTypeFloatOrVec(output.type);
629 bool isInt = glu::isDataTypeIntOrIVec(output.type);
630 bool isUint = glu::isDataTypeUintOrUVec(output.type);
631 int numVecs = output.arrayLength > 0 ? output.arrayLength : 1;
632 int numScalars = glu::getDataTypeScalarSize(output.type);
633
634 for (int vecNdx = 0; vecNdx < numVecs; vecNdx++)
635 {
636 inputs[curInVec].resize(numVertices*numScalars);
637
638 // Record how many outputs are written in attachment.
639 DE_ASSERT(output.location+vecNdx < (int)attachments.size());
640 attachments[output.location+vecNdx].numWrittenChannels = numScalars;
641 attachments[output.location+vecNdx].outPrecision = output.precision;
642
643 if (isFloat)
644 {
645 Vec2 range = getFloatRange(output.precision);
646 Vec4 minVal (range.x());
647 Vec4 maxVal (range.y());
648 float* dst = (float*)&inputs[curInVec][0];
649
650 if (de::inBounds(output.location+vecNdx, 0, (int)attachments.size()))
651 {
652 // \note Floating-point precision conversion is not well-defined. For that reason we must
653 // limit value range to intersection of both data type and render target value ranges.
654 const tcu::TextureFormatInfo fmtInfo = tcu::getTextureFormatInfo(attachments[output.location+vecNdx].format);
655 minVal = tcu::max(minVal, fmtInfo.valueMin);
656 maxVal = tcu::min(maxVal, fmtInfo.valueMax);
657 }
658
659 m_testCtx.getLog() << TestLog::Message << "out" << curInVec << " value range: " << valueRangeToString(numScalars, minVal, maxVal) << TestLog::EndMessage;
660
661 for (int y = 0; y < gridHeight; y++)
662 {
663 for (int x = 0; x < gridWidth; x++)
664 {
665 float xf = (float)x / (float)(gridWidth-1);
666 float yf = (float)y / (float)(gridHeight-1);
667
668 float f0 = (xf + yf) * 0.5f;
669 float f1 = 0.5f + (xf - yf) * 0.5f;
670 Vec4 f = swizzleVec(Vec4(f0, f1, 1.0f-f0, 1.0f-f1), curInVec);
671 Vec4 c = minVal + (maxVal-minVal)*f;
672 float* v = dst + (y*gridWidth + x)*numScalars;
673
674 for (int ndx = 0; ndx < numScalars; ndx++)
675 v[ndx] = c[ndx];
676 }
677 }
678 }
679 else if (isInt)
680 {
681 const IVec2 range = getIntRange(output.precision);
682 IVec4 minVal (range.x());
683 IVec4 maxVal (range.y());
684
685 if (de::inBounds(output.location+vecNdx, 0, (int)attachments.size()))
686 {
687 // Limit to range of output format as conversion mode is not specified.
688 const IVec4 fmtBits = tcu::getTextureFormatBitDepth(attachments[output.location+vecNdx].format);
689 const BVec4 isZero = lessThanEqual(fmtBits, IVec4(0));
690 const IVec4 shift = tcu::clamp(fmtBits-1, tcu::IVec4(0), tcu::IVec4(256));
691 const IVec4 fmtMinVal = (-(tcu::Vector<deInt64, 4>(1) << shift.cast<deInt64>())).asInt();
692 const IVec4 fmtMaxVal = ((tcu::Vector<deInt64, 4>(1) << shift.cast<deInt64>())-deInt64(1)).asInt();
693
694 minVal = select(minVal, tcu::max(minVal, fmtMinVal), isZero);
695 maxVal = select(maxVal, tcu::min(maxVal, fmtMaxVal), isZero);
696 }
697
698 m_testCtx.getLog() << TestLog::Message << "out" << curInVec << " value range: " << valueRangeToString(numScalars, minVal, maxVal) << TestLog::EndMessage;
699
700 const IVec4 rangeDiv = swizzleVec((IVec4(gridWidth, gridHeight, gridWidth, gridHeight)-1), curInVec);
701 const IVec4 step = ((maxVal.cast<deInt64>() - minVal.cast<deInt64>()) / (rangeDiv.cast<deInt64>())).asInt();
702 deInt32* dst = (deInt32*)&inputs[curInVec][0];
703
704 for (int y = 0; y < gridHeight; y++)
705 {
706 for (int x = 0; x < gridWidth; x++)
707 {
708 int ix = gridWidth - x - 1;
709 int iy = gridHeight - y - 1;
710 IVec4 c = minVal + step*swizzleVec(IVec4(x, y, ix, iy), curInVec);
711 deInt32* v = dst + (y*gridWidth + x)*numScalars;
712
713 DE_ASSERT(boolAll(logicalAnd(greaterThanEqual(c, minVal), lessThanEqual(c, maxVal))));
714
715 for (int ndx = 0; ndx < numScalars; ndx++)
716 v[ndx] = c[ndx];
717 }
718 }
719 }
720 else if (isUint)
721 {
722 const UVec2 range = getUintRange(output.precision);
723 UVec4 maxVal (range.y());
724
725 if (de::inBounds(output.location+vecNdx, 0, (int)attachments.size()))
726 {
727 // Limit to range of output format as conversion mode is not specified.
728 const IVec4 fmtBits = tcu::getTextureFormatBitDepth(attachments[output.location+vecNdx].format);
729 const UVec4 fmtMaxVal = ((tcu::Vector<deUint64, 4>(1) << fmtBits.cast<deUint64>())-deUint64(1)).asUint();
730
731 maxVal = tcu::min(maxVal, fmtMaxVal);
732 }
733
734 m_testCtx.getLog() << TestLog::Message << "out" << curInVec << " value range: " << valueRangeToString(numScalars, UVec4(0), maxVal) << TestLog::EndMessage;
735
736 const IVec4 rangeDiv = swizzleVec((IVec4(gridWidth, gridHeight, gridWidth, gridHeight)-1), curInVec);
737 const UVec4 step = maxVal / rangeDiv.asUint();
738 deUint32* dst = &inputs[curInVec][0];
739
740 DE_ASSERT(range.x() == 0);
741
742 for (int y = 0; y < gridHeight; y++)
743 {
744 for (int x = 0; x < gridWidth; x++)
745 {
746 int ix = gridWidth - x - 1;
747 int iy = gridHeight - y - 1;
748 UVec4 c = step*swizzleVec(IVec4(x, y, ix, iy).asUint(), curInVec);
749 deUint32* v = dst + (y*gridWidth + x)*numScalars;
750
751 DE_ASSERT(boolAll(lessThanEqual(c, maxVal)));
752
753 for (int ndx = 0; ndx < numScalars; ndx++)
754 v[ndx] = c[ndx];
755 }
756 }
757 }
758 else
759 DE_ASSERT(false);
760
761 curInVec += 1;
762 }
763 }
764 }
765
766 // Render using gl.
767 gl.useProgram(m_program->getProgram());
768 gl.bindFramebuffer(GL_FRAMEBUFFER, m_framebuffer);
769 gl.viewport(0, 0, viewportW, viewportH);
770 gl.drawBuffers((int)drawBuffers.size(), &drawBuffers[0]);
771 gl.disable(GL_DITHER); // Dithering causes issues with unorm formats. Those issues could be worked around in threshold, but it makes validation less accurate.
772 GLU_EXPECT_NO_ERROR(gl.getError(), "After program setup");
773
774 {
775 int curInVec = 0;
776 for (int outputNdx = 0; outputNdx < (int)m_outputs.size(); outputNdx++)
777 {
778 const FragmentOutput& output = m_outputs[outputNdx];
779 bool isArray = output.arrayLength > 0;
780 bool isFloat = glu::isDataTypeFloatOrVec(output.type);
781 bool isInt = glu::isDataTypeIntOrIVec(output.type);
782 bool isUint = glu::isDataTypeUintOrUVec(output.type);
783 int scalarSize = glu::getDataTypeScalarSize(output.type);
784 deUint32 glScalarType = isFloat ? GL_FLOAT :
785 isInt ? GL_INT :
786 isUint ? GL_UNSIGNED_INT : GL_NONE;
787 int numVecs = isArray ? output.arrayLength : 1;
788
789 for (int vecNdx = 0; vecNdx < numVecs; vecNdx++)
790 {
791 string name = string("in") + de::toString(outputNdx) + (isArray ? string("_") + de::toString(vecNdx) : string());
792 int loc = gl.getAttribLocation(m_program->getProgram(), name.c_str());
793
794 if (loc >= 0)
795 {
796 gl.enableVertexAttribArray(loc);
797 if (isFloat)
798 gl.vertexAttribPointer(loc, scalarSize, glScalarType, GL_FALSE, 0, &inputs[curInVec][0]);
799 else
800 gl.vertexAttribIPointer(loc, scalarSize, glScalarType, 0, &inputs[curInVec][0]);
801 }
802 else
803 log << TestLog::Message << "Warning: No location for attribute '" << name << "' found." << TestLog::EndMessage;
804
805 curInVec += 1;
806 }
807 }
808 }
809 {
810 int posLoc = gl.getAttribLocation(m_program->getProgram(), "a_position");
811 TCU_CHECK(posLoc >= 0);
812 gl.enableVertexAttribArray(posLoc);
813 gl.vertexAttribPointer(posLoc, 4, GL_FLOAT, GL_FALSE, 0, &positions[0]);
814 }
815 GLU_EXPECT_NO_ERROR(gl.getError(), "After attribute setup");
816
817 gl.drawElements(GL_TRIANGLES, numIndices, GL_UNSIGNED_SHORT, &indices[0]);
818 GLU_EXPECT_NO_ERROR(gl.getError(), "glDrawElements");
819
820 // Read all attachment points.
821 for (int ndx = 0; ndx < numAttachments; ndx++)
822 {
823 const glu::TransferFormat transferFmt = glu::getTransferFormat(attachments[ndx].readFormat);
824 void* dst = &attachments[ndx].renderedData[0];
825 const int attachmentW = m_fboSpec[ndx].width;
826 const int attachmentH = m_fboSpec[ndx].height;
827 const int numValidChannels = attachments[ndx].numWrittenChannels;
828 const tcu::PixelBufferAccess rendered (attachments[ndx].readFormat, attachmentW, attachmentH, 1, deAlign32(attachments[ndx].readFormat.getPixelSize()*attachmentW, readAlignment), 0, &attachments[ndx].renderedData[0]);
829
830 gl.readBuffer(GL_COLOR_ATTACHMENT0+ndx);
831 gl.readPixels(0, 0, minBufSize.x(), minBufSize.y(), transferFmt.format, transferFmt.dataType, dst);
832
833 clearUndefined(rendered, numValidChannels);
834 }
835
836 // Render reference images.
837 {
838 int curInNdx = 0;
839 for (int outputNdx = 0; outputNdx < (int)m_outputs.size(); outputNdx++)
840 {
841 const FragmentOutput& output = m_outputs[outputNdx];
842 const bool isArray = output.arrayLength > 0;
843 const bool isFloat = glu::isDataTypeFloatOrVec(output.type);
844 const bool isInt = glu::isDataTypeIntOrIVec(output.type);
845 const bool isUint = glu::isDataTypeUintOrUVec(output.type);
846 const int scalarSize = glu::getDataTypeScalarSize(output.type);
847 const int numVecs = isArray ? output.arrayLength : 1;
848
849 for (int vecNdx = 0; vecNdx < numVecs; vecNdx++)
850 {
851 const int location = output.location+vecNdx;
852 const void* inputData = &inputs[curInNdx][0];
853
854 DE_ASSERT(de::inBounds(location, 0, (int)m_fboSpec.size()));
855
856 const int bufW = m_fboSpec[location].width;
857 const int bufH = m_fboSpec[location].height;
858 const tcu::PixelBufferAccess buf (attachments[location].referenceFormat, bufW, bufH, 1, &attachments[location].referenceData[0]);
859 const tcu::PixelBufferAccess viewportBuf = getSubregion(buf, 0, 0, 0, viewportW, viewportH, 1);
860
861 if (isInt || isUint)
862 renderIntReference(viewportBuf, gridWidth, gridHeight, scalarSize, (const int*)inputData);
863 else if (isFloat)
864 renderFloatReference(viewportBuf, gridWidth, gridHeight, scalarSize, (const float*)inputData);
865 else
866 DE_ASSERT(false);
867
868 curInNdx += 1;
869 }
870 }
871 }
872
873 // Compare all images.
874 bool allLevelsOk = true;
875 for (int attachNdx = 0; attachNdx < numAttachments; attachNdx++)
876 {
877 const int attachmentW = m_fboSpec[attachNdx].width;
878 const int attachmentH = m_fboSpec[attachNdx].height;
879 const int numValidChannels = attachments[attachNdx].numWrittenChannels;
880 const tcu::BVec4 cmpMask (numValidChannels >= 1, numValidChannels >= 2, numValidChannels >= 3, numValidChannels >= 4);
881 const glu::Precision outPrecision = attachments[attachNdx].outPrecision;
882 const tcu::TextureFormat& format = attachments[attachNdx].format;
883 tcu::ConstPixelBufferAccess rendered (attachments[attachNdx].readFormat, attachmentW, attachmentH, 1, deAlign32(attachments[attachNdx].readFormat.getPixelSize()*attachmentW, readAlignment), 0, &attachments[attachNdx].renderedData[0]);
884 tcu::ConstPixelBufferAccess reference (attachments[attachNdx].referenceFormat, attachmentW, attachmentH, 1, &attachments[attachNdx].referenceData[0]);
885 tcu::TextureChannelClass texClass = tcu::getTextureChannelClass(format.type);
886 bool isOk = true;
887 const string name = string("Attachment") + de::toString(attachNdx);
888 const string desc = string("Color attachment ") + de::toString(attachNdx);
889
890 log << TestLog::Message << "Attachment " << attachNdx << ": " << numValidChannels << " channels have defined values and used for comparison" << TestLog::EndMessage;
891
892 switch (texClass)
893 {
894 case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
895 {
896 const deUint32 interpThreshold = 4; //!< 4 ULP interpolation threshold (interpolation always in highp)
897 deUint32 outTypeThreshold = 0; //!< Threshold based on output type
898 UVec4 formatThreshold; //!< Threshold computed based on format.
899 UVec4 finalThreshold;
900
901 // 1 ULP rounding error is allowed for smaller floating-point formats
902 switch (format.type)
903 {
904 case tcu::TextureFormat::FLOAT: formatThreshold = UVec4(0); break;
905 case tcu::TextureFormat::HALF_FLOAT: formatThreshold = UVec4((1<<(23-10))); break;
906 case tcu::TextureFormat::UNSIGNED_INT_11F_11F_10F_REV: formatThreshold = UVec4((1<<(23-6)), (1<<(23-6)), (1<<(23-5)), 0); break;
907 default:
908 DE_ASSERT(false);
909 break;
910 }
911
912 // 1 ULP rounding error for highp -> output precision cast
913 switch (outPrecision)
914 {
915 case glu::PRECISION_LOWP: outTypeThreshold = (1<<(23-8)); break;
916 case glu::PRECISION_MEDIUMP: outTypeThreshold = (1<<(23-10)); break;
917 case glu::PRECISION_HIGHP: outTypeThreshold = 0; break;
918 default:
919 DE_ASSERT(false);
920 }
921
922 finalThreshold = select(max(formatThreshold, UVec4(deMax32(interpThreshold, outTypeThreshold))), UVec4(~0u), cmpMask);
923
924 isOk = tcu::floatUlpThresholdCompare(log, name.c_str(), desc.c_str(), reference, rendered, finalThreshold, tcu::COMPARE_LOG_RESULT);
925 break;
926 }
927
928 case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
929 {
930 // \note glReadPixels() allows only 8 bits to be read. This means that RGB10_A2 will loose some
931 // bits in the process and it must be taken into account when computing threshold.
932 const IVec4 bits = min(IVec4(8), tcu::getTextureFormatBitDepth(format));
933 const Vec4 baseThreshold = 1.0f / ((IVec4(1) << bits)-1).asFloat();
934 const Vec4 threshold = select(baseThreshold, Vec4(2.0f), cmpMask);
935
936 isOk = tcu::floatThresholdCompare(log, name.c_str(), desc.c_str(), reference, rendered, threshold, tcu::COMPARE_LOG_RESULT);
937 break;
938 }
939
940 case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
941 case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
942 {
943 const tcu::UVec4 threshold = select(UVec4(0u), UVec4(~0u), cmpMask);
944 isOk = tcu::intThresholdCompare(log, name.c_str(), desc.c_str(), reference, rendered, threshold, tcu::COMPARE_LOG_RESULT);
945 break;
946 }
947
948 default:
949 TCU_FAIL("Unsupported comparison");
950 }
951
952 if (!isOk)
953 allLevelsOk = false;
954 }
955
956 m_testCtx.setTestResult(allLevelsOk ? QP_TEST_RESULT_PASS : QP_TEST_RESULT_FAIL,
957 allLevelsOk ? "Pass" : "Image comparison failed");
958 return STOP;
959 }
960
FragmentOutputTests(Context & context)961 FragmentOutputTests::FragmentOutputTests (Context& context)
962 : TestCaseGroup(context, "fragment_out", "Fragment output tests")
963 {
964 }
965
~FragmentOutputTests(void)966 FragmentOutputTests::~FragmentOutputTests (void)
967 {
968 }
969
createRandomCase(Context & context,int minRenderTargets,int maxRenderTargets,deUint32 seed)970 static FragmentOutputCase* createRandomCase (Context& context, int minRenderTargets, int maxRenderTargets, deUint32 seed)
971 {
972 static const glu::DataType outputTypes[] =
973 {
974 glu::TYPE_FLOAT,
975 glu::TYPE_FLOAT_VEC2,
976 glu::TYPE_FLOAT_VEC3,
977 glu::TYPE_FLOAT_VEC4,
978 glu::TYPE_INT,
979 glu::TYPE_INT_VEC2,
980 glu::TYPE_INT_VEC3,
981 glu::TYPE_INT_VEC4,
982 glu::TYPE_UINT,
983 glu::TYPE_UINT_VEC2,
984 glu::TYPE_UINT_VEC3,
985 glu::TYPE_UINT_VEC4
986 };
987 static const glu::Precision precisions[] =
988 {
989 glu::PRECISION_LOWP,
990 glu::PRECISION_MEDIUMP,
991 glu::PRECISION_HIGHP
992 };
993 static const deUint32 floatFormats[] =
994 {
995 GL_RGBA32F,
996 GL_RGBA16F,
997 GL_R11F_G11F_B10F,
998 GL_RG32F,
999 GL_RG16F,
1000 GL_R32F,
1001 GL_R16F,
1002 GL_RGBA8,
1003 GL_SRGB8_ALPHA8,
1004 GL_RGB10_A2,
1005 GL_RGBA4,
1006 GL_RGB5_A1,
1007 GL_RGB8,
1008 GL_RGB565,
1009 GL_RG8,
1010 GL_R8
1011 };
1012 static const deUint32 intFormats[] =
1013 {
1014 GL_RGBA32I,
1015 GL_RGBA16I,
1016 GL_RGBA8I,
1017 GL_RG32I,
1018 GL_RG16I,
1019 GL_RG8I,
1020 GL_R32I,
1021 GL_R16I,
1022 GL_R8I
1023 };
1024 static const deUint32 uintFormats[] =
1025 {
1026 GL_RGBA32UI,
1027 GL_RGBA16UI,
1028 GL_RGBA8UI,
1029 GL_RGB10_A2UI,
1030 GL_RG32UI,
1031 GL_RG16UI,
1032 GL_RG8UI,
1033 GL_R32UI,
1034 GL_R16UI,
1035 GL_R8UI
1036 };
1037
1038 de::Random rnd (seed);
1039 vector<FragmentOutput> outputs;
1040 vector<BufferSpec> targets;
1041 vector<glu::DataType> outTypes;
1042
1043 int numTargets = rnd.getInt(minRenderTargets, maxRenderTargets);
1044 const int width = 128; // \todo [2012-04-10 pyry] Separate randomized sizes per target?
1045 const int height = 64;
1046 const int samples = 0;
1047
1048 // Compute outputs.
1049 int curLoc = 0;
1050 while (curLoc < numTargets)
1051 {
1052 bool useArray = rnd.getFloat() < 0.3f;
1053 int maxArrayLen = numTargets-curLoc;
1054 int arrayLen = useArray ? rnd.getInt(1, maxArrayLen) : 0;
1055 glu::DataType basicType = rnd.choose<glu::DataType>(&outputTypes[0], &outputTypes[0] + DE_LENGTH_OF_ARRAY(outputTypes));
1056 glu::Precision precision = rnd.choose<glu::Precision>(&precisions[0], &precisions[0] + DE_LENGTH_OF_ARRAY(precisions));
1057 int numLocations = useArray ? arrayLen : 1;
1058
1059 outputs.push_back(FragmentOutput(basicType, precision, curLoc, arrayLen));
1060
1061 for (int ndx = 0; ndx < numLocations; ndx++)
1062 outTypes.push_back(basicType);
1063
1064 curLoc += numLocations;
1065 }
1066 DE_ASSERT(curLoc == numTargets);
1067 DE_ASSERT((int)outTypes.size() == numTargets);
1068
1069 // Compute buffers.
1070 while ((int)targets.size() < numTargets)
1071 {
1072 glu::DataType outType = outTypes[targets.size()];
1073 bool isFloat = glu::isDataTypeFloatOrVec(outType);
1074 bool isInt = glu::isDataTypeIntOrIVec(outType);
1075 bool isUint = glu::isDataTypeUintOrUVec(outType);
1076 deUint32 format = 0;
1077
1078 if (isFloat)
1079 format = rnd.choose<deUint32>(&floatFormats[0], &floatFormats[0] + DE_LENGTH_OF_ARRAY(floatFormats));
1080 else if (isInt)
1081 format = rnd.choose<deUint32>(&intFormats[0], &intFormats[0] + DE_LENGTH_OF_ARRAY(intFormats));
1082 else if (isUint)
1083 format = rnd.choose<deUint32>(&uintFormats[0], &uintFormats[0] + DE_LENGTH_OF_ARRAY(uintFormats));
1084 else
1085 DE_ASSERT(false);
1086
1087 targets.push_back(BufferSpec(format, width, height, samples));
1088 }
1089
1090 return new FragmentOutputCase(context, de::toString(seed).c_str(), "", targets, outputs);
1091 }
1092
init(void)1093 void FragmentOutputTests::init (void)
1094 {
1095 static const deUint32 requiredFloatFormats[] =
1096 {
1097 GL_RGBA32F,
1098 GL_RGBA16F,
1099 GL_R11F_G11F_B10F,
1100 GL_RG32F,
1101 GL_RG16F,
1102 GL_R32F,
1103 GL_R16F
1104 };
1105 static const deUint32 requiredFixedFormats[] =
1106 {
1107 GL_RGBA8,
1108 GL_SRGB8_ALPHA8,
1109 GL_RGB10_A2,
1110 GL_RGBA4,
1111 GL_RGB5_A1,
1112 GL_RGB8,
1113 GL_RGB565,
1114 GL_RG8,
1115 GL_R8
1116 };
1117 static const deUint32 requiredIntFormats[] =
1118 {
1119 GL_RGBA32I,
1120 GL_RGBA16I,
1121 GL_RGBA8I,
1122 GL_RG32I,
1123 GL_RG16I,
1124 GL_RG8I,
1125 GL_R32I,
1126 GL_R16I,
1127 GL_R8I
1128 };
1129 static const deUint32 requiredUintFormats[] =
1130 {
1131 GL_RGBA32UI,
1132 GL_RGBA16UI,
1133 GL_RGBA8UI,
1134 GL_RGB10_A2UI,
1135 GL_RG32UI,
1136 GL_RG16UI,
1137 GL_RG8UI,
1138 GL_R32UI,
1139 GL_R16UI,
1140 GL_R8UI
1141 };
1142
1143 static const glu::Precision precisions[] =
1144 {
1145 glu::PRECISION_LOWP,
1146 glu::PRECISION_MEDIUMP,
1147 glu::PRECISION_HIGHP
1148 };
1149
1150 // .basic.
1151 {
1152 tcu::TestCaseGroup* basicGroup = new tcu::TestCaseGroup(m_testCtx, "basic", "Basic fragment output tests");
1153 addChild(basicGroup);
1154
1155 const int width = 64;
1156 const int height = 64;
1157 const int samples = 0;
1158
1159 // .float
1160 tcu::TestCaseGroup* floatGroup = new tcu::TestCaseGroup(m_testCtx, "float", "Floating-point output tests");
1161 basicGroup->addChild(floatGroup);
1162 for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(requiredFloatFormats); fmtNdx++)
1163 {
1164 deUint32 format = requiredFloatFormats[fmtNdx];
1165 string fmtName = getFormatName(format);
1166 vector<BufferSpec> fboSpec;
1167
1168 fboSpec.push_back(BufferSpec(format, width, height, samples));
1169
1170 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1171 {
1172 glu::Precision prec = precisions[precNdx];
1173 string precName = glu::getPrecisionName(prec);
1174
1175 floatGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_float").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT, prec, 0)).toVec()));
1176 floatGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec2").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC2, prec, 0)).toVec()));
1177 floatGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec3").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC3, prec, 0)).toVec()));
1178 floatGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec4").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC4, prec, 0)).toVec()));
1179 }
1180 }
1181
1182 // .fixed
1183 tcu::TestCaseGroup* fixedGroup = new tcu::TestCaseGroup(m_testCtx, "fixed", "Fixed-point output tests");
1184 basicGroup->addChild(fixedGroup);
1185 for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(requiredFixedFormats); fmtNdx++)
1186 {
1187 deUint32 format = requiredFixedFormats[fmtNdx];
1188 string fmtName = getFormatName(format);
1189 vector<BufferSpec> fboSpec;
1190
1191 fboSpec.push_back(BufferSpec(format, width, height, samples));
1192
1193 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1194 {
1195 glu::Precision prec = precisions[precNdx];
1196 string precName = glu::getPrecisionName(prec);
1197
1198 fixedGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_float").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT, prec, 0)).toVec()));
1199 fixedGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec2").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC2, prec, 0)).toVec()));
1200 fixedGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec3").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC3, prec, 0)).toVec()));
1201 fixedGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec4").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC4, prec, 0)).toVec()));
1202 }
1203 }
1204
1205 // .int
1206 tcu::TestCaseGroup* intGroup = new tcu::TestCaseGroup(m_testCtx, "int", "Integer output tests");
1207 basicGroup->addChild(intGroup);
1208 for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(requiredIntFormats); fmtNdx++)
1209 {
1210 deUint32 format = requiredIntFormats[fmtNdx];
1211 string fmtName = getFormatName(format);
1212 vector<BufferSpec> fboSpec;
1213
1214 fboSpec.push_back(BufferSpec(format, width, height, samples));
1215
1216 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1217 {
1218 glu::Precision prec = precisions[precNdx];
1219 string precName = glu::getPrecisionName(prec);
1220
1221 intGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_int").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_INT, prec, 0)).toVec()));
1222 intGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_ivec2").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_INT_VEC2, prec, 0)).toVec()));
1223 intGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_ivec3").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_INT_VEC3, prec, 0)).toVec()));
1224 intGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_ivec4").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_INT_VEC4, prec, 0)).toVec()));
1225 }
1226 }
1227
1228 // .uint
1229 tcu::TestCaseGroup* uintGroup = new tcu::TestCaseGroup(m_testCtx, "uint", "Usigned integer output tests");
1230 basicGroup->addChild(uintGroup);
1231 for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(requiredUintFormats); fmtNdx++)
1232 {
1233 deUint32 format = requiredUintFormats[fmtNdx];
1234 string fmtName = getFormatName(format);
1235 vector<BufferSpec> fboSpec;
1236
1237 fboSpec.push_back(BufferSpec(format, width, height, samples));
1238
1239 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1240 {
1241 glu::Precision prec = precisions[precNdx];
1242 string precName = glu::getPrecisionName(prec);
1243
1244 uintGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_uint").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_UINT, prec, 0)).toVec()));
1245 uintGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_uvec2").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_UINT_VEC2, prec, 0)).toVec()));
1246 uintGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_uvec3").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_UINT_VEC3, prec, 0)).toVec()));
1247 uintGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_uvec4").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_UINT_VEC4, prec, 0)).toVec()));
1248 }
1249 }
1250 }
1251
1252 // .array
1253 {
1254 tcu::TestCaseGroup* arrayGroup = new tcu::TestCaseGroup(m_testCtx, "array", "Array outputs");
1255 addChild(arrayGroup);
1256
1257 const int width = 64;
1258 const int height = 64;
1259 const int samples = 0;
1260 const int numTargets = 3;
1261
1262 // .float
1263 tcu::TestCaseGroup* floatGroup = new tcu::TestCaseGroup(m_testCtx, "float", "Floating-point output tests");
1264 arrayGroup->addChild(floatGroup);
1265 for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(requiredFloatFormats); fmtNdx++)
1266 {
1267 deUint32 format = requiredFloatFormats[fmtNdx];
1268 string fmtName = getFormatName(format);
1269 vector<BufferSpec> fboSpec;
1270
1271 for (int ndx = 0; ndx < numTargets; ndx++)
1272 fboSpec.push_back(BufferSpec(format, width, height, samples));
1273
1274 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1275 {
1276 glu::Precision prec = precisions[precNdx];
1277 string precName = glu::getPrecisionName(prec);
1278
1279 floatGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_float").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT, prec, 0, numTargets)).toVec()));
1280 floatGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec2").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC2, prec, 0, numTargets)).toVec()));
1281 floatGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec3").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC3, prec, 0, numTargets)).toVec()));
1282 floatGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec4").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC4, prec, 0, numTargets)).toVec()));
1283 }
1284 }
1285
1286 // .fixed
1287 tcu::TestCaseGroup* fixedGroup = new tcu::TestCaseGroup(m_testCtx, "fixed", "Fixed-point output tests");
1288 arrayGroup->addChild(fixedGroup);
1289 for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(requiredFixedFormats); fmtNdx++)
1290 {
1291 deUint32 format = requiredFixedFormats[fmtNdx];
1292 string fmtName = getFormatName(format);
1293 vector<BufferSpec> fboSpec;
1294
1295 for (int ndx = 0; ndx < numTargets; ndx++)
1296 fboSpec.push_back(BufferSpec(format, width, height, samples));
1297
1298 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1299 {
1300 glu::Precision prec = precisions[precNdx];
1301 string precName = glu::getPrecisionName(prec);
1302
1303 fixedGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_float").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT, prec, 0, numTargets)).toVec()));
1304 fixedGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec2").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC2, prec, 0, numTargets)).toVec()));
1305 fixedGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec3").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC3, prec, 0, numTargets)).toVec()));
1306 fixedGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_vec4").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_FLOAT_VEC4, prec, 0, numTargets)).toVec()));
1307 }
1308 }
1309
1310 // .int
1311 tcu::TestCaseGroup* intGroup = new tcu::TestCaseGroup(m_testCtx, "int", "Integer output tests");
1312 arrayGroup->addChild(intGroup);
1313 for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(requiredIntFormats); fmtNdx++)
1314 {
1315 deUint32 format = requiredIntFormats[fmtNdx];
1316 string fmtName = getFormatName(format);
1317 vector<BufferSpec> fboSpec;
1318
1319 for (int ndx = 0; ndx < numTargets; ndx++)
1320 fboSpec.push_back(BufferSpec(format, width, height, samples));
1321
1322 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1323 {
1324 glu::Precision prec = precisions[precNdx];
1325 string precName = glu::getPrecisionName(prec);
1326
1327 intGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_int").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_INT, prec, 0, numTargets)).toVec()));
1328 intGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_ivec2").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_INT_VEC2, prec, 0, numTargets)).toVec()));
1329 intGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_ivec3").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_INT_VEC3, prec, 0, numTargets)).toVec()));
1330 intGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_ivec4").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_INT_VEC4, prec, 0, numTargets)).toVec()));
1331 }
1332 }
1333
1334 // .uint
1335 tcu::TestCaseGroup* uintGroup = new tcu::TestCaseGroup(m_testCtx, "uint", "Usigned integer output tests");
1336 arrayGroup->addChild(uintGroup);
1337 for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(requiredUintFormats); fmtNdx++)
1338 {
1339 deUint32 format = requiredUintFormats[fmtNdx];
1340 string fmtName = getFormatName(format);
1341 vector<BufferSpec> fboSpec;
1342
1343 for (int ndx = 0; ndx < numTargets; ndx++)
1344 fboSpec.push_back(BufferSpec(format, width, height, samples));
1345
1346 for (int precNdx = 0; precNdx < DE_LENGTH_OF_ARRAY(precisions); precNdx++)
1347 {
1348 glu::Precision prec = precisions[precNdx];
1349 string precName = glu::getPrecisionName(prec);
1350
1351 uintGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_uint").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_UINT, prec, 0, numTargets)).toVec()));
1352 uintGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_uvec2").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_UINT_VEC2, prec, 0, numTargets)).toVec()));
1353 uintGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_uvec3").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_UINT_VEC3, prec, 0, numTargets)).toVec()));
1354 uintGroup->addChild(new FragmentOutputCase(m_context, (fmtName + "_" + precName + "_uvec4").c_str(), "", fboSpec, (OutputVec() << FragmentOutput(glu::TYPE_UINT_VEC4, prec, 0, numTargets)).toVec()));
1355 }
1356 }
1357 }
1358
1359 // .random
1360 {
1361 tcu::TestCaseGroup* randomGroup = new tcu::TestCaseGroup(m_testCtx, "random", "Random fragment output cases");
1362 addChild(randomGroup);
1363
1364 for (deUint32 seed = 0; seed < 100; seed++)
1365 randomGroup->addChild(createRandomCase(m_context, 2, 4, seed));
1366 }
1367 }
1368
1369 } // Functional
1370 } // gles3
1371 } // deqp
1372