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 Framebuffer Object Tests.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es3fFboRenderTest.hpp"
25 #include "sglrContextUtil.hpp"
26 #include "sglrGLContext.hpp"
27 #include "sglrReferenceContext.hpp"
28 #include "es3fFboTestUtil.hpp"
29 #include "tcuSurface.hpp"
30 #include "tcuImageCompare.hpp"
31 #include "tcuTextureUtil.hpp"
32 #include "tcuVectorUtil.hpp"
33 #include "tcuRenderTarget.hpp"
34 #include "gluPixelTransfer.hpp"
35 #include "gluTextureUtil.hpp"
36 #include "gluStrUtil.hpp"
37 #include "deRandom.h"
38 #include "deString.h"
39 #include "glwDefs.hpp"
40 #include "glwEnums.hpp"
41 
42 #include <sstream>
43 
44 using std::vector;
45 using std::string;
46 using tcu::TestLog;
47 using tcu::Vec2;
48 using tcu::Vec3;
49 using tcu::Vec4;
50 using tcu::IVec2;
51 using tcu::IVec3;
52 using tcu::IVec4;
53 using tcu::RGBA;
54 using tcu::Surface;
55 
56 namespace deqp
57 {
58 namespace gles3
59 {
60 namespace Functional
61 {
62 
63 using glw::GLenum;
64 using namespace FboTestUtil;
65 
66 class FboConfig
67 {
68 public:
FboConfig(deUint32 buffers_,deUint32 colorType_,deUint32 colorFormat_,deUint32 depthStencilType_,deUint32 depthStencilFormat_,int width_=0,int height_=0,int samples_=0)69 	FboConfig (deUint32 buffers_, deUint32 colorType_, deUint32 colorFormat_, deUint32 depthStencilType_, deUint32 depthStencilFormat_, int width_ = 0, int height_ = 0, int samples_ = 0)
70 		: buffers				(buffers_)
71 		, colorType				(colorType_)
72 		, colorFormat			(colorFormat_)
73 		, depthStencilType		(depthStencilType_)
74 		, depthStencilFormat	(depthStencilFormat_)
75 		, width					(width_)
76 		, height				(height_)
77 		, samples				(samples_)
78 	{
79 	}
80 
FboConfig(void)81 	FboConfig (void)
82 		: buffers				(0)
83 		, colorType				(GL_NONE)
84 		, colorFormat			(GL_NONE)
85 		, depthStencilType		(GL_NONE)
86 		, depthStencilFormat	(GL_NONE)
87 		, width					(0)
88 		, height				(0)
89 		, samples				(0)
90 	{
91 	}
92 
93 	std::string				getName			(void) const;
94 
95 	deUint32				buffers;		//!< Buffer bit mask (GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|...)
96 
97 	GLenum					colorType;		//!< GL_TEXTURE_2D, GL_TEXTURE_CUBE_MAP, GL_RENDERBUFFER
98 	GLenum					colorFormat;	//!< Internal format for color buffer texture or renderbuffer
99 
100 	GLenum					depthStencilType;
101 	GLenum					depthStencilFormat;
102 
103 	int						width;
104 	int						height;
105 	int						samples;
106 };
107 
getTypeName(GLenum type)108 static const char* getTypeName (GLenum type)
109 {
110 	switch (type)
111 	{
112 		case GL_TEXTURE_2D:		return "tex2d";
113 		case GL_RENDERBUFFER:	return "rbo";
114 		default:
115 			TCU_FAIL("Unknown type");
116 	}
117 }
118 
getName(void) const119 std::string FboConfig::getName (void) const
120 {
121 	std::ostringstream name;
122 
123 	DE_ASSERT(buffers & GL_COLOR_BUFFER_BIT);
124 	name << getTypeName(colorType) << "_" << getFormatName(colorFormat);
125 
126 	if (buffers & GL_DEPTH_BUFFER_BIT)
127 		name << "_depth";
128 	if (buffers & GL_STENCIL_BUFFER_BIT)
129 		name << "_stencil";
130 
131 	if (buffers & (GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT))
132 		name << "_" << getTypeName(depthStencilType) << "_" << getFormatName(depthStencilFormat);
133 
134 	return name.str();
135 }
136 
137 class Framebuffer
138 {
139 public:
140 						Framebuffer				(sglr::Context& context, const FboConfig& config, int width, int height, deUint32 fbo = 0, deUint32 colorBuffer = 0, deUint32 depthStencilBuffer = 0);
141 						~Framebuffer			(void);
142 
getConfig(void) const143 	const FboConfig&	getConfig				(void) const { return m_config;				}
getFramebuffer(void) const144 	deUint32			getFramebuffer			(void) const { return m_framebuffer;		}
getColorBuffer(void) const145 	deUint32			getColorBuffer			(void) const { return m_colorBuffer;		}
getDepthStencilBuffer(void) const146 	deUint32			getDepthStencilBuffer	(void) const { return m_depthStencilBuffer;	}
147 
148 	void				checkCompleteness		(void);
149 
150 private:
151 	deUint32			createTex2D				(deUint32 name, GLenum format, int width, int height);
152 	deUint32			createRbo				(deUint32 name, GLenum format, int width, int height);
153 	void				destroyBuffer			(deUint32 name, GLenum type);
154 
155 	FboConfig			m_config;
156 	sglr::Context&		m_context;
157 	deUint32			m_framebuffer;
158 	deUint32			m_colorBuffer;
159 	deUint32			m_depthStencilBuffer;
160 };
161 
getEnablingExtensions(deUint32 format)162 static std::vector<std::string> getEnablingExtensions (deUint32 format)
163 {
164 	std::vector<std::string> out;
165 
166 	switch (format)
167 	{
168 		case GL_RGB16F:
169 			out.push_back("GL_EXT_color_buffer_half_float");
170 			break;
171 
172 		case GL_RGBA16F:
173 		case GL_RG16F:
174 		case GL_R16F:
175 			out.push_back("GL_EXT_color_buffer_half_float");
176 		// Fallthrough
177 
178 		case GL_RGBA32F:
179 		case GL_RGB32F:
180 		case GL_R11F_G11F_B10F:
181 		case GL_RG32F:
182 		case GL_R32F:
183 			out.push_back("GL_EXT_color_buffer_float");
184 			break;
185 
186 		default:
187 			break;
188 	}
189 
190 	return out;
191 }
192 
isExtensionSupported(sglr::Context & context,const char * name)193 static bool isExtensionSupported (sglr::Context& context, const char* name)
194 {
195 	std::istringstream extensions(context.getString(GL_EXTENSIONS));
196 	std::string extension;
197 
198 	while (std::getline(extensions, extension, ' '))
199 	{
200 		if (extension == name)
201 			return true;
202 	}
203 
204 	return false;
205 }
206 
isAnyExtensionSupported(sglr::Context & context,const std::vector<std::string> & requiredExts)207 static bool isAnyExtensionSupported (sglr::Context& context, const std::vector<std::string>& requiredExts)
208 {
209 	if (requiredExts.empty())
210 		return true;
211 
212 	for (std::vector<std::string>::const_iterator iter = requiredExts.begin(); iter != requiredExts.end(); iter++)
213 	{
214 		const std::string& extension = *iter;
215 
216 		if (isExtensionSupported(context, extension.c_str()))
217 			return true;
218 	}
219 
220 	return false;
221 }
222 
223 template<typename T>
join(const std::vector<T> & list,const std::string & sep)224 static std::string join (const std::vector<T>& list, const std::string& sep)
225 {
226 	std::ostringstream	out;
227 
228 	for (typename std::vector<T>::const_iterator iter = list.begin(); iter != list.end(); iter++)
229 	{
230 		if (iter != list.begin())
231 			out << sep;
232 		out << *iter;
233 	}
234 
235 	return out.str();
236 }
237 
checkColorFormatSupport(sglr::Context & context,deUint32 sizedFormat)238 static void checkColorFormatSupport (sglr::Context& context, deUint32 sizedFormat)
239 {
240 	const std::vector<std::string> requiredExts = getEnablingExtensions(sizedFormat);
241 
242 	if (!isAnyExtensionSupported(context, requiredExts))
243 	{
244 		std::string	errMsg	= "Format not supported, requires "
245 							+ ((requiredExts.size() == 1) ? requiredExts[0] : " one of the following: " + join(requiredExts, ", "));
246 
247 		throw tcu::NotSupportedError(errMsg);
248 	}
249 }
250 
Framebuffer(sglr::Context & context,const FboConfig & config,int width,int height,deUint32 fbo,deUint32 colorBufferName,deUint32 depthStencilBufferName)251 Framebuffer::Framebuffer (sglr::Context& context, const FboConfig& config, int width, int height, deUint32 fbo, deUint32 colorBufferName, deUint32 depthStencilBufferName)
252 	: m_config				(config)
253 	, m_context				(context)
254 	, m_framebuffer			(fbo)
255 	, m_colorBuffer			(0)
256 	, m_depthStencilBuffer	(0)
257 {
258 	// Verify that color format is supported
259 	checkColorFormatSupport(context, config.colorFormat);
260 
261 	if (m_framebuffer == 0)
262 		context.genFramebuffers(1, &m_framebuffer);
263 	context.bindFramebuffer(GL_FRAMEBUFFER, m_framebuffer);
264 
265 	if (m_config.buffers & (GL_COLOR_BUFFER_BIT))
266 	{
267 		switch (m_config.colorType)
268 		{
269 			case GL_TEXTURE_2D:
270 				m_colorBuffer = createTex2D(colorBufferName, m_config.colorFormat, width, height);
271 				context.framebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, m_colorBuffer, 0);
272 				break;
273 
274 			case GL_RENDERBUFFER:
275 				m_colorBuffer = createRbo(colorBufferName, m_config.colorFormat, width, height);
276 				context.framebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, m_colorBuffer);
277 				break;
278 
279 			default:
280 				TCU_FAIL("Unsupported type");
281 		}
282 	}
283 
284 	if (m_config.buffers & (GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT))
285 	{
286 		switch (m_config.depthStencilType)
287 		{
288 			case GL_TEXTURE_2D:		m_depthStencilBuffer = createTex2D(depthStencilBufferName, m_config.depthStencilFormat, width, height);		break;
289 			case GL_RENDERBUFFER:	m_depthStencilBuffer = createRbo(depthStencilBufferName, m_config.depthStencilFormat, width, height);		break;
290 			default:
291 				TCU_FAIL("Unsupported type");
292 		}
293 	}
294 
295 	for (int ndx = 0; ndx < 2; ndx++)
296 	{
297 		deUint32	bit		= ndx ? GL_STENCIL_BUFFER_BIT : GL_DEPTH_BUFFER_BIT;
298 		deUint32	point	= ndx ? GL_STENCIL_ATTACHMENT : GL_DEPTH_ATTACHMENT;
299 
300 		if ((m_config.buffers & bit) == 0)
301 			continue; /* Not used. */
302 
303 		switch (m_config.depthStencilType)
304 		{
305 			case GL_TEXTURE_2D:		context.framebufferTexture2D(GL_FRAMEBUFFER, point, GL_TEXTURE_2D, m_depthStencilBuffer, 0);	break;
306 			case GL_RENDERBUFFER:	context.framebufferRenderbuffer(GL_FRAMEBUFFER, point, GL_RENDERBUFFER, m_depthStencilBuffer);	break;
307 			default:
308 				DE_ASSERT(false);
309 		}
310 	}
311 
312 	GLenum err = m_context.getError();
313 	if (err != GL_NO_ERROR)
314 		throw glu::Error(err, glu::getErrorStr(err).toString().c_str(), "", __FILE__, __LINE__);
315 
316 	context.bindFramebuffer(GL_FRAMEBUFFER, 0);
317 }
318 
~Framebuffer(void)319 Framebuffer::~Framebuffer (void)
320 {
321 	m_context.deleteFramebuffers(1, &m_framebuffer);
322 	destroyBuffer(m_colorBuffer, m_config.colorType);
323 	destroyBuffer(m_depthStencilBuffer, m_config.depthStencilType);
324 }
325 
checkCompleteness(void)326 void Framebuffer::checkCompleteness (void)
327 {
328 	m_context.bindFramebuffer(GL_FRAMEBUFFER, m_framebuffer);
329 	GLenum status = m_context.checkFramebufferStatus(GL_FRAMEBUFFER);
330 	m_context.bindFramebuffer(GL_FRAMEBUFFER, 0);
331 	if (status != GL_FRAMEBUFFER_COMPLETE)
332 		throw FboIncompleteException(status, __FILE__, __LINE__);
333 }
334 
createTex2D(deUint32 name,GLenum format,int width,int height)335 deUint32 Framebuffer::createTex2D (deUint32 name, GLenum format, int width, int height)
336 {
337 	if (name == 0)
338 		m_context.genTextures(1, &name);
339 
340 	m_context.bindTexture(GL_TEXTURE_2D, name);
341 	m_context.texImage2D(GL_TEXTURE_2D, 0, format, width, height);
342 
343 	if (!deIsPowerOfTwo32(width) || !deIsPowerOfTwo32(height))
344 	{
345 		// Set wrap mode to clamp for NPOT FBOs
346 		m_context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
347 		m_context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
348 	}
349 
350 	m_context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
351 	m_context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
352 
353 	return name;
354 }
355 
createRbo(deUint32 name,GLenum format,int width,int height)356 deUint32 Framebuffer::createRbo (deUint32 name, GLenum format, int width, int height)
357 {
358 	if (name == 0)
359 		m_context.genRenderbuffers(1, &name);
360 
361 	m_context.bindRenderbuffer(GL_RENDERBUFFER, name);
362 	m_context.renderbufferStorage(GL_RENDERBUFFER, format, width, height);
363 
364 	return name;
365 }
366 
destroyBuffer(deUint32 name,GLenum type)367 void Framebuffer::destroyBuffer (deUint32 name, GLenum type)
368 {
369 	if (type == GL_TEXTURE_2D || type == GL_TEXTURE_CUBE_MAP)
370 		m_context.deleteTextures(1, &name);
371 	else if (type == GL_RENDERBUFFER)
372 		m_context.deleteRenderbuffers(1, &name);
373 	else
374 		DE_ASSERT(type == GL_NONE);
375 }
376 
createMetaballsTex2D(sglr::Context & context,deUint32 name,GLenum format,GLenum dataType,int width,int height)377 static void createMetaballsTex2D (sglr::Context& context, deUint32 name, GLenum format, GLenum dataType, int width, int height)
378 {
379 	tcu::TextureFormat	texFormat	= glu::mapGLTransferFormat(format, dataType);
380 	tcu::TextureLevel	level		(texFormat, width, height);
381 
382 	tcu::fillWithMetaballs(level.getAccess(), 5, name ^ width ^ height);
383 
384 	context.bindTexture(GL_TEXTURE_2D, name);
385 	context.texImage2D(GL_TEXTURE_2D, 0, format, width, height, 0, format, dataType, level.getAccess().getDataPtr());
386 	context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
387 }
388 
createQuadsTex2D(sglr::Context & context,deUint32 name,GLenum format,GLenum dataType,int width,int height)389 static void createQuadsTex2D (sglr::Context& context, deUint32 name, GLenum format, GLenum dataType, int width, int height)
390 {
391 	tcu::TextureFormat	texFormat	= glu::mapGLTransferFormat(format, dataType);
392 	tcu::TextureLevel	level		(texFormat, width, height);
393 
394 	tcu::fillWithRGBAQuads(level.getAccess());
395 
396 	context.bindTexture(GL_TEXTURE_2D, name);
397 	context.texImage2D(GL_TEXTURE_2D, 0, format, width, height, 0, format, dataType, level.getAccess().getDataPtr());
398 	context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
399 }
400 
401 class FboRenderCase : public TestCase
402 {
403 public:
404 								FboRenderCase			(Context& context, const char* name, const char* description, const FboConfig& config);
~FboRenderCase(void)405 	virtual						~FboRenderCase			(void) {}
406 
407 	virtual IterateResult		iterate					(void);
408 	virtual void				render					(sglr::Context& fboContext, Surface& dst) = DE_NULL;
409 
410 	bool						compare					(const tcu::Surface& reference, const tcu::Surface& result);
411 
412 protected:
413 	const FboConfig				m_config;
414 };
415 
FboRenderCase(Context & context,const char * name,const char * description,const FboConfig & config)416 FboRenderCase::FboRenderCase (Context& context, const char* name, const char* description, const FboConfig& config)
417 	: TestCase	(context, name, description)
418 	, m_config	(config)
419 {
420 }
421 
iterate(void)422 TestCase::IterateResult FboRenderCase::iterate (void)
423 {
424 	tcu::Vec4					clearColor				= tcu::Vec4(0.125f, 0.25f, 0.5f, 1.0f);
425 	glu::RenderContext&			renderCtx				= m_context.getRenderContext();
426 	const tcu::RenderTarget&	renderTarget			= renderCtx.getRenderTarget();
427 	tcu::TestLog&				log						= m_testCtx.getLog();
428 	const char*					failReason				= DE_NULL;
429 
430 	// Position & size for context
431 	deRandom rnd;
432 	deRandom_init(&rnd, deStringHash(getName()));
433 
434 	int		width	= deMin32(renderTarget.getWidth(), 128);
435 	int		height	= deMin32(renderTarget.getHeight(), 128);
436 	int		xMax	= renderTarget.getWidth()-width+1;
437 	int		yMax	= renderTarget.getHeight()-height+1;
438 	int		x		= deRandom_getUint32(&rnd) % xMax;
439 	int		y		= deRandom_getUint32(&rnd) % yMax;
440 
441 	tcu::Surface	gles3Frame	(width, height);
442 	tcu::Surface	refFrame	(width, height);
443 	GLenum			gles3Error;
444 	GLenum			refError;
445 
446 	// Render using GLES3
447 	try
448 	{
449 		sglr::GLContext context(renderCtx, log, sglr::GLCONTEXT_LOG_CALLS, tcu::IVec4(x, y, width, height));
450 
451 		context.clearColor(clearColor.x(), clearColor.y(), clearColor.z(), clearColor.w());
452 		context.clear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
453 
454 		render(context, gles3Frame); // Call actual render func
455 		gles3Error = context.getError();
456 	}
457 	catch (const FboIncompleteException& e)
458 	{
459 		if (e.getReason() == GL_FRAMEBUFFER_UNSUPPORTED)
460 		{
461 			// Mark test case as unsupported
462 			log << e;
463 			m_testCtx.setTestResult(QP_TEST_RESULT_NOT_SUPPORTED, "Not supported");
464 			return STOP;
465 		}
466 		else
467 			throw; // Propagate error
468 	}
469 
470 	// Render reference image
471 	{
472 		sglr::ReferenceContextBuffers	buffers	(tcu::PixelFormat(8,8,8,renderTarget.getPixelFormat().alphaBits?8:0), renderTarget.getDepthBits(), renderTarget.getStencilBits(), width, height);
473 		sglr::ReferenceContext			context	(sglr::ReferenceContextLimits(renderCtx), buffers.getColorbuffer(), buffers.getDepthbuffer(), buffers.getStencilbuffer());
474 
475 		context.clearColor(clearColor.x(), clearColor.y(), clearColor.z(), clearColor.w());
476 		context.clear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
477 
478 		render(context, refFrame);
479 		refError = context.getError();
480 	}
481 
482 	// Compare error codes
483 	bool errorCodesOk = (gles3Error == refError);
484 
485 	if (!errorCodesOk)
486 	{
487 		log << tcu::TestLog::Message << "Error code mismatch: got " << glu::getErrorStr(gles3Error) << ", expected " << glu::getErrorStr(refError) << tcu::TestLog::EndMessage;
488 		failReason = "Got unexpected error";
489 	}
490 
491 	// Compare images
492 	bool imagesOk = compare(refFrame, gles3Frame);
493 
494 	if (!imagesOk && !failReason)
495 		failReason = "Image comparison failed";
496 
497 	// Store test result
498 	bool isOk = errorCodesOk && imagesOk;
499 	m_testCtx.setTestResult(isOk ? QP_TEST_RESULT_PASS	: QP_TEST_RESULT_FAIL,
500 							isOk ? "Pass"				: failReason);
501 
502 	return STOP;
503 }
504 
compare(const tcu::Surface & reference,const tcu::Surface & result)505 bool FboRenderCase::compare (const tcu::Surface& reference, const tcu::Surface& result)
506 {
507 	const tcu::RGBA threshold (tcu::max(getFormatThreshold(m_config.colorFormat), tcu::RGBA(12, 12, 12, 12)));
508 
509 	return tcu::bilinearCompare(m_testCtx.getLog(), "ComparisonResult", "Image comparison result", reference.getAccess(), result.getAccess(), threshold, tcu::COMPARE_LOG_RESULT);
510 }
511 
512 namespace FboCases
513 {
514 
515 class StencilClearsTest : public FboRenderCase
516 {
517 public:
518 						StencilClearsTest		(Context& context, const FboConfig& config);
~StencilClearsTest(void)519 	virtual				~StencilClearsTest		(void) {};
520 
521 	void				render					(sglr::Context& context, Surface& dst);
522 };
523 
StencilClearsTest(Context & context,const FboConfig & config)524 StencilClearsTest::StencilClearsTest (Context& context, const FboConfig& config)
525 	: FboRenderCase	(context, config.getName().c_str(), "Stencil clears", config)
526 {
527 }
528 
render(sglr::Context & context,Surface & dst)529 void StencilClearsTest::render (sglr::Context& context, Surface& dst)
530 {
531 	tcu::TextureFormat		colorFormat			= glu::mapGLInternalFormat(m_config.colorFormat);
532 	glu::DataType			fboSamplerType		= glu::getSampler2DType(colorFormat);
533 	glu::DataType			fboOutputType		= getFragmentOutputType(colorFormat);
534 	tcu::TextureFormatInfo	fboRangeInfo		= tcu::getTextureFormatInfo(colorFormat);
535 	Vec4					fboOutScale			= fboRangeInfo.valueMax - fboRangeInfo.valueMin;
536 	Vec4					fboOutBias			= fboRangeInfo.valueMin;
537 
538 	Texture2DShader			texToFboShader		(DataTypes() << glu::TYPE_SAMPLER_2D, fboOutputType);
539 	Texture2DShader			texFromFboShader	(DataTypes() << fboSamplerType, glu::TYPE_FLOAT_VEC4);
540 
541 	deUint32				texToFboShaderID	= context.createProgram(&texToFboShader);
542 	deUint32				texFromFboShaderID	= context.createProgram(&texFromFboShader);
543 
544 	deUint32				metaballsTex		= 1;
545 	deUint32				quadsTex			= 2;
546 	int						width				= 128;
547 	int						height				= 128;
548 
549 	texToFboShader.setOutScaleBias(fboOutScale, fboOutBias);
550 	texFromFboShader.setTexScaleBias(0, fboRangeInfo.lookupScale, fboRangeInfo.lookupBias);
551 
552 	createQuadsTex2D(context, quadsTex, GL_RGBA, GL_UNSIGNED_BYTE, width, height);
553 	createMetaballsTex2D(context, metaballsTex, GL_RGBA, GL_UNSIGNED_BYTE, width, height);
554 
555 	Framebuffer fbo(context, m_config, width, height);
556 	fbo.checkCompleteness();
557 
558 	// Bind framebuffer and clear
559 	context.bindFramebuffer(GL_FRAMEBUFFER, fbo.getFramebuffer());
560 	context.viewport(0, 0, width, height);
561 	context.clearColor(0.0f, 0.0f, 0.0f, 1.0f);
562 	context.clear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
563 
564 	// Do stencil clears
565 	context.enable(GL_SCISSOR_TEST);
566 	context.scissor(10, 16, 32, 120);
567 	context.clearStencil(1);
568 	context.clear(GL_STENCIL_BUFFER_BIT);
569 	context.scissor(16, 32, 100, 64);
570 	context.clearStencil(2);
571 	context.clear(GL_STENCIL_BUFFER_BIT);
572 	context.disable(GL_SCISSOR_TEST);
573 
574 	// Draw 2 textures with stecil tests
575 	context.enable(GL_STENCIL_TEST);
576 
577 	context.bindTexture(GL_TEXTURE_2D, quadsTex);
578 	context.stencilFunc(GL_EQUAL, 1, 0xffu);
579 
580 	texToFboShader.setUniforms(context, texToFboShaderID);
581 	sglr::drawQuad(context, texToFboShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(+1.0f, +1.0f, 0.0f));
582 
583 	context.bindTexture(GL_TEXTURE_2D, metaballsTex);
584 	context.stencilFunc(GL_EQUAL, 2, 0xffu);
585 
586 	texToFboShader.setUniforms(context, texToFboShaderID);
587 	sglr::drawQuad(context, texToFboShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(+1.0f, +1.0f, 0.0f));
588 
589 	context.disable(GL_STENCIL_TEST);
590 
591 	if (fbo.getConfig().colorType == GL_TEXTURE_2D)
592 	{
593 		context.bindFramebuffer(GL_FRAMEBUFFER, 0);
594 		context.bindTexture(GL_TEXTURE_2D, fbo.getColorBuffer());
595 		context.viewport(0, 0, context.getWidth(), context.getHeight());
596 
597 		texFromFboShader.setUniforms(context, texFromFboShaderID);
598 		sglr::drawQuad(context, texFromFboShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
599 
600 		context.readPixels(dst, 0, 0, context.getWidth(), context.getHeight());
601 	}
602 	else
603 		readPixels(context, dst, 0, 0, width, height, colorFormat, fboRangeInfo.lookupScale, fboRangeInfo.lookupBias);
604 }
605 
606 class SharedColorbufferTest : public FboRenderCase
607 {
608 public:
609 						SharedColorbufferTest			(Context& context, const FboConfig& config);
~SharedColorbufferTest(void)610 	virtual				~SharedColorbufferTest			(void) {};
611 
612 	void				render							(sglr::Context& context, Surface& dst);
613 };
614 
SharedColorbufferTest(Context & context,const FboConfig & config)615 SharedColorbufferTest::SharedColorbufferTest (Context& context, const FboConfig& config)
616 	: FboRenderCase	(context, config.getName().c_str(), "Shared colorbuffer", config)
617 {
618 }
619 
render(sglr::Context & context,Surface & dst)620 void SharedColorbufferTest::render (sglr::Context& context, Surface& dst)
621 {
622 	Texture2DShader			texShader		(DataTypes() << glu::TYPE_SAMPLER_2D, glu::TYPE_FLOAT_VEC4);
623 	FlatColorShader			flatShader		(glu::TYPE_FLOAT_VEC4);
624 	deUint32				texShaderID		= context.createProgram(&texShader);
625 	deUint32				flatShaderID	= context.createProgram(&flatShader);
626 
627 	int						width			= 128;
628 	int						height			= 128;
629 	deUint32				quadsTex		= 1;
630 	deUint32				metaballsTex	= 2;
631 	bool					stencil			= (m_config.buffers & GL_STENCIL_BUFFER_BIT) != 0;
632 
633 	context.disable(GL_DITHER);
634 
635 	// Textures
636 	createQuadsTex2D(context, quadsTex, GL_RGB, GL_UNSIGNED_BYTE, 64, 64);
637 	createMetaballsTex2D(context, metaballsTex, GL_RGBA, GL_UNSIGNED_BYTE, 64, 64);
638 
639 	context.viewport(0, 0, width, height);
640 
641 	// Fbo A
642 	Framebuffer fboA(context, m_config, width, height);
643 	fboA.checkCompleteness();
644 
645 	// Fbo B - don't create colorbuffer
646 	FboConfig cfg = m_config;
647 	cfg.buffers		&= ~GL_COLOR_BUFFER_BIT;
648 	cfg.colorType	 = GL_NONE;
649 	cfg.colorFormat	 = GL_NONE;
650 	Framebuffer fboB(context, cfg, width, height);
651 
652 	// Attach color buffer from fbo A
653 	context.bindFramebuffer(GL_FRAMEBUFFER, fboB.getFramebuffer());
654 	switch (m_config.colorType)
655 	{
656 		case GL_TEXTURE_2D:
657 			context.framebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, fboA.getColorBuffer(), 0);
658 			break;
659 
660 		case GL_RENDERBUFFER:
661 			context.framebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, fboA.getColorBuffer());
662 			break;
663 
664 		default:
665 			DE_ASSERT(DE_FALSE);
666 	}
667 
668 	// Clear depth and stencil in fbo B
669 	context.clear(GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
670 
671 	// Render quads to fbo 1, with depth 0.0
672 	context.bindFramebuffer(GL_FRAMEBUFFER, fboA.getFramebuffer());
673 	context.bindTexture(GL_TEXTURE_2D, quadsTex);
674 	context.clearColor(0.0f, 0.0f, 0.0f, 1.0f);
675 	context.clear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
676 
677 	if (stencil)
678 	{
679 		// Stencil to 1 in fbo A
680 		context.clearStencil(1);
681 		context.clear(GL_STENCIL_BUFFER_BIT);
682 	}
683 
684 	texShader.setUniforms(context, texShaderID);
685 
686 	context.enable(GL_DEPTH_TEST);
687 	sglr::drawQuad(context, texShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
688 	context.disable(GL_DEPTH_TEST);
689 
690 	// Blend metaballs to fbo 2
691 	context.bindFramebuffer(GL_FRAMEBUFFER, fboB.getFramebuffer());
692 	context.bindTexture(GL_TEXTURE_2D, metaballsTex);
693 	context.enable(GL_BLEND);
694 	context.blendFuncSeparate(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA, GL_ZERO, GL_ONE);
695 	sglr::drawQuad(context, texShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
696 
697 	// Render small quad that is only visible if depth buffer is not shared with fbo A - or there is no depth bits
698 	context.bindTexture(GL_TEXTURE_2D, quadsTex);
699 	context.enable(GL_DEPTH_TEST);
700 	sglr::drawQuad(context, texShaderID, Vec3(0.5f, 0.5f, 0.5f), Vec3(1.0f, 1.0f, 0.5f));
701 	context.disable(GL_DEPTH_TEST);
702 
703 	if (stencil)
704 	{
705 		flatShader.setColor(context, flatShaderID, Vec4(0.0f, 1.0f, 0.0f, 1.0f));
706 
707 		// Clear subset of stencil buffer to 1
708 		context.enable(GL_SCISSOR_TEST);
709 		context.scissor(10, 10, 12, 25);
710 		context.clearStencil(1);
711 		context.clear(GL_STENCIL_BUFFER_BIT);
712 		context.disable(GL_SCISSOR_TEST);
713 
714 		// Render quad with stencil mask == 1
715 		context.enable(GL_STENCIL_TEST);
716 		context.stencilFunc(GL_EQUAL, 1, 0xffu);
717 		sglr::drawQuad(context, flatShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
718 		context.disable(GL_STENCIL_TEST);
719 	}
720 
721 	// Get results
722 	if (fboA.getConfig().colorType == GL_TEXTURE_2D)
723 	{
724 		texShader.setUniforms(context, texShaderID);
725 
726 		context.bindFramebuffer(GL_FRAMEBUFFER, 0);
727 		context.bindTexture(GL_TEXTURE_2D, fboA.getColorBuffer());
728 		context.viewport(0, 0, context.getWidth(), context.getHeight());
729 		sglr::drawQuad(context, texShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
730 		context.readPixels(dst, 0, 0, context.getWidth(), context.getHeight());
731 	}
732 	else
733 		readPixels(context, dst, 0, 0, width, height, glu::mapGLInternalFormat(fboA.getConfig().colorFormat), Vec4(1.0f), Vec4(0.0f));
734 }
735 
736 class SharedColorbufferClearsTest : public FboRenderCase
737 {
738 public:
739 					SharedColorbufferClearsTest		(Context& context, const FboConfig& config);
~SharedColorbufferClearsTest(void)740 	virtual			~SharedColorbufferClearsTest	(void) {}
741 
742 	void			render							(sglr::Context& context, Surface& dst);
743 };
744 
SharedColorbufferClearsTest(Context & context,const FboConfig & config)745 SharedColorbufferClearsTest::SharedColorbufferClearsTest (Context& context, const FboConfig& config)
746 	: FboRenderCase	(context, config.getName().c_str(), "Shared colorbuffer clears", config)
747 {
748 }
749 
render(sglr::Context & context,Surface & dst)750 void SharedColorbufferClearsTest::render (sglr::Context& context, Surface& dst)
751 {
752 	tcu::TextureFormat		colorFormat		= glu::mapGLInternalFormat(m_config.colorFormat);
753 	glu::DataType			fboSamplerType	= glu::getSampler2DType(colorFormat);
754 	int						width			= 128;
755 	int						height			= 128;
756 	deUint32				colorbuffer		= 1;
757 
758 	// Check for format support.
759 	checkColorFormatSupport(context, m_config.colorFormat);
760 
761 	// Single colorbuffer
762 	if (m_config.colorType == GL_TEXTURE_2D)
763 	{
764 		context.bindTexture(GL_TEXTURE_2D, colorbuffer);
765 		context.texImage2D(GL_TEXTURE_2D, 0, m_config.colorFormat, width, height);
766 		context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
767 		context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
768 	}
769 	else
770 	{
771 		DE_ASSERT(m_config.colorType == GL_RENDERBUFFER);
772 		context.bindRenderbuffer(GL_RENDERBUFFER, colorbuffer);
773 		context.renderbufferStorage(GL_RENDERBUFFER, m_config.colorFormat, width, height);
774 	}
775 
776 	// Multiple framebuffers sharing the colorbuffer
777 	for (int fbo = 1; fbo <= 3; fbo++)
778 	{
779 		context.bindFramebuffer(GL_FRAMEBUFFER, fbo);
780 
781 		if (m_config.colorType == GL_TEXTURE_2D)
782 			context.framebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, colorbuffer, 0);
783 		else
784 			context.framebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, colorbuffer);
785 	}
786 
787 	context.bindFramebuffer(GL_FRAMEBUFFER, 1);
788 
789 	// Check completeness
790 	{
791 		GLenum status = context.checkFramebufferStatus(GL_FRAMEBUFFER);
792 		if (status != GL_FRAMEBUFFER_COMPLETE)
793 			throw FboIncompleteException(status, __FILE__, __LINE__);
794 	}
795 
796 	// Render to them
797 	context.viewport(0, 0, width, height);
798 	context.clearColor(0.0f, 0.0f, 1.0f, 1.0f);
799 	context.clear(GL_COLOR_BUFFER_BIT);
800 
801 	context.enable(GL_SCISSOR_TEST);
802 
803 	context.bindFramebuffer(GL_FRAMEBUFFER, 2);
804 	context.clearColor(0.6f, 0.0f, 0.0f, 1.0f);
805 	context.scissor(10, 10, 64, 64);
806 	context.clear(GL_COLOR_BUFFER_BIT);
807 	context.clearColor(0.0f, 0.6f, 0.0f, 1.0f);
808 	context.scissor(60, 60, 40, 20);
809 	context.clear(GL_COLOR_BUFFER_BIT);
810 
811 	context.bindFramebuffer(GL_FRAMEBUFFER, 3);
812 	context.clearColor(0.0f, 0.0f, 0.6f, 1.0f);
813 	context.scissor(20, 20, 100, 10);
814 	context.clear(GL_COLOR_BUFFER_BIT);
815 
816 	context.bindFramebuffer(GL_FRAMEBUFFER, 1);
817 	context.clearColor(0.6f, 0.0f, 0.6f, 1.0f);
818 	context.scissor(20, 20, 5, 100);
819 	context.clear(GL_COLOR_BUFFER_BIT);
820 
821 	context.disable(GL_SCISSOR_TEST);
822 
823 	if (m_config.colorType == GL_TEXTURE_2D)
824 	{
825 		Texture2DShader shader(DataTypes() << fboSamplerType, glu::TYPE_FLOAT_VEC4);
826 		deUint32 shaderID = context.createProgram(&shader);
827 
828 		shader.setUniforms(context, shaderID);
829 
830 		context.bindFramebuffer(GL_FRAMEBUFFER, 0);
831 		context.viewport(0, 0, context.getWidth(), context.getHeight());
832 		sglr::drawQuad(context, shaderID, Vec3(-0.9f, -0.9f, 0.0f), Vec3(0.9f, 0.9f, 0.0f));
833 		context.readPixels(dst, 0, 0, context.getWidth(), context.getHeight());
834 	}
835 	else
836 		readPixels(context, dst, 0, 0, width, height, colorFormat, Vec4(1.0f), Vec4(0.0f));
837 }
838 
839 class SharedDepthStencilTest : public FboRenderCase
840 {
841 public:
842 					SharedDepthStencilTest		(Context& context, const FboConfig& config);
~SharedDepthStencilTest(void)843 	virtual			~SharedDepthStencilTest		(void) {};
844 
845 	static bool		isConfigSupported			(const FboConfig& config);
846 	void			render						(sglr::Context& context, Surface& dst);
847 };
848 
SharedDepthStencilTest(Context & context,const FboConfig & config)849 SharedDepthStencilTest::SharedDepthStencilTest (Context& context, const FboConfig& config)
850 	: FboRenderCase	(context, config.getName().c_str(), "Shared depth/stencilbuffer", config)
851 {
852 }
853 
isConfigSupported(const FboConfig & config)854 bool SharedDepthStencilTest::isConfigSupported (const FboConfig& config)
855 {
856 	return (config.buffers & (GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT)) != 0;
857 }
858 
render(sglr::Context & context,Surface & dst)859 void SharedDepthStencilTest::render (sglr::Context& context, Surface& dst)
860 {
861 	Texture2DShader	texShader		(DataTypes() << glu::TYPE_SAMPLER_2D, glu::TYPE_FLOAT_VEC4);
862 	FlatColorShader	flatShader		(glu::TYPE_FLOAT_VEC4);
863 	deUint32		texShaderID		= context.createProgram(&texShader);
864 	deUint32		flatShaderID	= context.createProgram(&flatShader);
865 	int				width			= 128;
866 	int				height			= 128;
867 //	bool			depth			= (m_config.buffers & GL_DEPTH_BUFFER_BIT) != 0;
868 	bool			stencil			= (m_config.buffers & GL_STENCIL_BUFFER_BIT) != 0;
869 
870 	// Textures
871 	deUint32 metaballsTex	= 5;
872 	deUint32 quadsTex		= 6;
873 	createMetaballsTex2D(context, metaballsTex, GL_RGB, GL_UNSIGNED_BYTE, 64, 64);
874 	createQuadsTex2D(context, quadsTex, GL_RGB, GL_UNSIGNED_BYTE, 64, 64);
875 
876 	context.viewport(0, 0, width, height);
877 
878 	// Fbo A
879 	Framebuffer fboA(context, m_config, width, height);
880 	fboA.checkCompleteness();
881 
882 	// Fbo B
883 	FboConfig cfg = m_config;
884 	cfg.buffers				&= ~(GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
885 	cfg.depthStencilType	 = GL_NONE;
886 	cfg.depthStencilFormat	 = GL_NONE;
887 	Framebuffer fboB(context, cfg, width, height);
888 
889 	// Bind depth/stencil buffers from fbo A to fbo B
890 	context.bindFramebuffer(GL_FRAMEBUFFER, fboB.getFramebuffer());
891 	for (int ndx = 0; ndx < 2; ndx++)
892 	{
893 		deUint32	bit		= ndx ? GL_STENCIL_BUFFER_BIT : GL_DEPTH_BUFFER_BIT;
894 		deUint32	point	= ndx ? GL_STENCIL_ATTACHMENT : GL_DEPTH_ATTACHMENT;
895 
896 		if ((m_config.buffers & bit) == 0)
897 			continue;
898 
899 		switch (m_config.depthStencilType)
900 		{
901 			case GL_TEXTURE_2D:		context.framebufferTexture2D(GL_FRAMEBUFFER, point, GL_TEXTURE_2D, fboA.getDepthStencilBuffer(), 0);	break;
902 			case GL_RENDERBUFFER:	context.framebufferRenderbuffer(GL_FRAMEBUFFER, point, GL_RENDERBUFFER, fboA.getDepthStencilBuffer());	break;
903 			default:
904 				TCU_FAIL("Not implemented");
905 		}
906 	}
907 
908 	// Setup uniforms
909 	texShader.setUniforms(context, texShaderID);
910 
911 	// Clear color to red and stencil to 1 in fbo B.
912 	context.clearColor(1.0f, 0.0f, 0.0f, 1.0f);
913 	context.clearStencil(1);
914 	context.clear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
915 
916 	context.enable(GL_DEPTH_TEST);
917 
918 	// Render quad to fbo A
919 	context.bindFramebuffer(GL_FRAMEBUFFER, fboA.getFramebuffer());
920 	context.bindTexture(GL_TEXTURE_2D, quadsTex);
921 	sglr::drawQuad(context, texShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
922 
923 	if (stencil)
924 	{
925 		// Clear subset of stencil buffer to 0 in fbo A
926 		context.enable(GL_SCISSOR_TEST);
927 		context.scissor(10, 10, 12, 25);
928 		context.clearStencil(0);
929 		context.clear(GL_STENCIL_BUFFER_BIT);
930 		context.disable(GL_SCISSOR_TEST);
931 	}
932 
933 	// Render metaballs to fbo B
934 	context.bindFramebuffer(GL_FRAMEBUFFER, fboB.getFramebuffer());
935 	context.bindTexture(GL_TEXTURE_2D, metaballsTex);
936 	sglr::drawQuad(context, texShaderID, Vec3(-1.0f, -1.0f, -1.0f), Vec3(1.0f, 1.0f, 1.0f));
937 
938 	context.disable(GL_DEPTH_TEST);
939 
940 	if (stencil)
941 	{
942 		// Render quad with stencil mask == 0
943 		context.enable(GL_STENCIL_TEST);
944 		context.stencilFunc(GL_EQUAL, 0, 0xffu);
945 		context.useProgram(flatShaderID);
946 		flatShader.setColor(context, flatShaderID, Vec4(0.0f, 1.0f, 0.0f, 1.0f));
947 		sglr::drawQuad(context, flatShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(+1.0f, +1.0f, 0.0f));
948 		context.disable(GL_STENCIL_TEST);
949 	}
950 
951 	if (m_config.colorType == GL_TEXTURE_2D)
952 	{
953 		// Render both to screen
954 		context.bindFramebuffer(GL_FRAMEBUFFER, 0);
955 		context.viewport(0, 0, context.getWidth(), context.getHeight());
956 		context.bindTexture(GL_TEXTURE_2D, fboA.getColorBuffer());
957 		sglr::drawQuad(context, texShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(0.0f, 1.0f, 0.0f));
958 		context.bindTexture(GL_TEXTURE_2D, fboB.getColorBuffer());
959 		sglr::drawQuad(context, texShaderID, Vec3(0.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
960 
961 		context.readPixels(dst, 0, 0, context.getWidth(), context.getHeight());
962 	}
963 	else
964 	{
965 		// Read results from fbo B
966 		readPixels(context, dst, 0, 0, width, height, glu::mapGLInternalFormat(m_config.colorFormat), Vec4(1.0f), Vec4(0.0f));
967 	}
968 }
969 
970 #if 0
971 class TexSubImageAfterRenderTest : public FboRenderCase
972 {
973 public:
974 					TexSubImageAfterRenderTest		(Context& context, const FboConfig& config);
975 	virtual			~TexSubImageAfterRenderTest		(void) {}
976 
977 	void			render							(sglr::Context& context, Surface& dst);
978 };
979 
980 TexSubImageAfterRenderTest::TexSubImageAfterRenderTest (Context& context, const FboConfig& config)
981 	: FboRenderCase(context, (string("after_render_") + config.getName()).c_str(), "TexSubImage after rendering to texture", config)
982 {
983 }
984 
985 void TexSubImageAfterRenderTest::render (sglr::Context& context, Surface& dst)
986 {
987 	using sglr::TexturedQuadOp;
988 
989 	bool isRGBA = true;
990 
991 	Surface fourQuads(Surface::PIXELFORMAT_RGB, 64, 64);
992 	tcu::SurfaceUtil::fillWithFourQuads(fourQuads);
993 
994 	Surface metaballs(isRGBA ? Surface::PIXELFORMAT_RGBA : Surface::PIXELFORMAT_RGB, 64, 64);
995 	tcu::SurfaceUtil::fillWithMetaballs(metaballs, 5, 3);
996 
997 	deUint32 fourQuadsTex = 1;
998 	context.bindTexture(GL_TEXTURE_2D, fourQuadsTex);
999 	context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
1000 	context.texImage2D(GL_TEXTURE_2D, 0, GL_RGB, fourQuads);
1001 
1002 	context.bindFramebuffer(GL_FRAMEBUFFER, 1);
1003 
1004 	deUint32 fboTex = 2;
1005 	context.bindTexture(GL_TEXTURE_2D, fboTex);
1006 	context.texImage2D(GL_TEXTURE_2D, 0, isRGBA ? GL_RGBA : GL_RGB, 128, 128);
1007 	context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
1008 	context.framebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, fboTex, 0);
1009 
1010 	// Render to fbo
1011 	context.viewport(0, 0, 128, 128);
1012 	context.bindTexture(GL_TEXTURE_2D, fourQuadsTex);
1013 	context.draw(TexturedQuadOp(-1.0f, -1.0f, 1.0f, 1.0f, 0));
1014 
1015 	// Update texture using TexSubImage2D
1016 	context.bindTexture(GL_TEXTURE_2D, fboTex);
1017 	context.texSubImage2D(GL_TEXTURE_2D, 0, 32, 32, metaballs);
1018 
1019 	// Draw to screen
1020 	context.bindFramebuffer(GL_FRAMEBUFFER, 0);
1021 	context.viewport(0, 0, context.getWidth(), context.getHeight());
1022 	context.draw(TexturedQuadOp(-1.0f, -1.0f, 1.0f, 1.0f, 0));
1023 	context.readPixels(dst, 0, 0, context.getWidth(), context.getHeight());
1024 }
1025 
1026 class TexSubImageBetweenRenderTest : public FboRenderCase
1027 {
1028 public:
1029 					TexSubImageBetweenRenderTest		(Context& context, const FboConfig& config);
1030 	virtual			~TexSubImageBetweenRenderTest		(void) {}
1031 
1032 	void			render								(sglr::Context& context, Surface& dst);
1033 };
1034 
1035 TexSubImageBetweenRenderTest::TexSubImageBetweenRenderTest (Context& context, const FboConfig& config)
1036 	: FboRenderCase(context, (string("between_render_") + config.getName()).c_str(), "TexSubImage between rendering calls", config)
1037 {
1038 }
1039 
1040 void TexSubImageBetweenRenderTest::render (sglr::Context& context, Surface& dst)
1041 {
1042 	using sglr::TexturedQuadOp;
1043 	using sglr::BlendTextureOp;
1044 
1045 	bool isRGBA = true;
1046 
1047 	Surface fourQuads(Surface::PIXELFORMAT_RGB, 64, 64);
1048 	tcu::SurfaceUtil::fillWithFourQuads(fourQuads);
1049 
1050 	Surface metaballs(isRGBA ? Surface::PIXELFORMAT_RGBA : Surface::PIXELFORMAT_RGB, 64, 64);
1051 	tcu::SurfaceUtil::fillWithMetaballs(metaballs, 5, 3);
1052 
1053 	Surface metaballs2(Surface::PIXELFORMAT_RGBA, 64, 64);
1054 	tcu::SurfaceUtil::fillWithMetaballs(metaballs2, 5, 4);
1055 
1056 	deUint32 metaballsTex = 3;
1057 	context.bindTexture(GL_TEXTURE_2D, metaballsTex);
1058 	context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
1059 	context.texImage2D(GL_TEXTURE_2D, 0, GL_RGBA, metaballs2);
1060 
1061 	deUint32 fourQuadsTex = 1;
1062 	context.bindTexture(GL_TEXTURE_2D, fourQuadsTex);
1063 	context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
1064 	context.texImage2D(GL_TEXTURE_2D, 0, GL_RGB, fourQuads);
1065 
1066 	context.bindFramebuffer(GL_FRAMEBUFFER, 1);
1067 
1068 	deUint32 fboTex = 2;
1069 	context.bindTexture(GL_TEXTURE_2D, fboTex);
1070 	context.texImage2D(GL_TEXTURE_2D, 0, isRGBA ? GL_RGBA : GL_RGB, 128, 128);
1071 	context.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
1072 	context.framebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, fboTex, 0);
1073 
1074 	// Render to fbo
1075 	context.viewport(0, 0, 128, 128);
1076 	context.bindTexture(GL_TEXTURE_2D, fourQuadsTex);
1077 	context.draw(TexturedQuadOp(-1.0f, -1.0f, 1.0f, 1.0f, 0));
1078 
1079 	// Update texture using TexSubImage2D
1080 	context.bindTexture(GL_TEXTURE_2D, fboTex);
1081 	context.texSubImage2D(GL_TEXTURE_2D, 0, 32, 32, metaballs);
1082 
1083 	// Render again to fbo
1084 	context.bindTexture(GL_TEXTURE_2D, metaballsTex);
1085 	context.draw(BlendTextureOp(0));
1086 
1087 	// Draw to screen
1088 	context.bindFramebuffer(GL_FRAMEBUFFER, 0);
1089 	context.viewport(0, 0, context.getWidth(), context.getHeight());
1090 	context.bindTexture(GL_TEXTURE_2D, fboTex);
1091 	context.draw(TexturedQuadOp(-1.0f, -1.0f, 1.0f, 1.0f, 0));
1092 
1093 	context.readPixels(dst, 0, 0, context.getWidth(), context.getHeight());
1094 }
1095 #endif
1096 
1097 class ResizeTest : public FboRenderCase
1098 {
1099 public:
1100 					ResizeTest				(Context& context, const FboConfig& config);
~ResizeTest(void)1101 	virtual			~ResizeTest				(void) {}
1102 
1103 	void			render					(sglr::Context& context, Surface& dst);
1104 };
1105 
ResizeTest(Context & context,const FboConfig & config)1106 ResizeTest::ResizeTest (Context& context, const FboConfig& config)
1107 	: FboRenderCase	(context, config.getName().c_str(), "Resize framebuffer", config)
1108 {
1109 }
1110 
render(sglr::Context & context,Surface & dst)1111 void ResizeTest::render (sglr::Context& context, Surface& dst)
1112 {
1113 	tcu::TextureFormat		colorFormat			= glu::mapGLInternalFormat(m_config.colorFormat);
1114 	glu::DataType			fboSamplerType		= glu::getSampler2DType(colorFormat);
1115 	glu::DataType			fboOutputType		= getFragmentOutputType(colorFormat);
1116 	tcu::TextureFormatInfo	fboRangeInfo		= tcu::getTextureFormatInfo(colorFormat);
1117 	Vec4					fboOutScale			= fboRangeInfo.valueMax - fboRangeInfo.valueMin;
1118 	Vec4					fboOutBias			= fboRangeInfo.valueMin;
1119 
1120 	Texture2DShader			texToFboShader		(DataTypes() << glu::TYPE_SAMPLER_2D, fboOutputType);
1121 	Texture2DShader			texFromFboShader	(DataTypes() << fboSamplerType, glu::TYPE_FLOAT_VEC4);
1122 	FlatColorShader			flatShader			(fboOutputType);
1123 	deUint32				texToFboShaderID	= context.createProgram(&texToFboShader);
1124 	deUint32				texFromFboShaderID	= context.createProgram(&texFromFboShader);
1125 	deUint32				flatShaderID		= context.createProgram(&flatShader);
1126 
1127 	deUint32				quadsTex			= 1;
1128 	deUint32				metaballsTex		= 2;
1129 	bool					depth				= (m_config.buffers & GL_DEPTH_BUFFER_BIT)		!= 0;
1130 	bool					stencil				= (m_config.buffers & GL_STENCIL_BUFFER_BIT)	!= 0;
1131 	int						initialWidth		= 128;
1132 	int						initialHeight		= 128;
1133 	int						newWidth			= 64;
1134 	int						newHeight			= 32;
1135 
1136 	texToFboShader.setOutScaleBias(fboOutScale, fboOutBias);
1137 	texFromFboShader.setTexScaleBias(0, fboRangeInfo.lookupScale, fboRangeInfo.lookupBias);
1138 
1139 	createQuadsTex2D(context, quadsTex, GL_RGB, GL_UNSIGNED_BYTE, 64, 64);
1140 	createMetaballsTex2D(context, metaballsTex, GL_RGB, GL_UNSIGNED_BYTE, 32, 32);
1141 
1142 	Framebuffer fbo(context, m_config, initialWidth, initialHeight);
1143 	fbo.checkCompleteness();
1144 
1145 	// Setup shaders
1146 	texToFboShader.setUniforms	(context, texToFboShaderID);
1147 	texFromFboShader.setUniforms(context, texFromFboShaderID);
1148 	flatShader.setColor			(context, flatShaderID, Vec4(0.0f, 1.0f, 0.0f, 1.0f) * fboOutScale + fboOutBias);
1149 
1150 	// Render quads
1151 	context.bindFramebuffer(GL_FRAMEBUFFER, fbo.getFramebuffer());
1152 	context.viewport(0, 0, initialWidth, initialHeight);
1153 	clearColorBuffer(context, colorFormat, tcu::Vec4(0.0f, 0.0f, 0.0f, 1.0f));
1154 	context.clear(GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
1155 	context.bindTexture(GL_TEXTURE_2D, quadsTex);
1156 	sglr::drawQuad(context, texToFboShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
1157 
1158 	if (fbo.getConfig().colorType == GL_TEXTURE_2D)
1159 	{
1160 		// Render fbo to screen
1161 		context.bindFramebuffer(GL_FRAMEBUFFER, 0);
1162 		context.viewport(0, 0, context.getWidth(), context.getHeight());
1163 		context.bindTexture(GL_TEXTURE_2D, fbo.getColorBuffer());
1164 		sglr::drawQuad(context, texFromFboShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
1165 
1166 		// Restore binding
1167 		context.bindFramebuffer(GL_FRAMEBUFFER, fbo.getFramebuffer());
1168 	}
1169 
1170 	// Resize buffers
1171 	switch (fbo.getConfig().colorType)
1172 	{
1173 		case GL_TEXTURE_2D:
1174 			context.bindTexture(GL_TEXTURE_2D, fbo.getColorBuffer());
1175 			context.texImage2D(GL_TEXTURE_2D, 0, fbo.getConfig().colorFormat, newWidth, newHeight);
1176 			break;
1177 
1178 		case GL_RENDERBUFFER:
1179 			context.bindRenderbuffer(GL_RENDERBUFFER, fbo.getColorBuffer());
1180 			context.renderbufferStorage(GL_RENDERBUFFER, fbo.getConfig().colorFormat, newWidth, newHeight);
1181 			break;
1182 
1183 		default:
1184 			DE_ASSERT(DE_FALSE);
1185 	}
1186 
1187 	if (depth || stencil)
1188 	{
1189 		switch (fbo.getConfig().depthStencilType)
1190 		{
1191 			case GL_TEXTURE_2D:
1192 				context.bindTexture(GL_TEXTURE_2D, fbo.getDepthStencilBuffer());
1193 				context.texImage2D(GL_TEXTURE_2D, 0, fbo.getConfig().depthStencilFormat, newWidth, newHeight);
1194 				break;
1195 
1196 			case GL_RENDERBUFFER:
1197 				context.bindRenderbuffer(GL_RENDERBUFFER, fbo.getDepthStencilBuffer());
1198 				context.renderbufferStorage(GL_RENDERBUFFER, fbo.getConfig().depthStencilFormat, newWidth, newHeight);
1199 				break;
1200 
1201 			default:
1202 				DE_ASSERT(false);
1203 		}
1204 	}
1205 
1206 	// Render to resized fbo
1207 	context.viewport(0, 0, newWidth, newHeight);
1208 	clearColorBuffer(context, colorFormat, tcu::Vec4(1.0f, 0.0f, 0.0f, 1.0f));
1209 	context.clear(GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
1210 
1211 	context.enable(GL_DEPTH_TEST);
1212 
1213 	context.bindTexture(GL_TEXTURE_2D, metaballsTex);
1214 	sglr::drawQuad(context, texToFboShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(+1.0f, +1.0f, 0.0f));
1215 
1216 	context.bindTexture(GL_TEXTURE_2D, quadsTex);
1217 	sglr::drawQuad(context, texToFboShaderID, Vec3(0.0f, 0.0f, -1.0f), Vec3(+1.0f, +1.0f, 1.0f));
1218 
1219 	context.disable(GL_DEPTH_TEST);
1220 
1221 	if (stencil)
1222 	{
1223 		context.enable(GL_SCISSOR_TEST);
1224 		context.clearStencil(1);
1225 		context.scissor(10, 10, 5, 15);
1226 		context.clear(GL_STENCIL_BUFFER_BIT);
1227 		context.disable(GL_SCISSOR_TEST);
1228 
1229 		context.enable(GL_STENCIL_TEST);
1230 		context.stencilFunc(GL_EQUAL, 1, 0xffu);
1231 		sglr::drawQuad(context, flatShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(+1.0f, +1.0f, 0.0f));
1232 		context.disable(GL_STENCIL_TEST);
1233 	}
1234 
1235 	if (m_config.colorType == GL_TEXTURE_2D)
1236 	{
1237 		context.bindFramebuffer(GL_FRAMEBUFFER, 0);
1238 		context.viewport(0, 0, context.getWidth(), context.getHeight());
1239 		context.bindTexture(GL_TEXTURE_2D, fbo.getColorBuffer());
1240 		sglr::drawQuad(context, texFromFboShaderID, Vec3(-0.5f, -0.5f, 0.0f), Vec3(0.5f, 0.5f, 0.0f));
1241 		context.readPixels(dst, 0, 0, context.getWidth(), context.getHeight());
1242 	}
1243 	else
1244 		readPixels(context, dst, 0, 0, newWidth, newHeight, colorFormat, fboRangeInfo.lookupScale, fboRangeInfo.lookupBias);
1245 }
1246 
1247 class RecreateBuffersTest : public FboRenderCase
1248 {
1249 public:
1250 					RecreateBuffersTest			(Context& context, const FboConfig& config, deUint32 buffers, bool rebind);
~RecreateBuffersTest(void)1251 	virtual			~RecreateBuffersTest		(void) {}
1252 
1253 	void			render						(sglr::Context& context, Surface& dst);
1254 
1255 private:
1256 	deUint32		m_buffers;
1257 	bool			m_rebind;
1258 };
1259 
RecreateBuffersTest(Context & context,const FboConfig & config,deUint32 buffers,bool rebind)1260 RecreateBuffersTest::RecreateBuffersTest (Context& context, const FboConfig& config, deUint32 buffers, bool rebind)
1261 	: FboRenderCase		(context, (string(config.getName()) + (rebind ? "" : "_no_rebind")).c_str(), "Recreate buffers", config)
1262 	, m_buffers			(buffers)
1263 	, m_rebind			(rebind)
1264 {
1265 }
1266 
render(sglr::Context & ctx,Surface & dst)1267 void RecreateBuffersTest::render (sglr::Context& ctx, Surface& dst)
1268 {
1269 	tcu::TextureFormat		colorFormat			= glu::mapGLInternalFormat(m_config.colorFormat);
1270 	glu::DataType			fboSamplerType		= glu::getSampler2DType(colorFormat);
1271 	glu::DataType			fboOutputType		= getFragmentOutputType(colorFormat);
1272 	tcu::TextureFormatInfo	fboRangeInfo		= tcu::getTextureFormatInfo(colorFormat);
1273 	Vec4					fboOutScale			= fboRangeInfo.valueMax - fboRangeInfo.valueMin;
1274 	Vec4					fboOutBias			= fboRangeInfo.valueMin;
1275 
1276 	Texture2DShader			texToFboShader		(DataTypes() << glu::TYPE_SAMPLER_2D, fboOutputType);
1277 	Texture2DShader			texFromFboShader	(DataTypes() << fboSamplerType, glu::TYPE_FLOAT_VEC4);
1278 	FlatColorShader			flatShader			(fboOutputType);
1279 	deUint32				texToFboShaderID	= ctx.createProgram(&texToFboShader);
1280 	deUint32				texFromFboShaderID	= ctx.createProgram(&texFromFboShader);
1281 	deUint32				flatShaderID		= ctx.createProgram(&flatShader);
1282 
1283 	int						width				= 128;
1284 	int						height				= 128;
1285 	deUint32				metaballsTex		= 1;
1286 	deUint32				quadsTex			= 2;
1287 	bool					stencil				= (m_config.buffers & GL_STENCIL_BUFFER_BIT) != 0;
1288 
1289 	createQuadsTex2D(ctx, quadsTex, GL_RGB, GL_UNSIGNED_BYTE, 64, 64);
1290 	createMetaballsTex2D(ctx, metaballsTex, GL_RGB, GL_UNSIGNED_BYTE, 64, 64);
1291 
1292 	Framebuffer fbo(ctx, m_config, width, height);
1293 	fbo.checkCompleteness();
1294 
1295 	// Setup shaders
1296 	texToFboShader.setOutScaleBias(fboOutScale, fboOutBias);
1297 	texFromFboShader.setTexScaleBias(0, fboRangeInfo.lookupScale, fboRangeInfo.lookupBias);
1298 	texToFboShader.setUniforms	(ctx, texToFboShaderID);
1299 	texFromFboShader.setUniforms(ctx, texFromFboShaderID);
1300 	flatShader.setColor			(ctx, flatShaderID, Vec4(0.0f, 0.0f, 1.0f, 1.0f) * fboOutScale + fboOutBias);
1301 
1302 	// Draw scene
1303 	ctx.bindFramebuffer(GL_FRAMEBUFFER, fbo.getFramebuffer());
1304 	ctx.viewport(0, 0, width, height);
1305 	clearColorBuffer(ctx, colorFormat, tcu::Vec4(1.0f, 0.0f, 0.0f, 1.0f));
1306 	ctx.clear(GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
1307 
1308 	ctx.enable(GL_DEPTH_TEST);
1309 
1310 	ctx.bindTexture(GL_TEXTURE_2D, quadsTex);
1311 	sglr::drawQuad(ctx, texToFboShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
1312 
1313 	ctx.disable(GL_DEPTH_TEST);
1314 
1315 	if (stencil)
1316 	{
1317 		ctx.enable(GL_SCISSOR_TEST);
1318 		ctx.scissor(width/4, height/4, width/2, height/2);
1319 		ctx.clearStencil(1);
1320 		ctx.clear(GL_STENCIL_BUFFER_BIT);
1321 		ctx.disable(GL_SCISSOR_TEST);
1322 	}
1323 
1324 	// Recreate buffers
1325 	if (!m_rebind)
1326 		ctx.bindFramebuffer(GL_FRAMEBUFFER, 0);
1327 
1328 	DE_ASSERT((m_buffers & (GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT)) == 0 ||
1329 			  (m_buffers & (GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT)) == (m_config.buffers & (GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT)));
1330 
1331 	// Recreate.
1332 	for (int ndx = 0; ndx < 2; ndx++)
1333 	{
1334 		deUint32	bit		= ndx == 0 ? GL_COLOR_BUFFER_BIT
1335 									   : (GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
1336 		deUint32	type	= ndx == 0 ? fbo.getConfig().colorType
1337 									   : fbo.getConfig().depthStencilType;
1338 		deUint32	format	= ndx == 0 ? fbo.getConfig().colorFormat
1339 									   : fbo.getConfig().depthStencilFormat;
1340 		deUint32	buf		= ndx == 0 ? fbo.getColorBuffer()
1341 									   : fbo.getDepthStencilBuffer();
1342 
1343 		if ((m_buffers & bit) == 0)
1344 			continue;
1345 
1346 		switch (type)
1347 		{
1348 			case GL_TEXTURE_2D:
1349 				ctx.deleteTextures(1, &buf);
1350 				ctx.bindTexture(GL_TEXTURE_2D, buf);
1351 				ctx.texImage2D(GL_TEXTURE_2D, 0, format, width, height);
1352 				ctx.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
1353 				ctx.texParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
1354 				break;
1355 
1356 			case GL_RENDERBUFFER:
1357 				ctx.deleteRenderbuffers(1, &buf);
1358 				ctx.bindRenderbuffer(GL_RENDERBUFFER, buf);
1359 				ctx.renderbufferStorage(GL_RENDERBUFFER, format, width, height);
1360 				break;
1361 
1362 			default:
1363 				DE_ASSERT(false);
1364 		}
1365 	}
1366 
1367 	// Rebind.
1368 	if (m_rebind)
1369 	{
1370 		for (int ndx = 0; ndx < 3; ndx++)
1371 		{
1372 			deUint32	bit		= ndx == 0 ? GL_COLOR_BUFFER_BIT	:
1373 								  ndx == 1 ? GL_DEPTH_BUFFER_BIT	:
1374 								  ndx == 2 ? GL_STENCIL_BUFFER_BIT	: 0;
1375 			deUint32	point	= ndx == 0 ? GL_COLOR_ATTACHMENT0	:
1376 								  ndx == 1 ? GL_DEPTH_ATTACHMENT	:
1377 								  ndx == 2 ? GL_STENCIL_ATTACHMENT	: 0;
1378 			deUint32	type	= ndx == 0 ? fbo.getConfig().colorType
1379 										   : fbo.getConfig().depthStencilType;
1380 			deUint32	buf		= ndx == 0 ? fbo.getColorBuffer()
1381 										   : fbo.getDepthStencilBuffer();
1382 
1383 			if ((m_buffers & bit) == 0)
1384 				continue;
1385 
1386 			switch (type)
1387 			{
1388 				case GL_TEXTURE_2D:
1389 					ctx.framebufferTexture2D(GL_FRAMEBUFFER, point, GL_TEXTURE_2D, buf, 0);
1390 					break;
1391 
1392 				case GL_RENDERBUFFER:
1393 					ctx.framebufferRenderbuffer(GL_FRAMEBUFFER, point, GL_RENDERBUFFER, buf);
1394 					break;
1395 
1396 				default:
1397 					DE_ASSERT(false);
1398 			}
1399 		}
1400 	}
1401 
1402 	if (!m_rebind)
1403 		ctx.bindFramebuffer(GL_FRAMEBUFFER, fbo.getFramebuffer());
1404 
1405 	ctx.clearStencil(0);
1406 	ctx.clear(m_buffers & (GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT)); // \note Clear only buffers that were re-created
1407 	if (m_buffers & GL_COLOR_BUFFER_BIT)
1408 	{
1409 		// Clearing of integer buffers is undefined so do clearing by rendering flat color.
1410 		sglr::drawQuad(ctx, flatShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
1411 	}
1412 
1413 	ctx.enable(GL_DEPTH_TEST);
1414 
1415 	if (stencil)
1416 	{
1417 		// \note Stencil test enabled only if we have stencil buffer
1418 		ctx.enable(GL_STENCIL_TEST);
1419 		ctx.stencilFunc(GL_EQUAL, 0, 0xffu);
1420 	}
1421 	ctx.bindTexture(GL_TEXTURE_2D, metaballsTex);
1422 	sglr::drawQuad(ctx, texToFboShaderID, Vec3(-1.0f, -1.0f, 1.0f), Vec3(1.0f, 1.0f, -1.0f));
1423 	if (stencil)
1424 		ctx.disable(GL_STENCIL_TEST);
1425 
1426 	ctx.disable(GL_DEPTH_TEST);
1427 
1428 	if (fbo.getConfig().colorType == GL_TEXTURE_2D)
1429 	{
1430 		// Unbind fbo
1431 		ctx.bindFramebuffer(GL_FRAMEBUFFER, 0);
1432 
1433 		// Draw to screen
1434 		ctx.bindTexture(GL_TEXTURE_2D, fbo.getColorBuffer());
1435 		ctx.viewport(0, 0, ctx.getWidth(), ctx.getHeight());
1436 		sglr::drawQuad(ctx, texFromFboShaderID, Vec3(-1.0f, -1.0f, 0.0f), Vec3(1.0f, 1.0f, 0.0f));
1437 
1438 		// Read from screen
1439 		ctx.readPixels(dst, 0, 0, ctx.getWidth(), ctx.getHeight());
1440 	}
1441 	else
1442 	{
1443 		// Read from fbo
1444 		readPixels(ctx, dst, 0, 0, width, height, colorFormat, fboRangeInfo.lookupScale, fboRangeInfo.lookupBias);
1445 	}
1446 }
1447 
1448 } // FboCases
1449 
FboRenderTestGroup(Context & context)1450 FboRenderTestGroup::FboRenderTestGroup (Context& context)
1451 	: TestCaseGroup(context, "render", "Rendering Tests")
1452 {
1453 }
1454 
~FboRenderTestGroup(void)1455 FboRenderTestGroup::~FboRenderTestGroup (void)
1456 {
1457 }
1458 
init(void)1459 void FboRenderTestGroup::init (void)
1460 {
1461 	static const deUint32 objectTypes[] =
1462 	{
1463 		GL_TEXTURE_2D,
1464 		GL_RENDERBUFFER
1465 	};
1466 
1467 	enum FormatType
1468 	{
1469 		FORMATTYPE_FLOAT = 0,
1470 		FORMATTYPE_FIXED,
1471 		FORMATTYPE_INT,
1472 		FORMATTYPE_UINT,
1473 
1474 		FORMATTYPE_LAST
1475 	};
1476 
1477 	// Required by specification.
1478 	static const struct
1479 	{
1480 		deUint32	format;
1481 		FormatType	type;
1482 	} colorFormats[] =
1483 	{
1484 		{ GL_RGBA32F,			FORMATTYPE_FLOAT	},
1485 		{ GL_RGBA32I,			FORMATTYPE_INT		},
1486 		{ GL_RGBA32UI,			FORMATTYPE_UINT		},
1487 		{ GL_RGBA16F,			FORMATTYPE_FLOAT	},
1488 		{ GL_RGBA16I,			FORMATTYPE_INT		},
1489 		{ GL_RGBA16UI,			FORMATTYPE_UINT		},
1490 		{ GL_RGB16F,			FORMATTYPE_FLOAT	},
1491 		{ GL_RGBA8,				FORMATTYPE_FIXED	},
1492 		{ GL_RGBA8I,			FORMATTYPE_INT		},
1493 		{ GL_RGBA8UI,			FORMATTYPE_UINT		},
1494 		{ GL_SRGB8_ALPHA8,		FORMATTYPE_FIXED	},
1495 		{ GL_RGB10_A2,			FORMATTYPE_FIXED	},
1496 		{ GL_RGB10_A2UI,		FORMATTYPE_UINT		},
1497 		{ GL_RGBA4,				FORMATTYPE_FIXED	},
1498 		{ GL_RGB5_A1,			FORMATTYPE_FIXED	},
1499 		{ GL_RGB8,				FORMATTYPE_FIXED	},
1500 		{ GL_RGB565,			FORMATTYPE_FIXED	},
1501 		{ GL_R11F_G11F_B10F,	FORMATTYPE_FLOAT	},
1502 		{ GL_RG32F,				FORMATTYPE_FLOAT	},
1503 		{ GL_RG32I,				FORMATTYPE_INT		},
1504 		{ GL_RG32UI,			FORMATTYPE_UINT		},
1505 		{ GL_RG16F,				FORMATTYPE_FLOAT	},
1506 		{ GL_RG16I,				FORMATTYPE_INT		},
1507 		{ GL_RG16UI,			FORMATTYPE_UINT		},
1508 		{ GL_RG8,				FORMATTYPE_FLOAT	},
1509 		{ GL_RG8I,				FORMATTYPE_INT		},
1510 		{ GL_RG8UI,				FORMATTYPE_UINT		},
1511 		{ GL_R32F,				FORMATTYPE_FLOAT	},
1512 		{ GL_R32I,				FORMATTYPE_INT		},
1513 		{ GL_R32UI,				FORMATTYPE_UINT		},
1514 		{ GL_R16F,				FORMATTYPE_FLOAT	},
1515 		{ GL_R16I,				FORMATTYPE_INT		},
1516 		{ GL_R16UI,				FORMATTYPE_UINT		},
1517 		{ GL_R8,				FORMATTYPE_FLOAT	},
1518 		{ GL_R8I,				FORMATTYPE_INT		},
1519 		{ GL_R8UI,				FORMATTYPE_UINT		}
1520 	};
1521 
1522 	static const struct
1523 	{
1524 		deUint32	format;
1525 		bool		depth;
1526 		bool		stencil;
1527 	} depthStencilFormats[] =
1528 	{
1529 		{ GL_DEPTH_COMPONENT32F,	true,	false	},
1530 		{ GL_DEPTH_COMPONENT24,		true,	false	},
1531 		{ GL_DEPTH_COMPONENT16,		true,	false	},
1532 		{ GL_DEPTH32F_STENCIL8,		true,	true	},
1533 		{ GL_DEPTH24_STENCIL8,		true,	true	},
1534 		{ GL_STENCIL_INDEX8,		false,	true	}
1535 	};
1536 
1537 	using namespace FboCases;
1538 
1539 	// .stencil_clear
1540 	tcu::TestCaseGroup* stencilClearGroup = new tcu::TestCaseGroup(m_testCtx, "stencil_clear", "Stencil buffer clears");
1541 	addChild(stencilClearGroup);
1542 	for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(depthStencilFormats); fmtNdx++)
1543 	{
1544 		deUint32	colorType	= GL_TEXTURE_2D;
1545 		deUint32	stencilType	= GL_RENDERBUFFER;
1546 		deUint32	colorFmt	= GL_RGBA8;
1547 
1548 		if (!depthStencilFormats[fmtNdx].stencil)
1549 			continue;
1550 
1551 		FboConfig config(GL_COLOR_BUFFER_BIT|GL_STENCIL_BUFFER_BIT, colorType, colorFmt, stencilType, depthStencilFormats[fmtNdx].format);
1552 		stencilClearGroup->addChild(new StencilClearsTest(m_context, config));
1553 	}
1554 
1555 	// .shared_colorbuffer_clear
1556 	tcu::TestCaseGroup* sharedColorbufferClearGroup = new tcu::TestCaseGroup(m_testCtx, "shared_colorbuffer_clear", "Shader colorbuffer clears");
1557 	addChild(sharedColorbufferClearGroup);
1558 	for (int colorFmtNdx = 0; colorFmtNdx < DE_LENGTH_OF_ARRAY(colorFormats); colorFmtNdx++)
1559 	{
1560 		// Clearing of integer buffers is undefined.
1561 		if (colorFormats[colorFmtNdx].type == FORMATTYPE_INT || colorFormats[colorFmtNdx].type == FORMATTYPE_UINT)
1562 			continue;
1563 
1564 		for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(objectTypes); typeNdx++)
1565 		{
1566 			FboConfig config(GL_COLOR_BUFFER_BIT, objectTypes[typeNdx], colorFormats[colorFmtNdx].format, GL_NONE, GL_NONE);
1567 			sharedColorbufferClearGroup->addChild(new SharedColorbufferClearsTest(m_context, config));
1568 		}
1569 	}
1570 
1571 	// .shared_colorbuffer
1572 	tcu::TestCaseGroup* sharedColorbufferGroup = new tcu::TestCaseGroup(m_testCtx, "shared_colorbuffer", "Shared colorbuffer tests");
1573 	addChild(sharedColorbufferGroup);
1574 	for (int colorFmtNdx = 0; colorFmtNdx < DE_LENGTH_OF_ARRAY(colorFormats); colorFmtNdx++)
1575 	{
1576 		deUint32	depthStencilType	= GL_RENDERBUFFER;
1577 		deUint32	depthStencilFormat	= GL_DEPTH24_STENCIL8;
1578 
1579 		// Blending with integer buffers and fp32 targets is not supported.
1580 		if (colorFormats[colorFmtNdx].type == FORMATTYPE_INT	||
1581 			colorFormats[colorFmtNdx].type == FORMATTYPE_UINT	||
1582 			colorFormats[colorFmtNdx].format == GL_RGBA32F		||
1583 			colorFormats[colorFmtNdx].format == GL_RGB32F		||
1584 			colorFormats[colorFmtNdx].format == GL_RG32F		||
1585 			colorFormats[colorFmtNdx].format == GL_R32F)
1586 			continue;
1587 
1588 		for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(objectTypes); typeNdx++)
1589 		{
1590 			FboConfig colorOnlyConfig			(GL_COLOR_BUFFER_BIT,											objectTypes[typeNdx], colorFormats[colorFmtNdx].format, GL_NONE, GL_NONE);
1591 			FboConfig colorDepthConfig			(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT,						objectTypes[typeNdx], colorFormats[colorFmtNdx].format, depthStencilType, depthStencilFormat);
1592 			FboConfig colorDepthStencilConfig	(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT,	objectTypes[typeNdx], colorFormats[colorFmtNdx].format, depthStencilType, depthStencilFormat);
1593 
1594 			sharedColorbufferGroup->addChild(new SharedColorbufferTest(m_context, colorOnlyConfig));
1595 			sharedColorbufferGroup->addChild(new SharedColorbufferTest(m_context, colorDepthConfig));
1596 			sharedColorbufferGroup->addChild(new SharedColorbufferTest(m_context, colorDepthStencilConfig));
1597 		}
1598 	}
1599 
1600 	// .shared_depth_stencil
1601 	tcu::TestCaseGroup* sharedDepthStencilGroup = new tcu::TestCaseGroup(m_testCtx, "shared_depth_stencil", "Shared depth and stencil buffers");
1602 	addChild(sharedDepthStencilGroup);
1603 	for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(depthStencilFormats); fmtNdx++)
1604 	{
1605 		deUint32	colorType		= GL_TEXTURE_2D;
1606 		deUint32	colorFmt		= GL_RGBA8;
1607 		bool		depth			= depthStencilFormats[fmtNdx].depth;
1608 		bool		stencil			= depthStencilFormats[fmtNdx].stencil;
1609 
1610 		if (!depth)
1611 			continue; // Not verified.
1612 
1613 		// Depth and stencil: both rbo and textures
1614 		for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(objectTypes); typeNdx++)
1615 		{
1616 			FboConfig config(GL_COLOR_BUFFER_BIT|(depth ? GL_DEPTH_BUFFER_BIT : 0)|(stencil ? GL_STENCIL_BUFFER_BIT : 0), colorType, colorFmt, objectTypes[typeNdx], depthStencilFormats[fmtNdx].format);
1617 			sharedDepthStencilGroup->addChild(new SharedDepthStencilTest(m_context, config));
1618 		}
1619 	}
1620 
1621 	// .resize
1622 	tcu::TestCaseGroup* resizeGroup = new tcu::TestCaseGroup(m_testCtx, "resize", "FBO resize tests");
1623 	addChild(resizeGroup);
1624 	for (int colorFmtNdx = 0; colorFmtNdx < DE_LENGTH_OF_ARRAY(colorFormats); colorFmtNdx++)
1625 	{
1626 		deUint32 colorFormat = colorFormats[colorFmtNdx].format;
1627 
1628 		// Color-only.
1629 		for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(objectTypes); typeNdx++)
1630 		{
1631 			FboConfig config(GL_COLOR_BUFFER_BIT, objectTypes[typeNdx], colorFormat, GL_NONE, GL_NONE);
1632 			resizeGroup->addChild(new ResizeTest(m_context, config));
1633 		}
1634 
1635 		// For selected color formats tests depth & stencil variants.
1636 		if (colorFormat == GL_RGBA8 || colorFormat == GL_RGBA16F)
1637 		{
1638 			for (int depthStencilFmtNdx = 0; depthStencilFmtNdx < DE_LENGTH_OF_ARRAY(depthStencilFormats); depthStencilFmtNdx++)
1639 			{
1640 				deUint32	colorType		= GL_TEXTURE_2D;
1641 				bool		depth			= depthStencilFormats[depthStencilFmtNdx].depth;
1642 				bool		stencil			= depthStencilFormats[depthStencilFmtNdx].stencil;
1643 
1644 				// Depth and stencil: both rbo and textures
1645 				for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(objectTypes); typeNdx++)
1646 				{
1647 					if (!depth && objectTypes[typeNdx] != GL_RENDERBUFFER)
1648 						continue; // Not supported.
1649 
1650 					FboConfig config(GL_COLOR_BUFFER_BIT|(depth ? GL_DEPTH_BUFFER_BIT : 0)|(stencil ? GL_STENCIL_BUFFER_BIT : 0),
1651 									 colorType, colorFormat,
1652 									 objectTypes[typeNdx], depthStencilFormats[depthStencilFmtNdx].format);
1653 					resizeGroup->addChild(new ResizeTest(m_context, config));
1654 				}
1655 			}
1656 		}
1657 	}
1658 
1659 	// .recreate_color
1660 	tcu::TestCaseGroup* recreateColorGroup = new tcu::TestCaseGroup(m_testCtx, "recreate_color", "Recreate colorbuffer tests");
1661 	addChild(recreateColorGroup);
1662 	for (int colorFmtNdx = 0; colorFmtNdx < DE_LENGTH_OF_ARRAY(colorFormats); colorFmtNdx++)
1663 	{
1664 		deUint32	colorFormat			= colorFormats[colorFmtNdx].format;
1665 		deUint32	depthStencilFormat	= GL_DEPTH24_STENCIL8;
1666 		deUint32	depthStencilType	= GL_RENDERBUFFER;
1667 
1668 		// Color-only.
1669 		for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(objectTypes); typeNdx++)
1670 		{
1671 			FboConfig config(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT, objectTypes[typeNdx], colorFormat, depthStencilType, depthStencilFormat);
1672 			recreateColorGroup->addChild(new RecreateBuffersTest(m_context, config, GL_COLOR_BUFFER_BIT, true /* rebind */));
1673 		}
1674 	}
1675 
1676 	// .recreate_depth_stencil
1677 	tcu::TestCaseGroup* recreateDepthStencilGroup = new tcu::TestCaseGroup(m_testCtx, "recreate_depth_stencil", "Recreate depth and stencil buffers");
1678 	addChild(recreateDepthStencilGroup);
1679 	for (int fmtNdx = 0; fmtNdx < DE_LENGTH_OF_ARRAY(depthStencilFormats); fmtNdx++)
1680 	{
1681 		deUint32	colorType		= GL_TEXTURE_2D;
1682 		deUint32	colorFmt		= GL_RGBA8;
1683 		bool		depth			= depthStencilFormats[fmtNdx].depth;
1684 		bool		stencil			= depthStencilFormats[fmtNdx].stencil;
1685 
1686 		// Depth and stencil: both rbo and textures
1687 		for (int typeNdx = 0; typeNdx < DE_LENGTH_OF_ARRAY(objectTypes); typeNdx++)
1688 		{
1689 			if (!depth && objectTypes[typeNdx] != GL_RENDERBUFFER)
1690 				continue;
1691 
1692 			FboConfig config(GL_COLOR_BUFFER_BIT|(depth ? GL_DEPTH_BUFFER_BIT : 0)|(stencil ? GL_STENCIL_BUFFER_BIT : 0), colorType, colorFmt, objectTypes[typeNdx], depthStencilFormats[fmtNdx].format);
1693 			recreateDepthStencilGroup->addChild(new RecreateBuffersTest(m_context, config, (depth ? GL_DEPTH_BUFFER_BIT : 0)|(stencil ? GL_STENCIL_BUFFER_BIT : 0), true /* rebind */));
1694 		}
1695 	}
1696 }
1697 
1698 } // Functional
1699 } // gles3
1700 } // deqp
1701