• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 Multisampling tests.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es3fMultisampleTests.hpp"
25 #include "gluStrUtil.hpp"
26 #include "gluShaderProgram.hpp"
27 #include "gluPixelTransfer.hpp"
28 #include "tcuSurface.hpp"
29 #include "tcuImageCompare.hpp"
30 #include "tcuRenderTarget.hpp"
31 #include "tcuTestLog.hpp"
32 #include "tcuTextureUtil.hpp"
33 #include "tcuCommandLine.hpp"
34 #include "deStringUtil.hpp"
35 #include "deRandom.hpp"
36 #include "deMath.h"
37 #include "deString.h"
38 
39 #include <string>
40 #include <vector>
41 
42 #include "glw.h"
43 
44 namespace deqp
45 {
46 namespace gles3
47 {
48 namespace Functional
49 {
50 
51 using tcu::Vec2;
52 using tcu::Vec3;
53 using tcu::Vec4;
54 using tcu::IVec2;
55 using tcu::IVec4;
56 using tcu::TestLog;
57 using std::vector;
58 
59 static const GLenum		FBO_COLOR_FORMAT	= GL_RGBA8;
60 static const float		SQRT_HALF			= 0.707107f;
61 
62 namespace
63 {
64 
65 struct QuadCorners
66 {
67 	Vec2 p0;
68 	Vec2 p1;
69 	Vec2 p2;
70 	Vec2 p3;
71 
QuadCornersdeqp::gles3::Functional::__anonc42434cf0111::QuadCorners72 	QuadCorners(const Vec2& p0_, const Vec2& p1_, const Vec2& p2_, const Vec2& p3_) : p0(p0_), p1(p1_), p2(p2_), p3(p3_) {}
73 };
74 
75 } // anonymous
76 
getIterationCount(const tcu::TestContext & ctx,int defaultCount)77 static inline int getIterationCount (const tcu::TestContext& ctx, int defaultCount)
78 {
79 	int cmdLineValue = ctx.getCommandLine().getTestIterationCount();
80 	return cmdLineValue > 0 ? cmdLineValue : defaultCount;
81 }
82 
getGLInteger(GLenum name)83 static inline int getGLInteger (GLenum name)
84 {
85 	int result;
86 	GLU_CHECK_CALL(glGetIntegerv(name, &result));
87 	return result;
88 }
89 
90 template<typename T>
min4(T a,T b,T c,T d)91 static inline T min4 (T a, T b, T c, T d)
92 {
93 	return de::min(de::min(de::min(a, b), c), d);
94 }
95 
96 template<typename T>
max4(T a,T b,T c,T d)97 static inline T max4 (T a, T b, T c, T d)
98 {
99 	return de::max(de::max(de::max(a, b), c), d);
100 }
101 
isInsideQuad(const IVec2 & point,const IVec2 & p0,const IVec2 & p1,const IVec2 & p2,const IVec2 & p3)102 static inline bool isInsideQuad (const IVec2& point, const IVec2& p0, const IVec2& p1, const IVec2& p2, const IVec2& p3)
103 {
104 	int dot0 = (point.x()-p0.x()) * (p1.y()-p0.y()) + (point.y()-p0.y()) * (p0.x()-p1.x());
105 	int dot1 = (point.x()-p1.x()) * (p2.y()-p1.y()) + (point.y()-p1.y()) * (p1.x()-p2.x());
106 	int dot2 = (point.x()-p2.x()) * (p3.y()-p2.y()) + (point.y()-p2.y()) * (p2.x()-p3.x());
107 	int dot3 = (point.x()-p3.x()) * (p0.y()-p3.y()) + (point.y()-p3.y()) * (p3.x()-p0.x());
108 
109 	return (dot0 > 0) == (dot1 > 0) && (dot1 > 0) == (dot2 > 0) && (dot2 > 0) == (dot3 > 0);
110 }
111 
112 /*--------------------------------------------------------------------*//*!
113  * \brief Check if a region in an image is unicolored.
114  *
115  * Checks if the pixels in img inside the convex quadilateral defined by
116  * p0, p1, p2 and p3 are all (approximately) of the same color.
117  *//*--------------------------------------------------------------------*/
isPixelRegionUnicolored(const tcu::Surface & img,const IVec2 & p0,const IVec2 & p1,const IVec2 & p2,const IVec2 & p3)118 static bool isPixelRegionUnicolored (const tcu::Surface& img, const IVec2& p0, const IVec2& p1, const IVec2& p2, const IVec2& p3)
119 {
120 	int			xMin				= de::clamp(min4(p0.x(), p1.x(), p2.x(), p3.x()), 0, img.getWidth()-1);
121 	int			yMin				= de::clamp(min4(p0.y(), p1.y(), p2.y(), p3.y()), 0, img.getHeight()-1);
122 	int			xMax				= de::clamp(max4(p0.x(), p1.x(), p2.x(), p3.x()), 0, img.getWidth()-1);
123 	int			yMax				= de::clamp(max4(p0.y(), p1.y(), p2.y(), p3.y()), 0, img.getHeight()-1);
124 	bool		insideEncountered	= false;	//!< Whether we have already seen at least one pixel inside the region.
125 	tcu::RGBA	insideColor;					//!< Color of the first pixel inside the region.
126 
127 	for (int y = yMin; y <= yMax; y++)
128 	for (int x = xMin; x <= xMax; x++)
129 	{
130 		if (isInsideQuad(IVec2(x, y), p0, p1, p2, p3))
131 		{
132 			tcu::RGBA pixColor = img.getPixel(x, y);
133 
134 			if (insideEncountered)
135 			{
136 				if (!tcu::compareThreshold(pixColor, insideColor, tcu::RGBA(3, 3, 3, 3))) // Pixel color differs from already-detected color inside same region - region not unicolored.
137 					return false;
138 			}
139 			else
140 			{
141 				insideEncountered = true;
142 				insideColor = pixColor;
143 			}
144 		}
145 	}
146 
147 	return true;
148 }
149 
drawUnicolorTestErrors(tcu::Surface & img,const tcu::PixelBufferAccess & errorImg,const IVec2 & p0,const IVec2 & p1,const IVec2 & p2,const IVec2 & p3)150 static bool drawUnicolorTestErrors (tcu::Surface& img, const tcu::PixelBufferAccess& errorImg, const IVec2& p0, const IVec2& p1, const IVec2& p2, const IVec2& p3)
151 {
152 	int			xMin		= de::clamp(min4(p0.x(), p1.x(), p2.x(), p3.x()), 0, img.getWidth()-1);
153 	int			yMin		= de::clamp(min4(p0.y(), p1.y(), p2.y(), p3.y()), 0, img.getHeight()-1);
154 	int			xMax		= de::clamp(max4(p0.x(), p1.x(), p2.x(), p3.x()), 0, img.getWidth()-1);
155 	int			yMax		= de::clamp(max4(p0.y(), p1.y(), p2.y(), p3.y()), 0, img.getHeight()-1);
156 	tcu::RGBA	refColor	= img.getPixel((xMin + xMax) / 2, (yMin + yMax) / 2);
157 
158 	for (int y = yMin; y <= yMax; y++)
159 	for (int x = xMin; x <= xMax; x++)
160 	{
161 		if (isInsideQuad(IVec2(x, y), p0, p1, p2, p3))
162 		{
163 			if (!tcu::compareThreshold(img.getPixel(x, y), refColor, tcu::RGBA(3, 3, 3, 3)))
164 			{
165 				img.setPixel(x, y, tcu::RGBA::red());
166 				errorImg.setPixel(Vec4(1.0f, 0.0f, 0.0f, 1.0f), x, y);
167 			}
168 		}
169 	}
170 
171 	return true;
172 }
173 
174 /*--------------------------------------------------------------------*//*!
175  * \brief Abstract base class handling common stuff for multisample cases.
176  *//*--------------------------------------------------------------------*/
177 class MultisampleCase : public TestCase
178 {
179 public:
180 	struct FboParams
181 	{
182 		bool		useFbo;
183 		int			numSamples; //!< If 0, use implementation-defined maximum.
184 		bool		useDepth;
185 		bool		useStencil;
186 
FboParamsdeqp::gles3::Functional::MultisampleCase::FboParams187 		FboParams (int numSamples_, bool useDepth_, bool useStencil_)
188 			: useFbo			(true)
189 			, numSamples		(numSamples_)
190 			, useDepth			(useDepth_)
191 			, useStencil		(useStencil_)
192 		{
193 		}
194 
FboParamsdeqp::gles3::Functional::MultisampleCase::FboParams195 		FboParams (void)
196 			: useFbo			(false)
197 			, numSamples		(-1)
198 			, useDepth			(false)
199 			, useStencil		(false)
200 		{
201 		}
202 	};
203 
204 						MultisampleCase			(Context& context, const char* name, const char* desc, int desiredViewportSize, const FboParams& fboParams = FboParams());
205 	virtual				~MultisampleCase		(void);
206 
207 	virtual void		init					(void);
208 	virtual void		deinit					(void);
209 
210 protected:
211 	void				renderTriangle			(const Vec3& p0, const Vec3& p1, const Vec3& p2, const Vec4& c0, const Vec4& c1, const Vec4& c2) const;
212 	void				renderTriangle			(const Vec3& p0, const Vec3& p1, const Vec3& p2, const Vec4& color) const;
213 	void				renderTriangle			(const Vec2& p0, const Vec2& p1, const Vec2& p2, const Vec4& c0, const Vec4& c1, const Vec4& c2) const;
214 	void				renderTriangle			(const Vec2& p0, const Vec2& p1, const Vec2& p2, const Vec4& color) const;
215 	void				renderQuad				(const Vec2& p0, const Vec2& p1, const Vec2& p2, const Vec2& p3, const Vec4& c0, const Vec4& c1, const Vec4& c2, const Vec4& c3) const;
216 	void				renderQuad				(const Vec2& p0, const Vec2& p1, const Vec2& p2, const Vec2& p3, const Vec4& color) const;
217 	void				renderLine				(const Vec2& p0, const Vec2& p1, const Vec4& color) const;
218 
219 	void				randomizeViewport		(void);
220 	void				readImage				(tcu::Surface& dst) const;
221 
getRenderTargetSize(void) const222 	IVec2				getRenderTargetSize		(void) const				{ return IVec2(m_renderWidth, m_renderHeight); }
223 
224 	int					m_numSamples;
225 
226 	int					m_viewportSize;
227 
228 private:
229 						MultisampleCase			(const MultisampleCase& other);
230 	MultisampleCase&	operator=				(const MultisampleCase& other);
231 
232 	const int			m_desiredViewportSize;
233 
234 	const FboParams		m_fboParams;
235 	deUint32			m_msColorRbo;
236 	deUint32			m_msDepthStencilRbo;
237 	deUint32			m_resolveColorRbo;
238 	deUint32			m_msFbo;
239 	deUint32			m_resolveFbo;
240 
241 	glu::ShaderProgram*	m_program;
242 	int					m_attrPositionLoc;
243 	int					m_attrColorLoc;
244 
245 	int					m_renderWidth;
246 	int					m_renderHeight;
247 	int					m_viewportX;
248 	int					m_viewportY;
249 	de::Random			m_rnd;
250 };
251 
MultisampleCase(Context & context,const char * name,const char * desc,int desiredViewportSize,const FboParams & fboParams)252 MultisampleCase::MultisampleCase (Context& context, const char* name, const char* desc, int desiredViewportSize, const FboParams& fboParams)
253 	: TestCase				(context, name, desc)
254 	, m_numSamples			(0)
255 	, m_viewportSize		(0)
256 	, m_desiredViewportSize	(desiredViewportSize)
257 	, m_fboParams			(fboParams)
258 	, m_msColorRbo			(0)
259 	, m_msDepthStencilRbo	(0)
260 	, m_resolveColorRbo		(0)
261 	, m_msFbo				(0)
262 	, m_resolveFbo			(0)
263 	, m_program				(DE_NULL)
264 	, m_attrPositionLoc		(-1)
265 	, m_attrColorLoc		(-1)
266 	, m_renderWidth			(fboParams.useFbo ? 2*desiredViewportSize : context.getRenderTarget().getWidth())
267 	, m_renderHeight		(fboParams.useFbo ? 2*desiredViewportSize : context.getRenderTarget().getHeight())
268 	, m_viewportX			(0)
269 	, m_viewportY			(0)
270 	, m_rnd					(deStringHash(name))
271 {
272 	if (m_fboParams.useFbo)
273 		DE_ASSERT(m_fboParams.numSamples >= 0);
274 }
275 
~MultisampleCase(void)276 MultisampleCase::~MultisampleCase (void)
277 {
278 	MultisampleCase::deinit();
279 }
280 
deinit(void)281 void MultisampleCase::deinit (void)
282 {
283 	delete m_program;
284 	m_program = DE_NULL;
285 
286 	GLU_CHECK_CALL(glBindRenderbuffer(GL_RENDERBUFFER, 0));
287 	GLU_CHECK_CALL(glBindFramebuffer(GL_FRAMEBUFFER, 0));
288 
289 	if (m_msColorRbo != 0)
290 	{
291 		GLU_CHECK_CALL(glDeleteRenderbuffers(1, &m_msColorRbo));
292 		m_msColorRbo = 0;
293 	}
294 	if (m_msDepthStencilRbo != 0)
295 	{
296 		GLU_CHECK_CALL(glDeleteRenderbuffers(1, &m_msDepthStencilRbo));
297 		m_msDepthStencilRbo = 0;
298 	}
299 	if (m_resolveColorRbo != 0)
300 	{
301 		GLU_CHECK_CALL(glDeleteRenderbuffers(1, &m_resolveColorRbo));
302 		m_resolveColorRbo = 0;
303 	}
304 
305 	if (m_msFbo != 0)
306 	{
307 		GLU_CHECK_CALL(glDeleteFramebuffers(1, &m_msFbo));
308 		m_msFbo = 0;
309 	}
310 	if (m_resolveFbo != 0)
311 	{
312 		GLU_CHECK_CALL(glDeleteFramebuffers(1, &m_resolveFbo));
313 		m_resolveFbo = 0;
314 	}
315 }
316 
renderTriangle(const Vec3 & p0,const Vec3 & p1,const Vec3 & p2,const Vec4 & c0,const Vec4 & c1,const Vec4 & c2) const317 void MultisampleCase::renderTriangle (const Vec3& p0, const Vec3& p1, const Vec3& p2, const Vec4& c0, const Vec4& c1, const Vec4& c2) const
318 {
319 	float vertexPositions[] =
320 	{
321 		p0.x(), p0.y(), p0.z(), 1.0f,
322 		p1.x(), p1.y(), p1.z(), 1.0f,
323 		p2.x(), p2.y(), p2.z(), 1.0f
324 	};
325 	float vertexColors[] =
326 	{
327 		c0.x(), c0.y(), c0.z(), c0.w(),
328 		c1.x(), c1.y(), c1.z(), c1.w(),
329 		c2.x(), c2.y(), c2.z(), c2.w(),
330 	};
331 
332 	GLU_CHECK_CALL(glEnableVertexAttribArray(m_attrPositionLoc));
333 	GLU_CHECK_CALL(glVertexAttribPointer(m_attrPositionLoc, 4, GL_FLOAT, false, 0, &vertexPositions[0]));
334 
335 	GLU_CHECK_CALL(glEnableVertexAttribArray(m_attrColorLoc));
336 	GLU_CHECK_CALL(glVertexAttribPointer(m_attrColorLoc, 4, GL_FLOAT, false, 0, &vertexColors[0]));
337 
338 	GLU_CHECK_CALL(glUseProgram(m_program->getProgram()));
339 	GLU_CHECK_CALL(glDrawArrays(GL_TRIANGLES, 0, 3));
340 }
341 
renderTriangle(const Vec3 & p0,const Vec3 & p1,const Vec3 & p2,const Vec4 & color) const342 void MultisampleCase::renderTriangle (const Vec3& p0, const Vec3& p1, const Vec3& p2, const Vec4& color) const
343 {
344 	renderTriangle(p0, p1, p2, color, color, color);
345 }
346 
renderTriangle(const Vec2 & p0,const Vec2 & p1,const Vec2 & p2,const Vec4 & c0,const Vec4 & c1,const Vec4 & c2) const347 void MultisampleCase::renderTriangle (const Vec2& p0, const Vec2& p1, const Vec2& p2, const Vec4& c0, const Vec4& c1, const Vec4& c2) const
348 {
349 	renderTriangle(Vec3(p0.x(), p0.y(), 0.0f),
350 				   Vec3(p1.x(), p1.y(), 0.0f),
351 				   Vec3(p2.x(), p2.y(), 0.0f),
352 				   c0, c1, c2);
353 }
354 
renderTriangle(const Vec2 & p0,const Vec2 & p1,const Vec2 & p2,const Vec4 & color) const355 void MultisampleCase::renderTriangle (const Vec2& p0, const Vec2& p1, const Vec2& p2, const Vec4& color) const
356 {
357 	renderTriangle(p0, p1, p2, color, color, color);
358 }
359 
renderQuad(const Vec2 & p0,const Vec2 & p1,const Vec2 & p2,const Vec2 & p3,const Vec4 & c0,const Vec4 & c1,const Vec4 & c2,const Vec4 & c3) const360 void MultisampleCase::renderQuad (const Vec2& p0, const Vec2& p1, const Vec2& p2, const Vec2& p3, const Vec4& c0, const Vec4& c1, const Vec4& c2, const Vec4& c3) const
361 {
362 	renderTriangle(p0, p1, p2, c0, c1, c2);
363 	renderTriangle(p2, p1, p3, c2, c1, c3);
364 }
365 
renderQuad(const Vec2 & p0,const Vec2 & p1,const Vec2 & p2,const Vec2 & p3,const Vec4 & color) const366 void MultisampleCase::renderQuad (const Vec2& p0, const Vec2& p1, const Vec2& p2, const Vec2& p3, const Vec4& color) const
367 {
368 	renderQuad(p0, p1, p2, p3, color, color, color, color);
369 }
370 
renderLine(const Vec2 & p0,const Vec2 & p1,const Vec4 & color) const371 void MultisampleCase::renderLine (const Vec2& p0, const Vec2& p1, const Vec4& color) const
372 {
373 	float vertexPositions[] =
374 	{
375 		p0.x(), p0.y(), 0.0f, 1.0f,
376 		p1.x(), p1.y(), 0.0f, 1.0f
377 	};
378 	float vertexColors[] =
379 	{
380 		color.x(), color.y(), color.z(), color.w(),
381 		color.x(), color.y(), color.z(), color.w()
382 	};
383 
384 	GLU_CHECK_CALL(glEnableVertexAttribArray(m_attrPositionLoc));
385 	GLU_CHECK_CALL(glVertexAttribPointer(m_attrPositionLoc, 4, GL_FLOAT, false, 0, &vertexPositions[0]));
386 
387 	GLU_CHECK_CALL(glEnableVertexAttribArray(m_attrColorLoc));
388 	GLU_CHECK_CALL(glVertexAttribPointer(m_attrColorLoc, 4, GL_FLOAT, false, 0, &vertexColors[0]));
389 
390 	GLU_CHECK_CALL(glUseProgram(m_program->getProgram()));
391 	GLU_CHECK_CALL(glDrawArrays(GL_LINES, 0, 2));
392 }
393 
randomizeViewport(void)394 void MultisampleCase::randomizeViewport (void)
395 {
396 	m_viewportX = m_rnd.getInt(0, m_renderWidth - m_viewportSize);
397 	m_viewportY = m_rnd.getInt(0, m_renderHeight - m_viewportSize);
398 
399 	GLU_CHECK_CALL(glViewport(m_viewportX, m_viewportY, m_viewportSize, m_viewportSize));
400 }
401 
readImage(tcu::Surface & dst) const402 void MultisampleCase::readImage (tcu::Surface& dst) const
403 {
404 	if (m_fboParams.useFbo)
405 	{
406 		GLU_CHECK_CALL(glBindFramebuffer(GL_DRAW_FRAMEBUFFER, m_resolveFbo));
407 		GLU_CHECK_CALL(glBlitFramebuffer(0, 0, m_renderWidth, m_renderHeight, 0, 0, m_renderWidth, m_renderHeight, GL_COLOR_BUFFER_BIT, GL_NEAREST));
408 		GLU_CHECK_CALL(glBindFramebuffer(GL_READ_FRAMEBUFFER, m_resolveFbo));
409 
410 		glu::readPixels(m_context.getRenderContext(), m_viewportX, m_viewportY, dst.getAccess());
411 
412 		GLU_CHECK_CALL(glBindFramebuffer(GL_FRAMEBUFFER, m_msFbo));
413 	}
414 	else
415 		glu::readPixels(m_context.getRenderContext(), m_viewportX, m_viewportY, dst.getAccess());
416 }
417 
init(void)418 void MultisampleCase::init (void)
419 {
420 	static const char* vertShaderSource =
421 		"#version 300 es\n"
422 		"in highp vec4 a_position;\n"
423 		"in mediump vec4 a_color;\n"
424 		"out mediump vec4 v_color;\n"
425 		"void main()\n"
426 		"{\n"
427 		"	gl_Position = a_position;\n"
428 		"	v_color = a_color;\n"
429 		"}\n";
430 
431 	static const char* fragShaderSource =
432 		"#version 300 es\n"
433 		"in mediump vec4 v_color;\n"
434 		"layout(location = 0) out mediump vec4 o_color;\n"
435 		"void main()\n"
436 		"{\n"
437 		"	o_color = v_color;\n"
438 		"}\n";
439 
440 	TestLog& log = m_testCtx.getLog();
441 
442 	if (!m_fboParams.useFbo && m_context.getRenderTarget().getNumSamples() <= 1)
443 		throw tcu::NotSupportedError("No multisample buffers");
444 
445 	if (m_fboParams.useFbo)
446 	{
447 		if (m_fboParams.numSamples > 0)
448 			m_numSamples = m_fboParams.numSamples;
449 		else
450 		{
451 			log << TestLog::Message << "Querying maximum number of samples for " << glu::getTextureFormatName(FBO_COLOR_FORMAT) << " with glGetInternalformativ()" << TestLog::EndMessage;
452 			GLU_CHECK_CALL(glGetInternalformativ(GL_RENDERBUFFER, FBO_COLOR_FORMAT, GL_SAMPLES, 1, &m_numSamples));
453 		}
454 
455 		log << TestLog::Message << "Using FBO of size (" << m_renderWidth << ", " << m_renderHeight << ") with " << m_numSamples << " samples" << TestLog::EndMessage;
456 	}
457 	else
458 	{
459 		// Query and log number of samples per pixel.
460 
461 		m_numSamples = getGLInteger(GL_SAMPLES);
462 		log << TestLog::Message << "GL_SAMPLES = " << m_numSamples << TestLog::EndMessage;
463 	}
464 
465 	// Prepare program.
466 
467 	DE_ASSERT(!m_program);
468 
469 	m_program = new glu::ShaderProgram(m_context.getRenderContext(), glu::makeVtxFragSources(vertShaderSource, fragShaderSource));
470 	if (!m_program->isOk())
471 		throw tcu::TestError("Failed to compile program", DE_NULL, __FILE__, __LINE__);
472 
473 	GLU_CHECK_CALL(m_attrPositionLoc	= glGetAttribLocation(m_program->getProgram(), "a_position"));
474 	GLU_CHECK_CALL(m_attrColorLoc		= glGetAttribLocation(m_program->getProgram(), "a_color"));
475 
476 	if (m_attrPositionLoc < 0 || m_attrColorLoc < 0)
477 	{
478 		delete m_program;
479 		throw tcu::TestError("Invalid attribute locations", DE_NULL, __FILE__, __LINE__);
480 	}
481 
482 	if (m_fboParams.useFbo)
483 	{
484 		DE_STATIC_ASSERT(sizeof(deUint32) == sizeof(GLuint));
485 
486 		// Setup ms color RBO.
487 		GLU_CHECK_CALL(glGenRenderbuffers(1, &m_msColorRbo));
488 		GLU_CHECK_CALL(glBindRenderbuffer(GL_RENDERBUFFER, m_msColorRbo));
489 
490 		// If glRenderbufferStorageMultisample() fails, check if it's because of a too high sample count.
491 		// \note We don't do the check until now because some implementations can't handle the GL_SAMPLES query with glGetInternalformativ(),
492 		//		 and we don't want that to be the cause of test case failure.
493 		try
494 		{
495 			GLU_CHECK_CALL(glRenderbufferStorageMultisample(GL_RENDERBUFFER, m_numSamples, FBO_COLOR_FORMAT, m_renderWidth, m_renderHeight));
496 		}
497 		catch (const glu::Error&)
498 		{
499 			GLint maxSampleCount = -1;
500 			GLU_CHECK_CALL(glGetInternalformativ(GL_RENDERBUFFER, FBO_COLOR_FORMAT, GL_SAMPLES, 1, &maxSampleCount));
501 			if (maxSampleCount < m_numSamples)
502 				throw tcu::NotSupportedError(std::string("") + "Maximum sample count returned by glGetInternalformativ() for " + glu::getTextureFormatName(FBO_COLOR_FORMAT) + " is only " + de::toString(maxSampleCount));
503 			else
504 				throw;
505 		}
506 
507 		if (m_fboParams.useDepth || m_fboParams.useStencil)
508 		{
509 			// Setup ms depth & stencil RBO.
510 			GLU_CHECK_CALL(glGenRenderbuffers(1, &m_msDepthStencilRbo));
511 			GLU_CHECK_CALL(glBindRenderbuffer(GL_RENDERBUFFER, m_msDepthStencilRbo));
512 			GLU_CHECK_CALL(glRenderbufferStorageMultisample(GL_RENDERBUFFER, m_numSamples, GL_DEPTH24_STENCIL8, m_renderWidth, m_renderHeight));
513 		}
514 
515 		// Setup ms FBO.
516 		GLU_CHECK_CALL(glGenFramebuffers(1, &m_msFbo));
517 		GLU_CHECK_CALL(glBindFramebuffer(GL_FRAMEBUFFER, m_msFbo));
518 		GLU_CHECK_CALL(glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, m_msColorRbo));
519 		GLU_CHECK_CALL(glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_RENDERBUFFER, m_msDepthStencilRbo));
520 
521 		// Setup resolve color RBO.
522 		GLU_CHECK_CALL(glGenRenderbuffers(1, &m_resolveColorRbo));
523 		GLU_CHECK_CALL(glBindRenderbuffer(GL_RENDERBUFFER, m_resolveColorRbo));
524 		GLU_CHECK_CALL(glRenderbufferStorage(GL_RENDERBUFFER, FBO_COLOR_FORMAT, m_renderWidth, m_renderHeight));
525 
526 		// Setup resolve FBO.
527 		GLU_CHECK_CALL(glGenFramebuffers(1, &m_resolveFbo));
528 		GLU_CHECK_CALL(glBindFramebuffer(GL_FRAMEBUFFER, m_resolveFbo));
529 		GLU_CHECK_CALL(glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, m_resolveColorRbo));
530 
531 		// Use ms FBO.
532 		GLU_CHECK_CALL(glBindFramebuffer(GL_FRAMEBUFFER, m_msFbo));
533 	}
534 
535 	// Get suitable viewport size.
536 
537 	m_viewportSize = de::min<int>(m_desiredViewportSize, de::min(m_renderWidth, m_renderHeight));
538 	randomizeViewport();
539 }
540 
541 /*--------------------------------------------------------------------*//*!
542  * \brief Base class for cases testing the value of sample count.
543  *
544  * Draws a test pattern (defined by renderPattern() of an inheriting class)
545  * and counts the number of distinct colors in the resulting image. That
546  * number should be at least the value of sample count plus one. This is
547  * repeated with increased values of m_currentIteration until this correct
548  * number of colors is detected or m_currentIteration reaches
549  * m_maxNumIterations.
550  *//*--------------------------------------------------------------------*/
551 class NumSamplesCase : public MultisampleCase
552 {
553 public:
554 						NumSamplesCase			(Context& context, const char* name, const char* description, const FboParams& fboParams = FboParams());
~NumSamplesCase(void)555 						~NumSamplesCase			(void) {}
556 
557 	IterateResult		iterate					(void);
558 
559 protected:
560 	virtual void		renderPattern			(void) const = 0;
561 
562 	int					m_currentIteration;
563 
564 private:
565 	enum { DEFAULT_MAX_NUM_ITERATIONS = 16 };
566 
567 	const int			m_maxNumIterations;
568 	vector<tcu::RGBA>	m_detectedColors;
569 };
570 
NumSamplesCase(Context & context,const char * name,const char * description,const FboParams & fboParams)571 NumSamplesCase::NumSamplesCase (Context& context, const char* name, const char* description, const FboParams& fboParams)
572 	: MultisampleCase		(context, name, description, 256, fboParams)
573 	, m_currentIteration	(0)
574 	, m_maxNumIterations	(getIterationCount(m_testCtx, DEFAULT_MAX_NUM_ITERATIONS))
575 {
576 }
577 
iterate(void)578 NumSamplesCase::IterateResult NumSamplesCase::iterate (void)
579 {
580 	TestLog&		log				= m_testCtx.getLog();
581 	tcu::Surface	renderedImg		(m_viewportSize, m_viewportSize);
582 
583 	randomizeViewport();
584 
585 	GLU_CHECK_CALL(glClearColor(0.0f, 0.0f, 0.0f, 1.0f));
586 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
587 
588 	renderPattern();
589 
590 	// Read and log rendered image.
591 
592 	readImage(renderedImg);
593 
594 	log << TestLog::Image("RenderedImage", "Rendered image", renderedImg, QP_IMAGE_COMPRESSION_MODE_PNG);
595 
596 	// Detect new, previously unseen colors from image.
597 
598 	int requiredNumDistinctColors = m_numSamples + 1;
599 
600 	for (int y = 0; y < renderedImg.getHeight() && (int)m_detectedColors.size() < requiredNumDistinctColors; y++)
601 	for (int x = 0; x < renderedImg.getWidth() && (int)m_detectedColors.size() < requiredNumDistinctColors; x++)
602 	{
603 		tcu::RGBA color = renderedImg.getPixel(x, y);
604 
605 		int i;
606 		for (i = 0; i < (int)m_detectedColors.size(); i++)
607 		{
608 			if (tcu::compareThreshold(color, m_detectedColors[i], tcu::RGBA(3, 3, 3, 3)))
609 				break;
610 		}
611 
612 		if (i == (int)m_detectedColors.size())
613 			m_detectedColors.push_back(color); // Color not previously detected.
614 	}
615 
616 	// Log results.
617 
618 	log << TestLog::Message
619 		<< "Number of distinct colors detected so far: "
620 		<< ((int)m_detectedColors.size() >= requiredNumDistinctColors ? "at least " : "")
621 		<< de::toString(m_detectedColors.size())
622 		<< TestLog::EndMessage;
623 
624 	if ((int)m_detectedColors.size() < requiredNumDistinctColors)
625 	{
626 		// Haven't detected enough different colors yet.
627 
628 		m_currentIteration++;
629 
630 		if (m_currentIteration >= m_maxNumIterations)
631 		{
632 			const IVec2 targetSize			= getRenderTargetSize();
633 			const int	detectedNumSamples	= (int)m_detectedColors.size() - 1; // One color is the background
634 
635 			log << TestLog::Message << "Failure: Number of distinct colors detected is lower than sample count+1" << TestLog::EndMessage;
636 
637 			// For high resolution render targets the lack of samples is not likely detected by a human
638 			// and for GLES 3.0 the application cannot observe the sample count directly. So, it only
639 			// warrants a quality warning.
640 			if ((targetSize.x() >= 2048 || targetSize.y() >= 2048) && (detectedNumSamples >= (m_numSamples/2)))
641 				m_context.getTestContext().setTestResult(QP_TEST_RESULT_QUALITY_WARNING, "Measured sample count below the advertised count");
642 			else
643 				m_context.getTestContext().setTestResult(QP_TEST_RESULT_FAIL, "Failed");
644 			return STOP;
645 		}
646 		else
647 		{
648 			log << TestLog::Message << "The number of distinct colors detected is lower than sample count+1 - trying again with a slightly altered pattern" << TestLog::EndMessage;
649 			return CONTINUE;
650 		}
651 	}
652 	else
653 	{
654 		log << TestLog::Message << "Success: The number of distinct colors detected is at least sample count+1" << TestLog::EndMessage;
655 		m_context.getTestContext().setTestResult(QP_TEST_RESULT_PASS, "Passed");
656 		return STOP;
657 	}
658 }
659 
660 class PolygonNumSamplesCase : public NumSamplesCase
661 {
662 public:
663 			PolygonNumSamplesCase	(Context& context, const char* name, const char* description, int numFboSamples = 0);
~PolygonNumSamplesCase(void)664 			~PolygonNumSamplesCase	(void) {}
665 
666 protected:
667 	void	renderPattern			(void) const;
668 };
669 
PolygonNumSamplesCase(Context & context,const char * name,const char * description,int numFboSamples)670 PolygonNumSamplesCase::PolygonNumSamplesCase (Context& context, const char* name, const char* description, int numFboSamples)
671 	: NumSamplesCase(context, name, description, numFboSamples >= 0 ? FboParams(numFboSamples, false, false) : FboParams())
672 {
673 }
674 
renderPattern(void) const675 void PolygonNumSamplesCase::renderPattern (void) const
676 {
677 	// The test pattern consists of several triangles with edges at different angles.
678 
679 	const int numTriangles = 25;
680 	for (int i = 0; i < numTriangles; i++)
681 	{
682 		float angle0 = 2.0f*DE_PI * (float)i			/ (float)numTriangles + 0.001f*(float)m_currentIteration;
683 		float angle1 = 2.0f*DE_PI * ((float)i + 0.5f)	/ (float)numTriangles + 0.001f*(float)m_currentIteration;
684 
685 		renderTriangle(Vec2(0.0f, 0.0f),
686 					   Vec2(deFloatCos(angle0)*0.95f, deFloatSin(angle0)*0.95f),
687 					   Vec2(deFloatCos(angle1)*0.95f, deFloatSin(angle1)*0.95f),
688 					   Vec4(1.0f));
689 	}
690 }
691 
692 class LineNumSamplesCase : public NumSamplesCase
693 {
694 public:
695 			LineNumSamplesCase		(Context& context, const char* name, const char* description, int numFboSamples = 0);
~LineNumSamplesCase(void)696 			~LineNumSamplesCase		(void) {}
697 
698 protected:
699 	void	renderPattern			(void) const;
700 };
701 
LineNumSamplesCase(Context & context,const char * name,const char * description,int numFboSamples)702 LineNumSamplesCase::LineNumSamplesCase (Context& context, const char* name, const char* description, int numFboSamples)
703 	: NumSamplesCase (context, name, description, numFboSamples >= 0 ? FboParams(numFboSamples, false, false) : FboParams())
704 {
705 }
706 
renderPattern(void) const707 void LineNumSamplesCase::renderPattern (void) const
708 {
709 	// The test pattern consists of several lines at different angles.
710 
711 	// We scale the number of lines based on the viewport size. This is because a gl line's thickness is
712 	// constant in pixel units, i.e. they get relatively thicker as viewport size decreases. Thus we must
713 	// decrease the number of lines in order to decrease the extent of overlap among the lines in the
714 	// center of the pattern.
715 	const int numLines = (int)(100.0f * deFloatSqrt((float)m_viewportSize / 256.0f));
716 
717 	for (int i = 0; i < numLines; i++)
718 	{
719 		float angle = 2.0f*DE_PI * (float)i / (float)numLines + 0.001f*(float)m_currentIteration;
720 		renderLine(Vec2(0.0f, 0.0f), Vec2(deFloatCos(angle)*0.95f, deFloatSin(angle)*0.95f), Vec4(1.0f));
721 	}
722 }
723 
724 /*--------------------------------------------------------------------*//*!
725  * \brief Case testing behaviour of common edges when multisampling.
726  *
727  * Draws a number of test patterns, each with a number of quads, each made
728  * of two triangles, rotated at different angles. The inner edge inside the
729  * quad (i.e. the common edge of the two triangles) still should not be
730  * visible, despite multisampling - i.e. the two triangles forming the quad
731  * should never get any common coverage bits in any pixel.
732  *//*--------------------------------------------------------------------*/
733 class CommonEdgeCase : public MultisampleCase
734 {
735 public:
736 	enum CaseType
737 	{
738 		CASETYPE_SMALL_QUADS = 0,				//!< Draw several small quads per iteration.
739 		CASETYPE_BIGGER_THAN_VIEWPORT_QUAD,		//!< Draw one bigger-than-viewport quad per iteration.
740 		CASETYPE_FIT_VIEWPORT_QUAD,				//!< Draw one exactly viewport-sized, axis aligned quad per iteration.
741 
742 		CASETYPE_LAST
743 	};
744 
745 					CommonEdgeCase			(Context& context, const char* name, const char* description, CaseType caseType, int numFboSamples = 0);
~CommonEdgeCase(void)746 					~CommonEdgeCase			(void) {}
747 
748 	void			init					(void);
749 
750 	IterateResult	iterate					(void);
751 
752 private:
753 	enum
754 	{
755 		DEFAULT_SMALL_QUADS_ITERATIONS					= 16,
756 		DEFAULT_BIGGER_THAN_VIEWPORT_QUAD_ITERATIONS	= 8*8
757 		// \note With CASETYPE_FIT_VIEWPORT_QUAD, we don't do rotations other than multiples of 90 deg -> constant number of iterations.
758 	};
759 
760 	const CaseType	m_caseType;
761 
762 	const int		m_numIterations;
763 	int				m_currentIteration;
764 };
765 
CommonEdgeCase(Context & context,const char * name,const char * description,CaseType caseType,int numFboSamples)766 CommonEdgeCase::CommonEdgeCase (Context& context, const char* name, const char* description, CaseType caseType, int numFboSamples)
767 	: MultisampleCase		(context, name, description, caseType == CASETYPE_SMALL_QUADS ? 128 : 32, numFboSamples >= 0 ? FboParams(numFboSamples, false, false) : FboParams())
768 	, m_caseType			(caseType)
769 	, m_numIterations		(caseType == CASETYPE_SMALL_QUADS					? getIterationCount(m_testCtx, DEFAULT_SMALL_QUADS_ITERATIONS)
770 							: caseType == CASETYPE_BIGGER_THAN_VIEWPORT_QUAD	? getIterationCount(m_testCtx, DEFAULT_BIGGER_THAN_VIEWPORT_QUAD_ITERATIONS)
771 							: 8)
772 	, m_currentIteration	(0)
773 {
774 }
775 
init(void)776 void CommonEdgeCase::init (void)
777 {
778 	MultisampleCase::init();
779 
780 	if (m_caseType == CASETYPE_SMALL_QUADS)
781 	{
782 		// Check for a big enough viewport. With too small viewports the test case can't analyze the resulting image well enough.
783 
784 		const int minViewportSize = 32;
785 
786 		if (m_viewportSize < minViewportSize)
787 			throw tcu::InternalError("Render target width or height too low (is " + de::toString(m_viewportSize) + ", should be at least " + de::toString(minViewportSize) + ")");
788 	}
789 
790 	GLU_CHECK_CALL(glEnable(GL_BLEND));
791 	GLU_CHECK_CALL(glBlendEquation(GL_FUNC_ADD));
792 	GLU_CHECK_CALL(glBlendFunc(GL_ONE, GL_ONE));
793 	m_testCtx.getLog() << TestLog::Message << "Additive blending enabled in order to detect (erroneously) overlapping samples" << TestLog::EndMessage;
794 }
795 
iterate(void)796 CommonEdgeCase::IterateResult CommonEdgeCase::iterate (void)
797 {
798 	TestLog&		log				= m_testCtx.getLog();
799 	tcu::Surface	renderedImg		(m_viewportSize, m_viewportSize);
800 	tcu::Surface	errorImg		(m_viewportSize, m_viewportSize);
801 
802 	randomizeViewport();
803 
804 	GLU_CHECK_CALL(glClearColor(0.0f, 0.0f, 0.0f, 1.0f));
805 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
806 
807 	// Draw test pattern. Test patterns consist of quads formed with two triangles.
808 	// After drawing the pattern, we check that the interior pixels of each quad are
809 	// all the same color - this is meant to verify that there are no artifacts on the inner edge.
810 
811 	vector<QuadCorners> unicoloredRegions;
812 
813 	if (m_caseType == CASETYPE_SMALL_QUADS)
814 	{
815 		// Draw several quads, rotated at different angles.
816 
817 		const float		quadDiagLen = 2.0f / 3.0f * 0.9f; // \note Fit 3 quads in both x and y directions.
818 		float			angleCos;
819 		float			angleSin;
820 
821 		// \note First and second iteration get exact 0 (and 90, 180, 270) and 45 (and 135, 225, 315) angle quads, as they are kind of a special case.
822 
823 		if (m_currentIteration == 0)
824 		{
825 			angleCos = 1.0f;
826 			angleSin = 0.0f;
827 		}
828 		else if (m_currentIteration == 1)
829 		{
830 			angleCos = SQRT_HALF;
831 			angleSin = SQRT_HALF;
832 		}
833 		else
834 		{
835 			float angle = 0.5f * DE_PI * (float)(m_currentIteration-1) / (float)(m_numIterations-1);
836 			angleCos = deFloatCos(angle);
837 			angleSin = deFloatSin(angle);
838 		}
839 
840 		Vec2 corners[4] =
841 		{
842 			0.5f * quadDiagLen * Vec2( angleCos,  angleSin),
843 			0.5f * quadDiagLen * Vec2(-angleSin,  angleCos),
844 			0.5f * quadDiagLen * Vec2(-angleCos, -angleSin),
845 			0.5f * quadDiagLen * Vec2( angleSin, -angleCos)
846 		};
847 
848 		unicoloredRegions.reserve(8);
849 
850 		// Draw 8 quads.
851 		// First four are rotated at angles angle+0, angle+90, angle+180 and angle+270.
852 		// Last four are rotated the same angles as the first four, but the ordering of the last triangle's vertices is reversed.
853 
854 		for (int quadNdx = 0; quadNdx < 8; quadNdx++)
855 		{
856 			Vec2 center = (2.0f-quadDiagLen) * Vec2((float)(quadNdx%3), (float)(quadNdx/3)) / 2.0f - 0.5f*(2.0f-quadDiagLen);
857 
858 			renderTriangle(corners[(0+quadNdx) % 4] + center,
859 						   corners[(1+quadNdx) % 4] + center,
860 						   corners[(2+quadNdx) % 4] + center,
861 						   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
862 
863 			if (quadNdx >= 4)
864 			{
865 				renderTriangle(corners[(3+quadNdx) % 4] + center,
866 							   corners[(2+quadNdx) % 4] + center,
867 							   corners[(0+quadNdx) % 4] + center,
868 							   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
869 			}
870 			else
871 			{
872 				renderTriangle(corners[(0+quadNdx) % 4] + center,
873 							   corners[(2+quadNdx) % 4] + center,
874 							   corners[(3+quadNdx) % 4] + center,
875 							   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
876 			}
877 
878 			// The size of the "interior" of a quad is assumed to be approximately unicolorRegionScale*<actual size of quad>.
879 			// By "interior" we here mean the region of non-boundary pixels of the rendered quad for which we can safely assume
880 			// that it has all coverage bits set to 1, for every pixel.
881 			float unicolorRegionScale = 1.0f - 6.0f*2.0f / (float)m_viewportSize / quadDiagLen;
882 			unicoloredRegions.push_back(QuadCorners((center + corners[0]*unicolorRegionScale),
883 													(center + corners[1]*unicolorRegionScale),
884 													(center + corners[2]*unicolorRegionScale),
885 													(center + corners[3]*unicolorRegionScale)));
886 		}
887 	}
888 	else if (m_caseType == CASETYPE_BIGGER_THAN_VIEWPORT_QUAD)
889 	{
890 		// Draw a bigger-than-viewport quad, rotated at an angle depending on m_currentIteration.
891 
892 		int				quadBaseAngleNdx		= m_currentIteration / 8;
893 		int				quadSubAngleNdx			= m_currentIteration % 8;
894 		float			angleCos;
895 		float			angleSin;
896 
897 		if (quadBaseAngleNdx == 0)
898 		{
899 			angleCos = 1.0f;
900 			angleSin = 0.0f;
901 		}
902 		else if (quadBaseAngleNdx == 1)
903 		{
904 			angleCos = SQRT_HALF;
905 			angleSin = SQRT_HALF;
906 		}
907 		else
908 		{
909 			float angle = 0.5f * DE_PI * (float)(m_currentIteration-1) / (float)(m_numIterations-1);
910 			angleCos = deFloatCos(angle);
911 			angleSin = deFloatSin(angle);
912 		}
913 
914 		float quadDiagLen = 2.5f / de::max(angleCos, angleSin);
915 
916 		Vec2 corners[4] =
917 		{
918 			0.5f * quadDiagLen * Vec2( angleCos,  angleSin),
919 			0.5f * quadDiagLen * Vec2(-angleSin,  angleCos),
920 			0.5f * quadDiagLen * Vec2(-angleCos, -angleSin),
921 			0.5f * quadDiagLen * Vec2( angleSin, -angleCos)
922 		};
923 
924 		renderTriangle(corners[(0+quadSubAngleNdx) % 4],
925 					   corners[(1+quadSubAngleNdx) % 4],
926 					   corners[(2+quadSubAngleNdx) % 4],
927 					   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
928 
929 		if (quadSubAngleNdx >= 4)
930 		{
931 			renderTriangle(corners[(3+quadSubAngleNdx) % 4],
932 						   corners[(2+quadSubAngleNdx) % 4],
933 						   corners[(0+quadSubAngleNdx) % 4],
934 						   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
935 		}
936 		else
937 		{
938 			renderTriangle(corners[(0+quadSubAngleNdx) % 4],
939 						   corners[(2+quadSubAngleNdx) % 4],
940 						   corners[(3+quadSubAngleNdx) % 4],
941 						   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
942 		}
943 
944 		float unicolorRegionScale = 1.0f - 6.0f*2.0f / (float)m_viewportSize / quadDiagLen;
945 		unicoloredRegions.push_back(QuadCorners((corners[0]*unicolorRegionScale),
946 												(corners[1]*unicolorRegionScale),
947 												(corners[2]*unicolorRegionScale),
948 												(corners[3]*unicolorRegionScale)));
949 	}
950 	else if (m_caseType == CASETYPE_FIT_VIEWPORT_QUAD)
951 	{
952 		// Draw an exactly viewport-sized quad, rotated by multiples of 90 degrees angle depending on m_currentIteration.
953 
954 		int quadSubAngleNdx = m_currentIteration % 8;
955 
956 		Vec2 corners[4] =
957 		{
958 			Vec2( 1.0f,  1.0f),
959 			Vec2(-1.0f,  1.0f),
960 			Vec2(-1.0f, -1.0f),
961 			Vec2( 1.0f, -1.0f)
962 		};
963 
964 		renderTriangle(corners[(0+quadSubAngleNdx) % 4],
965 					   corners[(1+quadSubAngleNdx) % 4],
966 					   corners[(2+quadSubAngleNdx) % 4],
967 					   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
968 
969 		if (quadSubAngleNdx >= 4)
970 		{
971 			renderTriangle(corners[(3+quadSubAngleNdx) % 4],
972 						   corners[(2+quadSubAngleNdx) % 4],
973 						   corners[(0+quadSubAngleNdx) % 4],
974 						   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
975 		}
976 		else
977 		{
978 			renderTriangle(corners[(0+quadSubAngleNdx) % 4],
979 						   corners[(2+quadSubAngleNdx) % 4],
980 						   corners[(3+quadSubAngleNdx) % 4],
981 						   Vec4(0.5f, 0.5f, 0.5f, 1.0f));
982 		}
983 
984 		unicoloredRegions.push_back(QuadCorners(corners[0], corners[1], corners[2], corners[3]));
985 	}
986 	else
987 		DE_ASSERT(false);
988 
989 	// Read pixels and check unicolored regions.
990 
991 	readImage(renderedImg);
992 
993 	tcu::clear(errorImg.getAccess(), Vec4(0.0f, 1.0f, 0.0f, 1.0f));
994 
995 	log << TestLog::Image("RenderedImage", "Rendered image", renderedImg, QP_IMAGE_COMPRESSION_MODE_PNG);
996 
997 	bool errorsDetected = false;
998 	for (int i = 0; i < (int)unicoloredRegions.size(); i++)
999 	{
1000 		const QuadCorners&	region					= unicoloredRegions[i];
1001 		IVec2				p0Win					= ((region.p0+1.0f) * 0.5f * (float)(m_viewportSize-1) + 0.5f).asInt();
1002 		IVec2				p1Win					= ((region.p1+1.0f) * 0.5f * (float)(m_viewportSize-1) + 0.5f).asInt();
1003 		IVec2				p2Win					= ((region.p2+1.0f) * 0.5f * (float)(m_viewportSize-1) + 0.5f).asInt();
1004 		IVec2				p3Win					= ((region.p3+1.0f) * 0.5f * (float)(m_viewportSize-1) + 0.5f).asInt();
1005 		bool				errorsInCurrentRegion	= !isPixelRegionUnicolored(renderedImg, p0Win, p1Win, p2Win, p3Win);
1006 
1007 		if (errorsInCurrentRegion)
1008 			drawUnicolorTestErrors(renderedImg, errorImg.getAccess(), p0Win, p1Win, p2Win, p3Win);
1009 
1010 		errorsDetected = errorsDetected || errorsInCurrentRegion;
1011 	}
1012 
1013 	m_currentIteration++;
1014 
1015 	if (errorsDetected)
1016 	{
1017 		log << TestLog::Message << "Failure: Not all quad interiors seem unicolored - common-edge artifacts?" << TestLog::EndMessage;
1018 		log << TestLog::Message << "Erroneous pixels are drawn red in the following image" << TestLog::EndMessage;
1019 		log << TestLog::Image("RenderedImageWithErrors",	"Rendered image with errors marked",	renderedImg,	QP_IMAGE_COMPRESSION_MODE_PNG);
1020 		log << TestLog::Image("ErrorsOnly",					"Image with error pixels only",			errorImg,		QP_IMAGE_COMPRESSION_MODE_PNG);
1021 		m_context.getTestContext().setTestResult(QP_TEST_RESULT_FAIL, "Failed");
1022 		return STOP;
1023 	}
1024 	else if (m_currentIteration < m_numIterations)
1025 	{
1026 		log << TestLog::Message << "Quads seem OK - moving on to next pattern" << TestLog::EndMessage;
1027 		return CONTINUE;
1028 	}
1029 	else
1030 	{
1031 		log << TestLog::Message << "Success: All quad interiors seem unicolored (no common-edge artifacts)" << TestLog::EndMessage;
1032 		m_context.getTestContext().setTestResult(QP_TEST_RESULT_PASS, "Passed");
1033 		return STOP;
1034 	}
1035 }
1036 
1037 /*--------------------------------------------------------------------*//*!
1038  * \brief Test that depth values are per-sample.
1039  *
1040  * Draws intersecting, differently-colored polygons and checks that there
1041  * are at least sample count+1 distinct colors present, due to some of the
1042  * samples at the intersection line belonging to one and some to another
1043  * polygon.
1044  *//*--------------------------------------------------------------------*/
1045 class SampleDepthCase : public NumSamplesCase
1046 {
1047 public:
1048 						SampleDepthCase			(Context& context, const char* name, const char* description, int numFboSamples = 0);
~SampleDepthCase(void)1049 						~SampleDepthCase		(void) {}
1050 
1051 	void				init					(void);
1052 
1053 protected:
1054 	void				renderPattern			(void) const;
1055 };
1056 
SampleDepthCase(Context & context,const char * name,const char * description,int numFboSamples)1057 SampleDepthCase::SampleDepthCase (Context& context, const char* name, const char* description, int numFboSamples)
1058 	: NumSamplesCase (context, name, description, numFboSamples >= 0 ? FboParams(numFboSamples, true, false) : FboParams())
1059 {
1060 }
1061 
init(void)1062 void SampleDepthCase::init (void)
1063 {
1064 	TestLog& log = m_testCtx.getLog();
1065 
1066 	if (m_context.getRenderTarget().getDepthBits() == 0)
1067 		TCU_THROW(NotSupportedError, "Test requires depth buffer");
1068 
1069 	MultisampleCase::init();
1070 
1071 	GLU_CHECK_CALL(glEnable(GL_DEPTH_TEST));
1072 	GLU_CHECK_CALL(glDepthFunc(GL_LESS));
1073 
1074 	log << TestLog::Message << "Depth test enabled, depth func is GL_LESS" << TestLog::EndMessage;
1075 	log << TestLog::Message << "Drawing several bigger-than-viewport black or white polygons intersecting each other" << TestLog::EndMessage;
1076 }
1077 
renderPattern(void) const1078 void SampleDepthCase::renderPattern (void) const
1079 {
1080 	GLU_CHECK_CALL(glClearColor(0.0f, 0.0f, 0.0f, 0.0f));
1081 	GLU_CHECK_CALL(glClearDepthf(1.0f));
1082 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT));
1083 
1084 	{
1085 		const int numPolygons = 50;
1086 
1087 		for (int i = 0; i < numPolygons; i++)
1088 		{
1089 			Vec4	color	= i % 2 == 0 ? Vec4(1.0f, 1.0f, 1.0f, 1.0f) : Vec4(0.0f, 0.0f, 0.0f, 1.0f);
1090 			float	angle	= 2.0f * DE_PI * (float)i / (float)numPolygons + 0.001f*(float)m_currentIteration;
1091 			Vec3	pt0		(3.0f*deFloatCos(angle + 2.0f*DE_PI*0.0f/3.0f), 3.0f*deFloatSin(angle + 2.0f*DE_PI*0.0f/3.0f), 1.0f);
1092 			Vec3	pt1		(3.0f*deFloatCos(angle + 2.0f*DE_PI*1.0f/3.0f), 3.0f*deFloatSin(angle + 2.0f*DE_PI*1.0f/3.0f), 0.0f);
1093 			Vec3	pt2		(3.0f*deFloatCos(angle + 2.0f*DE_PI*2.0f/3.0f), 3.0f*deFloatSin(angle + 2.0f*DE_PI*2.0f/3.0f), 0.0f);
1094 
1095 			renderTriangle(pt0, pt1, pt2, color);
1096 		}
1097 	}
1098 }
1099 
1100 /*--------------------------------------------------------------------*//*!
1101  * \brief Test that stencil buffer values are per-sample.
1102  *
1103  * Draws a unicolored pattern and marks drawn samples in stencil buffer;
1104  * then clears and draws a viewport-size quad with that color and with
1105  * proper stencil test such that the resulting image should be exactly the
1106  * same as after the pattern was first drawn.
1107  *//*--------------------------------------------------------------------*/
1108 class SampleStencilCase : public MultisampleCase
1109 {
1110 public:
1111 						SampleStencilCase		(Context& context, const char* name, const char* description, int numFboSamples = 0);
~SampleStencilCase(void)1112 						~SampleStencilCase		(void) {}
1113 
1114 	void				init					(void);
1115 	IterateResult		iterate					(void);
1116 };
1117 
SampleStencilCase(Context & context,const char * name,const char * description,int numFboSamples)1118 SampleStencilCase::SampleStencilCase (Context& context, const char* name, const char* description, int numFboSamples)
1119 	: MultisampleCase (context, name, description, 256, numFboSamples >= 0 ? FboParams(numFboSamples, false, true) : FboParams())
1120 {
1121 }
1122 
init(void)1123 void SampleStencilCase::init (void)
1124 {
1125 	if (m_context.getRenderTarget().getStencilBits() == 0)
1126 		TCU_THROW(NotSupportedError, "Test requires stencil buffer");
1127 
1128 	MultisampleCase::init();
1129 }
1130 
iterate(void)1131 SampleStencilCase::IterateResult SampleStencilCase::iterate (void)
1132 {
1133 	TestLog&		log					= m_testCtx.getLog();
1134 	tcu::Surface	renderedImgFirst	(m_viewportSize, m_viewportSize);
1135 	tcu::Surface	renderedImgSecond	(m_viewportSize, m_viewportSize);
1136 
1137 	randomizeViewport();
1138 
1139 	GLU_CHECK_CALL(glClearColor(0.0f, 0.0f, 0.0f, 1.0f));
1140 	GLU_CHECK_CALL(glClearStencil(0));
1141 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT | GL_STENCIL_BUFFER_BIT));
1142 	GLU_CHECK_CALL(glEnable(GL_STENCIL_TEST));
1143 	GLU_CHECK_CALL(glStencilFunc(GL_ALWAYS, 1, 1));
1144 	GLU_CHECK_CALL(glStencilOp(GL_KEEP, GL_KEEP, GL_REPLACE));
1145 
1146 	log << TestLog::Message << "Drawing a pattern with glStencilFunc(GL_ALWAYS, 1, 1) and glStencilOp(GL_KEEP, GL_KEEP, GL_REPLACE)" << TestLog::EndMessage;
1147 
1148 	{
1149 		const int numTriangles = 25;
1150 		for (int i = 0; i < numTriangles; i++)
1151 		{
1152 			float angle0 = 2.0f*DE_PI * (float)i			/ (float)numTriangles;
1153 			float angle1 = 2.0f*DE_PI * ((float)i + 0.5f)	/ (float)numTriangles;
1154 
1155 			renderTriangle(Vec2(0.0f, 0.0f),
1156 						   Vec2(deFloatCos(angle0)*0.95f, deFloatSin(angle0)*0.95f),
1157 						   Vec2(deFloatCos(angle1)*0.95f, deFloatSin(angle1)*0.95f),
1158 						   Vec4(1.0f));
1159 		}
1160 	}
1161 
1162 	readImage(renderedImgFirst);
1163 	log << TestLog::Image("RenderedImgFirst", "First image rendered", renderedImgFirst, QP_IMAGE_COMPRESSION_MODE_PNG);
1164 
1165 	log << TestLog::Message << "Clearing color buffer to black" << TestLog::EndMessage;
1166 
1167 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
1168 	GLU_CHECK_CALL(glStencilFunc(GL_EQUAL, 1, 1));
1169 	GLU_CHECK_CALL(glStencilOp(GL_KEEP, GL_KEEP, GL_KEEP));
1170 
1171 	{
1172 		log << TestLog::Message << "Checking that color buffer was actually cleared to black" << TestLog::EndMessage;
1173 
1174 		tcu::Surface clearedImg(m_viewportSize, m_viewportSize);
1175 		readImage(clearedImg);
1176 
1177 		for (int y = 0; y < clearedImg.getHeight(); y++)
1178 		for (int x = 0; x < clearedImg.getWidth(); x++)
1179 		{
1180 			const tcu::RGBA& clr = clearedImg.getPixel(x, y);
1181 			if (clr != tcu::RGBA::black())
1182 			{
1183 				log << TestLog::Message << "Failure: first non-black pixel, color " << clr << ", detected at coordinates (" << x << ", " << y << ")" << TestLog::EndMessage;
1184 				log << TestLog::Image("ClearedImg", "Image after clearing, erroneously non-black", clearedImg);
1185 				m_context.getTestContext().setTestResult(QP_TEST_RESULT_FAIL, "Failed");
1186 				return STOP;
1187 			}
1188 		}
1189 	}
1190 
1191 	log << TestLog::Message << "Drawing a viewport-sized quad with glStencilFunc(GL_EQUAL, 1, 1) and glStencilOp(GL_KEEP, GL_KEEP, GL_KEEP) - should result in same image as the first" << TestLog::EndMessage;
1192 
1193 	renderQuad(Vec2(-1.0f, -1.0f),
1194 			   Vec2( 1.0f, -1.0f),
1195 			   Vec2(-1.0f,  1.0f),
1196 			   Vec2( 1.0f,  1.0f),
1197 			   Vec4(1.0f));
1198 
1199 	readImage(renderedImgSecond);
1200 	log << TestLog::Image("RenderedImgSecond", "Second image rendered", renderedImgSecond, QP_IMAGE_COMPRESSION_MODE_PNG);
1201 
1202 	bool passed = tcu::pixelThresholdCompare(log,
1203 											 "ImageCompare",
1204 											 "Image comparison",
1205 											 renderedImgFirst,
1206 											 renderedImgSecond,
1207 											 tcu::RGBA(0),
1208 											 tcu::COMPARE_LOG_ON_ERROR);
1209 
1210 	if (passed)
1211 		log << TestLog::Message << "Success: The two images rendered are identical" << TestLog::EndMessage;
1212 
1213 	m_context.getTestContext().setTestResult(passed ? QP_TEST_RESULT_PASS	: QP_TEST_RESULT_FAIL,
1214 											 passed ? "Passed"				: "Failed");
1215 
1216 	return STOP;
1217 }
1218 
1219 /*--------------------------------------------------------------------*//*!
1220  * \brief Tests coverage mask generation proportionality property.
1221  *
1222  * Tests that the number of coverage bits in a coverage mask created by
1223  * GL_SAMPLE_ALPHA_TO_COVERAGE or GL_SAMPLE_COVERAGE is, on average,
1224  * proportional to the alpha or coverage value, respectively. Draws
1225  * multiple frames, each time increasing the alpha or coverage value used,
1226  * and checks that the average color is changing appropriately.
1227  *//*--------------------------------------------------------------------*/
1228 class MaskProportionalityCase : public MultisampleCase
1229 {
1230 public:
1231 	enum CaseType
1232 	{
1233 		CASETYPE_ALPHA_TO_COVERAGE = 0,
1234 		CASETYPE_SAMPLE_COVERAGE,
1235 		CASETYPE_SAMPLE_COVERAGE_INVERTED,
1236 
1237 		CASETYPE_LAST
1238 	};
1239 
1240 					MaskProportionalityCase				(Context& context, const char* name, const char* description, CaseType type, int numFboSamples = 0);
~MaskProportionalityCase(void)1241 					~MaskProportionalityCase			(void) {}
1242 
1243 	void			init								(void);
1244 
1245 	IterateResult	iterate								(void);
1246 
1247 private:
1248 	const CaseType	m_type;
1249 
1250 	int				m_numIterations;
1251 	int				m_currentIteration;
1252 
1253 	deInt32			m_previousIterationColorSum;
1254 };
1255 
MaskProportionalityCase(Context & context,const char * name,const char * description,CaseType type,int numFboSamples)1256 MaskProportionalityCase::MaskProportionalityCase (Context& context, const char* name, const char* description, CaseType type, int numFboSamples)
1257 	: MultisampleCase				(context, name, description, 32, numFboSamples >= 0 ? FboParams(numFboSamples, false, false) : FboParams())
1258 	, m_type						(type)
1259 	, m_numIterations				(0)
1260 	, m_currentIteration			(0)
1261 	, m_previousIterationColorSum	(-1)
1262 {
1263 }
1264 
init(void)1265 void MaskProportionalityCase::init (void)
1266 {
1267 	TestLog& log = m_testCtx.getLog();
1268 
1269 	MultisampleCase::init();
1270 
1271 	if (m_type == CASETYPE_ALPHA_TO_COVERAGE)
1272 	{
1273 		GLU_CHECK_CALL(glEnable(GL_SAMPLE_ALPHA_TO_COVERAGE));
1274 		log << TestLog::Message << "GL_SAMPLE_ALPHA_TO_COVERAGE is enabled" << TestLog::EndMessage;
1275 	}
1276 	else
1277 	{
1278 		DE_ASSERT(m_type == CASETYPE_SAMPLE_COVERAGE || m_type == CASETYPE_SAMPLE_COVERAGE_INVERTED);
1279 
1280 		GLU_CHECK_CALL(glEnable(GL_SAMPLE_COVERAGE));
1281 		log << TestLog::Message << "GL_SAMPLE_COVERAGE is enabled" << TestLog::EndMessage;
1282 	}
1283 
1284 	m_numIterations = de::max(2, getIterationCount(m_testCtx, m_numSamples * 5));
1285 
1286 	randomizeViewport(); // \note Using the same viewport for every iteration since coverage mask may depend on window-relative pixel coordinate.
1287 }
1288 
iterate(void)1289 MaskProportionalityCase::IterateResult MaskProportionalityCase::iterate (void)
1290 {
1291 	TestLog&		log				= m_testCtx.getLog();
1292 	tcu::Surface	renderedImg		(m_viewportSize, m_viewportSize);
1293 	deInt32			numPixels		= (deInt32)renderedImg.getWidth()*(deInt32)renderedImg.getHeight();
1294 
1295 	log << TestLog::Message << "Clearing color to black" << TestLog::EndMessage;
1296 	GLU_CHECK_CALL(glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE));
1297 	GLU_CHECK_CALL(glClearColor(0.0f, 0.0f, 0.0f, 1.0f));
1298 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
1299 
1300 	if (m_type == CASETYPE_ALPHA_TO_COVERAGE)
1301 	{
1302 		GLU_CHECK_CALL(glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_FALSE));
1303 		log << TestLog::Message << "Using color mask TRUE, TRUE, TRUE, FALSE" << TestLog::EndMessage;
1304 	}
1305 
1306 	// Draw quad.
1307 
1308 	{
1309 		const Vec2		pt0						(-1.0f, -1.0f);
1310 		const Vec2		pt1						( 1.0f, -1.0f);
1311 		const Vec2		pt2						(-1.0f,  1.0f);
1312 		const Vec2		pt3						( 1.0f,  1.0f);
1313 		Vec4			quadColor				(1.0f, 0.0f, 0.0f, 1.0f);
1314 		float			alphaOrCoverageValue	= (float)m_currentIteration / (float)(m_numIterations-1);
1315 
1316 		if (m_type == CASETYPE_ALPHA_TO_COVERAGE)
1317 		{
1318 			log << TestLog::Message << "Drawing a red quad using alpha value " + de::floatToString(alphaOrCoverageValue, 2) << TestLog::EndMessage;
1319 			quadColor.w() = alphaOrCoverageValue;
1320 		}
1321 		else
1322 		{
1323 			DE_ASSERT(m_type == CASETYPE_SAMPLE_COVERAGE || m_type == CASETYPE_SAMPLE_COVERAGE_INVERTED);
1324 
1325 			bool	isInverted		= m_type == CASETYPE_SAMPLE_COVERAGE_INVERTED;
1326 			float	coverageValue	= isInverted ? 1.0f - alphaOrCoverageValue : alphaOrCoverageValue;
1327 			log << TestLog::Message << "Drawing a red quad using sample coverage value " + de::floatToString(coverageValue, 2) << (isInverted ? " (inverted)" : "") << TestLog::EndMessage;
1328 			GLU_CHECK_CALL(glSampleCoverage(coverageValue, isInverted ? GL_TRUE : GL_FALSE));
1329 		}
1330 
1331 		renderQuad(pt0, pt1, pt2, pt3, quadColor);
1332 	}
1333 
1334 	// Read ang log image.
1335 
1336 	readImage(renderedImg);
1337 
1338 	log << TestLog::Image("RenderedImage", "Rendered image", renderedImg, QP_IMAGE_COMPRESSION_MODE_PNG);
1339 
1340 	// Compute average red component in rendered image.
1341 
1342 	deInt32 sumRed = 0;
1343 
1344 	for (int y = 0; y < renderedImg.getHeight(); y++)
1345 	for (int x = 0; x < renderedImg.getWidth(); x++)
1346 		sumRed += renderedImg.getPixel(x, y).getRed();
1347 
1348 	log << TestLog::Message << "Average red color component: " << de::floatToString((float)sumRed / 255.0f / (float)numPixels, 2) << TestLog::EndMessage;
1349 
1350 	// Check if average color has decreased from previous frame's color.
1351 
1352 	if (sumRed < m_previousIterationColorSum)
1353 	{
1354 		log << TestLog::Message << "Failure: Current average red color component is lower than previous" << TestLog::EndMessage;
1355 		m_context.getTestContext().setTestResult(QP_TEST_RESULT_FAIL, "Failed");
1356 		return STOP;
1357 	}
1358 
1359 	// Check if coverage mask is not all-zeros if alpha or coverage value is 0 (or 1, if inverted).
1360 
1361 	if (m_currentIteration == 0 && sumRed != 0)
1362 	{
1363 		log << TestLog::Message << "Failure: Image should be completely black" << TestLog::EndMessage;
1364 		m_context.getTestContext().setTestResult(QP_TEST_RESULT_FAIL, "Failed");
1365 		return STOP;
1366 	}
1367 
1368 	if (m_currentIteration == m_numIterations-1 && sumRed != 0xff*numPixels)
1369 	{
1370 		log << TestLog::Message << "Failure: Image should be completely red" << TestLog::EndMessage;
1371 
1372 		m_context.getTestContext().setTestResult(QP_TEST_RESULT_FAIL, "Failed");
1373 		return STOP;
1374 	}
1375 
1376 	m_previousIterationColorSum = sumRed;
1377 
1378 	m_currentIteration++;
1379 
1380 	if (m_currentIteration >= m_numIterations)
1381 	{
1382 		log << TestLog::Message
1383 			<< "Success: Number of coverage mask bits set appears to be, on average, proportional to "
1384 			<< (m_type == CASETYPE_ALPHA_TO_COVERAGE ? "alpha" : m_type == CASETYPE_SAMPLE_COVERAGE ? "sample coverage value" : "inverted sample coverage value")
1385 			<< TestLog::EndMessage;
1386 
1387 		m_context.getTestContext().setTestResult(QP_TEST_RESULT_PASS, "Passed");
1388 		return STOP;
1389 	}
1390 	else
1391 		return CONTINUE;
1392 }
1393 
1394 /*--------------------------------------------------------------------*//*!
1395  * \brief Tests coverage mask generation constancy property.
1396  *
1397  * Tests that the coverage mask created by GL_SAMPLE_ALPHA_TO_COVERAGE or
1398  * GL_SAMPLE_COVERAGE is constant at given pixel coordinates, with a given
1399  * alpha component or coverage value, respectively. Draws two quads, with
1400  * the second one fully overlapping the first one such that at any given
1401  * pixel, both quads have the same alpha or coverage value. This way, if
1402  * the constancy property is fulfilled, only the second quad should be
1403  * visible.
1404  *//*--------------------------------------------------------------------*/
1405 class MaskConstancyCase : public MultisampleCase
1406 {
1407 public:
1408 	enum CaseType
1409 	{
1410 		CASETYPE_ALPHA_TO_COVERAGE = 0,		//!< Use only alpha-to-coverage.
1411 		CASETYPE_SAMPLE_COVERAGE,			//!< Use only sample coverage.
1412 		CASETYPE_SAMPLE_COVERAGE_INVERTED,	//!< Use only inverted sample coverage.
1413 		CASETYPE_BOTH,						//!< Use both alpha-to-coverage and sample coverage.
1414 		CASETYPE_BOTH_INVERTED,				//!< Use both alpha-to-coverage and inverted sample coverage.
1415 
1416 		CASETYPE_LAST
1417 	};
1418 
1419 					MaskConstancyCase			(Context& context, const char* name, const char* description, CaseType type, int numFboSamples = 0);
~MaskConstancyCase(void)1420 					~MaskConstancyCase			(void) {}
1421 
1422 	IterateResult	iterate						(void);
1423 
1424 private:
1425 	const bool		m_isAlphaToCoverageCase;
1426 	const bool		m_isSampleCoverageCase;
1427 	const bool		m_isInvertedSampleCoverageCase;
1428 };
1429 
MaskConstancyCase(Context & context,const char * name,const char * description,CaseType type,int numFboSamples)1430 MaskConstancyCase::MaskConstancyCase (Context& context, const char* name, const char* description, CaseType type, int numFboSamples)
1431 	: MultisampleCase					(context, name, description, 256, numFboSamples >= 0 ? FboParams(numFboSamples, false, false) : FboParams())
1432 	, m_isAlphaToCoverageCase			(type == CASETYPE_ALPHA_TO_COVERAGE			|| type == CASETYPE_BOTH						|| type == CASETYPE_BOTH_INVERTED)
1433 	, m_isSampleCoverageCase			(type == CASETYPE_SAMPLE_COVERAGE			|| type == CASETYPE_SAMPLE_COVERAGE_INVERTED	|| type == CASETYPE_BOTH			|| type == CASETYPE_BOTH_INVERTED)
1434 	, m_isInvertedSampleCoverageCase	(type == CASETYPE_SAMPLE_COVERAGE_INVERTED	|| type == CASETYPE_BOTH_INVERTED)
1435 {
1436 }
1437 
iterate(void)1438 MaskConstancyCase::IterateResult MaskConstancyCase::iterate (void)
1439 {
1440 	TestLog&		log				= m_testCtx.getLog();
1441 	tcu::Surface	renderedImg		(m_viewportSize, m_viewportSize);
1442 
1443 	randomizeViewport();
1444 
1445 	log << TestLog::Message << "Clearing color to black" << TestLog::EndMessage;
1446 	GLU_CHECK_CALL(glClearColor(0.0f, 0.0f, 0.0f, 1.0f));
1447 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
1448 
1449 	if (m_isAlphaToCoverageCase)
1450 	{
1451 		GLU_CHECK_CALL(glEnable(GL_SAMPLE_ALPHA_TO_COVERAGE));
1452 		GLU_CHECK_CALL(glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_FALSE));
1453 		log << TestLog::Message << "GL_SAMPLE_ALPHA_TO_COVERAGE is enabled" << TestLog::EndMessage;
1454 		log << TestLog::Message << "Color mask is TRUE, TRUE, TRUE, FALSE" << TestLog::EndMessage;
1455 	}
1456 
1457 	if (m_isSampleCoverageCase)
1458 	{
1459 		GLU_CHECK_CALL(glEnable(GL_SAMPLE_COVERAGE));
1460 		log << TestLog::Message << "GL_SAMPLE_COVERAGE is enabled" << TestLog::EndMessage;
1461 	}
1462 
1463 	log << TestLog::Message
1464 		<< "Drawing several green quads, each fully overlapped by a red quad with the same "
1465 		<< (m_isAlphaToCoverageCase ? "alpha" : "")
1466 		<< (m_isAlphaToCoverageCase && m_isSampleCoverageCase ? " and " : "")
1467 		<< (m_isInvertedSampleCoverageCase ? "inverted " : "")
1468 		<< (m_isSampleCoverageCase ? "sample coverage" : "")
1469 		<< " values"
1470 		<< TestLog::EndMessage;
1471 
1472 	const int numQuadRowsCols = m_numSamples*4;
1473 
1474 	for (int row = 0; row < numQuadRowsCols; row++)
1475 	{
1476 		for (int col = 0; col < numQuadRowsCols; col++)
1477 		{
1478 			float		x0			= (float)(col+0) / (float)numQuadRowsCols * 2.0f - 1.0f;
1479 			float		x1			= (float)(col+1) / (float)numQuadRowsCols * 2.0f - 1.0f;
1480 			float		y0			= (float)(row+0) / (float)numQuadRowsCols * 2.0f - 1.0f;
1481 			float		y1			= (float)(row+1) / (float)numQuadRowsCols * 2.0f - 1.0f;
1482 			const Vec4	baseGreen	(0.0f, 1.0f, 0.0f, 0.0f);
1483 			const Vec4	baseRed		(1.0f, 0.0f, 0.0f, 0.0f);
1484 			Vec4		alpha0		(0.0f, 0.0f, 0.0f, m_isAlphaToCoverageCase ? (float)col / (float)(numQuadRowsCols-1) : 1.0f);
1485 			Vec4		alpha1		(0.0f, 0.0f, 0.0f, m_isAlphaToCoverageCase ? (float)row / (float)(numQuadRowsCols-1) : 1.0f);
1486 
1487 			if (m_isSampleCoverageCase)
1488 			{
1489 				float value = (float)(row*numQuadRowsCols + col) / (float)(numQuadRowsCols*numQuadRowsCols-1);
1490 				GLU_CHECK_CALL(glSampleCoverage(m_isInvertedSampleCoverageCase ? 1.0f - value : value, m_isInvertedSampleCoverageCase ? GL_TRUE : GL_FALSE));
1491 			}
1492 
1493 			renderQuad(Vec2(x0, y0), Vec2(x1, y0), Vec2(x0, y1), Vec2(x1, y1), baseGreen + alpha0,	baseGreen + alpha1,	baseGreen + alpha0,	baseGreen + alpha1);
1494 			renderQuad(Vec2(x0, y0), Vec2(x1, y0), Vec2(x0, y1), Vec2(x1, y1), baseRed + alpha0,	baseRed + alpha1,	baseRed + alpha0,	baseRed + alpha1);
1495 		}
1496 	}
1497 
1498 	readImage(renderedImg);
1499 
1500 	log << TestLog::Image("RenderedImage", "Rendered image", renderedImg, QP_IMAGE_COMPRESSION_MODE_PNG);
1501 
1502 	for (int y = 0; y < renderedImg.getHeight(); y++)
1503 	for (int x = 0; x < renderedImg.getWidth(); x++)
1504 	{
1505 		if (renderedImg.getPixel(x, y).getGreen() > 0)
1506 		{
1507 			log << TestLog::Message << "Failure: Non-zero green color component detected - should have been completely overwritten by red quad" << TestLog::EndMessage;
1508 			m_context.getTestContext().setTestResult(QP_TEST_RESULT_FAIL, "Failed");
1509 			return STOP;
1510 		}
1511 	}
1512 
1513 	log << TestLog::Message
1514 		<< "Success: Coverage mask appears to be constant at a given pixel coordinate with a given "
1515 		<< (m_isAlphaToCoverageCase ? "alpha" : "")
1516 		<< (m_isAlphaToCoverageCase && m_isSampleCoverageCase ? " and " : "")
1517 		<< (m_isSampleCoverageCase ? "coverage value" : "")
1518 		<< TestLog::EndMessage;
1519 
1520 	m_context.getTestContext().setTestResult(QP_TEST_RESULT_PASS, "Passed");
1521 
1522 	return STOP;
1523 }
1524 
1525 /*--------------------------------------------------------------------*//*!
1526  * \brief Tests coverage mask inversion validity.
1527  *
1528  * Tests that the coverage masks obtained by glSampleCoverage(..., GL_TRUE)
1529  * and glSampleCoverage(..., GL_FALSE) are indeed each others' inverses.
1530  * This is done by drawing a pattern, with varying coverage values,
1531  * overlapped by a pattern that has inverted masks and is otherwise
1532  * identical. The resulting image is compared to one obtained by drawing
1533  * the same pattern but with all-ones coverage masks.
1534  *//*--------------------------------------------------------------------*/
1535 class CoverageMaskInvertCase : public MultisampleCase
1536 {
1537 public:
1538 					CoverageMaskInvertCase		(Context& context, const char* name, const char* description, int numFboSamples = 0);
~CoverageMaskInvertCase(void)1539 					~CoverageMaskInvertCase		(void) {}
1540 
1541 	IterateResult	iterate						(void);
1542 
1543 private:
1544 	void			drawPattern					(bool invertSampleCoverage) const;
1545 };
1546 
CoverageMaskInvertCase(Context & context,const char * name,const char * description,int numFboSamples)1547 CoverageMaskInvertCase::CoverageMaskInvertCase (Context& context, const char* name, const char* description, int numFboSamples)
1548 	: MultisampleCase (context, name, description, 256, numFboSamples >= 0 ? FboParams(numFboSamples, false, false) : FboParams())
1549 {
1550 }
1551 
drawPattern(bool invertSampleCoverage) const1552 void CoverageMaskInvertCase::drawPattern (bool invertSampleCoverage) const
1553 {
1554 	const int numTriangles = 25;
1555 	for (int i = 0; i < numTriangles; i++)
1556 	{
1557 		GLU_CHECK_CALL(glSampleCoverage((float)i / (float)(numTriangles-1), invertSampleCoverage ? GL_TRUE : GL_FALSE));
1558 
1559 		float angle0 = 2.0f*DE_PI * (float)i			/ (float)numTriangles;
1560 		float angle1 = 2.0f*DE_PI * ((float)i + 0.5f)	/ (float)numTriangles;
1561 
1562 		renderTriangle(Vec2(0.0f, 0.0f),
1563 					   Vec2(deFloatCos(angle0)*0.95f, deFloatSin(angle0)*0.95f),
1564 					   Vec2(deFloatCos(angle1)*0.95f, deFloatSin(angle1)*0.95f),
1565 					   Vec4(0.4f + (float)i/(float)numTriangles*0.6f,
1566 							0.5f + (float)i/(float)numTriangles*0.3f,
1567 							0.6f - (float)i/(float)numTriangles*0.5f,
1568 							0.7f - (float)i/(float)numTriangles*0.7f));
1569 	}
1570 }
1571 
iterate(void)1572 CoverageMaskInvertCase::IterateResult CoverageMaskInvertCase::iterate (void)
1573 {
1574 	TestLog&		log								= m_testCtx.getLog();
1575 	tcu::Surface	renderedImgNoSampleCoverage		(m_viewportSize, m_viewportSize);
1576 	tcu::Surface	renderedImgSampleCoverage		(m_viewportSize, m_viewportSize);
1577 
1578 	randomizeViewport();
1579 
1580 	GLU_CHECK_CALL(glEnable(GL_BLEND));
1581 	GLU_CHECK_CALL(glBlendEquation(GL_FUNC_ADD));
1582 	GLU_CHECK_CALL(glBlendFunc(GL_ONE, GL_ONE));
1583 	log << TestLog::Message << "Additive blending enabled in order to detect (erroneously) overlapping samples" << TestLog::EndMessage;
1584 
1585 	log << TestLog::Message << "Clearing color to all-zeros" << TestLog::EndMessage;
1586 	GLU_CHECK_CALL(glClearColor(0.0f, 0.0f, 0.0f, 0.0f));
1587 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
1588 	log << TestLog::Message << "Drawing the pattern with GL_SAMPLE_COVERAGE disabled" << TestLog::EndMessage;
1589 	drawPattern(false);
1590 	readImage(renderedImgNoSampleCoverage);
1591 
1592 	log << TestLog::Image("RenderedImageNoSampleCoverage", "Rendered image with GL_SAMPLE_COVERAGE disabled", renderedImgNoSampleCoverage, QP_IMAGE_COMPRESSION_MODE_PNG);
1593 
1594 	log << TestLog::Message << "Clearing color to all-zeros" << TestLog::EndMessage;
1595 	GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
1596 	GLU_CHECK_CALL(glEnable(GL_SAMPLE_COVERAGE));
1597 	log << TestLog::Message << "Drawing the pattern with GL_SAMPLE_COVERAGE enabled, using non-inverted masks" << TestLog::EndMessage;
1598 	drawPattern(false);
1599 	log << TestLog::Message << "Drawing the pattern with GL_SAMPLE_COVERAGE enabled, using same sample coverage values but inverted masks" << TestLog::EndMessage;
1600 	drawPattern(true);
1601 	readImage(renderedImgSampleCoverage);
1602 
1603 	log << TestLog::Image("RenderedImageSampleCoverage", "Rendered image with GL_SAMPLE_COVERAGE enabled", renderedImgSampleCoverage, QP_IMAGE_COMPRESSION_MODE_PNG);
1604 
1605 	bool passed = tcu::pixelThresholdCompare(log,
1606 											 "CoverageVsNoCoverage",
1607 											 "Comparison of same pattern with GL_SAMPLE_COVERAGE disabled and enabled",
1608 											 renderedImgNoSampleCoverage,
1609 											 renderedImgSampleCoverage,
1610 											 tcu::RGBA(0),
1611 											 tcu::COMPARE_LOG_ON_ERROR);
1612 
1613 	if (passed)
1614 		log << TestLog::Message << "Success: The two images rendered are identical" << TestLog::EndMessage;
1615 
1616 	m_context.getTestContext().setTestResult(passed ? QP_TEST_RESULT_PASS	: QP_TEST_RESULT_FAIL,
1617 											 passed ? "Passed"				: "Failed");
1618 
1619 	return STOP;
1620 }
1621 
MultisampleTests(Context & context)1622 MultisampleTests::MultisampleTests (Context& context)
1623 	: TestCaseGroup(context, "multisample", "Multisampling tests")
1624 {
1625 }
1626 
~MultisampleTests(void)1627 MultisampleTests::~MultisampleTests (void)
1628 {
1629 }
1630 
init(void)1631 void MultisampleTests::init (void)
1632 {
1633 	enum CaseType
1634 	{
1635 		CASETYPE_DEFAULT_FRAMEBUFFER = 0,
1636 		CASETYPE_FBO_4_SAMPLES,
1637 		CASETYPE_FBO_8_SAMPLES,
1638 		CASETYPE_FBO_MAX_SAMPLES,
1639 
1640 		CASETYPE_LAST
1641 	};
1642 
1643 	for (int caseTypeI = 0; caseTypeI < (int)CASETYPE_LAST; caseTypeI++)
1644 	{
1645 		CaseType		caseType		= (CaseType)caseTypeI;
1646 		int				numFboSamples	= caseType == CASETYPE_DEFAULT_FRAMEBUFFER	? -1
1647 										: caseType == CASETYPE_FBO_4_SAMPLES		? 4
1648 										: caseType == CASETYPE_FBO_8_SAMPLES		? 8
1649 										: caseType == CASETYPE_FBO_MAX_SAMPLES		? 0
1650 										: -2;
1651 
1652 		TestCaseGroup*	group			= new TestCaseGroup(m_context,
1653 															caseType == CASETYPE_DEFAULT_FRAMEBUFFER	? "default_framebuffer" :
1654 															caseType == CASETYPE_FBO_4_SAMPLES			? "fbo_4_samples" :
1655 															caseType == CASETYPE_FBO_8_SAMPLES			? "fbo_8_samples" :
1656 															caseType == CASETYPE_FBO_MAX_SAMPLES		? "fbo_max_samples" :
1657 															DE_NULL,
1658 															caseType == CASETYPE_DEFAULT_FRAMEBUFFER	? "Render into default framebuffer" :
1659 															caseType == CASETYPE_FBO_4_SAMPLES			? "Render into a framebuffer object with 4 samples" :
1660 															caseType == CASETYPE_FBO_8_SAMPLES			? "Render into a framebuffer object with 8 samples" :
1661 															caseType == CASETYPE_FBO_MAX_SAMPLES		? "Render into a framebuffer object with the maximum number of samples" :
1662 															DE_NULL);
1663 		DE_ASSERT(group->getName() != DE_NULL);
1664 		DE_ASSERT(group->getDescription() != DE_NULL);
1665 		DE_ASSERT(numFboSamples >= -1);
1666 		addChild(group);
1667 
1668 		group->addChild(new PolygonNumSamplesCase		(m_context, "num_samples_polygon",			"Test sanity of the sample count, with polygons",										numFboSamples));
1669 		group->addChild(new LineNumSamplesCase			(m_context, "num_samples_line",				"Test sanity of the sample count, with lines",											numFboSamples));
1670 		group->addChild(new CommonEdgeCase				(m_context, "common_edge_small_quads",		"Test polygons' common edges with small quads",											CommonEdgeCase::CASETYPE_SMALL_QUADS,				numFboSamples));
1671 		group->addChild(new CommonEdgeCase				(m_context, "common_edge_big_quad",			"Test polygons' common edges with bigger-than-viewport quads",							CommonEdgeCase::CASETYPE_BIGGER_THAN_VIEWPORT_QUAD,	numFboSamples));
1672 		group->addChild(new CommonEdgeCase				(m_context, "common_edge_viewport_quad",	"Test polygons' common edges with exactly viewport-sized quads",						CommonEdgeCase::CASETYPE_FIT_VIEWPORT_QUAD,			numFboSamples));
1673 		group->addChild(new SampleDepthCase				(m_context, "depth",						"Test that depth values are per-sample",												numFboSamples));
1674 		group->addChild(new SampleStencilCase			(m_context, "stencil",						"Test that stencil values are per-sample",												numFboSamples));
1675 		group->addChild(new CoverageMaskInvertCase		(m_context, "sample_coverage_invert",		"Test that non-inverted and inverted sample coverage masks are each other's negations",	numFboSamples));
1676 
1677 		group->addChild(new MaskProportionalityCase		(m_context, "proportionality_alpha_to_coverage",
1678 																	"Test the proportionality property of GL_SAMPLE_ALPHA_TO_COVERAGE",
1679 																	MaskProportionalityCase::CASETYPE_ALPHA_TO_COVERAGE, numFboSamples));
1680 		group->addChild(new MaskProportionalityCase		(m_context, "proportionality_sample_coverage",
1681 																	"Test the proportionality property of GL_SAMPLE_COVERAGE",
1682 																	MaskProportionalityCase::CASETYPE_SAMPLE_COVERAGE, numFboSamples));
1683 		group->addChild(new MaskProportionalityCase		(m_context, "proportionality_sample_coverage_inverted",
1684 																	"Test the proportionality property of inverted-mask GL_SAMPLE_COVERAGE",
1685 																	MaskProportionalityCase::CASETYPE_SAMPLE_COVERAGE_INVERTED, numFboSamples));
1686 
1687 		group->addChild(new MaskConstancyCase			(m_context, "constancy_alpha_to_coverage",
1688 																	"Test that coverage mask is constant at given coordinates with a given alpha or coverage value, using GL_SAMPLE_ALPHA_TO_COVERAGE",
1689 																	MaskConstancyCase::CASETYPE_ALPHA_TO_COVERAGE, numFboSamples));
1690 		group->addChild(new MaskConstancyCase			(m_context, "constancy_sample_coverage",
1691 																	"Test that coverage mask is constant at given coordinates with a given alpha or coverage value, using GL_SAMPLE_COVERAGE",
1692 																	MaskConstancyCase::CASETYPE_SAMPLE_COVERAGE, numFboSamples));
1693 		group->addChild(new MaskConstancyCase			(m_context, "constancy_sample_coverage_inverted",
1694 																	"Test that coverage mask is constant at given coordinates with a given alpha or coverage value, using inverted-mask GL_SAMPLE_COVERAGE",
1695 																	MaskConstancyCase::CASETYPE_SAMPLE_COVERAGE_INVERTED, numFboSamples));
1696 		group->addChild(new MaskConstancyCase			(m_context, "constancy_both",
1697 																	"Test that coverage mask is constant at given coordinates with a given alpha or coverage value, using GL_SAMPLE_ALPHA_TO_COVERAGE and GL_SAMPLE_COVERAGE",
1698 																	MaskConstancyCase::CASETYPE_BOTH, numFboSamples));
1699 		group->addChild(new MaskConstancyCase			(m_context, "constancy_both_inverted",
1700 																	"Test that coverage mask is constant at given coordinates with a given alpha or coverage value, using GL_SAMPLE_ALPHA_TO_COVERAGE and inverted-mask GL_SAMPLE_COVERAGE",
1701 																	MaskConstancyCase::CASETYPE_BOTH_INVERTED, numFboSamples));
1702 	}
1703 }
1704 
1705 } // Functional
1706 } // gles3
1707 } // deqp
1708