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 GLSL ES 1.0 gl_FragData[] tests.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es3fShaderFragDataTests.hpp"
25 
26 #include "glsShaderLibrary.hpp"
27 
28 #include "gluRenderContext.hpp"
29 #include "gluShaderProgram.hpp"
30 #include "gluDrawUtil.hpp"
31 #include "gluPixelTransfer.hpp"
32 #include "gluObjectWrapper.hpp"
33 
34 #include "tcuRenderTarget.hpp"
35 #include "tcuStringTemplate.hpp"
36 #include "tcuTestLog.hpp"
37 #include "tcuSurface.hpp"
38 
39 #include "glwFunctions.hpp"
40 #include "glwEnums.hpp"
41 
42 namespace deqp
43 {
44 namespace gles3
45 {
46 namespace Functional
47 {
48 
49 using std::string;
50 using tcu::TestLog;
51 
52 namespace
53 {
54 
55 enum IndexExprType
56 {
57 	INDEX_EXPR_STATIC	= 0,
58 	INDEX_EXPR_UNIFORM,
59 	INDEX_EXPR_DYNAMIC,
60 
61 	INDEX_EXPR_TYPE_LAST
62 };
63 
compareSingleColor(tcu::TestLog & log,const tcu::Surface & surface,tcu::RGBA expectedColor,tcu::RGBA threshold)64 static bool compareSingleColor (tcu::TestLog& log, const tcu::Surface& surface, tcu::RGBA expectedColor, tcu::RGBA threshold)
65 {
66 	const int	maxPrints			= 10;
67 	int			numFailedPixels		= 0;
68 
69 	log << TestLog::Message << "Expecting " << expectedColor << " with threshold " << threshold << TestLog::EndMessage;
70 
71 	for (int y = 0; y < surface.getHeight(); y++)
72 	{
73 		for (int x = 0; x < surface.getWidth(); x++)
74 		{
75 			const tcu::RGBA		resultColor		= surface.getPixel(x, y);
76 			const bool			isOk			= compareThreshold(resultColor, expectedColor, threshold);
77 
78 			if (!isOk)
79 			{
80 				if (numFailedPixels < maxPrints)
81 					log << TestLog::Message << "ERROR: Got " << resultColor << " at (" << x << ", " << y << ")!" << TestLog::EndMessage;
82 				else if (numFailedPixels == maxPrints)
83 					log << TestLog::Message << "..." << TestLog::EndMessage;
84 
85 				numFailedPixels += 1;
86 			}
87 		}
88 	}
89 
90 	if (numFailedPixels > 0)
91 	{
92 		log << TestLog::Message << "Found " << numFailedPixels << " invalid pixels, comparison FAILED!" << TestLog::EndMessage;
93 		log << TestLog::Image("ResultImage", "Result Image", surface);
94 		return false;
95 	}
96 	else
97 	{
98 		log << TestLog::Message << "Image comparison passed." << TestLog::EndMessage;
99 		return true;
100 	}
101 }
102 
103 class FragDataIndexingCase : public TestCase
104 {
105 public:
FragDataIndexingCase(Context & context,const char * name,const char * description,IndexExprType indexExprType)106 	FragDataIndexingCase (Context& context, const char* name, const char* description, IndexExprType indexExprType)
107 		: TestCase			(context, name, description)
108 		, m_indexExprType	(indexExprType)
109 	{
110 	}
111 
genSources(const IndexExprType indexExprType)112 	static glu::ProgramSources genSources (const IndexExprType indexExprType)
113 	{
114 		const char* const	fragIndexExpr	= indexExprType == INDEX_EXPR_STATIC	? "0"				:
115 											  indexExprType == INDEX_EXPR_UNIFORM	? "u_index"			:
116 											  indexExprType == INDEX_EXPR_DYNAMIC	? "int(v_index)"	: DE_NULL;
117 		glu::ProgramSources	sources;
118 
119 		DE_ASSERT(fragIndexExpr);
120 
121 		sources << glu::VertexSource(
122 			"attribute highp vec4 a_position;\n"
123 			"attribute highp float a_index;\n"
124 			"attribute highp vec4 a_color;\n"
125 			"varying mediump float v_index;\n"
126 			"varying mediump vec4 v_color;\n"
127 			"void main (void)\n"
128 			"{\n"
129 			"	gl_Position = a_position;\n"
130 			"	v_color = a_color;\n"
131 			"	v_index = a_index;\n"
132 			"}\n");
133 
134 		sources << glu::FragmentSource(string(
135 			"varying mediump vec4 v_color;\n"
136 			"varying mediump float v_index;\n"
137 			"uniform mediump int u_index;\n"
138 			"void main (void)\n"
139 			"{\n"
140 			"	gl_FragData[") + fragIndexExpr + "] = v_color;\n"
141 			"}\n");
142 
143 		return sources;
144 	}
145 
iterate(void)146 	IterateResult iterate (void)
147 	{
148 		const glu::RenderContext&		renderCtx		= m_context.getRenderContext();
149 		const glw::Functions&			gl				= renderCtx.getFunctions();
150 		const glu::ShaderProgram		program			(renderCtx, genSources(m_indexExprType));
151 		const int						viewportW		= de::min(renderCtx.getRenderTarget().getWidth(), 128);
152 		const int						viewportH		= de::min(renderCtx.getRenderTarget().getHeight(), 128);
153 
154 		const float positions[] =
155 		{
156 			-1.0f, -1.0f,
157 			+1.0f, -1.0f,
158 			-1.0f, +1.0f,
159 			+1.0f, +1.0f
160 		};
161 		const float colors[] =
162 		{
163 			0.0f, 1.0f, 0.0f, 1.0f,
164 			0.0f, 1.0f, 0.0f, 1.0f,
165 			0.0f, 1.0f, 0.0f, 1.0f,
166 			0.0f, 1.0f, 0.0f, 1.0f
167 		};
168 		const float		indexValues[]	= { 0.0f, 0.0f, 0.0f, 0.0f };
169 		const deUint8	indices[]		= { 0, 1, 2, 2, 1, 3 };
170 
171 		const glu::VertexArrayBinding vertexArrays[] =
172 		{
173 			glu::va::Float("a_position",	2, 4, 0, &positions[0]),
174 			glu::va::Float("a_color",		4, 4, 0, &colors[0]),
175 			glu::va::Float("a_index",		1, 4, 0, &indexValues[0])
176 		};
177 
178 		m_testCtx.getLog() << program;
179 
180 		if (!program.isOk())
181 		{
182 			if (m_indexExprType == INDEX_EXPR_STATIC)
183 				TCU_FAIL("Compile failed");
184 			else
185 				throw tcu::NotSupportedError("Dynamic indexing of gl_FragData[] not supported");
186 		}
187 
188 		gl.clearColor	(1.0f, 0.0f, 0.0f, 1.0f);
189 		gl.clear		(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
190 
191 		gl.viewport		(0, 0, viewportW, viewportH);
192 		gl.useProgram	(program.getProgram());
193 		gl.uniform1i	(gl.getUniformLocation(program.getProgram(), "u_index"), 0);
194 
195 		glu::draw(renderCtx, program.getProgram(), DE_LENGTH_OF_ARRAY(vertexArrays), &vertexArrays[0],
196 				  glu::pr::Triangles(DE_LENGTH_OF_ARRAY(indices), &indices[0]));
197 		GLU_EXPECT_NO_ERROR(gl.getError(), "Rendering failed");
198 
199 		{
200 			tcu::Surface		result		(viewportW, viewportH);
201 			const tcu::RGBA		threshold	= renderCtx.getRenderTarget().getPixelFormat().getColorThreshold() + tcu::RGBA(1,1,1,1);
202 			bool				isOk;
203 
204 			glu::readPixels(renderCtx, 0, 0, result.getAccess());
205 			GLU_EXPECT_NO_ERROR(gl.getError(), "Reading pixels failed");
206 
207 			isOk = compareSingleColor(m_testCtx.getLog(), result, tcu::RGBA::green(), threshold);
208 
209 			m_testCtx.setTestResult(isOk ? QP_TEST_RESULT_PASS	: QP_TEST_RESULT_FAIL,
210 									isOk ? "Pass"				: "Image comparison failed");
211 		}
212 
213 		return STOP;
214 	}
215 
216 private:
217 	const IndexExprType m_indexExprType;
218 };
219 
220 class FragDataDrawBuffersCase : public TestCase
221 {
222 public:
FragDataDrawBuffersCase(Context & context)223 	FragDataDrawBuffersCase (Context& context)
224 		: TestCase(context, "draw_buffers", "gl_FragData[] and glDrawBuffers() interaction")
225 	{
226 	}
227 
iterate(void)228 	IterateResult iterate (void)
229 	{
230 		const glu::RenderContext&		renderCtx		= m_context.getRenderContext();
231 		const glu::ShaderProgram		program			(renderCtx, glu::ProgramSources()
232 															<< glu::VertexSource(
233 																"attribute highp vec4 a_position;\n"
234 																"attribute highp vec4 a_color;\n"
235 																"varying mediump vec4 v_color;\n"
236 																"void main (void)\n"
237 																"{\n"
238 																"	gl_Position = a_position;\n"
239 																"	v_color = a_color;\n"
240 																"}\n")
241 															<< glu::FragmentSource(
242 																"varying mediump vec4 v_color;\n"
243 																"uniform mediump int u_index;\n"
244 																"void main (void)\n"
245 																"{\n"
246 																"	gl_FragData[u_index] = v_color;\n"
247 																"}\n"));
248 		const glw::Functions&			gl				= renderCtx.getFunctions();
249 		const int						width			= 128;
250 		const int						height			= 128;
251 		const int						indexLoc		= program.isOk() ? gl.getUniformLocation(program.getProgram(), "u_index") : -1;
252 		const glu::Framebuffer			fbo				(renderCtx);
253 		const glu::Renderbuffer			colorBuf0		(renderCtx);
254 		const glu::Renderbuffer			colorBuf1		(renderCtx);
255 
256 		const float positions[] =
257 		{
258 			-1.0f, -1.0f,
259 			+1.0f, -1.0f,
260 			-1.0f, +1.0f,
261 			+1.0f, +1.0f
262 		};
263 		const float colors[] =
264 		{
265 			0.0f, 1.0f, 0.0f, 1.0f,
266 			0.0f, 1.0f, 0.0f, 1.0f,
267 			0.0f, 1.0f, 0.0f, 1.0f,
268 			0.0f, 1.0f, 0.0f, 1.0f
269 		};
270 		const deUint8	indices[]		= { 0, 1, 2, 2, 1, 3 };
271 
272 		const glu::VertexArrayBinding vertexArrays[] =
273 		{
274 			glu::va::Float("a_position",	2, 4, 0, &positions[0]),
275 			glu::va::Float("a_color",		4, 4, 0, &colors[0])
276 		};
277 
278 		m_testCtx.getLog() << program;
279 
280 		if (!program.isOk())
281 			throw tcu::NotSupportedError("Dynamic indexing of gl_FragData[] not supported");
282 
283 		gl.bindFramebuffer(GL_FRAMEBUFFER, *fbo);
284 		for (int ndx = 0; ndx < 2; ndx++)
285 		{
286 			const deUint32	rbo	= ndx == 0 ? *colorBuf0 : *colorBuf1;
287 
288 			gl.bindRenderbuffer(GL_RENDERBUFFER, rbo);
289 			gl.renderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, width, height);
290 			gl.framebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0+ndx, GL_RENDERBUFFER, rbo);
291 		}
292 		TCU_CHECK(gl.checkFramebufferStatus(GL_FRAMEBUFFER) == GL_FRAMEBUFFER_COMPLETE);
293 
294 		{
295 			const deUint32 drawBuffers[] = { GL_COLOR_ATTACHMENT0, GL_COLOR_ATTACHMENT1 };
296 			gl.drawBuffers(DE_LENGTH_OF_ARRAY(drawBuffers), &drawBuffers[0]);
297 		}
298 
299 		gl.clearBufferfv(GL_COLOR, 0, tcu::RGBA::red().toVec().getPtr());
300 		gl.clearBufferfv(GL_COLOR, 1, tcu::RGBA::red().toVec().getPtr());
301 
302 		gl.viewport		(0, 0, width, height);
303 		gl.useProgram	(program.getProgram());
304 
305 		GLU_EXPECT_NO_ERROR(gl.getError(), "Setup failed");
306 
307 		m_testCtx.getLog() << TestLog::Message << "Drawing to attachments 0 and 1, expecting only attachment 0 to change." << TestLog::EndMessage;
308 
309 		for (int ndx = 0; ndx < 2; ndx++)
310 		{
311 			gl.uniform1i(indexLoc, ndx);
312 			glu::draw(renderCtx, program.getProgram(), DE_LENGTH_OF_ARRAY(vertexArrays), &vertexArrays[0],
313 					  glu::pr::Triangles(DE_LENGTH_OF_ARRAY(indices), &indices[0]));
314 		}
315 		GLU_EXPECT_NO_ERROR(gl.getError(), "Rendering failed");
316 
317 		{
318 			tcu::Surface		result		(width, height);
319 			const tcu::RGBA		threshold	= renderCtx.getRenderTarget().getPixelFormat().getColorThreshold() + tcu::RGBA(1,1,1,1);
320 			bool				allOk		= true;
321 
322 			for (int ndx = 0; ndx < 2; ndx++)
323 			{
324 				m_testCtx.getLog() << TestLog::Message << "Verifying attachment " << ndx << "..." << TestLog::EndMessage;
325 
326 				gl.readBuffer(GL_COLOR_ATTACHMENT0+ndx);
327 				glu::readPixels(renderCtx, 0, 0, result.getAccess());
328 				GLU_EXPECT_NO_ERROR(gl.getError(), "Reading pixels failed");
329 
330 				if (!compareSingleColor(m_testCtx.getLog(), result, ndx == 0 ? tcu::RGBA::green() : tcu::RGBA::red(), threshold))
331 					allOk = false;
332 			}
333 
334 			m_testCtx.setTestResult(allOk ? QP_TEST_RESULT_PASS	: QP_TEST_RESULT_FAIL,
335 									allOk ? "Pass"				: "Image comparison failed");
336 		}
337 
338 		return STOP;
339 	}
340 };
341 
342 } // anonymous
343 
ShaderFragDataTests(Context & context)344 ShaderFragDataTests::ShaderFragDataTests (Context& context)
345 	: TestCaseGroup(context, "fragdata", "gl_FragData[] Tests")
346 {
347 }
348 
~ShaderFragDataTests(void)349 ShaderFragDataTests::~ShaderFragDataTests (void)
350 {
351 }
352 
init(void)353 void ShaderFragDataTests::init (void)
354 {
355 	addChild(new FragDataIndexingCase		(m_context, "valid_static_index",	"Valid gl_FragData[] assignment using static index",	INDEX_EXPR_STATIC));
356 	addChild(new FragDataIndexingCase		(m_context, "valid_uniform_index",	"Valid gl_FragData[] assignment using uniform index",	INDEX_EXPR_UNIFORM));
357 	addChild(new FragDataIndexingCase		(m_context, "valid_dynamic_index",	"Valid gl_FragData[] assignment using dynamic index",	INDEX_EXPR_DYNAMIC));
358 	addChild(new FragDataDrawBuffersCase	(m_context));
359 
360 	// Negative cases.
361 	{
362 		gls::ShaderLibrary library(m_testCtx, m_context.getRenderContext(), m_context.getContextInfo());
363 		std::vector<tcu::TestNode*> negativeCases = library.loadShaderFile("shaders/fragdata.test");
364 
365 		for (std::vector<tcu::TestNode*>::iterator i = negativeCases.begin(); i != negativeCases.end(); i++)
366 			addChild(*i);
367 	}
368 }
369 
370 } // Functional
371 } // gles3
372 } // deqp
373