• 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 Rbo state query tests.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es3fRboStateQueryTests.hpp"
25 #include "glsStateQueryUtil.hpp"
26 #include "es3fApiCase.hpp"
27 #include "gluRenderContext.hpp"
28 #include "glwEnums.hpp"
29 #include "glwFunctions.hpp"
30 #include "deRandom.hpp"
31 #include "deMath.h"
32 
33 using namespace glw; // GLint and other GL types
34 using deqp::gls::StateQueryUtil::StateQueryMemoryWriteGuard;
35 
36 
37 namespace deqp
38 {
39 namespace gles3
40 {
41 namespace Functional
42 {
43 namespace
44 {
45 
checkRenderbufferComponentSize(tcu::TestContext & testCtx,glu::CallLogWrapper & gl,int r,int g,int b,int a,int d,int s)46 void checkRenderbufferComponentSize (tcu::TestContext& testCtx, glu::CallLogWrapper& gl, int r, int g, int b, int a, int d, int s)
47 {
48 	using tcu::TestLog;
49 
50 	const int referenceSizes[] = {r, g, b, a, d, s};
51 	const GLenum paramNames[] =
52 	{
53 		GL_RENDERBUFFER_RED_SIZE,
54 		GL_RENDERBUFFER_GREEN_SIZE,
55 		GL_RENDERBUFFER_BLUE_SIZE,
56 		GL_RENDERBUFFER_ALPHA_SIZE,
57 		GL_RENDERBUFFER_DEPTH_SIZE,
58 		GL_RENDERBUFFER_STENCIL_SIZE
59 	};
60 
61 	DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(referenceSizes) == DE_LENGTH_OF_ARRAY(paramNames));
62 
63 	for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(referenceSizes); ++ndx)
64 	{
65 		if (referenceSizes[ndx] == -1)
66 			continue;
67 
68 		StateQueryMemoryWriteGuard<GLint> state;
69 		gl.glGetRenderbufferParameteriv(GL_RENDERBUFFER, paramNames[ndx], &state);
70 
71 		if (!state.verifyValidity(testCtx))
72 			return;
73 
74 		if (state < referenceSizes[ndx])
75 		{
76 			testCtx.getLog() << TestLog::Message << "// ERROR: Expected greater or equal to " << referenceSizes[ndx] << "; got " << state << TestLog::EndMessage;
77 			if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
78 				testCtx.setTestResult(QP_TEST_RESULT_FAIL, "got invalid value");
79 		}
80 	}
81 }
82 
checkIntEquals(tcu::TestContext & testCtx,GLint got,GLint expected)83 void checkIntEquals (tcu::TestContext& testCtx, GLint got, GLint expected)
84 {
85 	using tcu::TestLog;
86 
87 	if (got != expected)
88 	{
89 		testCtx.getLog() << TestLog::Message << "// ERROR: Expected " << expected << "; got " << got << TestLog::EndMessage;
90 		if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
91 			testCtx.setTestResult(QP_TEST_RESULT_FAIL, "got invalid value");
92 	}
93 }
94 
checkIntGreaterOrEqual(tcu::TestContext & testCtx,GLint got,GLint expected)95 void checkIntGreaterOrEqual (tcu::TestContext& testCtx, GLint got, GLint expected)
96 {
97 	using tcu::TestLog;
98 
99 	if (got < expected)
100 	{
101 		testCtx.getLog() << TestLog::Message << "// ERROR: Expected greater or equal to " << expected << "; got " << got << TestLog::EndMessage;
102 		if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
103 			testCtx.setTestResult(QP_TEST_RESULT_FAIL, "got invalid value");
104 	}
105 }
106 
checkRenderbufferParam(tcu::TestContext & testCtx,glu::CallLogWrapper & gl,GLenum pname,GLenum reference)107 void checkRenderbufferParam (tcu::TestContext& testCtx, glu::CallLogWrapper& gl, GLenum pname, GLenum reference)
108 {
109 	StateQueryMemoryWriteGuard<GLint> state;
110 	gl.glGetRenderbufferParameteriv(GL_RENDERBUFFER, pname, &state);
111 
112 	if (state.verifyValidity(testCtx))
113 		checkIntEquals(testCtx, state, reference);
114 }
115 
checkRenderbufferParamGreaterOrEqual(tcu::TestContext & testCtx,glu::CallLogWrapper & gl,GLenum pname,GLenum reference)116 void checkRenderbufferParamGreaterOrEqual (tcu::TestContext& testCtx, glu::CallLogWrapper& gl, GLenum pname, GLenum reference)
117 {
118 	StateQueryMemoryWriteGuard<GLint> state;
119 	gl.glGetRenderbufferParameteriv(GL_RENDERBUFFER, pname, &state);
120 
121 	if (state.verifyValidity(testCtx))
122 		checkIntGreaterOrEqual(testCtx, state, reference);
123 }
124 
125 class RboSizeCase : public ApiCase
126 {
127 public:
RboSizeCase(Context & context,const char * name,const char * description)128 	RboSizeCase (Context& context, const char* name, const char* description)
129 		: ApiCase(context, name, description)
130 	{
131 	}
132 
test(void)133 	void test (void)
134 	{
135 		de::Random rnd(0xabcdef);
136 
137 		GLuint renderbufferID = 0;
138 		glGenRenderbuffers(1, &renderbufferID);
139 		glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
140 		expectError(GL_NO_ERROR);
141 
142 		checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_WIDTH,		0);
143 		checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_HEIGHT,	0);
144 		expectError(GL_NO_ERROR);
145 
146 		const int numIterations = 60;
147 		for (int i = 0; i < numIterations; ++i)
148 		{
149 			const GLint w = rnd.getInt(0, 128);
150 			const GLint h = rnd.getInt(0, 128);
151 
152 			glRenderbufferStorage(GL_RENDERBUFFER, GL_RGB8, w, h);
153 
154 			checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_WIDTH,		w);
155 			checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_HEIGHT,	h);
156 		}
157 
158 		glDeleteRenderbuffers(1, &renderbufferID);
159 	}
160 };
161 
162 class RboInternalFormatCase : public ApiCase
163 {
164 public:
RboInternalFormatCase(Context & context,const char * name,const char * description)165 	RboInternalFormatCase (Context& context, const char* name, const char* description)
166 		: ApiCase(context, name, description)
167 	{
168 	}
169 
test(void)170 	void test (void)
171 	{
172 		GLuint renderbufferID = 0;
173 		glGenRenderbuffers(1, &renderbufferID);
174 		glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
175 		expectError(GL_NO_ERROR);
176 
177 		const glu::ContextType& contextType = m_context.getRenderContext().getType();
178 		const bool isCoreGL45 = glu::contextSupports(contextType, glu::ApiType::core(4, 5));
179 
180 		GLenum initialValue = isCoreGL45 ? GL_RGBA : GL_RGBA4;
181 		checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_INTERNAL_FORMAT, initialValue);
182 		expectError(GL_NO_ERROR);
183 
184 		const GLenum requiredColorformats[] =
185 		{
186 			GL_R8, GL_RG8, GL_RGB8, GL_RGB565, GL_RGBA4, GL_RGB5_A1, GL_RGBA8, GL_RGB10_A2,
187 			GL_RGB10_A2UI, GL_SRGB8_ALPHA8, GL_R8I, GL_R8UI, GL_R16I, GL_R16UI, GL_R32I, GL_R32UI,
188 			GL_RG8I, GL_RG8UI, GL_RG16I, GL_RG16UI, GL_RG32I, GL_RG32UI, GL_RGBA8I, GL_RGBA8UI,
189 			GL_RGBA16I, GL_RGBA16UI, GL_RGBA32I, GL_RGBA32UI
190 		};
191 
192 		for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(requiredColorformats); ++ndx)
193 		{
194 			glRenderbufferStorage(GL_RENDERBUFFER, requiredColorformats[ndx], 128, 128);
195 			expectError(GL_NO_ERROR);
196 
197 			checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_INTERNAL_FORMAT, requiredColorformats[ndx]);
198 		}
199 
200 		glDeleteRenderbuffers(1, &renderbufferID);
201 	}
202 };
203 
204 class RboComponentSizeColorCase : public ApiCase
205 {
206 public:
RboComponentSizeColorCase(Context & context,const char * name,const char * description)207 	RboComponentSizeColorCase (Context& context, const char* name, const char* description)
208 		: ApiCase(context, name, description)
209 	{
210 	}
211 
test(void)212 	void test (void)
213 	{
214 		GLuint renderbufferID = 0;
215 		glGenRenderbuffers(1, &renderbufferID);
216 		glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
217 		expectError(GL_NO_ERROR);
218 
219 		checkRenderbufferComponentSize(m_testCtx, *this, 0, 0, 0, 0, 0, 0);
220 		expectError(GL_NO_ERROR);
221 
222 		const struct ColorFormat
223 		{
224 			GLenum	internalFormat;
225 			int		bitsR, bitsG, bitsB, bitsA;
226 		} requiredColorFormats[] =
227 		{
228 			{ GL_R8,			8,	0,	0,	0	},
229 			{ GL_RG8,			8,	8,	0,	0	},
230 			{ GL_RGB8,			8,	8,	8,	0	},
231 			{ GL_RGB565,		5,	6,	5,	0	},
232 			{ GL_RGBA4,			4,	4,	4,	4	},
233 			{ GL_RGB5_A1,		5,	5,	5,	1	},
234 			{ GL_RGBA8,			8,	8,	8,	8	},
235 			{ GL_RGB10_A2,		10, 10, 10, 2	},
236 			{ GL_RGB10_A2UI,	10, 10, 10, 2	},
237 			{ GL_SRGB8_ALPHA8,	8,	8,	8,	8	},
238 			{ GL_R8I,			8,	0,	0,	0	},
239 			{ GL_R8UI,			8,	0,	0,	0	},
240 			{ GL_R16I,			16, 0,	0,	0	},
241 			{ GL_R16UI,			16, 0,	0,	0	},
242 			{ GL_R32I,			32, 0,	0,	0	},
243 			{ GL_R32UI,			32, 0,	0,	0	},
244 			{ GL_RG8I,			8,	8,	0,	0	},
245 			{ GL_RG8UI,			8,	8,	0,	0	},
246 			{ GL_RG16I,			16, 16, 0,	0	},
247 			{ GL_RG16UI,		16, 16, 0,	0	},
248 			{ GL_RG32I,			32, 32, 0,	0	},
249 			{ GL_RG32UI,		32, 32, 0,	0	},
250 			{ GL_RGBA8I,		8,	8,	8,	8	},
251 			{ GL_RGBA8UI,		8,	8,	8,	8	},
252 			{ GL_RGBA16I,		16, 16, 16, 16	},
253 			{ GL_RGBA16UI,		16, 16, 16, 16	},
254 			{ GL_RGBA32I,		32, 32, 32, 32	},
255 			{ GL_RGBA32UI,		32, 32, 32, 32	}
256 		};
257 
258 		for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(requiredColorFormats); ++ndx)
259 		{
260 			glRenderbufferStorage(GL_RENDERBUFFER, requiredColorFormats[ndx].internalFormat, 128, 128);
261 			expectError(GL_NO_ERROR);
262 
263 			checkRenderbufferComponentSize(m_testCtx, *this, requiredColorFormats[ndx].bitsR, requiredColorFormats[ndx].bitsG, requiredColorFormats[ndx].bitsB, requiredColorFormats[ndx].bitsA, -1, -1);
264 		}
265 
266 		glDeleteRenderbuffers(1, &renderbufferID);
267 	}
268 };
269 
270 class RboComponentSizeDepthCase : public ApiCase
271 {
272 public:
RboComponentSizeDepthCase(Context & context,const char * name,const char * description)273 	RboComponentSizeDepthCase (Context& context, const char* name, const char* description)
274 		: ApiCase(context, name, description)
275 	{
276 	}
277 
test(void)278 	void test (void)
279 	{
280 		using tcu::TestLog;
281 
282 		GLuint renderbufferID = 0;
283 		glGenRenderbuffers(1, &renderbufferID);
284 		glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
285 		expectError(GL_NO_ERROR);
286 
287 		const struct DepthFormat
288 		{
289 			GLenum	internalFormat;
290 			int		dbits;
291 			int		sbits;
292 		} requiredDepthFormats[] =
293 		{
294 			{ GL_DEPTH_COMPONENT16,		16, 0 },
295 			{ GL_DEPTH_COMPONENT24,		24, 0 },
296 			{ GL_DEPTH_COMPONENT32F,	32, 0 },
297 			{ GL_DEPTH24_STENCIL8,		24, 8 },
298 			{ GL_DEPTH32F_STENCIL8,		32, 8 },
299 		};
300 
301 		for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(requiredDepthFormats); ++ndx)
302 		{
303 			glRenderbufferStorage(GL_RENDERBUFFER, requiredDepthFormats[ndx].internalFormat, 128, 128);
304 			expectError(GL_NO_ERROR);
305 
306 			checkRenderbufferComponentSize(m_testCtx, *this, -1, -1, -1, -1, requiredDepthFormats[ndx].dbits, requiredDepthFormats[ndx].sbits);
307 		}
308 
309 		// STENCIL_INDEX8 is required, in that case sBits >= 8
310 		{
311 			glRenderbufferStorage(GL_RENDERBUFFER, GL_STENCIL_INDEX8, 128, 128);
312 			expectError(GL_NO_ERROR);
313 
314 			StateQueryMemoryWriteGuard<GLint> state;
315 			glGetRenderbufferParameteriv(GL_RENDERBUFFER, GL_RENDERBUFFER_STENCIL_SIZE, &state);
316 
317 			if (state.verifyValidity(m_testCtx) && state < 8)
318 			{
319 				m_testCtx.getLog() << TestLog::Message << "// ERROR: Expected greater or equal to 8; got " << state << TestLog::EndMessage;
320 				if (m_testCtx.getTestResult() == QP_TEST_RESULT_PASS)
321 					m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "got invalid value");
322 			}
323 		}
324 
325 		glDeleteRenderbuffers(1, &renderbufferID);
326 	}
327 };
328 
329 class RboSamplesCase : public ApiCase
330 {
331 public:
RboSamplesCase(Context & context,const char * name,const char * description)332 	RboSamplesCase (Context& context, const char* name, const char* description)
333 		: ApiCase(context, name, description)
334 	{
335 	}
336 
test(void)337 	void test (void)
338 	{
339 		GLuint renderbufferID = 0;
340 		glGenRenderbuffers(1, &renderbufferID);
341 		glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
342 		expectError(GL_NO_ERROR);
343 
344 		checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_SAMPLES, 0);
345 		expectError(GL_NO_ERROR);
346 
347 		StateQueryMemoryWriteGuard<GLint> max_samples;
348 		glGetIntegerv(GL_MAX_SAMPLES, &max_samples);
349 		if (!max_samples.verifyValidity(m_testCtx))
350 			return;
351 
352 		// 0 samples is a special case
353 		{
354 			glRenderbufferStorageMultisample(GL_RENDERBUFFER, 0, GL_RGBA8, 128, 128);
355 			expectError(GL_NO_ERROR);
356 
357 			checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_SAMPLES, 0);
358 		}
359 
360 		// test [1, n] samples
361 		for (int samples = 1; samples <= max_samples; ++samples)
362 		{
363 			glRenderbufferStorageMultisample(GL_RENDERBUFFER, samples, GL_RGBA8, 128, 128);
364 			expectError(GL_NO_ERROR);
365 
366 			checkRenderbufferParamGreaterOrEqual(m_testCtx, *this, GL_RENDERBUFFER_SAMPLES, samples);
367 		}
368 
369 		glDeleteRenderbuffers(1, &renderbufferID);
370 	}
371 };
372 
373 } // anonymous
374 
375 
RboStateQueryTests(Context & context)376 RboStateQueryTests::RboStateQueryTests (Context& context)
377 	: TestCaseGroup(context, "rbo", "Rbo State Query tests")
378 {
379 }
380 
init(void)381 void RboStateQueryTests::init (void)
382 {
383 	addChild(new RboSizeCase				(m_context, "renderbuffer_size",					"RENDERBUFFER_WIDTH and RENDERBUFFER_HEIGHT"));
384 	addChild(new RboInternalFormatCase		(m_context, "renderbuffer_internal_format",			"RENDERBUFFER_INTERNAL_FORMAT"));
385 	addChild(new RboComponentSizeColorCase	(m_context, "renderbuffer_component_size_color",	"RENDERBUFFER_x_SIZE"));
386 	addChild(new RboComponentSizeDepthCase	(m_context, "renderbuffer_component_size_depth",	"RENDERBUFFER_x_SIZE"));
387 	addChild(new RboSamplesCase				(m_context, "renderbuffer_samples",					"RENDERBUFFER_SAMPLES"));
388 }
389 
390 } // Functional
391 } // gles3
392 } // deqp
393