1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program OpenGL ES 3.1 Module
3  * -------------------------------------------------
4  *
5  * Copyright 2015 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 Indexed blend operation tests (GL_EXT_draw_buffers_indexed)
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es31fDrawBuffersIndexedTests.hpp"
25 
26 #include "gluContextInfo.hpp"
27 #include "gluDrawUtil.hpp"
28 #include "gluObjectWrapper.hpp"
29 #include "gluPixelTransfer.hpp"
30 #include "gluShaderProgram.hpp"
31 #include "gluStrUtil.hpp"
32 #include "gluTextureUtil.hpp"
33 
34 #include "sglrReferenceUtils.hpp"
35 
36 #include "rrMultisamplePixelBufferAccess.hpp"
37 #include "rrRenderer.hpp"
38 
39 #include "glwEnums.hpp"
40 #include "glwFunctions.hpp"
41 
42 #include "tcuEither.hpp"
43 #include "tcuImageCompare.hpp"
44 #include "tcuMaybe.hpp"
45 #include "tcuResultCollector.hpp"
46 #include "tcuStringTemplate.hpp"
47 #include "tcuTestLog.hpp"
48 #include "tcuTexture.hpp"
49 #include "tcuTextureUtil.hpp"
50 #include "tcuVector.hpp"
51 #include "tcuVectorUtil.hpp"
52 #include "tcuFloat.hpp"
53 
54 #include "deRandom.hpp"
55 #include "deArrayUtil.hpp"
56 #include "deStringUtil.hpp"
57 #include "deUniquePtr.hpp"
58 
59 #include "deInt32.h"
60 
61 #include <string>
62 #include <vector>
63 #include <map>
64 
65 using tcu::BVec4;
66 using tcu::Either;
67 using tcu::IVec2;
68 using tcu::IVec4;
69 using tcu::Maybe;
70 using tcu::TestLog;
71 using tcu::TextureFormat;
72 using tcu::TextureLevel;
73 using tcu::UVec4;
74 using tcu::Vec2;
75 using tcu::Vec4;
76 using tcu::just;
77 
78 using std::string;
79 using std::vector;
80 using std::map;
81 
82 using sglr::rr_util::mapGLBlendEquation;
83 using sglr::rr_util::mapGLBlendFunc;
84 using sglr::rr_util::mapGLBlendEquationAdvanced;
85 
86 namespace deqp
87 {
88 namespace gles31
89 {
90 namespace Functional
91 {
92 namespace
93 {
94 
95 typedef deUint32 BlendEq;
96 
isAdvancedBlendEq(BlendEq eq)97 bool isAdvancedBlendEq (BlendEq eq)
98 {
99 	switch (eq)
100 	{
101 		case GL_MULTIPLY:		return true;
102 		case GL_SCREEN:			return true;
103 		case GL_OVERLAY:		return true;
104 		case GL_DARKEN:			return true;
105 		case GL_LIGHTEN:		return true;
106 		case GL_COLORDODGE:		return true;
107 		case GL_COLORBURN:		return true;
108 		case GL_HARDLIGHT:		return true;
109 		case GL_SOFTLIGHT:		return true;
110 		case GL_DIFFERENCE:		return true;
111 		case GL_EXCLUSION:		return true;
112 		case GL_HSL_HUE:		return true;
113 		case GL_HSL_SATURATION:	return true;
114 		case GL_HSL_COLOR:		return true;
115 		case GL_HSL_LUMINOSITY:	return true;
116 		default:
117 			return false;
118 	}
119 }
120 
121 struct SeparateBlendEq
122 {
SeparateBlendEqdeqp::gles31::Functional::__anon169a63d40111::SeparateBlendEq123 	SeparateBlendEq (BlendEq rgb_, BlendEq alpha_)
124 		: rgb	(rgb_)
125 		, alpha	(alpha_)
126 	{
127 	}
128 
129 	BlendEq rgb;
130 	BlendEq alpha;
131 };
132 
133 struct BlendFunc
134 {
BlendFuncdeqp::gles31::Functional::__anon169a63d40111::BlendFunc135 	BlendFunc (deUint32 src_, deUint32 dst_)
136 		: src (src_)
137 		, dst (dst_)
138 	{
139 	}
140 
141 	deUint32 src;
142 	deUint32 dst;
143 };
144 
145 struct SeparateBlendFunc
146 {
SeparateBlendFuncdeqp::gles31::Functional::__anon169a63d40111::SeparateBlendFunc147 	SeparateBlendFunc (BlendFunc rgb_, BlendFunc alpha_)
148 		: rgb	(rgb_)
149 		, alpha	(alpha_)
150 	{
151 	}
152 
153 	BlendFunc rgb;
154 	BlendFunc alpha;
155 };
156 
157 typedef deUint32 DrawBuffer;
158 
159 struct BlendState
160 {
BlendStatedeqp::gles31::Functional::__anon169a63d40111::BlendState161 	BlendState (void) {}
162 
BlendStatedeqp::gles31::Functional::__anon169a63d40111::BlendState163 	BlendState (const Maybe<bool>&									enableBlend_,
164 				const Maybe<Either<BlendEq, SeparateBlendEq> >&		blendEq_,
165 				const Maybe<Either<BlendFunc, SeparateBlendFunc> >&	blendFunc_,
166 				const Maybe<BVec4>&									colorMask_)
167 		: enableBlend	(enableBlend_)
168 		, blendEq		(blendEq_)
169 		, blendFunc		(blendFunc_)
170 		, colorMask		(colorMask_)
171 	{
172 	}
173 
isEmptydeqp::gles31::Functional::__anon169a63d40111::BlendState174 	bool isEmpty (void) const
175 	{
176 		return (!enableBlend) && (!blendEq) && (!blendFunc) && (!colorMask);
177 	}
178 
179 	Maybe<bool>										enableBlend;
180 	Maybe<Either<BlendEq, SeparateBlendEq> >		blendEq;
181 	Maybe<Either<BlendFunc, SeparateBlendFunc> >	blendFunc;
182 	Maybe<BVec4>									colorMask;
183 };
184 
checkES32orGL45Support(Context & ctx)185 static bool checkES32orGL45Support(Context& ctx)
186 {
187 	auto ctxType = ctx.getRenderContext().getType();
188 	return contextSupports(ctxType, glu::ApiType::es(3, 2)) ||
189 		   contextSupports(ctxType, glu::ApiType::core(4, 5));
190 }
191 
setCommonBlendState(const glw::Functions & gl,const BlendState & blend)192 void setCommonBlendState (const glw::Functions& gl, const BlendState& blend)
193 {
194 	if (blend.enableBlend)
195 	{
196 		if (*blend.enableBlend)
197 			gl.enable(GL_BLEND);
198 		else
199 			gl.disable(GL_BLEND);
200 	}
201 
202 	if (blend.colorMask)
203 	{
204 		const BVec4& mask = *blend.colorMask;
205 
206 		gl.colorMask(mask.x(), mask.y(), mask.z(), mask.w());
207 	}
208 
209 	if (blend.blendEq)
210 	{
211 		const Either<BlendEq, SeparateBlendEq>& blendEq = *blend.blendEq;
212 
213 		if (blendEq.is<BlendEq>())
214 			gl.blendEquation(blendEq.get<BlendEq>());
215 		else if (blendEq.is<SeparateBlendEq>())
216 			gl.blendEquationSeparate(blendEq.get<SeparateBlendEq>().rgb, blendEq.get<SeparateBlendEq>().alpha);
217 		else
218 			DE_ASSERT(false);
219 	}
220 
221 	if (blend.blendFunc)
222 	{
223 		const Either<BlendFunc, SeparateBlendFunc>& blendFunc = *blend.blendFunc;
224 
225 		if (blendFunc.is<BlendFunc>())
226 			gl.blendFunc(blendFunc.get<BlendFunc>().src, blendFunc.get<BlendFunc>().dst);
227 		else if (blendFunc.is<SeparateBlendFunc>())
228 			gl.blendFuncSeparate(blendFunc.get<SeparateBlendFunc>().rgb.src, blendFunc.get<SeparateBlendFunc>().rgb.dst, blendFunc.get<SeparateBlendFunc>().alpha.src, blendFunc.get<SeparateBlendFunc>().alpha.dst);
229 		else
230 			DE_ASSERT(false);
231 	}
232 
233 	GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to set common blend state.");
234 }
235 
setIndexedBlendState(const glw::Functions & gl,const BlendState & blend,deUint32 index)236 void setIndexedBlendState (const glw::Functions& gl, const BlendState& blend, deUint32 index)
237 {
238 	if (blend.enableBlend)
239 	{
240 		if (*blend.enableBlend)
241 			gl.enablei(GL_BLEND, index);
242 		else
243 			gl.disablei(GL_BLEND, index);
244 	}
245 
246 	if (blend.colorMask)
247 	{
248 		const BVec4 mask = *blend.colorMask;
249 
250 		gl.colorMaski(index, mask.x(), mask.y(), mask.z(), mask.w());
251 	}
252 
253 	if (blend.blendEq)
254 	{
255 		const Either<BlendEq, SeparateBlendEq>& blendEq = *blend.blendEq;
256 
257 		if (blendEq.is<BlendEq>())
258 			gl.blendEquationi(index, blendEq.get<BlendEq>());
259 		else if (blendEq.is<SeparateBlendEq>())
260 			gl.blendEquationSeparatei(index, blendEq.get<SeparateBlendEq>().rgb, blendEq.get<SeparateBlendEq>().alpha);
261 		else
262 			DE_ASSERT(false);
263 	}
264 
265 	if (blend.blendFunc)
266 	{
267 		const Either<BlendFunc, SeparateBlendFunc>& blendFunc = *blend.blendFunc;
268 
269 		if (blendFunc.is<BlendFunc>())
270 			gl.blendFunci(index, blendFunc.get<BlendFunc>().src, blendFunc.get<BlendFunc>().dst);
271 		else if (blendFunc.is<SeparateBlendFunc>())
272 			gl.blendFuncSeparatei(index, blendFunc.get<SeparateBlendFunc>().rgb.src, blendFunc.get<SeparateBlendFunc>().rgb.dst, blendFunc.get<SeparateBlendFunc>().alpha.src, blendFunc.get<SeparateBlendFunc>().alpha.dst);
273 		else
274 			DE_ASSERT(false);
275 	}
276 
277 	GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to set draw buffer specifig blend state.");
278 }
279 
280 class DrawBufferInfo
281 {
282 public:
283 							DrawBufferInfo	(bool					render,
284 											 const IVec2&			size,
285 											 const BlendState&		blendState,
286 											 const TextureFormat&	format);
287 
getFormat(void) const288 	const TextureFormat&	getFormat		(void) const { return m_format;		}
getSize(void) const289 	const IVec2&			getSize			(void) const { return m_size;		}
getBlendState(void) const290 	const BlendState&		getBlendState	(void) const { return m_blendState;	}
getRender(void) const291 	bool					getRender		(void) const { return m_render;		}
292 
293 private:
294 	bool					m_render;
295 	IVec2					m_size;
296 	TextureFormat			m_format;
297 	BlendState				m_blendState;
298 };
299 
DrawBufferInfo(bool render,const IVec2 & size,const BlendState & blendState,const TextureFormat & format)300 DrawBufferInfo::DrawBufferInfo (bool render, const IVec2& size, const BlendState& blendState, const TextureFormat& format)
301 	: m_render		(render)
302 	, m_size		(size)
303 	, m_format		(format)
304 	, m_blendState	(blendState)
305 {
306 }
307 
clearRenderbuffer(const glw::Functions & gl,const tcu::TextureFormat & format,int renderbufferNdx,int renderbufferCount,tcu::TextureLevel & refRenderbuffer)308 void clearRenderbuffer (const glw::Functions&			gl,
309 						const tcu::TextureFormat&		format,
310 						int								renderbufferNdx,
311 						int								renderbufferCount,
312 						tcu::TextureLevel&				refRenderbuffer)
313 {
314 	const tcu::TextureFormatInfo	info		= tcu::getTextureFormatInfo(format);
315 
316 	// Clear each buffer to different color
317 	const float						redScale	= float(renderbufferNdx + 1) / float(renderbufferCount);
318 	const float						blueScale	= float(renderbufferCount - renderbufferNdx) / float(renderbufferCount);
319 	const float						greenScale	= float(((renderbufferCount/2) + renderbufferNdx) % renderbufferCount) / float(renderbufferCount);
320 	// Alpha should never be zero as advanced blend equations assume premultiplied alpha.
321 	const float						alphaScale	= float(1 + (((renderbufferCount/2) + renderbufferCount - renderbufferNdx) % renderbufferCount)) / float(renderbufferCount);
322 
323 	switch (tcu::getTextureChannelClass(format.type))
324 	{
325 		case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
326 		{
327 			const float red		= -1000.0f + 2000.0f * redScale;
328 			const float green	= -1000.0f + 2000.0f * greenScale;
329 			const float blue	= -1000.0f + 2000.0f * blueScale;
330 			const float alpha	= -1000.0f + 2000.0f * alphaScale;
331 			const Vec4	color	(red, green, blue, alpha);
332 
333 			tcu::clear(refRenderbuffer, color);
334 			gl.clearBufferfv(GL_COLOR, renderbufferNdx, color.getPtr());
335 			break;
336 		}
337 
338 		case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
339 		{
340 			const deInt32	red		= deInt32(info.valueMin.x() + (info.valueMax.x() - info.valueMin.x()) * redScale);
341 			const deInt32	green	= deInt32(info.valueMin.y() + (info.valueMax.y() - info.valueMin.y()) * greenScale);
342 			const deInt32	blue	= deInt32(info.valueMin.z() + (info.valueMax.z() - info.valueMin.z()) * blueScale);
343 			const deInt32	alpha	= deInt32(info.valueMin.w() + (info.valueMax.w() - info.valueMin.w()) * alphaScale);
344 			const IVec4		color	(red, green, blue, alpha);
345 
346 			tcu::clear(refRenderbuffer, color);
347 			gl.clearBufferiv(GL_COLOR, renderbufferNdx, color.getPtr());
348 			break;
349 		}
350 
351 		case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
352 		{
353 			const deUint32	red		= deUint32(info.valueMax.x() * redScale);
354 			const deUint32	green	= deUint32(info.valueMax.y() * greenScale);
355 			const deUint32	blue	= deUint32(info.valueMax.z() * blueScale);
356 			const deUint32	alpha	= deUint32(info.valueMax.w() * alphaScale);
357 			const UVec4		color	(red, green, blue, alpha);
358 
359 			tcu::clear(refRenderbuffer, color);
360 			gl.clearBufferuiv(GL_COLOR, renderbufferNdx, color.getPtr());
361 			break;
362 		}
363 
364 		case tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT:
365 		{
366 			const float red		= info.valueMin.x() + (info.valueMax.x() - info.valueMin.x()) * redScale;
367 			const float green	= info.valueMin.y() + (info.valueMax.y() - info.valueMin.y()) * greenScale;
368 			const float blue	= info.valueMin.z() + (info.valueMax.z() - info.valueMin.z()) * blueScale;
369 			const float alpha	= info.valueMin.w() + (info.valueMax.w() - info.valueMin.w()) * alphaScale;
370 			const Vec4	color	(red, green, blue, alpha);
371 
372 			tcu::clear(refRenderbuffer, color);
373 			gl.clearBufferfv(GL_COLOR, renderbufferNdx, color.getPtr());
374 			break;
375 		}
376 
377 		case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
378 		{
379 			const float red		= info.valueMax.x() * redScale;
380 			const float green	= info.valueMax.y() * greenScale;
381 			const float blue	= info.valueMax.z() * blueScale;
382 			const float alpha	= info.valueMax.w() * alphaScale;
383 			const Vec4	color	(red, green, blue, alpha);
384 
385 			tcu::clear(refRenderbuffer, color);
386 			gl.clearBufferfv(GL_COLOR, renderbufferNdx, color.getPtr());
387 			break;
388 		}
389 
390 		default:
391 			DE_ASSERT(DE_FALSE);
392 	}
393 
394 	GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to clear renderbuffer.");
395 }
396 
genRenderbuffers(const glw::Functions & gl,const vector<DrawBufferInfo> & drawBuffers,const glu::Framebuffer & framebuffer,const glu::RenderbufferVector & renderbuffers,vector<TextureLevel> & refRenderbuffers)397 void genRenderbuffers (const glw::Functions&			gl,
398 						const vector<DrawBufferInfo>&	drawBuffers,
399 						const glu::Framebuffer&			framebuffer,
400 						const glu::RenderbufferVector&	renderbuffers,
401 						vector<TextureLevel>&			refRenderbuffers)
402 {
403 	vector<deUint32> bufs;
404 
405 	bufs.resize(drawBuffers.size());
406 
407 	DE_ASSERT(drawBuffers.size() == renderbuffers.size());
408 	DE_ASSERT(drawBuffers.size() == refRenderbuffers.size());
409 
410 	gl.bindFramebuffer(GL_FRAMEBUFFER, *framebuffer);
411 
412 	for (int renderbufferNdx = 0; renderbufferNdx < (int)drawBuffers.size(); renderbufferNdx++)
413 	{
414 		const DrawBufferInfo&		drawBuffer	= drawBuffers[renderbufferNdx];
415 		const TextureFormat&		format		= drawBuffer.getFormat();
416 		const IVec2&				size		= drawBuffer.getSize();
417 		const deUint32				glFormat	= glu::getInternalFormat(format);
418 
419 		bufs[renderbufferNdx]					= GL_COLOR_ATTACHMENT0 + renderbufferNdx;
420 		refRenderbuffers[renderbufferNdx]		= TextureLevel(drawBuffer.getFormat(), size.x(), size.y());
421 
422 		gl.bindRenderbuffer(GL_RENDERBUFFER, renderbuffers[renderbufferNdx]);
423 		gl.renderbufferStorage(GL_RENDERBUFFER, glFormat, size.x(), size.y());
424 		gl.framebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0 + renderbufferNdx, GL_RENDERBUFFER, renderbuffers[renderbufferNdx]);
425 		GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to create renderbuffer.");
426 	}
427 
428 	gl.drawBuffers((glw::GLsizei)bufs.size(), &(bufs[0]));
429 
430 	for (int renderbufferNdx = 0; renderbufferNdx < (int)drawBuffers.size(); renderbufferNdx++)
431 	{
432 		const DrawBufferInfo&		drawBuffer	= drawBuffers[renderbufferNdx];
433 		const TextureFormat&		format		= drawBuffer.getFormat();
434 
435 		clearRenderbuffer(gl, format, renderbufferNdx, (int)refRenderbuffers.size(),  refRenderbuffers[renderbufferNdx]);
436 	}
437 
438 	gl.bindRenderbuffer(GL_RENDERBUFFER, 0);
439 	gl.bindFramebuffer(GL_FRAMEBUFFER, 0);
440 }
441 
getFixedPointFormatThreshold(const tcu::TextureFormat & sourceFormat,const tcu::TextureFormat & readPixelsFormat)442 Vec4 getFixedPointFormatThreshold (const tcu::TextureFormat& sourceFormat, const tcu::TextureFormat& readPixelsFormat)
443 {
444 	DE_ASSERT(tcu::getTextureChannelClass(sourceFormat.type) != tcu::TEXTURECHANNELCLASS_FLOATING_POINT);
445 	DE_ASSERT(tcu::getTextureChannelClass(readPixelsFormat.type) != tcu::TEXTURECHANNELCLASS_FLOATING_POINT);
446 
447 	DE_ASSERT(tcu::getTextureChannelClass(sourceFormat.type) != tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER);
448 	DE_ASSERT(tcu::getTextureChannelClass(readPixelsFormat.type) != tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER);
449 
450 	DE_ASSERT(tcu::getTextureChannelClass(sourceFormat.type) != tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER);
451 	DE_ASSERT(tcu::getTextureChannelClass(readPixelsFormat.type) != tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER);
452 
453 	const tcu::IVec4	srcBits		= tcu::getTextureFormatBitDepth(sourceFormat);
454 	const tcu::IVec4	readBits	= tcu::getTextureFormatBitDepth(readPixelsFormat);
455 
456 	return Vec4(3.0f) / ((tcu::Vector<deUint64, 4>(1) << (tcu::min(srcBits, readBits).cast<deUint64>())) - tcu::Vector<deUint64, 4>(1)).cast<float>();
457 }
458 
getFloatULPThreshold(const tcu::TextureFormat & sourceFormat,const tcu::TextureFormat & readPixelsFormat)459 UVec4 getFloatULPThreshold (const tcu::TextureFormat& sourceFormat, const tcu::TextureFormat& readPixelsFormat)
460 {
461 	const tcu::IVec4	srcMantissaBits		= tcu::getTextureFormatMantissaBitDepth(sourceFormat);
462 	const tcu::IVec4	readMantissaBits	= tcu::getTextureFormatMantissaBitDepth(readPixelsFormat);
463 	tcu::IVec4			ULPDiff(0);
464 
465 	for (int i = 0; i < 4; i++)
466 		if (readMantissaBits[i] >= srcMantissaBits[i])
467 			ULPDiff[i] = readMantissaBits[i] - srcMantissaBits[i];
468 
469 	return UVec4(4) * (UVec4(1) << (ULPDiff.cast<deUint32>()));
470 }
471 
verifyRenderbuffer(TestLog & log,tcu::ResultCollector & results,const tcu::TextureFormat & format,int renderbufferNdx,const tcu::TextureLevel & refRenderbuffer,const tcu::TextureLevel & result)472 void verifyRenderbuffer (TestLog&					log,
473 						 tcu::ResultCollector&		results,
474 						 const tcu::TextureFormat&	format,
475 						 int						renderbufferNdx,
476 						 const tcu::TextureLevel&	refRenderbuffer,
477 						 const tcu::TextureLevel&	result)
478 {
479 	switch (tcu::getTextureChannelClass(format.type))
480 	{
481 		case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
482 		{
483 			const string	name		= "Renderbuffer" + de::toString(renderbufferNdx);
484 			const string	desc		= "Compare renderbuffer " + de::toString(renderbufferNdx);
485 			const UVec4		threshold	= getFloatULPThreshold(format, result.getFormat());
486 
487 			if (!tcu::floatUlpThresholdCompare(log, name.c_str(), desc.c_str(), refRenderbuffer, result, threshold, tcu::COMPARE_LOG_RESULT))
488 				results.fail("Verification of renderbuffer " + de::toString(renderbufferNdx) + " failed.");
489 
490 			break;
491 		}
492 
493 		case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
494 		case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
495 		{
496 			const string	name		= "Renderbuffer" + de::toString(renderbufferNdx);
497 			const string	desc		= "Compare renderbuffer " + de::toString(renderbufferNdx);
498 			const UVec4		threshold	(1, 1, 1, 1);
499 
500 			if (!tcu::intThresholdCompare(log, name.c_str(), desc.c_str(), refRenderbuffer, result, threshold, tcu::COMPARE_LOG_RESULT))
501 				results.fail("Verification of renderbuffer " + de::toString(renderbufferNdx) + " failed.");
502 
503 			break;
504 		}
505 
506 		case tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT:
507 		case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
508 		{
509 			const string	name		= "Renderbuffer" + de::toString(renderbufferNdx);
510 			const string	desc		= "Compare renderbuffer " + de::toString(renderbufferNdx);
511 			const Vec4		threshold	= getFixedPointFormatThreshold(format, result.getFormat());
512 
513 			if (!tcu::floatThresholdCompare(log, name.c_str(), desc.c_str(), refRenderbuffer, result, threshold, tcu::COMPARE_LOG_RESULT))
514 				results.fail("Verification of renderbuffer " + de::toString(renderbufferNdx) + " failed.");
515 
516 			break;
517 		}
518 
519 		default:
520 			DE_ASSERT(DE_FALSE);
521 	}
522 }
523 
getReadPixelFormat(const TextureFormat & format)524 TextureFormat getReadPixelFormat (const TextureFormat& format)
525 {
526 	switch (tcu::getTextureChannelClass(format.type))
527 	{
528 		case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
529 			return TextureFormat(TextureFormat::RGBA, TextureFormat::UNSIGNED_INT32);
530 
531 		case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
532 			return TextureFormat(TextureFormat::RGBA, TextureFormat::SIGNED_INT32);
533 
534 		case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
535 		case tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT:
536 			return TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8);
537 
538 		case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
539 			return TextureFormat(TextureFormat::RGBA, TextureFormat::FLOAT);
540 
541 		default:
542 			DE_ASSERT(false);
543 			return TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8);
544 	}
545 }
546 
verifyRenderbuffers(TestLog & log,tcu::ResultCollector & results,glu::RenderContext & renderContext,const glu::RenderbufferVector & renderbuffers,const glu::Framebuffer & framebuffer,const vector<TextureLevel> & refRenderbuffers)547 void verifyRenderbuffers (TestLog&							log,
548 							tcu::ResultCollector&				results,
549 							glu::RenderContext&				renderContext,
550 							const glu::RenderbufferVector&	renderbuffers,
551 							const glu::Framebuffer&			framebuffer,
552 							const vector<TextureLevel>&		refRenderbuffers)
553 {
554 	const glw::Functions& gl = renderContext.getFunctions();
555 
556 	DE_ASSERT(renderbuffers.size() == refRenderbuffers.size());
557 
558 	gl.bindFramebuffer(GL_FRAMEBUFFER, *framebuffer);
559 
560 	for (int renderbufferNdx = 0; renderbufferNdx < (int)renderbuffers.size(); renderbufferNdx++)
561 	{
562 		const TextureLevel&	refRenderbuffer	= refRenderbuffers[renderbufferNdx];
563 		const int			width			= refRenderbuffer.getWidth();
564 		const int			height			= refRenderbuffer.getHeight();
565 		const TextureFormat	format			= refRenderbuffer.getFormat();
566 
567 		tcu::TextureLevel	result			(getReadPixelFormat(format), width, height);
568 
569 		gl.readBuffer(GL_COLOR_ATTACHMENT0 + renderbufferNdx);
570 		glu::readPixels(renderContext, 0, 0, result.getAccess());
571 		GLU_EXPECT_NO_ERROR(gl.getError(), "Reading pixels from renderbuffer failed.");
572 
573 		verifyRenderbuffer(log, results, format, renderbufferNdx, refRenderbuffer, result);
574 	}
575 
576 	gl.bindFramebuffer(GL_FRAMEBUFFER, 0);
577 }
578 
579 static const float s_quadCoords[] =
580 {
581 	-0.5f, -0.5f,
582 	 0.5f, -0.5f,
583 	 0.5f,  0.5f,
584 
585 	 0.5f,  0.5f,
586 	-0.5f,  0.5f,
587 	-0.5f, -0.5f
588 };
589 
setBlendState(rr::FragmentOperationState & fragOps,const BlendState & state)590 void setBlendState (rr::FragmentOperationState& fragOps, const BlendState& state)
591 {
592 	if (state.blendEq)
593 	{
594 		if (state.blendEq->is<BlendEq>())
595 		{
596 			if (isAdvancedBlendEq(state.blendEq->get<BlendEq>()))
597 			{
598 				const rr::BlendEquationAdvanced	equation = mapGLBlendEquationAdvanced(state.blendEq->get<BlendEq>());
599 
600 				fragOps.blendMode				= rr::BLENDMODE_ADVANCED;
601 				fragOps.blendEquationAdvaced	= equation;
602 			}
603 			else
604 			{
605 				const rr::BlendEquation equation = mapGLBlendEquation(state.blendEq->get<BlendEq>());
606 
607 				fragOps.blendMode				= rr::BLENDMODE_STANDARD;
608 				fragOps.blendRGBState.equation	= equation;
609 				fragOps.blendAState.equation	= equation;
610 			}
611 		}
612 		else
613 		{
614 			DE_ASSERT(state.blendEq->is<SeparateBlendEq>());
615 
616 			fragOps.blendMode				= rr::BLENDMODE_STANDARD;
617 			fragOps.blendRGBState.equation	= mapGLBlendEquation(state.blendEq->get<SeparateBlendEq>().rgb);
618 			fragOps.blendAState.equation	= mapGLBlendEquation(state.blendEq->get<SeparateBlendEq>().alpha);
619 		}
620 	}
621 
622 	if (state.blendFunc)
623 	{
624 		if (state.blendFunc->is<BlendFunc>())
625 		{
626 			const rr::BlendFunc srcFunction = mapGLBlendFunc(state.blendFunc->get<BlendFunc>().src);
627 			const rr::BlendFunc dstFunction = mapGLBlendFunc(state.blendFunc->get<BlendFunc>().dst);
628 
629 			fragOps.blendRGBState.srcFunc	= srcFunction;
630 			fragOps.blendRGBState.dstFunc	= dstFunction;
631 
632 			fragOps.blendAState.srcFunc		= srcFunction;
633 			fragOps.blendAState.dstFunc		= dstFunction;
634 		}
635 		else
636 		{
637 			DE_ASSERT(state.blendFunc->is<SeparateBlendFunc>());
638 
639 			fragOps.blendRGBState.srcFunc	= mapGLBlendFunc(state.blendFunc->get<SeparateBlendFunc>().rgb.src);
640 			fragOps.blendRGBState.dstFunc	= mapGLBlendFunc(state.blendFunc->get<SeparateBlendFunc>().rgb.dst);
641 
642 			fragOps.blendAState.srcFunc		= mapGLBlendFunc(state.blendFunc->get<SeparateBlendFunc>().alpha.src);
643 			fragOps.blendAState.dstFunc		= mapGLBlendFunc(state.blendFunc->get<SeparateBlendFunc>().alpha.dst);
644 		}
645 	}
646 
647 	if (state.colorMask)
648 		fragOps.colorMask = *state.colorMask;
649 }
650 
createRenderState(const BlendState & preCommonBlendState,const BlendState & postCommonBlendState,const DrawBufferInfo & info,int subpixelBits)651 rr::RenderState createRenderState (const BlendState& preCommonBlendState, const BlendState& postCommonBlendState, const DrawBufferInfo& info, int subpixelBits)
652 {
653 	const IVec2		size	= info.getSize();
654 	rr::RenderState	state	(rr::ViewportState(rr::WindowRectangle(0, 0, size.x(), size.y())), subpixelBits);
655 
656 	state.fragOps.blendMode = rr::BLENDMODE_STANDARD;
657 
658 	setBlendState(state.fragOps, preCommonBlendState);
659 	setBlendState(state.fragOps, info.getBlendState());
660 	setBlendState(state.fragOps, postCommonBlendState);
661 
662 	if (postCommonBlendState.enableBlend)
663 		state.fragOps.blendMode = (*(postCommonBlendState.enableBlend) ? state.fragOps.blendMode : rr::BLENDMODE_NONE);
664 	else  if (info.getBlendState().enableBlend)
665 		state.fragOps.blendMode = (*(info.getBlendState().enableBlend) ? state.fragOps.blendMode : rr::BLENDMODE_NONE);
666 	else if (preCommonBlendState.enableBlend)
667 		state.fragOps.blendMode = (*(preCommonBlendState.enableBlend) ? state.fragOps.blendMode : rr::BLENDMODE_NONE);
668 	else
669 		state.fragOps.blendMode = rr::BLENDMODE_NONE;
670 
671 	if (tcu::getTextureChannelClass(info.getFormat().type) != tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT
672 		&& tcu::getTextureChannelClass(info.getFormat().type) != tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT
673 		&& tcu::getTextureChannelClass(info.getFormat().type) != tcu::TEXTURECHANNELCLASS_FLOATING_POINT)
674 		state.fragOps.blendMode = rr::BLENDMODE_NONE;
675 
676 	return state;
677 }
678 
679 class VertexShader : public rr::VertexShader
680 {
681 public:
682 					VertexShader	(void);
683 	virtual void	shadeVertices	(const rr::VertexAttrib* inputs, rr::VertexPacket* const* packets, const int numPackets) const;
684 };
685 
VertexShader(void)686 VertexShader::VertexShader (void)
687 	: rr::VertexShader	(1, 1)
688 {
689 	m_inputs[0].type	= rr::GENERICVECTYPE_FLOAT;
690 	m_outputs[0].type	= rr::GENERICVECTYPE_FLOAT;
691 }
692 
shadeVertices(const rr::VertexAttrib * inputs,rr::VertexPacket * const * packets,const int numPackets) const693 void VertexShader::shadeVertices (const rr::VertexAttrib* inputs, rr::VertexPacket* const* packets, const int numPackets) const
694 {
695 	for (int packetNdx = 0; packetNdx < numPackets; ++packetNdx)
696 	{
697 		rr::VertexPacket& packet = *packets[packetNdx];
698 
699 		packet.position		= rr::readVertexAttribFloat(inputs[0], packet.instanceNdx, packet.vertexNdx);
700 		packet.outputs[0]	= 0.5f * (Vec4(1.0f) + rr::readVertexAttribFloat(inputs[0], packet.instanceNdx, packet.vertexNdx));
701 	}
702 }
703 
704 class FragmentShader : public rr::FragmentShader
705 {
706 public:
707 			FragmentShader	(int drawBufferNdx, const DrawBufferInfo& info);
708 	void	shadeFragments	(rr::FragmentPacket* packets, const int numPackets, const rr::FragmentShadingContext& context) const;
709 
710 private:
711 	const int				m_drawBufferNdx;
712 	const DrawBufferInfo	m_info;
713 };
714 
FragmentShader(int drawBufferNdx,const DrawBufferInfo & info)715 FragmentShader::FragmentShader (int drawBufferNdx, const DrawBufferInfo& info)
716 	: rr::FragmentShader	(1, 1)
717 	, m_drawBufferNdx		(drawBufferNdx)
718 	, m_info				(info)
719 {
720 	m_inputs[0].type = rr::GENERICVECTYPE_FLOAT;
721 
722 	switch (tcu::getTextureChannelClass(m_info.getFormat().type))
723 	{
724 		case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
725 		case tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT:
726 		case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
727 			m_outputs[0].type = rr::GENERICVECTYPE_FLOAT;
728 			break;
729 
730 		case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
731 			m_outputs[0].type = rr::GENERICVECTYPE_UINT32;
732 			break;
733 
734 		case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
735 			m_outputs[0].type = rr::GENERICVECTYPE_INT32;
736 			break;
737 
738 		default:
739 			DE_ASSERT(false);
740 	};
741 }
742 
shadeFragments(rr::FragmentPacket * packets,const int numPackets,const rr::FragmentShadingContext & context) const743 void FragmentShader::shadeFragments (rr::FragmentPacket* packets, const int numPackets, const rr::FragmentShadingContext& context) const
744 {
745 	for (int packetNdx = 0; packetNdx < numPackets; ++packetNdx)
746 	{
747 		rr::FragmentPacket& packet = packets[packetNdx];
748 
749 		DE_ASSERT(m_drawBufferNdx >= 0);
750 		DE_UNREF(m_info);
751 
752 		for (int fragNdx = 0; fragNdx < 4; ++fragNdx)
753 		{
754 			const Vec2	vColor		= rr::readVarying<float>(packet, context, 0, fragNdx).xy();
755 			const float	values[]	=
756 			{
757 				vColor.x(),
758 				vColor.y(),
759 				(1.0f - vColor.x()),
760 				(1.0f - vColor.y())
761 			};
762 
763 			switch (tcu::getTextureChannelClass(m_info.getFormat().type))
764 			{
765 				case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
766 				case tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT:
767 				case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
768 				{
769 					const Vec4 color (values[(m_drawBufferNdx + 0) % 4],
770 									  values[(m_drawBufferNdx + 1) % 4],
771 									  values[(m_drawBufferNdx + 2) % 4],
772 									  values[(m_drawBufferNdx + 3) % 4]);
773 
774 					rr::writeFragmentOutput(context, packetNdx, fragNdx, 0, color);
775 					break;
776 				}
777 
778 				case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
779 				{
780 					const UVec4 color ((deUint32)(values[(m_drawBufferNdx + 0) % 4]),
781 									   (deUint32)(values[(m_drawBufferNdx + 1) % 4]),
782 									   (deUint32)(values[(m_drawBufferNdx + 2) % 4]),
783 									   (deUint32)(values[(m_drawBufferNdx + 3) % 4]));
784 
785 					rr::writeFragmentOutput(context, packetNdx, fragNdx, 0, color);
786 					break;
787 				}
788 
789 				case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
790 				{
791 					const IVec4 color ((deInt32)(values[(m_drawBufferNdx + 0) % 4]),
792 									   (deInt32)(values[(m_drawBufferNdx + 1) % 4]),
793 									   (deInt32)(values[(m_drawBufferNdx + 2) % 4]),
794 									   (deInt32)(values[(m_drawBufferNdx + 3) % 4]));
795 
796 					rr::writeFragmentOutput(context, packetNdx, fragNdx, 0, color);
797 					break;
798 				}
799 
800 				default:
801 					DE_ASSERT(DE_FALSE);
802 			};
803 		}
804 	}
805 }
806 
createVertexAttrib(const float * coords)807 rr::VertexAttrib createVertexAttrib (const float* coords)
808 {
809 	rr::VertexAttrib attrib;
810 
811 	attrib.type		= rr::VERTEXATTRIBTYPE_FLOAT;
812 	attrib.size		= 2;
813 	attrib.pointer	= coords;
814 
815 	return attrib;
816 }
817 
renderRefQuad(const BlendState & preCommonBlendState,const BlendState & postCommonBlendState,const vector<DrawBufferInfo> & drawBuffers,const int subpixelBits,vector<TextureLevel> & refRenderbuffers)818 void renderRefQuad (const BlendState&				preCommonBlendState,
819 					const BlendState&				postCommonBlendState,
820 					const vector<DrawBufferInfo>&	drawBuffers,
821 					const int						subpixelBits,
822 					vector<TextureLevel>&			refRenderbuffers)
823 {
824 	const rr::Renderer			renderer;
825 	const rr::PrimitiveList		primitives		(rr::PRIMITIVETYPE_TRIANGLES, 6, 0);
826 	const rr::VertexAttrib		vertexAttribs[] =
827 	{
828 		createVertexAttrib(s_quadCoords)
829 	};
830 
831 	for (int drawBufferNdx = 0; drawBufferNdx < (int)drawBuffers.size(); drawBufferNdx++)
832 	{
833 		if (drawBuffers[drawBufferNdx].getRender())
834 		{
835 			const rr::RenderState	renderState		(createRenderState(preCommonBlendState, postCommonBlendState, drawBuffers[drawBufferNdx], subpixelBits));
836 			const rr::RenderTarget	renderTarget	(rr::MultisamplePixelBufferAccess::fromSinglesampleAccess(refRenderbuffers[drawBufferNdx].getAccess()));
837 			const VertexShader		vertexShader;
838 			const FragmentShader	fragmentShader	(drawBufferNdx, drawBuffers[drawBufferNdx]);
839 			const rr::Program		program			(&vertexShader, &fragmentShader);
840 			const rr::DrawCommand	command			(renderState, renderTarget, program, DE_LENGTH_OF_ARRAY(vertexAttribs), vertexAttribs, primitives);
841 
842 			renderer.draw(command);
843 		}
844 	}
845 }
846 
requiresAdvancedBlendEq(const BlendState & pre,const BlendState post,const vector<DrawBufferInfo> & drawBuffers)847 bool requiresAdvancedBlendEq (const BlendState& pre, const BlendState post, const vector<DrawBufferInfo>& drawBuffers)
848 {
849 	bool requiresAdvancedBlendEq = false;
850 
851 	if (pre.blendEq && pre.blendEq->is<BlendEq>())
852 		requiresAdvancedBlendEq |= isAdvancedBlendEq(pre.blendEq->get<BlendEq>());
853 
854 	if (post.blendEq && post.blendEq->is<BlendEq>())
855 		requiresAdvancedBlendEq |= isAdvancedBlendEq(post.blendEq->get<BlendEq>());
856 
857 	for (int drawBufferNdx = 0; drawBufferNdx < (int)drawBuffers.size(); drawBufferNdx++)
858 	{
859 		const BlendState& drawBufferBlendState = drawBuffers[drawBufferNdx].getBlendState();
860 
861 		if (drawBufferBlendState.blendEq && drawBufferBlendState.blendEq->is<BlendEq>())
862 			requiresAdvancedBlendEq |= isAdvancedBlendEq(drawBufferBlendState.blendEq->get<BlendEq>());
863 	}
864 
865 	return requiresAdvancedBlendEq;
866 }
867 
genVertexSource(glu::RenderContext & renderContext)868 glu::VertexSource genVertexSource (glu::RenderContext& renderContext)
869 {
870 	const bool supportsES32 = glu::contextSupports(renderContext.getType(), glu::ApiType::es(3, 2));
871 
872 	const char* const vertexSource =
873 		"${GLSL_VERSION_DECL}\n"
874 		"layout(location=0) in highp vec2 i_coord;\n"
875 		"out highp vec2 v_color;\n"
876 		"void main (void)\n"
877 		"{\n"
878 		"\tv_color = 0.5 * (vec2(1.0) + i_coord);\n"
879 		"\tgl_Position = vec4(i_coord, 0.0, 1.0);\n"
880 		"}";
881 
882 	map<string, string> args;
883 	args["GLSL_VERSION_DECL"] = supportsES32 ? getGLSLVersionDeclaration(glu::GLSL_VERSION_320_ES) : getGLSLVersionDeclaration(glu::GLSL_VERSION_310_ES);
884 
885 	return glu::VertexSource(tcu::StringTemplate(vertexSource).specialize(args));
886 }
887 
genFragmentSource(const BlendState & preCommonBlendState,const BlendState & postCommonBlendState,const vector<DrawBufferInfo> & drawBuffers,glu::RenderContext & renderContext)888 glu::FragmentSource genFragmentSource (const BlendState& preCommonBlendState, const BlendState& postCommonBlendState, const vector<DrawBufferInfo>& drawBuffers, glu::RenderContext& renderContext)
889 {
890 	std::ostringstream stream;
891 	const bool supportsES32 = glu::contextSupports(renderContext.getType(), glu::ApiType::es(3, 2));
892 
893 	stream << "${GLSL_VERSION_DECL}\n";
894 
895 	if (requiresAdvancedBlendEq(preCommonBlendState, postCommonBlendState, drawBuffers))
896 	{
897 		stream << "${GLSL_EXTENSION}"
898 			   <<  "layout(blend_support_all_equations) out;\n";
899 	}
900 
901 	stream << "in highp vec2 v_color;\n";
902 
903 	for (int drawBufferNdx = 0; drawBufferNdx < (int)drawBuffers.size(); drawBufferNdx++)
904 	{
905 		const DrawBufferInfo&	drawBuffer			= drawBuffers[drawBufferNdx];
906 		const TextureFormat&	format				= drawBuffer.getFormat();
907 
908 		stream << "layout(location=" << drawBufferNdx << ") out highp ";
909 
910 		switch (tcu::getTextureChannelClass(format.type))
911 		{
912 			case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
913 			case tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT:
914 			case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
915 				stream << "vec4";
916 				break;
917 
918 			case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
919 				stream << "uvec4";
920 				break;
921 
922 			case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
923 				stream << "ivec4";
924 				break;
925 
926 			default:
927 				DE_ASSERT(DE_FALSE);
928 		};
929 
930 		stream << " o_drawBuffer" <<  drawBufferNdx << ";\n";
931 	}
932 
933 	stream << "void main (void)\n"
934 		   << "{\n";
935 
936 	for (int drawBufferNdx = 0; drawBufferNdx < (int)drawBuffers.size(); drawBufferNdx++)
937 	{
938 		const DrawBufferInfo&	drawBuffer		= drawBuffers[drawBufferNdx];
939 		const TextureFormat&	format			= drawBuffer.getFormat();
940 		const char* const		values[]		=
941 		{
942 			"v_color.x",
943 			"v_color.y",
944 			"(1.0 - v_color.x)",
945 			"(1.0 - v_color.y)"
946 		};
947 
948 		stream << "\to_drawBuffer" <<  drawBufferNdx;
949 
950 		switch (tcu::getTextureChannelClass(format.type))
951 		{
952 			case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
953 			case tcu::TEXTURECHANNELCLASS_SIGNED_FIXED_POINT:
954 			case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
955 				stream << " = vec4(" << values[(drawBufferNdx + 0) % 4]
956 					   << ", " << values[(drawBufferNdx + 1) % 4]
957 					   << ", " << values[(drawBufferNdx + 2) % 4]
958 					   << ", " << values[(drawBufferNdx + 3) % 4] << ");\n";
959 				break;
960 
961 			case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
962 				stream << " = uvec4(uint(" << values[(drawBufferNdx + 0) % 4]
963 					   << "), uint(" << values[(drawBufferNdx + 1) % 4]
964 					   << "), uint(" << values[(drawBufferNdx + 2) % 4]
965 					   << "), uint(" << values[(drawBufferNdx + 3) % 4] << "));\n";
966 				break;
967 
968 			case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
969 				stream << " = ivec4(int(" << values[(drawBufferNdx + 0) % 4]
970 					   << "), int(" << values[(drawBufferNdx + 1) % 4]
971 					   << "), int(" << values[(drawBufferNdx + 2) % 4]
972 					   << "), int(" << values[(drawBufferNdx + 3) % 4] << "));\n";
973 				break;
974 
975 			default:
976 				DE_ASSERT(DE_FALSE);
977 		};
978 	}
979 
980 	stream << "}";
981 
982 	map<string, string> args;
983 	args["GLSL_VERSION_DECL"] = supportsES32 ? getGLSLVersionDeclaration(glu::GLSL_VERSION_320_ES) : getGLSLVersionDeclaration(glu::GLSL_VERSION_310_ES);
984 	args["GLSL_EXTENSION"] = supportsES32 ? "\n" : "#extension GL_KHR_blend_equation_advanced : require\n";
985 
986 	return glu::FragmentSource(tcu::StringTemplate(stream.str()).specialize(args));
987 }
988 
genShaderSources(const BlendState & preCommonBlendState,const BlendState & postCommonBlendState,const vector<DrawBufferInfo> & drawBuffers,glu::RenderContext & renderContext)989 glu::ProgramSources genShaderSources (const BlendState& preCommonBlendState, const BlendState& postCommonBlendState, const vector<DrawBufferInfo>& drawBuffers, glu::RenderContext& renderContext)
990 {
991 	return glu::ProgramSources() << genVertexSource(renderContext) << genFragmentSource(preCommonBlendState, postCommonBlendState, drawBuffers, renderContext);
992 }
993 
renderGLQuad(glu::RenderContext & renderContext,const glu::ShaderProgram & program)994 void renderGLQuad (glu::RenderContext&			renderContext,
995 				   const glu::ShaderProgram&	program)
996 {
997 	const glu::VertexArrayBinding vertexArrays[] =
998 	{
999 		glu::VertexArrayBinding(glu::BindingPoint(0), glu::VertexArrayPointer(glu::VTX_COMP_FLOAT, glu::VTX_COMP_CONVERT_NONE, 2, 6, 0, s_quadCoords))
1000 	};
1001 
1002 	glu::draw(renderContext, program.getProgram(), 1, vertexArrays, glu::pr::Triangles(6));
1003 }
1004 
renderQuad(TestLog & log,glu::RenderContext & renderContext,const BlendState & preCommonBlendState,const BlendState & postCommonBlendState,const vector<DrawBufferInfo> & drawBuffers,const glu::Framebuffer & framebuffer,vector<TextureLevel> & refRenderbuffers)1005 void renderQuad (TestLog&						log,
1006 				 glu::RenderContext&			renderContext,
1007 				 const BlendState&				preCommonBlendState,
1008 				 const BlendState&				postCommonBlendState,
1009 				 const vector<DrawBufferInfo>&	drawBuffers,
1010 				 const glu::Framebuffer&		framebuffer,
1011 				 vector<TextureLevel>&			refRenderbuffers)
1012 {
1013 	const glw::Functions&		gl						= renderContext.getFunctions();
1014 	const glu::ShaderProgram	program					(gl, genShaderSources(preCommonBlendState, postCommonBlendState, drawBuffers, renderContext));
1015 	const IVec2					size					= drawBuffers[0].getSize();
1016 	const bool					requiresBlendBarriers	= requiresAdvancedBlendEq(preCommonBlendState, postCommonBlendState, drawBuffers);
1017 
1018 	vector<deUint32> bufs;
1019 
1020 	bufs.resize(drawBuffers.size());
1021 
1022 	for (int bufNdx = 0; bufNdx < (int)bufs.size(); bufNdx++)
1023 		bufs[bufNdx] = (drawBuffers[bufNdx].getRender() ? GL_COLOR_ATTACHMENT0 + bufNdx : GL_NONE);
1024 
1025 	log << program;
1026 
1027 	gl.viewport(0, 0, size.x(), size.y());
1028 	gl.useProgram(program.getProgram());
1029 	gl.bindFramebuffer(GL_FRAMEBUFFER, *framebuffer);
1030 
1031 	setCommonBlendState(gl, preCommonBlendState);
1032 
1033 	for (int renderbufferNdx = 0; renderbufferNdx < (int)drawBuffers.size(); renderbufferNdx++)
1034 		setIndexedBlendState(gl, drawBuffers[renderbufferNdx].getBlendState(), renderbufferNdx);
1035 
1036 	setCommonBlendState(gl, postCommonBlendState);
1037 
1038 	gl.drawBuffers((glw::GLsizei)bufs.size(), &(bufs[0]));
1039 
1040 	if (requiresBlendBarriers)
1041 		gl.blendBarrier();
1042 
1043 	renderGLQuad(renderContext, program);
1044 
1045 	if (requiresBlendBarriers)
1046 		gl.blendBarrier();
1047 
1048 	gl.drawBuffers(0, 0);
1049 	gl.bindFramebuffer(GL_FRAMEBUFFER, 0);
1050 	gl.useProgram(0);
1051 
1052 	GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to render");
1053 
1054 	int subpixelBits = 0;
1055 	gl.getIntegerv(GL_SUBPIXEL_BITS, &subpixelBits);
1056 
1057 	renderRefQuad(preCommonBlendState, postCommonBlendState, drawBuffers, subpixelBits, refRenderbuffers);
1058 }
1059 
logBlendState(TestLog & log,const BlendState & blend)1060 void logBlendState (TestLog&			log,
1061 					const BlendState&	blend)
1062 {
1063 	if (blend.enableBlend)
1064 	{
1065 		if (*blend.enableBlend)
1066 			log << TestLog::Message << "Enable blending." << TestLog::EndMessage;
1067 		else
1068 			log << TestLog::Message << "Disable blending." << TestLog::EndMessage;
1069 	}
1070 
1071 	if (blend.colorMask)
1072 	{
1073 		const BVec4 mask = *blend.colorMask;
1074 
1075 		log << TestLog::Message << "Set color mask: " << mask << "." << TestLog::EndMessage;
1076 	}
1077 
1078 	if (blend.blendEq)
1079 	{
1080 		const Either<BlendEq, SeparateBlendEq>& blendEq = *blend.blendEq;
1081 
1082 		if (blendEq.is<BlendEq>())
1083 			log << TestLog::Message << "Set blend equation: " << glu::getBlendEquationStr(blendEq.get<BlendEq>()) << "." << TestLog::EndMessage;
1084 		else if (blendEq.is<SeparateBlendEq>())
1085 			log << TestLog::Message << "Set blend equation rgb: " << glu::getBlendEquationStr(blendEq.get<SeparateBlendEq>().rgb) << ", alpha: " << glu::getBlendEquationStr(blendEq.get<SeparateBlendEq>().alpha) << "." << TestLog::EndMessage;
1086 		else
1087 			DE_ASSERT(false);
1088 	}
1089 
1090 	if (blend.blendFunc)
1091 	{
1092 		const Either<BlendFunc, SeparateBlendFunc>& blendFunc = *blend.blendFunc;
1093 
1094 		if (blendFunc.is<BlendFunc>())
1095 			log << TestLog::Message << "Set blend function source: " << glu::getBlendFactorStr(blendFunc.get<BlendFunc>().src) << ", destination: " << glu::getBlendFactorStr(blendFunc.get<BlendFunc>().dst) << "." << TestLog::EndMessage;
1096 		else if (blendFunc.is<SeparateBlendFunc>())
1097 		{
1098 			log << TestLog::Message << "Set blend function rgb source: " << glu::getBlendFactorStr(blendFunc.get<SeparateBlendFunc>().rgb.src) << ", destination: " << glu::getBlendFactorStr(blendFunc.get<SeparateBlendFunc>().rgb.dst) << "." << TestLog::EndMessage;
1099 			log << TestLog::Message << "Set blend function alpha source: " << glu::getBlendFactorStr(blendFunc.get<SeparateBlendFunc>().alpha.src) << ", destination: " << glu::getBlendFactorStr(blendFunc.get<SeparateBlendFunc>().alpha.dst) << "." << TestLog::EndMessage;
1100 		}
1101 		else
1102 			DE_ASSERT(false);
1103 	}
1104 }
1105 
logTestCaseInfo(TestLog & log,const BlendState & preCommonBlendState,const BlendState & postCommonBlendState,const vector<DrawBufferInfo> & drawBuffers)1106 void logTestCaseInfo (TestLog&						log,
1107 					  const BlendState&				preCommonBlendState,
1108 					  const BlendState&				postCommonBlendState,
1109 					  const vector<DrawBufferInfo>&	drawBuffers)
1110 {
1111 	{
1112 		tcu::ScopedLogSection drawBuffersSection(log, "DrawBuffers", "Draw buffers");
1113 
1114 		for (int drawBufferNdx = 0; drawBufferNdx < (int)drawBuffers.size(); drawBufferNdx++)
1115 		{
1116 			const tcu::ScopedLogSection	drawBufferSection	(log, "DrawBuffer" + de::toString(drawBufferNdx), "Draw Buffer " + de::toString(drawBufferNdx));
1117 			const DrawBufferInfo&		drawBuffer			= drawBuffers[drawBufferNdx];
1118 
1119 			log << TestLog::Message << "Format: " << drawBuffer.getFormat() << TestLog::EndMessage;
1120 			log << TestLog::Message << "Size: " << drawBuffer.getSize() << TestLog::EndMessage;
1121 			log << TestLog::Message << "Render: " << (drawBuffer.getRender() ? "true" : "false") << TestLog::EndMessage;
1122 		}
1123 	}
1124 
1125 	if (!preCommonBlendState.isEmpty())
1126 	{
1127 		tcu::ScopedLogSection s(log, "PreCommonState", "First set common blend state");
1128 		logBlendState(log, preCommonBlendState);
1129 	}
1130 
1131 	for (int drawBufferNdx = 0; drawBufferNdx < (int)drawBuffers.size(); drawBufferNdx++)
1132 	{
1133 		if (!drawBuffers[drawBufferNdx].getBlendState().isEmpty())
1134 		{
1135 			const tcu::ScopedLogSection s(log, "DrawBufferState" + de::toString(drawBufferNdx), "Set DrawBuffer " + de::toString(drawBufferNdx) + " state to");
1136 
1137 			logBlendState(log, drawBuffers[drawBufferNdx].getBlendState());
1138 		}
1139 	}
1140 
1141 	if (!postCommonBlendState.isEmpty())
1142 	{
1143 		tcu::ScopedLogSection s(log, "PostCommonState", "After set common blend state");
1144 		logBlendState(log, postCommonBlendState);
1145 	}
1146 }
1147 
runTest(TestLog & log,tcu::ResultCollector & results,glu::RenderContext & renderContext,const BlendState & preCommonBlendState,const BlendState & postCommonBlendState,const vector<DrawBufferInfo> & drawBuffers)1148 void runTest (TestLog&						log,
1149 			  tcu::ResultCollector&			results,
1150 			  glu::RenderContext&			renderContext,
1151 
1152 			  const BlendState&				preCommonBlendState,
1153 			  const BlendState&				postCommonBlendState,
1154 			  const vector<DrawBufferInfo>&	drawBuffers)
1155 {
1156 	const glw::Functions&	gl					= renderContext.getFunctions();
1157 	glu::RenderbufferVector	renderbuffers		(gl, drawBuffers.size());
1158 	glu::Framebuffer		framebuffer			(gl);
1159 	vector<TextureLevel>	refRenderbuffers	(drawBuffers.size());
1160 
1161 	logTestCaseInfo(log, preCommonBlendState, postCommonBlendState, drawBuffers);
1162 
1163 	genRenderbuffers(gl, drawBuffers, framebuffer, renderbuffers, refRenderbuffers);
1164 
1165 	renderQuad(log, renderContext, preCommonBlendState, postCommonBlendState, drawBuffers, framebuffer, refRenderbuffers);
1166 
1167 	verifyRenderbuffers(log, results, renderContext, renderbuffers, framebuffer, refRenderbuffers);
1168 }
1169 
1170 class DrawBuffersIndexedTest : public TestCase
1171 {
1172 public:
1173 					DrawBuffersIndexedTest (Context&						context,
1174 											const BlendState&				preCommonBlendState,
1175 											const BlendState&				postCommonBlendState,
1176 											const vector<DrawBufferInfo>&	drawBuffers,
1177 											const string&					name,
1178 											const string&					description);
1179 
1180 	void			init					(void);
1181 	IterateResult	iterate					(void);
1182 
1183 private:
1184 	const BlendState				m_preCommonBlendState;
1185 	const BlendState				m_postCommonBlendState;
1186 	const vector<DrawBufferInfo>	m_drawBuffers;
1187 };
1188 
DrawBuffersIndexedTest(Context & context,const BlendState & preCommonBlendState,const BlendState & postCommonBlendState,const vector<DrawBufferInfo> & drawBuffers,const string & name,const string & description)1189 DrawBuffersIndexedTest::DrawBuffersIndexedTest (Context&						context,
1190 												const BlendState&				preCommonBlendState,
1191 												const BlendState&				postCommonBlendState,
1192 												const vector<DrawBufferInfo>&	drawBuffers,
1193 												const string&					name,
1194 												const string&					description)
1195 	: TestCase					(context, name.c_str(), description.c_str())
1196 	, m_preCommonBlendState		(preCommonBlendState)
1197 	, m_postCommonBlendState	(postCommonBlendState)
1198 	, m_drawBuffers				(drawBuffers)
1199 {
1200 }
1201 
init(void)1202 void DrawBuffersIndexedTest::init (void)
1203 {
1204 	const bool supportsES32orGL45 = checkES32orGL45Support(m_context);
1205 
1206 	if (!supportsES32orGL45)
1207 	{
1208 		if (requiresAdvancedBlendEq(m_preCommonBlendState, m_postCommonBlendState, m_drawBuffers) && !m_context.getContextInfo().isExtensionSupported("GL_KHR_blend_equation_advanced"))
1209 			TCU_THROW(NotSupportedError, "Extension GL_KHR_blend_equation_advanced not supported");
1210 
1211 		if (!m_context.getContextInfo().isExtensionSupported("GL_EXT_draw_buffers_indexed"))
1212 			TCU_THROW(NotSupportedError, "Extension GL_EXT_draw_buffers_indexed not supported");
1213 	}
1214 }
1215 
iterate(void)1216 TestCase::IterateResult DrawBuffersIndexedTest::iterate (void)
1217 {
1218 	TestLog&				log		= m_testCtx.getLog();
1219 	tcu::ResultCollector	results	(log);
1220 
1221 	runTest(log, results, m_context.getRenderContext(), m_preCommonBlendState, m_postCommonBlendState, m_drawBuffers);
1222 
1223 	results.setTestContextResult(m_testCtx);
1224 
1225 	return STOP;
1226 }
1227 
getRandomBlendEq(de::Random & rng)1228 BlendEq getRandomBlendEq (de::Random& rng)
1229 {
1230 	const BlendEq eqs[] =
1231 	{
1232 		GL_FUNC_ADD,
1233 		GL_FUNC_SUBTRACT,
1234 		GL_FUNC_REVERSE_SUBTRACT,
1235 		GL_MIN,
1236 		GL_MAX
1237 	};
1238 
1239 	return de::getSizedArrayElement<DE_LENGTH_OF_ARRAY(eqs)>(eqs, rng.getUint32() % DE_LENGTH_OF_ARRAY(eqs));
1240 }
1241 
getRandomBlendFunc(de::Random & rng)1242 BlendFunc getRandomBlendFunc (de::Random& rng)
1243 {
1244 	const deUint32 funcs[] =
1245 	{
1246 		GL_ZERO,
1247 		GL_ONE,
1248 		GL_SRC_COLOR,
1249 		GL_ONE_MINUS_SRC_COLOR,
1250 		GL_DST_COLOR,
1251 		GL_ONE_MINUS_DST_COLOR,
1252 		GL_SRC_ALPHA,
1253 		GL_ONE_MINUS_SRC_ALPHA,
1254 		GL_DST_ALPHA,
1255 		GL_ONE_MINUS_DST_ALPHA,
1256 		GL_CONSTANT_COLOR,
1257 		GL_ONE_MINUS_CONSTANT_COLOR,
1258 		GL_CONSTANT_ALPHA,
1259 		GL_ONE_MINUS_CONSTANT_ALPHA,
1260 		GL_SRC_ALPHA_SATURATE
1261 	};
1262 
1263 	const deUint32 src = de::getSizedArrayElement<DE_LENGTH_OF_ARRAY(funcs)>(funcs, rng.getUint32() % DE_LENGTH_OF_ARRAY(funcs));
1264 	const deUint32 dst = de::getSizedArrayElement<DE_LENGTH_OF_ARRAY(funcs)>(funcs, rng.getUint32() % DE_LENGTH_OF_ARRAY(funcs));
1265 
1266 	return BlendFunc(src, dst);
1267 }
1268 
genRandomBlendState(de::Random & rng,BlendState & blendState)1269 void genRandomBlendState (de::Random& rng, BlendState& blendState)
1270 {
1271 	if (rng.getBool())
1272 		blendState.enableBlend = rng.getBool();
1273 
1274 	if (rng.getBool())
1275 	{
1276 		if (rng.getBool())
1277 			blendState.blendEq = getRandomBlendEq(rng);
1278 		else
1279 		{
1280 			const BlendEq	rgb		= getRandomBlendEq(rng);
1281 			const BlendEq	alpha	= getRandomBlendEq(rng);
1282 
1283 			blendState.blendEq		= SeparateBlendEq(rgb, alpha);
1284 		}
1285 	}
1286 
1287 	if (rng.getBool())
1288 	{
1289 		if (rng.getBool())
1290 			blendState.blendFunc = getRandomBlendFunc(rng);
1291 		else
1292 		{
1293 			const BlendFunc	rgb		= getRandomBlendFunc(rng);
1294 			const BlendFunc	alpha	= getRandomBlendFunc(rng);
1295 
1296 			blendState.blendFunc	= SeparateBlendFunc(rgb, alpha);
1297 		}
1298 	}
1299 
1300 	if (rng.getBool())
1301 	{
1302 		const bool red		= rng.getBool();
1303 		const bool green	= rng.getBool();
1304 		const bool blue		= rng.getBool();
1305 		const bool alpha	= rng.getBool();
1306 
1307 		blendState.colorMask = BVec4(red, blue, green, alpha);
1308 	}
1309 }
1310 
getRandomFormat(de::Random & rng,Context & context)1311 TextureFormat getRandomFormat (de::Random& rng, Context& context)
1312 {
1313 	const bool supportsES32orGL45 = checkES32orGL45Support(context);
1314 
1315 	const deUint32 glFormats[] =
1316 	{
1317 		GL_R8,
1318 		GL_RG8,
1319 		GL_RGB8,
1320 		GL_RGB565,
1321 		GL_RGBA4,
1322 		GL_RGB5_A1,
1323 		GL_RGBA8,
1324 		GL_RGB10_A2,
1325 		GL_RGB10_A2UI,
1326 		GL_R8I,
1327 		GL_R8UI,
1328 		GL_R16I,
1329 		GL_R16UI,
1330 		GL_R32I,
1331 		GL_R32UI,
1332 		GL_RG8I,
1333 		GL_RG8UI,
1334 		GL_RG16I,
1335 		GL_RG16UI,
1336 		GL_RG32I,
1337 		GL_RG32UI,
1338 		GL_RGBA8I,
1339 		GL_RGBA8UI,
1340 		GL_RGBA16I,
1341 		GL_RGBA16UI,
1342 		GL_RGBA32I,
1343 		GL_RGBA32UI,
1344 		GL_RGBA16F,
1345 		GL_R32F,
1346 		GL_RG32F,
1347 		GL_RGBA32F,
1348 		GL_R11F_G11F_B10F
1349 	};
1350 
1351 	if (supportsES32orGL45)
1352 		return glu::mapGLInternalFormat(de::getArrayElement(glFormats, rng.getUint32() % DE_LENGTH_OF_ARRAY(glFormats)));
1353 	else
1354 	{
1355 		DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(glFormats) == 32);
1356 		return glu::mapGLInternalFormat(de::getArrayElement(glFormats, rng.getUint32() % (DE_LENGTH_OF_ARRAY(glFormats) - 5)));
1357 	}
1358 }
1359 
genRandomTest(de::Random & rng,BlendState & preCommon,BlendState & postCommon,vector<DrawBufferInfo> & drawBuffers,int maxDrawBufferCount,Context & context)1360 void genRandomTest (de::Random& rng, BlendState& preCommon, BlendState& postCommon, vector<DrawBufferInfo>& drawBuffers, int maxDrawBufferCount, Context& context)
1361 {
1362 	genRandomBlendState(rng, preCommon);
1363 	genRandomBlendState(rng, postCommon);
1364 
1365 	for (int drawBufferNdx = 0; drawBufferNdx < maxDrawBufferCount; drawBufferNdx++)
1366 	{
1367 		const bool			render		= rng.getFloat() > 0.1f;
1368 		const IVec2			size		(64, 64);
1369 		const TextureFormat	format		(getRandomFormat(rng, context));
1370 		BlendState			blendState;
1371 
1372 		genRandomBlendState(rng, blendState);
1373 
1374 		// 32bit float formats don't support blending in GLES32
1375 		if (format.type == tcu::TextureFormat::FLOAT)
1376 		{
1377 			// If format is 32bit float post common can't enable blending
1378 			if (postCommon.enableBlend && *postCommon.enableBlend)
1379 			{
1380 				// Either don't set enable blend or disable blending
1381 				if (rng.getBool())
1382 					postCommon.enableBlend = tcu::nothing<bool>();
1383 				else
1384 					postCommon.enableBlend = tcu::just(false);
1385 			}
1386 
1387 			// If post common doesn't disable blending, per attachment state or
1388 			// pre common must.
1389 			if (!postCommon.enableBlend)
1390 			{
1391 				// If pre common enables blend per attachment must disable it
1392 				// If per attachment state changes blend state it must disable it
1393 				if ((preCommon.enableBlend && *preCommon.enableBlend)
1394 					|| blendState.enableBlend)
1395 					blendState.enableBlend = tcu::just(false);
1396 			}
1397 		}
1398 
1399 		drawBuffers.push_back(DrawBufferInfo(render, size, blendState, format));
1400 	}
1401 }
1402 
1403 class MaxDrawBuffersIndexedTest : public TestCase
1404 {
1405 public:
1406 					MaxDrawBuffersIndexedTest	(Context& contet, int seed);
1407 
1408 	void			init						(void);
1409 	IterateResult	iterate						(void);
1410 
1411 private:
1412 	const int		m_seed;
1413 };
1414 
MaxDrawBuffersIndexedTest(Context & context,int seed)1415 MaxDrawBuffersIndexedTest::MaxDrawBuffersIndexedTest (Context& context, int seed)
1416 	: TestCase	(context, de::toString(seed).c_str(), de::toString(seed).c_str())
1417 	, m_seed	(deInt32Hash(seed) ^ 1558001307u)
1418 {
1419 }
1420 
init(void)1421 void MaxDrawBuffersIndexedTest::init (void)
1422 {
1423 	const bool supportsES32orGL45 = checkES32orGL45Support(m_context);
1424 
1425 	if (!supportsES32orGL45 && !m_context.getContextInfo().isExtensionSupported("GL_EXT_draw_buffers_indexed"))
1426 		TCU_THROW(NotSupportedError, "Extension GL_EXT_draw_buffers_indexed not supported");
1427 }
1428 
iterate(void)1429 TestCase::IterateResult MaxDrawBuffersIndexedTest::iterate (void)
1430 {
1431 	TestLog&				log						= m_testCtx.getLog();
1432 	tcu::ResultCollector	results					(log);
1433 	de::Random				rng						(m_seed);
1434 	BlendState				preCommonBlendState;
1435 	BlendState				postCommonBlendState;
1436 	vector<DrawBufferInfo>	drawBuffers;
1437 
1438 	genRandomTest(rng, preCommonBlendState, postCommonBlendState, drawBuffers, 4, m_context);
1439 
1440 	runTest(log, results, m_context.getRenderContext(), preCommonBlendState, postCommonBlendState, drawBuffers);
1441 
1442 	results.setTestContextResult(m_testCtx);
1443 
1444 	return STOP;
1445 }
1446 
1447 class ImplMaxDrawBuffersIndexedTest : public TestCase
1448 {
1449 public:
1450 					ImplMaxDrawBuffersIndexedTest	(Context& contet, int seed);
1451 
1452 	void			init							(void);
1453 	IterateResult	iterate							(void);
1454 
1455 private:
1456 	const int		m_seed;
1457 };
1458 
ImplMaxDrawBuffersIndexedTest(Context & context,int seed)1459 ImplMaxDrawBuffersIndexedTest::ImplMaxDrawBuffersIndexedTest (Context& context, int seed)
1460 	: TestCase	(context, de::toString(seed).c_str(), de::toString(seed).c_str())
1461 	, m_seed	(deInt32Hash(seed) ^ 2686315738u)
1462 {
1463 }
1464 
init(void)1465 void ImplMaxDrawBuffersIndexedTest::init (void)
1466 {
1467 	const bool supportsES32orGL45 = checkES32orGL45Support(m_context);
1468 
1469 	if (!supportsES32orGL45 && !m_context.getContextInfo().isExtensionSupported("GL_EXT_draw_buffers_indexed"))
1470 		TCU_THROW(NotSupportedError, "Extension GL_EXT_draw_buffers_indexed not supported");
1471 }
1472 
iterate(void)1473 TestCase::IterateResult ImplMaxDrawBuffersIndexedTest::iterate (void)
1474 {
1475 	TestLog&				log						= m_testCtx.getLog();
1476 	tcu::ResultCollector	results					(log);
1477 	const glw::Functions&	gl						= m_context.getRenderContext().getFunctions();
1478 	de::Random				rng						(m_seed);
1479 	deInt32					maxDrawBuffers			= 0;
1480 	BlendState				preCommonBlendState;
1481 	BlendState				postCommonBlendState;
1482 	vector<DrawBufferInfo>	drawBuffers;
1483 
1484 	gl.getIntegerv(GL_MAX_DRAW_BUFFERS, &maxDrawBuffers);
1485 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv(GL_MAX_DRAW_BUFFERS) failed");
1486 
1487 	TCU_CHECK(maxDrawBuffers > 0);
1488 
1489 	genRandomTest(rng, preCommonBlendState, postCommonBlendState, drawBuffers, maxDrawBuffers, m_context);
1490 
1491 	runTest(log, results, m_context.getRenderContext(), preCommonBlendState, postCommonBlendState, drawBuffers);
1492 
1493 	results.setTestContextResult(m_testCtx);
1494 
1495 	return STOP;
1496 }
1497 
1498 enum PrePost
1499 {
1500 	PRE,
1501 	POST
1502 };
1503 
createDiffTest(Context & context,PrePost prepost,const char * name,const BlendState & commonState,const BlendState & drawBufferState)1504 TestCase* createDiffTest (Context& context, PrePost prepost, const char* name, const BlendState& commonState, const BlendState& drawBufferState)
1505 {
1506 	const BlendState emptyState = BlendState(tcu::nothing<bool>(), tcu::nothing<Either<BlendEq, SeparateBlendEq> >(), tcu::nothing<Either<BlendFunc, SeparateBlendFunc> >(), tcu::nothing<BVec4>());
1507 
1508 	if (prepost == PRE)
1509 	{
1510 		const BlendState		preState	= BlendState((commonState.enableBlend ? commonState.enableBlend : just(true)),
1511 														 commonState.blendEq,
1512 														 (commonState.blendFunc ? commonState.blendFunc : just(Either<BlendFunc, SeparateBlendFunc>(BlendFunc(GL_ONE, GL_ONE)))),
1513 														 tcu::nothing<BVec4>());
1514 		vector<DrawBufferInfo>	drawBuffers;
1515 
1516 		drawBuffers.push_back(DrawBufferInfo(true, IVec2(64, 64), emptyState, TextureFormat(tcu::TextureFormat::RGBA, tcu::TextureFormat::UNORM_INT8)));
1517 		drawBuffers.push_back(DrawBufferInfo(true, IVec2(64, 64), drawBufferState, TextureFormat(tcu::TextureFormat::RGBA, tcu::TextureFormat::UNORM_INT8)));
1518 
1519 		return new DrawBuffersIndexedTest(context, preState, emptyState, drawBuffers, name, name);
1520 	}
1521 	else if (prepost == POST)
1522 	{
1523 		const BlendState		preState	= BlendState(just(true),
1524 														 tcu::nothing<Either<BlendEq, SeparateBlendEq> >(),
1525 														 Maybe<Either<BlendFunc, SeparateBlendFunc> >(BlendFunc(GL_ONE, GL_ONE)),
1526 														 tcu::nothing<BVec4>());
1527 		vector<DrawBufferInfo>	drawBuffers;
1528 
1529 		drawBuffers.push_back(DrawBufferInfo(true, IVec2(64, 64), emptyState, TextureFormat(tcu::TextureFormat::RGBA, tcu::TextureFormat::UNORM_INT8)));
1530 		drawBuffers.push_back(DrawBufferInfo(true, IVec2(64, 64), drawBufferState, TextureFormat(tcu::TextureFormat::RGBA, tcu::TextureFormat::UNORM_INT8)));
1531 
1532 		return new DrawBuffersIndexedTest(context, preState, commonState, drawBuffers, name, name);
1533 	}
1534 	else
1535 	{
1536 		DE_ASSERT(false);
1537 		return DE_NULL;
1538 	}
1539 }
1540 
createAdvancedEqDiffTest(Context & context,PrePost prepost,const char * name,const BlendState & commonState,const BlendState & drawBufferState)1541 TestCase* createAdvancedEqDiffTest (Context& context, PrePost prepost, const char* name, const BlendState& commonState, const BlendState& drawBufferState)
1542 {
1543 	const BlendState emptyState = BlendState(tcu::nothing<bool>(), tcu::nothing<Either<BlendEq, SeparateBlendEq> >(), tcu::nothing<Either<BlendFunc, SeparateBlendFunc> >(), tcu::nothing<BVec4>());
1544 
1545 	if (prepost == PRE)
1546 	{
1547 		const BlendState		preState	= BlendState((commonState.enableBlend ? commonState.enableBlend : just(true)),
1548 														 commonState.blendEq,
1549 														 (commonState.blendFunc ? commonState.blendFunc : just(Either<BlendFunc, SeparateBlendFunc>(BlendFunc(GL_ONE, GL_ONE)))),
1550 														 tcu::nothing<BVec4>());
1551 		vector<DrawBufferInfo>	drawBuffers;
1552 
1553 		drawBuffers.push_back(DrawBufferInfo(true, IVec2(64, 64), drawBufferState, TextureFormat(tcu::TextureFormat::RGBA, tcu::TextureFormat::UNORM_INT8)));
1554 
1555 		return new DrawBuffersIndexedTest(context, preState, emptyState, drawBuffers, name, name);
1556 	}
1557 	else if (prepost == POST)
1558 	{
1559 		const BlendState		preState	= BlendState(just(true),
1560 														 tcu::nothing<Either<BlendEq, SeparateBlendEq> >(),
1561 														 Maybe<Either<BlendFunc, SeparateBlendFunc> >(BlendFunc(GL_ONE, GL_ONE)),
1562 														 tcu::nothing<BVec4>());
1563 		vector<DrawBufferInfo>	drawBuffers;
1564 
1565 		drawBuffers.push_back(DrawBufferInfo(true, IVec2(64, 64), drawBufferState, TextureFormat(tcu::TextureFormat::RGBA, tcu::TextureFormat::UNORM_INT8)));
1566 
1567 		return new DrawBuffersIndexedTest(context, preState, commonState, drawBuffers, name, name);
1568 	}
1569 	else
1570 	{
1571 		DE_ASSERT(false);
1572 		return DE_NULL;
1573 	}
1574 }
1575 
addDrawBufferCommonTests(TestCaseGroup * root,PrePost prepost)1576 void addDrawBufferCommonTests (TestCaseGroup* root, PrePost prepost)
1577 {
1578 	const BlendState		emptyState	= BlendState(Maybe<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1579 
1580 	{
1581 		const BlendState	disableState	= BlendState(just(false), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1582 		const BlendState	enableState		= BlendState(just(true), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1583 
1584 		root->addChild(createDiffTest(root->getContext(), prepost, "common_enable_buffer_enable",	enableState,	enableState));
1585 		root->addChild(createDiffTest(root->getContext(), prepost, "common_disable_buffer_disable",	disableState,	disableState));
1586 		root->addChild(createDiffTest(root->getContext(), prepost, "common_disable_buffer_enable",	disableState,	enableState));
1587 		root->addChild(createDiffTest(root->getContext(), prepost, "common_enable_buffer_disable",	enableState,	disableState));
1588 	}
1589 
1590 	{
1591 		const BlendState	eqStateA			= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(GL_FUNC_ADD), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1592 		const BlendState	eqStateB			= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(GL_FUNC_SUBTRACT), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1593 
1594 		const BlendState	separateEqStateA	= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(SeparateBlendEq(GL_FUNC_ADD, GL_FUNC_SUBTRACT)), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1595 		const BlendState	separateEqStateB	= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(SeparateBlendEq(GL_FUNC_SUBTRACT, GL_FUNC_ADD)), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1596 
1597 		const BlendState	advancedEqStateA	= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(GL_DIFFERENCE), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1598 		const BlendState	advancedEqStateB	= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(GL_SCREEN), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1599 
1600 		root->addChild(createDiffTest(root->getContext(), prepost, "common_blend_eq_buffer_blend_eq", eqStateA, eqStateB));
1601 		root->addChild(createDiffTest(root->getContext(), prepost, "common_blend_eq_buffer_separate_blend_eq", eqStateA, separateEqStateB));
1602 		root->addChild(createAdvancedEqDiffTest(root->getContext(), prepost, "common_blend_eq_buffer_advanced_blend_eq", eqStateA, advancedEqStateB));
1603 
1604 		root->addChild(createDiffTest(root->getContext(), prepost, "common_separate_blend_eq_buffer_blend_eq", separateEqStateA, eqStateB));
1605 		root->addChild(createDiffTest(root->getContext(), prepost, "common_separate_blend_eq_buffer_separate_blend_eq", separateEqStateA, separateEqStateB));
1606 		root->addChild(createAdvancedEqDiffTest(root->getContext(), prepost, "common_separate_blend_eq_buffer_advanced_blend_eq", separateEqStateA, advancedEqStateB));
1607 
1608 		root->addChild(createAdvancedEqDiffTest(root->getContext(), prepost, "common_advanced_blend_eq_buffer_blend_eq", advancedEqStateA, eqStateB));
1609 		root->addChild(createAdvancedEqDiffTest(root->getContext(), prepost, "common_advanced_blend_eq_buffer_separate_blend_eq", advancedEqStateA, separateEqStateB));
1610 		root->addChild(createAdvancedEqDiffTest(root->getContext(), prepost, "common_advanced_blend_eq_buffer_advanced_blend_eq", advancedEqStateA, advancedEqStateB));
1611 	}
1612 
1613 	{
1614 		const BlendState	funcStateA			= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(BlendFunc(GL_SRC_ALPHA, GL_DST_ALPHA)), Maybe<BVec4>());
1615 		const BlendState	funcStateB			= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(BlendFunc(GL_DST_ALPHA, GL_SRC_ALPHA)), Maybe<BVec4>());
1616 		const BlendState	separateFuncStateA	= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(SeparateBlendFunc(BlendFunc(GL_SRC_ALPHA, GL_DST_ALPHA), BlendFunc(GL_ONE_MINUS_SRC_ALPHA, GL_ONE_MINUS_DST_ALPHA))), Maybe<BVec4>());
1617 		const BlendState	separateFuncStateB	= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(SeparateBlendFunc(BlendFunc(GL_DST_ALPHA, GL_SRC_ALPHA), BlendFunc(GL_ONE_MINUS_DST_ALPHA, GL_ONE_MINUS_SRC_ALPHA))), Maybe<BVec4>());
1618 
1619 		root->addChild(createDiffTest(root->getContext(), prepost, "common_blend_func_buffer_blend_func",					funcStateA,			funcStateB));
1620 		root->addChild(createDiffTest(root->getContext(), prepost, "common_blend_func_buffer_separate_blend_func",			funcStateA,			separateFuncStateB));
1621 		root->addChild(createDiffTest(root->getContext(), prepost, "common_separate_blend_func_buffer_blend_func",			separateFuncStateA,	funcStateB));
1622 		root->addChild(createDiffTest(root->getContext(), prepost, "common_separate_blend_func_buffer_separate_blend_func",	separateFuncStateA,	separateFuncStateB));
1623 	}
1624 
1625 	{
1626 		const BlendState	commonColorMaskState	= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>(BVec4(true, false, true, false)));
1627 		const BlendState	bufferColorMaskState	= BlendState(tcu::nothing<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>(BVec4(false, true, false, true)));
1628 
1629 		root->addChild(createDiffTest(root->getContext(), prepost, "common_color_mask_buffer_color_mask", commonColorMaskState, bufferColorMaskState));
1630 	}
1631 }
1632 
addRandomMaxTest(TestCaseGroup * root)1633 void addRandomMaxTest (TestCaseGroup* root)
1634 {
1635 	for (int i = 0; i < 20; i++)
1636 		root->addChild(new MaxDrawBuffersIndexedTest(root->getContext(), i));
1637 }
1638 
addRandomImplMaxTest(TestCaseGroup * root)1639 void addRandomImplMaxTest (TestCaseGroup* root)
1640 {
1641 	for (int i = 0; i < 20; i++)
1642 		root->addChild(new ImplMaxDrawBuffersIndexedTest(root->getContext(), i));
1643 }
1644 
1645 } // anonymous
1646 
createDrawBuffersIndexedTests(Context & context)1647 TestCaseGroup* createDrawBuffersIndexedTests (Context& context)
1648 {
1649 	const BlendState		emptyState		= BlendState(Maybe<bool>(), Maybe<Either<BlendEq, SeparateBlendEq> >(), Maybe<Either<BlendFunc, SeparateBlendFunc> >(), Maybe<BVec4>());
1650 	TestCaseGroup* const	group			= new TestCaseGroup(context, "draw_buffers_indexed", "Test for indexed draw buffers. GL_EXT_draw_buffers_indexed.");
1651 
1652 	TestCaseGroup* const	preGroup		= new TestCaseGroup(context, "overwrite_common", "Set common state and overwrite it with draw buffer blend state.");
1653 	TestCaseGroup* const	postGroup		= new TestCaseGroup(context, "overwrite_indexed", "Set indexed blend state and overwrite it with common state.");
1654 	TestCaseGroup* const	randomGroup		= new TestCaseGroup(context, "random", "Random indexed blend state tests.");
1655 	TestCaseGroup* const	maxGroup		= new TestCaseGroup(context, "max_required_draw_buffers", "Random tests using minimum maximum number of draw buffers.");
1656 	TestCaseGroup* const	maxImplGroup	= new TestCaseGroup(context, "max_implementation_draw_buffers", "Random tests using maximum number of draw buffers reported by implementation.");
1657 
1658 	group->addChild(preGroup);
1659 	group->addChild(postGroup);
1660 	group->addChild(randomGroup);
1661 
1662 	randomGroup->addChild(maxGroup);
1663 	randomGroup->addChild(maxImplGroup);
1664 
1665 	addDrawBufferCommonTests(preGroup, PRE);
1666 	addDrawBufferCommonTests(postGroup, POST);
1667 	addRandomMaxTest(maxGroup);
1668 	addRandomImplMaxTest(maxImplGroup);
1669 
1670 	return group;
1671 }
1672 
1673 } // Functional
1674 } // gles31
1675 } // deqp
1676