1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program OpenGL ES 2.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 Depth & stencil tests.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es2fDepthStencilTests.hpp"
25 #include "glsFragmentOpUtil.hpp"
26 #include "gluPixelTransfer.hpp"
27 #include "gluStrUtil.hpp"
28 #include "tcuSurface.hpp"
29 #include "tcuRenderTarget.hpp"
30 #include "tcuTestLog.hpp"
31 #include "tcuImageCompare.hpp"
32 #include "tcuCommandLine.hpp"
33 #include "tcuTextureUtil.hpp"
34 #include "deRandom.hpp"
35 #include "deStringUtil.hpp"
36 #include "deMemory.h"
37 #include "deString.h"
38 #include "rrFragmentOperations.hpp"
39 #include "sglrReferenceUtils.hpp"
40 
41 #include <algorithm>
42 #include <sstream>
43 
44 #include "glw.h"
45 
46 namespace deqp
47 {
48 namespace gles2
49 {
50 namespace Functional
51 {
52 
53 using std::vector;
54 using tcu::IVec2;
55 using tcu::Vec2;
56 using tcu::Vec4;
57 using tcu::TestLog;
58 using std::ostringstream;
59 
60 enum
61 {
62 	VIEWPORT_WIDTH			= 4*3*4,
63 	VIEWPORT_HEIGHT			= 4*12,
64 
65 	NUM_RANDOM_CASES		= 25,
66 	NUM_RANDOM_SUB_CASES	= 10
67 };
68 
69 namespace DepthStencilCaseUtil
70 {
71 
72 struct StencilParams
73 {
74 	deUint32	function;
75 	int			reference;
76 	deUint32	compareMask;
77 
78 	deUint32	stencilFailOp;
79 	deUint32	depthFailOp;
80 	deUint32	depthPassOp;
81 
82 	deUint32	writeMask;
83 
StencilParamsdeqp::gles2::Functional::DepthStencilCaseUtil::StencilParams84 	StencilParams (void)
85 		: function		(0)
86 		, reference		(0)
87 		, compareMask	(0)
88 		, stencilFailOp	(0)
89 		, depthFailOp	(0)
90 		, depthPassOp	(0)
91 		, writeMask		(0)
92 	{
93 	}
94 };
95 
96 struct DepthStencilParams
97 {
98 	rr::FaceType	visibleFace;			//!< Quad visible face.
99 
100 	bool			stencilTestEnabled;
101 	StencilParams	stencil[rr::FACETYPE_LAST];
102 
103 	bool			depthTestEnabled;
104 	deUint32		depthFunc;
105 	float			depth;
106 	bool			depthWriteMask;
107 
DepthStencilParamsdeqp::gles2::Functional::DepthStencilCaseUtil::DepthStencilParams108 	DepthStencilParams (void)
109 		: visibleFace			(rr::FACETYPE_LAST)
110 		, stencilTestEnabled	(false)
111 		, depthTestEnabled		(false)
112 		, depthFunc				(0)
113 		, depth					(0.0f)
114 		, depthWriteMask		(false)
115 	{
116 	}
117 };
118 
operator <<(tcu::TestLog & log,const StencilParams & params)119 tcu::TestLog& operator<< (tcu::TestLog& log, const StencilParams& params)
120 {
121 	log << TestLog::Message << "  func = " << glu::getCompareFuncStr(params.function) << "\n"
122 							<< "  ref = " << params.reference << "\n"
123 							<< "  compare mask = " << tcu::toHex(params.compareMask) << "\n"
124 							<< "  stencil fail = " << glu::getStencilOpStr(params.stencilFailOp) << "\n"
125 							<< "  depth fail = " << glu::getStencilOpStr(params.depthFailOp) << "\n"
126 							<< "  depth pass = " << glu::getStencilOpStr(params.depthPassOp) << "\n"
127 							<< "  write mask = " << tcu::toHex(params.writeMask) << "\n"
128 		<< TestLog::EndMessage;
129 	return log;
130 }
131 
operator <<(tcu::TestLog & log,const DepthStencilParams & params)132 tcu::TestLog& operator<< (tcu::TestLog& log, const DepthStencilParams& params)
133 {
134 	log << TestLog::Message << "Stencil test: " << (params.stencilTestEnabled ? "enabled" : "disabled") << TestLog::EndMessage;
135 	if (params.stencilTestEnabled)
136 	{
137 		log << TestLog::Message << "Front-face stencil state: " << TestLog::EndMessage;
138 		log << params.stencil[rr::FACETYPE_FRONT];
139 
140 		log << TestLog::Message << "Back-face stencil state: " << TestLog::EndMessage;
141 		log << params.stencil[rr::FACETYPE_BACK];
142 	}
143 
144 	log << TestLog::Message << "Depth test: " << (params.depthTestEnabled ? "enabled" : "disabled") << TestLog::EndMessage;
145 	if (params.depthTestEnabled)
146 	{
147 		log << TestLog::Message << "  func = " << glu::getCompareFuncStr(params.depthFunc) << "\n"
148 								   "  depth value = " << params.depth << "\n"
149 								   "  write mask = " << (params.depthWriteMask ? "true" : "false") << "\n"
150 			<< TestLog::EndMessage;
151 	}
152 
153 	log << TestLog::Message << "Triangles are " << (params.visibleFace == rr::FACETYPE_FRONT ? "front" : "back") << "-facing" << TestLog::EndMessage;
154 
155 	return log;
156 }
157 
158 struct ClearCommand
159 {
160 	rr::WindowRectangle	rect;
161 	deUint32			buffers;
162 	tcu::Vec4			color;
163 	int					stencil;
164 	// \note No depth here - don't use clears for setting depth values; use quad rendering instead. Cleared depths are in [0, 1] to begin with,
165 	//		 whereas rendered depths are given in [-1, 1] and then mapped to [0, 1]; this discrepancy could cause precision issues in depth tests.
166 
ClearCommanddeqp::gles2::Functional::DepthStencilCaseUtil::ClearCommand167 	ClearCommand (void)
168 		: rect		(0, 0, 0, 0)
169 		, buffers	(0)
170 		, stencil	(0)
171 	{
172 	}
173 
ClearCommanddeqp::gles2::Functional::DepthStencilCaseUtil::ClearCommand174 	ClearCommand (const rr::WindowRectangle&	rect_,
175 				  deUint32					buffers_,
176 				  const tcu::Vec4&			color_,
177 				  int						stencil_)
178 		: rect		(rect_)
179 		, buffers	(buffers_)
180 		, color		(color_)
181 		, stencil	(stencil_)
182 	{
183 	}
184 };
185 
186 struct RenderCommand
187 {
188 	DepthStencilParams		params;
189 	rr::WindowRectangle		rect;
190 	tcu::Vec4				color;
191 	tcu::BVec4				colorMask;
192 
RenderCommanddeqp::gles2::Functional::DepthStencilCaseUtil::RenderCommand193 	RenderCommand (void)
194 		: rect(0, 0, 0, 0)
195 	{
196 	}
197 };
198 
199 struct RefRenderCommand
200 {
201 	gls::FragmentOpUtil::IntegerQuad	quad;
202 	rr::FragmentOperationState			state;
203 };
204 
205 struct TestRenderTarget
206 {
207 	int		width;
208 	int		height;
209 	int		depthBits;
210 	int		stencilBits;
211 
TestRenderTargetdeqp::gles2::Functional::DepthStencilCaseUtil::TestRenderTarget212 	TestRenderTarget (int width_,
213 					  int height_,
214 					  int depthBits_,
215 					  int stencilBits_)
216 		: width			(width_)
217 		, height		(height_)
218 		, depthBits		(depthBits_)
219 		, stencilBits	(stencilBits_)
220 	{
221 	}
222 
TestRenderTargetdeqp::gles2::Functional::DepthStencilCaseUtil::TestRenderTarget223 	TestRenderTarget (void)
224 		: width			(0)
225 		, height		(0)
226 		, depthBits		(0)
227 		, stencilBits	(0)
228 	{
229 	}
230 };
231 
getStencilTestValues(int stencilBits,int numValues,int * values)232 void getStencilTestValues (int stencilBits, int numValues, int* values)
233 {
234 	int numLowest		= numValues/2;
235 	int	numHighest		= numValues-numLowest;
236 	int	maxVal			= (1<<stencilBits)-1;
237 
238 	for (int ndx = 0; ndx < numLowest; ndx++)
239 		values[ndx] = ndx;
240 
241 	for (int ndx = 0; ndx < numHighest; ndx++)
242 		values[numValues-ndx-1] = maxVal-ndx;
243 }
244 
generateBaseClearAndDepthCommands(const TestRenderTarget & target,vector<ClearCommand> & clearCommands,vector<RenderCommand> & renderCommands)245 void generateBaseClearAndDepthCommands (const TestRenderTarget& target, vector<ClearCommand>& clearCommands, vector<RenderCommand>& renderCommands)
246 {
247 	DE_ASSERT(clearCommands.empty());
248 	DE_ASSERT(renderCommands.empty());
249 
250 	const int		numL0CellsX		= 4;
251 	const int		numL0CellsY		= 4;
252 	const int		numL1CellsX		= 3;
253 	const int		numL1CellsY		= 1;
254 	int				cellL0Width		= target.width/numL0CellsX;
255 	int				cellL0Height	= target.height/numL0CellsY;
256 	int				cellL1Width		= cellL0Width/numL1CellsX;
257 	int				cellL1Height	= cellL0Height/numL1CellsY;
258 
259 	int				stencilValues[numL0CellsX*numL0CellsY];
260 	float			depthValues[numL1CellsX*numL1CellsY];
261 
262 	if (cellL0Width <= 0 || cellL1Width <= 0 || cellL0Height <= 0 || cellL1Height <= 0)
263 		throw tcu::NotSupportedError("Too small render target");
264 
265 	// Fullscreen clear to black.
266 	clearCommands.push_back(ClearCommand(rr::WindowRectangle(0, 0, target.width, target.height), GL_COLOR_BUFFER_BIT, Vec4(0.0f, 0.0f, 0.0f, 1.0f), 0));
267 
268 	// Compute stencil values: numL0CellsX*numL0CellsY combinations of lowest and highest bits.
269 	getStencilTestValues(target.stencilBits, numL0CellsX*numL0CellsY, &stencilValues[0]);
270 
271 	// Compute depth values
272 	{
273 		int		numValues		= DE_LENGTH_OF_ARRAY(depthValues);
274 		float	depthStep		= 2.0f/(float)(numValues-1);
275 
276 		for (int ndx = 0; ndx < numValues; ndx++)
277 			depthValues[ndx] = -1.0f + depthStep*(float)ndx;
278 	}
279 
280 	for (int y0 = 0; y0 < numL0CellsY; y0++)
281 	{
282 		for (int x0 = 0; x0 < numL0CellsX; x0++)
283 		{
284 			int stencilValue = stencilValues[y0*numL0CellsX + x0];
285 
286 			for (int y1 = 0; y1 < numL1CellsY; y1++)
287 			{
288 				for (int x1 = 0; x1 < numL1CellsX; x1++)
289 				{
290 					int					x			= x0*cellL0Width + x1*cellL1Width;
291 					int					y			= y0*cellL0Height + y1*cellL1Height;
292 					rr::WindowRectangle	cellL1Rect	(x, y, cellL1Width, cellL1Height);
293 
294 					clearCommands.push_back(ClearCommand(cellL1Rect, GL_STENCIL_BUFFER_BIT, Vec4(0), stencilValue));
295 
296 					RenderCommand renderCmd;
297 					renderCmd.params.visibleFace		= rr::FACETYPE_FRONT;
298 					renderCmd.params.depth				= depthValues[y1*numL1CellsX + x1];;
299 					renderCmd.params.depthTestEnabled	= true;
300 					renderCmd.params.depthFunc			= GL_ALWAYS;
301 					renderCmd.params.depthWriteMask		= true;
302 					renderCmd.colorMask					= tcu::BVec4(false);
303 					renderCmd.rect						= cellL1Rect;
304 
305 					renderCommands.push_back(renderCmd);
306 				}
307 			}
308 		}
309 	}
310 }
311 
generateDepthVisualizeCommands(const TestRenderTarget & target,vector<RenderCommand> & commands)312 void generateDepthVisualizeCommands (const TestRenderTarget& target, vector<RenderCommand>& commands)
313 {
314 	const float			epsilon			= -0.05f;
315 	static const float	depthSteps[]	= {-1.0f, -0.5f, 0.0f, 0.5f, 1.0f};
316 	int					numSteps		= DE_LENGTH_OF_ARRAY(depthSteps);
317 	const float			colorStep		= 1.0f / (float)(numSteps-1);
318 
319 	for (int ndx = 0; ndx < numSteps; ndx++)
320 	{
321 		RenderCommand cmd;
322 
323 		cmd.params.visibleFace		= rr::FACETYPE_FRONT;
324 		cmd.rect					= rr::WindowRectangle(0, 0, target.width, target.height);
325 		cmd.color					= Vec4(0.0f, 0.0f, colorStep*(float)ndx, 0.0f);
326 		cmd.colorMask				= tcu::BVec4(false, false, true, false);
327 		cmd.params.depth			= depthSteps[ndx]+epsilon;
328 		cmd.params.depthTestEnabled	= true;
329 		cmd.params.depthFunc		= GL_LESS;
330 		cmd.params.depthWriteMask	= false;
331 
332 		commands.push_back(cmd);
333 	}
334 }
335 
generateStencilVisualizeCommands(const TestRenderTarget & target,vector<RenderCommand> & commands)336 void generateStencilVisualizeCommands (const TestRenderTarget& target, vector<RenderCommand>& commands)
337 {
338 	const int	numValues		= 4*4;
339 	float		colorStep		= 1.0f / numValues; // 0 is reserved for non-matching.
340 	int			stencilValues[numValues];
341 
342 	getStencilTestValues(target.stencilBits, numValues, &stencilValues[0]);
343 
344 	for (int ndx = 0; ndx < numValues; ndx++)
345 	{
346 		RenderCommand cmd;
347 
348 		cmd.params.visibleFace							= rr::FACETYPE_FRONT;
349 		cmd.rect										= rr::WindowRectangle(0, 0, target.width, target.height);
350 		cmd.color										= Vec4(0.0f, colorStep*float(ndx+1), 0.0f, 0.0f);
351 		cmd.colorMask									= tcu::BVec4(false, true, false, false);
352 		cmd.params.stencilTestEnabled					= true;
353 
354 		cmd.params.stencil[rr::FACETYPE_FRONT].function			= GL_EQUAL;
355 		cmd.params.stencil[rr::FACETYPE_FRONT].reference		= stencilValues[ndx];
356 		cmd.params.stencil[rr::FACETYPE_FRONT].compareMask		= ~0u;
357 		cmd.params.stencil[rr::FACETYPE_FRONT].stencilFailOp	= GL_KEEP;
358 		cmd.params.stencil[rr::FACETYPE_FRONT].depthFailOp		= GL_KEEP;
359 		cmd.params.stencil[rr::FACETYPE_FRONT].depthPassOp		= GL_KEEP;
360 		cmd.params.stencil[rr::FACETYPE_FRONT].writeMask		= 0u;
361 
362 		cmd.params.stencil[rr::FACETYPE_BACK] = cmd.params.stencil[rr::FACETYPE_FRONT];
363 
364 		commands.push_back(cmd);
365 	}
366 }
367 
translateStencilState(const StencilParams & src,rr::StencilState & dst)368 void translateStencilState (const StencilParams& src, rr::StencilState& dst)
369 {
370 	dst.func		= sglr::rr_util::mapGLTestFunc(src.function);
371 	dst.ref			= src.reference;
372 	dst.compMask	= src.compareMask;
373 	dst.sFail		= sglr::rr_util::mapGLStencilOp(src.stencilFailOp);
374 	dst.dpFail		= sglr::rr_util::mapGLStencilOp(src.depthFailOp);
375 	dst.dpPass		= sglr::rr_util::mapGLStencilOp(src.depthPassOp);
376 	dst.writeMask	= src.writeMask;
377 }
378 
translateCommand(const RenderCommand & src,RefRenderCommand & dst,const TestRenderTarget & renderTarget)379 void translateCommand (const RenderCommand& src, RefRenderCommand& dst, const TestRenderTarget& renderTarget)
380 {
381 	const float		far				= 1.0f;
382 	const float		near			= 0.0f;
383 	bool			hasDepth		= renderTarget.depthBits > 0;
384 	bool			hasStencil		= renderTarget.stencilBits > 0;
385 	bool			isFrontFacing	= src.params.visibleFace == rr::FACETYPE_FRONT;
386 
387 	dst.quad.posA = IVec2(isFrontFacing ? src.rect.left : (src.rect.left+src.rect.width-1), src.rect.bottom);
388 	dst.quad.posB = IVec2(isFrontFacing ? (src.rect.left+src.rect.width-1) : src.rect.left, src.rect.bottom+src.rect.height-1);
389 
390 	std::fill(DE_ARRAY_BEGIN(dst.quad.color), DE_ARRAY_END(dst.quad.color), src.color);
391 	std::fill(DE_ARRAY_BEGIN(dst.quad.depth), DE_ARRAY_END(dst.quad.depth), ((far-near)/2.0f) * src.params.depth + (near+far)/2.0f);
392 
393 	dst.state.colorMask = src.colorMask;
394 
395 	dst.state.scissorTestEnabled		= false;
396 	dst.state.stencilTestEnabled		= hasStencil && src.params.stencilTestEnabled;
397 	dst.state.depthTestEnabled			= hasDepth && src.params.depthTestEnabled;
398 	dst.state.blendMode					= rr::BLENDMODE_NONE;
399 	dst.state.numStencilBits			= renderTarget.stencilBits;
400 
401 	if (dst.state.depthTestEnabled)
402 	{
403 		dst.state.depthFunc					= sglr::rr_util::mapGLTestFunc(src.params.depthFunc);
404 		dst.state.depthMask					= src.params.depthWriteMask;
405 	}
406 
407 	if (dst.state.stencilTestEnabled)
408 	{
409 		translateStencilState(src.params.stencil[rr::FACETYPE_BACK],	dst.state.stencilStates[rr::FACETYPE_BACK]);
410 		translateStencilState(src.params.stencil[rr::FACETYPE_FRONT],	dst.state.stencilStates[rr::FACETYPE_FRONT]);
411 	}
412 }
413 
render(const vector<ClearCommand> & clears,int viewportX,int viewportY)414 void render (const vector<ClearCommand>& clears, int viewportX, int viewportY)
415 {
416 	glEnable(GL_SCISSOR_TEST);
417 
418 	for (int ndx = 0; ndx < (int)clears.size(); ndx++)
419 	{
420 		const ClearCommand& clear = clears[ndx];
421 
422 		if (clear.buffers & GL_COLOR_BUFFER_BIT)	glClearColor(clear.color.x(), clear.color.y(), clear.color.z(), clear.color.w());
423 		if (clear.buffers & GL_STENCIL_BUFFER_BIT)	glClearStencil(clear.stencil);
424 
425 		DE_ASSERT(clear.buffers == (clear.buffers & (GL_COLOR_BUFFER_BIT|GL_STENCIL_BUFFER_BIT))); // \note Don't use clear for depths.
426 
427 		glScissor(clear.rect.left+viewportX, clear.rect.bottom+viewportY, clear.rect.width, clear.rect.height);
428 		glClear(clear.buffers);
429 	}
430 
431 	glDisable(GL_SCISSOR_TEST);
432 }
433 
render(gls::FragmentOpUtil::QuadRenderer & renderer,const RenderCommand & command,int viewportX,int viewportY)434 void render (gls::FragmentOpUtil::QuadRenderer& renderer, const RenderCommand& command, int viewportX, int viewportY)
435 {
436 	if (command.params.stencilTestEnabled)
437 	{
438 		glEnable(GL_STENCIL_TEST);
439 
440 		for (int face = 0; face < rr::FACETYPE_LAST; face++)
441 		{
442 			deUint32				glFace	= face == rr::FACETYPE_BACK ? GL_BACK : GL_FRONT;
443 			const StencilParams&	sParams	= command.params.stencil[face];
444 
445 			glStencilFuncSeparate(glFace, sParams.function, sParams.reference, sParams.compareMask);
446 			glStencilOpSeparate(glFace, sParams.stencilFailOp, sParams.depthFailOp, sParams.depthPassOp);
447 			glStencilMaskSeparate(glFace, sParams.writeMask);
448 		}
449 	}
450 	else
451 		glDisable(GL_STENCIL_TEST);
452 
453 	if (command.params.depthTestEnabled)
454 	{
455 		glEnable(GL_DEPTH_TEST);
456 		glDepthFunc(command.params.depthFunc);
457 		glDepthMask(command.params.depthWriteMask ? GL_TRUE : GL_FALSE);
458 	}
459 	else
460 		glDisable(GL_DEPTH_TEST);
461 
462 	glColorMask(command.colorMask[0] ? GL_TRUE : GL_FALSE,
463 				command.colorMask[1] ? GL_TRUE : GL_FALSE,
464 				command.colorMask[2] ? GL_TRUE : GL_FALSE,
465 				command.colorMask[3] ? GL_TRUE : GL_FALSE);
466 	glViewport(command.rect.left+viewportX, command.rect.bottom+viewportY, command.rect.width, command.rect.height);
467 
468 	gls::FragmentOpUtil::Quad quad;
469 
470 	bool isFrontFacing = command.params.visibleFace == rr::FACETYPE_FRONT;
471 	quad.posA = Vec2(isFrontFacing ? -1.0f :  1.0f, -1.0f);
472 	quad.posB = Vec2(isFrontFacing ?  1.0f : -1.0f,  1.0f);
473 
474 	std::fill(DE_ARRAY_BEGIN(quad.color), DE_ARRAY_END(quad.color), command.color);
475 	std::fill(DE_ARRAY_BEGIN(quad.depth), DE_ARRAY_END(quad.depth), command.params.depth);
476 
477 	renderer.render(quad);
478 	GLU_CHECK();
479 }
480 
renderReference(const vector<ClearCommand> & clears,const tcu::PixelBufferAccess & dstColor,const tcu::PixelBufferAccess & dstStencil,int stencilBits)481 void renderReference (const vector<ClearCommand>& clears, const tcu::PixelBufferAccess& dstColor, const tcu::PixelBufferAccess& dstStencil, int stencilBits)
482 {
483 	for (int ndx = 0; ndx < (int)clears.size(); ndx++)
484 	{
485 		const ClearCommand& clear = clears[ndx];
486 
487 		if (clear.buffers & GL_COLOR_BUFFER_BIT)
488 			tcu::clear(tcu::getSubregion(dstColor, clear.rect.left, clear.rect.bottom, clear.rect.width, clear.rect.height), clear.color);
489 
490 		if (clear.buffers & GL_STENCIL_BUFFER_BIT && stencilBits > 0)
491 		{
492 			int maskedVal = clear.stencil & ((1<<stencilBits)-1);
493 			tcu::clearStencil(tcu::getSubregion(dstStencil, clear.rect.left, clear.rect.bottom, clear.rect.width, clear.rect.height), maskedVal);
494 		}
495 
496 		DE_ASSERT(clear.buffers == (clear.buffers & (GL_COLOR_BUFFER_BIT|GL_STENCIL_BUFFER_BIT))); // \note Don't use clear for depths.
497 	}
498 }
499 
500 } // DepthStencilCaseUtil
501 
502 using namespace DepthStencilCaseUtil;
503 
504 class DepthStencilCase : public TestCase
505 {
506 public:
507 							DepthStencilCase		(Context& context, const char* name, const char* desc, const std::vector<DepthStencilParams>& cases);
508 							~DepthStencilCase		(void);
509 
510 	void					init					(void);
511 	void					deinit					(void);
512 
513 	IterateResult			iterate					(void);
514 
515 private:
516 							DepthStencilCase		(const DepthStencilCase& other);
517 	DepthStencilCase&		operator=				(const DepthStencilCase& other);
518 
519 	std::vector<DepthStencilParams>					m_cases;
520 
521 	TestRenderTarget								m_renderTarget;
522 	std::vector<ClearCommand>						m_baseClears;
523 	std::vector<RenderCommand>						m_baseDepthRenders;
524 	std::vector<RenderCommand>						m_visualizeCommands;
525 	std::vector<RefRenderCommand>					m_refBaseDepthRenders;
526 	std::vector<RefRenderCommand>					m_refVisualizeCommands;
527 
528 	gls::FragmentOpUtil::QuadRenderer*				m_renderer;
529 	tcu::Surface*									m_refColorBuffer;
530 	tcu::TextureLevel*								m_refDepthBuffer;
531 	tcu::TextureLevel*								m_refStencilBuffer;
532 	gls::FragmentOpUtil::ReferenceQuadRenderer*		m_refRenderer;
533 
534 	int												m_iterNdx;
535 };
536 
DepthStencilCase(Context & context,const char * name,const char * desc,const std::vector<DepthStencilParams> & cases)537 DepthStencilCase::DepthStencilCase (Context& context, const char* name, const char* desc, const std::vector<DepthStencilParams>& cases)
538 	: TestCase				(context, name, desc)
539 	, m_cases				(cases)
540 	, m_renderer			(DE_NULL)
541 	, m_refColorBuffer		(DE_NULL)
542 	, m_refDepthBuffer		(DE_NULL)
543 	, m_refStencilBuffer	(DE_NULL)
544 	, m_refRenderer			(DE_NULL)
545 	, m_iterNdx				(0)
546 {
547 }
548 
~DepthStencilCase(void)549 DepthStencilCase::~DepthStencilCase (void)
550 {
551 	delete m_renderer;
552 	delete m_refColorBuffer;
553 	delete m_refDepthBuffer;
554 	delete m_refStencilBuffer;
555 	delete m_refRenderer;
556 }
557 
init(void)558 void DepthStencilCase::init (void)
559 {
560 	DE_ASSERT(!m_renderer && !m_refColorBuffer && !m_refDepthBuffer && !m_refStencilBuffer && !m_refRenderer);
561 
562 	// Compute render target.
563 	int viewportW	= de::min<int>(m_context.getRenderTarget().getWidth(), VIEWPORT_WIDTH);
564 	int viewportH	= de::min<int>(m_context.getRenderTarget().getHeight(), VIEWPORT_HEIGHT);
565 	m_renderTarget	= TestRenderTarget(viewportW, viewportH, m_context.getRenderTarget().getDepthBits(), m_context.getRenderTarget().getStencilBits());
566 
567 	// Compute base clears & visualization commands.
568 	generateBaseClearAndDepthCommands(m_renderTarget, m_baseClears, m_baseDepthRenders);
569 	generateDepthVisualizeCommands(m_renderTarget, m_visualizeCommands);
570 	generateStencilVisualizeCommands(m_renderTarget, m_visualizeCommands);
571 
572 	// Translate to ref commands.
573 	m_refBaseDepthRenders.resize(m_baseDepthRenders.size());
574 	for (int ndx = 0; ndx < (int)m_baseDepthRenders.size(); ndx++)
575 		translateCommand(m_baseDepthRenders[ndx], m_refBaseDepthRenders[ndx], m_renderTarget);
576 
577 	m_refVisualizeCommands.resize(m_visualizeCommands.size());
578 	for (int ndx = 0; ndx < (int)m_visualizeCommands.size(); ndx++)
579 		translateCommand(m_visualizeCommands[ndx], m_refVisualizeCommands[ndx], m_renderTarget);
580 
581 	m_renderer			= new gls::FragmentOpUtil::QuadRenderer(m_context.getRenderContext(), glu::GLSL_VERSION_100_ES);
582 	m_refColorBuffer	= new tcu::Surface(viewportW, viewportH);
583 	m_refDepthBuffer	= new tcu::TextureLevel(tcu::TextureFormat(tcu::TextureFormat::D, tcu::TextureFormat::FLOAT),			viewportW, viewportH);
584 	m_refStencilBuffer	= new tcu::TextureLevel(tcu::TextureFormat(tcu::TextureFormat::S, tcu::TextureFormat::UNSIGNED_INT32),	viewportW, viewportH);
585 	m_refRenderer		= new gls::FragmentOpUtil::ReferenceQuadRenderer();
586 
587 	m_iterNdx			= 0;
588 	m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
589 }
590 
deinit(void)591 void DepthStencilCase::deinit (void)
592 {
593 	delete m_renderer;
594 	delete m_refColorBuffer;
595 	delete m_refDepthBuffer;
596 	delete m_refStencilBuffer;
597 	delete m_refRenderer;
598 
599 	m_renderer			= DE_NULL;
600 	m_refColorBuffer	= DE_NULL;
601 	m_refDepthBuffer	= DE_NULL;
602 	m_refStencilBuffer	= DE_NULL;
603 	m_refRenderer		= DE_NULL;
604 
605 	m_baseClears.clear();
606 	m_baseDepthRenders.clear();
607 	m_visualizeCommands.clear();
608 	m_refBaseDepthRenders.clear();
609 	m_refVisualizeCommands.clear();
610 }
611 
iterate(void)612 DepthStencilCase::IterateResult DepthStencilCase::iterate (void)
613 {
614 	de::Random				rnd				(deStringHash(getName()) ^ deInt32Hash(m_iterNdx));
615 	int						viewportX		= rnd.getInt(0, m_context.getRenderTarget().getWidth()-m_renderTarget.width);
616 	int						viewportY		= rnd.getInt(0, m_context.getRenderTarget().getHeight()-m_renderTarget.height);
617 	RenderCommand			testCmd;
618 
619 	tcu::Surface			renderedImg		(m_renderTarget.width, m_renderTarget.height);
620 	tcu::RGBA				threshold		= m_context.getRenderTarget().getPixelFormat().getColorThreshold();
621 
622 	// Fill in test command for this iteration.
623 	testCmd.color		= Vec4(1.0f, 0.0f, 0.0f, 1.0f);
624 	testCmd.colorMask	= tcu::BVec4(true);
625 	testCmd.rect		= rr::WindowRectangle(0, 0, m_renderTarget.width, m_renderTarget.height);
626 	testCmd.params		= m_cases[m_iterNdx];
627 
628 	if (m_iterNdx == 0)
629 	{
630 		m_testCtx.getLog() << TestLog::Message << "Channels:\n"
631 													"  RED: passing pixels\n"
632 													"  GREEN: stencil values\n"
633 													"  BLUE: depth values"
634 							<< TestLog::EndMessage;
635 	}
636 
637 	if (m_cases.size() > 1)
638 		m_testCtx.getLog() << TestLog::Message << "Iteration " << m_iterNdx << "..." << TestLog::EndMessage;
639 
640 	m_testCtx.getLog() << m_cases[m_iterNdx];
641 
642 	// Submit render commands to gl GL.
643 
644 	// Base clears.
645 	render(m_baseClears, viewportX, viewportY);
646 
647 	// Base depths.
648 	for (vector<RenderCommand>::const_iterator cmd = m_baseDepthRenders.begin(); cmd != m_baseDepthRenders.end(); ++cmd)
649 		render(*m_renderer, *cmd, viewportX, viewportY);
650 
651 	// Test command.
652 	render(*m_renderer, testCmd, viewportX, viewportY);
653 
654 	// Visualization commands.
655 	for (vector<RenderCommand>::const_iterator cmd = m_visualizeCommands.begin(); cmd != m_visualizeCommands.end(); ++cmd)
656 		render(*m_renderer, *cmd, viewportX, viewportY);
657 
658 	// Re-enable all write masks.
659 	glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE);
660 	glDepthMask(GL_TRUE);
661 	glStencilMask(~0u);
662 
663 	// Ask GPU to start rendering.
664 	glFlush();
665 
666 	// Render reference while GPU is doing work.
667 	{
668 		RefRenderCommand refTestCmd;
669 		translateCommand(testCmd, refTestCmd, m_renderTarget);
670 
671 		// Base clears.
672 		renderReference(m_baseClears, m_refColorBuffer->getAccess(), m_refStencilBuffer->getAccess(), m_renderTarget.stencilBits);
673 
674 		// Base depths.
675 		for (vector<RefRenderCommand>::const_iterator cmd = m_refBaseDepthRenders.begin(); cmd != m_refBaseDepthRenders.end(); ++cmd)
676 			m_refRenderer->render(gls::FragmentOpUtil::getMultisampleAccess(m_refColorBuffer->getAccess()),
677 								  gls::FragmentOpUtil::getMultisampleAccess(m_refDepthBuffer->getAccess()),
678 								  gls::FragmentOpUtil::getMultisampleAccess(m_refStencilBuffer->getAccess()),
679 								  cmd->quad, cmd->state);
680 
681 		// Test command.
682 		m_refRenderer->render(gls::FragmentOpUtil::getMultisampleAccess(m_refColorBuffer->getAccess()),
683 							  gls::FragmentOpUtil::getMultisampleAccess(m_refDepthBuffer->getAccess()),
684 							  gls::FragmentOpUtil::getMultisampleAccess(m_refStencilBuffer->getAccess()),
685 							  refTestCmd.quad, refTestCmd.state);
686 
687 		// Visualization commands.
688 		for (vector<RefRenderCommand>::const_iterator cmd = m_refVisualizeCommands.begin(); cmd != m_refVisualizeCommands.end(); ++cmd)
689 			m_refRenderer->render(gls::FragmentOpUtil::getMultisampleAccess(m_refColorBuffer->getAccess()),
690 								  gls::FragmentOpUtil::getMultisampleAccess(m_refDepthBuffer->getAccess()),
691 								  gls::FragmentOpUtil::getMultisampleAccess(m_refStencilBuffer->getAccess()),
692 								  cmd->quad, cmd->state);
693 	}
694 
695 	// Read rendered image.
696 	glu::readPixels(m_context.getRenderContext(), viewportX, viewportY, renderedImg.getAccess());
697 
698 	m_iterNdx += 1;
699 
700 	// Compare to reference.
701 	bool	isLastIter	= m_iterNdx >= (int)m_cases.size();
702 	bool	compareOk	= tcu::pixelThresholdCompare(m_testCtx.getLog(), "CompareResult", "Image Comparison Result", *m_refColorBuffer, renderedImg, threshold,
703 													 tcu::COMPARE_LOG_RESULT);
704 
705 	m_testCtx.getLog() << TestLog::Message << (compareOk ? "  Passed." : "  FAILED!") << TestLog::EndMessage;
706 	if (!compareOk)
707 		m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Image comparison failed");
708 
709 	if (compareOk && !isLastIter)
710 		return CONTINUE;
711 	else
712 		return STOP;
713 }
714 
DepthStencilTests(Context & context)715 DepthStencilTests::DepthStencilTests (Context& context)
716 	: TestCaseGroup(context, "depth_stencil", "Depth and Stencil Op Tests")
717 {
718 }
719 
~DepthStencilTests(void)720 DepthStencilTests::~DepthStencilTests (void)
721 {
722 }
723 
randomDepthStencilState(de::Random & rnd,DepthStencilParams & params)724 static void randomDepthStencilState (de::Random& rnd, DepthStencilParams& params)
725 {
726 	const float stencilTestProbability	= 0.8f;
727 	const float depthTestProbability	= 0.7f;
728 
729 	static const deUint32 compareFuncs[] =
730 	{
731 		GL_NEVER,
732 		GL_ALWAYS,
733 		GL_LESS,
734 		GL_LEQUAL,
735 		GL_EQUAL,
736 		GL_GEQUAL,
737 		GL_GREATER,
738 		GL_NOTEQUAL
739 	};
740 
741 	static const deUint32 stencilOps[] =
742 	{
743 		GL_KEEP,
744 		GL_ZERO,
745 		GL_REPLACE,
746 		GL_INCR,
747 		GL_DECR,
748 		GL_INVERT,
749 		GL_INCR_WRAP,
750 		GL_DECR_WRAP
751 	};
752 
753 	static const float depthValues[] = { -1.0f, -0.8f, -0.6f, -0.4f, -0.2f, 0.0f, 0.2f, 0.4f, 0.6f, 0.8f, 1.0f };
754 
755 	params.visibleFace			= rnd.getBool() ? rr::FACETYPE_FRONT : rr::FACETYPE_BACK;
756 	params.stencilTestEnabled	= rnd.getFloat() < stencilTestProbability;
757 	params.depthTestEnabled		= !params.stencilTestEnabled || (rnd.getFloat() < depthTestProbability);
758 
759 	if (params.stencilTestEnabled)
760 	{
761 		for (int ndx = 0; ndx < 2; ndx++)
762 		{
763 			params.stencil[ndx].function		= rnd.choose<deUint32>(DE_ARRAY_BEGIN(compareFuncs), DE_ARRAY_END(compareFuncs));
764 			params.stencil[ndx].reference		= rnd.getInt(-2, 260);
765 			params.stencil[ndx].compareMask		= rnd.getUint32();
766 			params.stencil[ndx].stencilFailOp	= rnd.choose<deUint32>(DE_ARRAY_BEGIN(stencilOps), DE_ARRAY_END(stencilOps));
767 			params.stencil[ndx].depthFailOp		= rnd.choose<deUint32>(DE_ARRAY_BEGIN(stencilOps), DE_ARRAY_END(stencilOps));
768 			params.stencil[ndx].depthPassOp		= rnd.choose<deUint32>(DE_ARRAY_BEGIN(stencilOps), DE_ARRAY_END(stencilOps));
769 			params.stencil[ndx].writeMask		= rnd.getUint32();
770 		}
771 	}
772 
773 	if (params.depthTestEnabled)
774 	{
775 		params.depthFunc		= rnd.choose<deUint32>(DE_ARRAY_BEGIN(compareFuncs), DE_ARRAY_END(compareFuncs));
776 		params.depth			= rnd.choose<float>(DE_ARRAY_BEGIN(depthValues), DE_ARRAY_END(depthValues));
777 		params.depthWriteMask	= rnd.getBool();
778 	}
779 }
780 
init(void)781 void DepthStencilTests::init (void)
782 {
783 	static const struct
784 	{
785 		const char*	name;
786 		deUint32	func;
787 	} compareFuncs[] =
788 	{
789 		{ "never",		GL_NEVER	},
790 		{ "always",		GL_ALWAYS	},
791 		{ "less",		GL_LESS		},
792 		{ "lequal",		GL_LEQUAL	},
793 		{ "equal",		GL_EQUAL	},
794 		{ "gequal",		GL_GEQUAL	},
795 		{ "greater",	GL_GREATER	},
796 		{ "notequal",	GL_NOTEQUAL	}
797 	};
798 
799 	static const struct
800 	{
801 		const char*	name;
802 		deUint32	op;
803 	} stencilOps[] =
804 	{
805 		{ "keep",		GL_KEEP			},
806 		{ "zero",		GL_ZERO			},
807 		{ "replace",	GL_REPLACE		},
808 		{ "incr",		GL_INCR			},
809 		{ "decr",		GL_DECR			},
810 		{ "invert",		GL_INVERT		},
811 		{ "incr_wrap",	GL_INCR_WRAP	},
812 		{ "decr_wrap",	GL_DECR_WRAP	}
813 	};
814 
815 	static const struct
816 	{
817 		rr::FaceType	visibleFace;
818 		deUint32		sFail;
819 		deUint32		dFail;
820 		deUint32		dPass;
821 		int				stencilRef;
822 		deUint32		compareMask;
823 		deUint32		writeMask;
824 		float			depth;
825 	} functionCases[] =
826 	{
827 		{ rr::FACETYPE_BACK,	GL_DECR,		GL_INCR,	GL_INVERT,		4,	~0u, ~0u,	-0.7f },
828 		{ rr::FACETYPE_FRONT,	GL_DECR,		GL_INCR,	GL_INVERT,		2,	~0u, ~0u,	0.0f },
829 		{ rr::FACETYPE_BACK,	GL_DECR,		GL_INCR,	GL_INVERT,		1,	~0u, ~0u,	0.2f },
830 		{ rr::FACETYPE_FRONT,	GL_DECR_WRAP,	GL_INVERT,	GL_REPLACE,		4,	~0u, ~0u,	1.0f }
831 	};
832 
833 	// All combinations of depth stencil functions.
834 	{
835 		tcu::TestCaseGroup* functionsGroup = new tcu::TestCaseGroup(m_testCtx, "stencil_depth_funcs", "Combinations of Depth and Stencil Functions");
836 		addChild(functionsGroup);
837 
838 		for (int stencilFunc = 0; stencilFunc < DE_LENGTH_OF_ARRAY(compareFuncs)+1; stencilFunc++)
839 		{
840 			// One extra: depth test disabled.
841 			for (int depthFunc = 0; depthFunc < DE_LENGTH_OF_ARRAY(compareFuncs)+1; depthFunc++)
842 			{
843 				DepthStencilParams	params;
844 				ostringstream		name;
845 				bool				hasStencilFunc	= de::inBounds(stencilFunc, 0, DE_LENGTH_OF_ARRAY(compareFuncs));
846 				bool				hasDepthFunc	= de::inBounds(depthFunc, 0, DE_LENGTH_OF_ARRAY(compareFuncs));
847 
848 				if (hasStencilFunc)
849 					name << "stencil_" << compareFuncs[stencilFunc].name << "_";
850 				else
851 					name << "no_stencil_";
852 
853 				if (hasDepthFunc)
854 					name << "depth_" << compareFuncs[depthFunc].name;
855 				else
856 					name << "no_depth";
857 
858 				params.depthFunc			= hasDepthFunc ? compareFuncs[depthFunc].func : 0;
859 				params.depthTestEnabled		= hasDepthFunc;
860 				params.depthWriteMask		= true;
861 
862 				params.stencilTestEnabled	= hasStencilFunc;
863 
864 				vector<DepthStencilParams> cases;
865 				for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(functionCases); ndx++)
866 				{
867 					rr::FaceType	visible		= functionCases[ndx].visibleFace;
868 					rr::FaceType	notVisible	= visible == rr::FACETYPE_FRONT ? rr::FACETYPE_BACK : rr::FACETYPE_FRONT;
869 
870 					params.depth								= functionCases[ndx].depth;
871 					params.visibleFace							= visible;
872 
873 					params.stencil[visible].function			= hasStencilFunc ? compareFuncs[stencilFunc].func : 0;
874 					params.stencil[visible].reference			= functionCases[ndx].stencilRef;
875 					params.stencil[visible].stencilFailOp		= functionCases[ndx].sFail;
876 					params.stencil[visible].depthFailOp			= functionCases[ndx].dFail;
877 					params.stencil[visible].depthPassOp			= functionCases[ndx].dPass;
878 					params.stencil[visible].compareMask			= functionCases[ndx].compareMask;
879 					params.stencil[visible].writeMask			= functionCases[ndx].writeMask;
880 
881 					params.stencil[notVisible].function			= GL_ALWAYS;
882 					params.stencil[notVisible].reference		= 0;
883 					params.stencil[notVisible].stencilFailOp	= GL_REPLACE;
884 					params.stencil[notVisible].depthFailOp		= GL_REPLACE;
885 					params.stencil[notVisible].depthPassOp		= GL_REPLACE;
886 					params.stencil[notVisible].compareMask		= 0u;
887 					params.stencil[notVisible].writeMask		= ~0u;
888 
889 
890 					cases.push_back(params);
891 				}
892 
893 				functionsGroup->addChild(new DepthStencilCase(m_context, name.str().c_str(), "", cases));
894 			}
895 		}
896 	}
897 
898 	static const struct
899 	{
900 		rr::FaceType	visibleFace;
901 		deUint32		func;
902 		int				ref;
903 		deUint32		compareMask;
904 		deUint32		writeMask;
905 	} opCombinationCases[] =
906 	{
907 		{ rr::FACETYPE_BACK,	GL_LESS,		4,		~0u,	~0u	},
908 		{ rr::FACETYPE_FRONT,	GL_GREATER,		2,		~0u,	~0u	},
909 		{ rr::FACETYPE_BACK,	GL_EQUAL,		3,		~2u,	~0u	},
910 		{ rr::FACETYPE_FRONT,	GL_NOTEQUAL,	1,		~0u,	~1u	}
911 	};
912 
913 	// All combinations of stencil ops.
914 	{
915 		tcu::TestCaseGroup* opCombinationGroup = new tcu::TestCaseGroup(m_testCtx, "stencil_ops", "Stencil Op Combinations");
916 		addChild(opCombinationGroup);
917 
918 		for (int sFail = 0; sFail < DE_LENGTH_OF_ARRAY(stencilOps); sFail++)
919 		{
920 			for (int dFail = 0; dFail < DE_LENGTH_OF_ARRAY(stencilOps); dFail++)
921 			{
922 				for (int dPass = 0; dPass < DE_LENGTH_OF_ARRAY(stencilOps); dPass++)
923 				{
924 					DepthStencilParams	params;
925 					ostringstream		name;
926 
927 					name << stencilOps[sFail].name << "_" << stencilOps[dFail].name << "_" << stencilOps[dPass].name;
928 
929 					params.depthFunc			= GL_LEQUAL;
930 					params.depth				= 0.0f;
931 					params.depthTestEnabled		= true;
932 					params.depthWriteMask		= true;
933 
934 					params.stencilTestEnabled	= true;
935 
936 					vector<DepthStencilParams> cases;
937 					for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(opCombinationCases); ndx++)
938 					{
939 						rr::FaceType	visible		= opCombinationCases[ndx].visibleFace;
940 						rr::FaceType	notVisible	= visible == rr::FACETYPE_FRONT ? rr::FACETYPE_BACK : rr::FACETYPE_FRONT;
941 
942 						params.visibleFace							= visible;
943 
944 						params.stencil[visible].function			= opCombinationCases[ndx].func;
945 						params.stencil[visible].reference			= opCombinationCases[ndx].ref;
946 						params.stencil[visible].stencilFailOp		= stencilOps[sFail].op;
947 						params.stencil[visible].depthFailOp			= stencilOps[dFail].op;
948 						params.stencil[visible].depthPassOp			= stencilOps[dPass].op;
949 						params.stencil[visible].compareMask			= opCombinationCases[ndx].compareMask;
950 						params.stencil[visible].writeMask			= opCombinationCases[ndx].writeMask;
951 
952 						params.stencil[notVisible].function			= GL_ALWAYS;
953 						params.stencil[notVisible].reference		= 0;
954 						params.stencil[notVisible].stencilFailOp	= GL_REPLACE;
955 						params.stencil[notVisible].depthFailOp		= GL_REPLACE;
956 						params.stencil[notVisible].depthPassOp		= GL_REPLACE;
957 						params.stencil[notVisible].compareMask		= 0u;
958 						params.stencil[notVisible].writeMask		= ~0u;
959 
960 
961 						cases.push_back(params);
962 					}
963 
964 					opCombinationGroup->addChild(new DepthStencilCase(m_context, name.str().c_str(), "", cases));
965 				}
966 			}
967 		}
968 	}
969 
970 	// Write masks
971 	{
972 		tcu::TestCaseGroup* writeMaskGroup = new tcu::TestCaseGroup(m_testCtx, "write_mask", "Depth and Stencil Write Masks");
973 		addChild(writeMaskGroup);
974 
975 		// Depth mask
976 		{
977 			DepthStencilParams params;
978 
979 			params.depthFunc			= GL_LEQUAL;
980 			params.depth				= 0.0f;
981 			params.depthTestEnabled		= true;
982 			params.stencilTestEnabled	= true;
983 
984 			params.stencil[rr::FACETYPE_FRONT].function			= GL_NOTEQUAL;
985 			params.stencil[rr::FACETYPE_FRONT].reference		= 1;
986 			params.stencil[rr::FACETYPE_FRONT].stencilFailOp	= GL_INVERT;
987 			params.stencil[rr::FACETYPE_FRONT].depthFailOp		= GL_INCR;
988 			params.stencil[rr::FACETYPE_FRONT].depthPassOp		= GL_DECR;
989 			params.stencil[rr::FACETYPE_FRONT].compareMask		= ~0u;
990 			params.stencil[rr::FACETYPE_FRONT].writeMask		= ~0u;
991 
992 			params.stencil[rr::FACETYPE_BACK].function			= GL_ALWAYS;
993 			params.stencil[rr::FACETYPE_BACK].reference			= 0;
994 			params.stencil[rr::FACETYPE_BACK].stencilFailOp		= GL_REPLACE;
995 			params.stencil[rr::FACETYPE_BACK].depthFailOp		= GL_INVERT;
996 			params.stencil[rr::FACETYPE_BACK].depthPassOp		= GL_INCR;
997 			params.stencil[rr::FACETYPE_BACK].compareMask		= ~0u;
998 			params.stencil[rr::FACETYPE_BACK].writeMask			= ~0u;
999 
1000 			vector<DepthStencilParams> cases;
1001 
1002 			// Case 1: front, depth write enabled
1003 			params.visibleFace		= rr::FACETYPE_FRONT;
1004 			params.depthWriteMask	= true;
1005 			cases.push_back(params);
1006 
1007 			// Case 2: front, depth write disabled
1008 			params.visibleFace		= rr::FACETYPE_FRONT;
1009 			params.depthWriteMask	= false;
1010 			cases.push_back(params);
1011 
1012 			// Case 3: back, depth write enabled
1013 			params.visibleFace		= rr::FACETYPE_BACK;
1014 			params.depthWriteMask	= true;
1015 			cases.push_back(params);
1016 
1017 			// Case 4: back, depth write disabled
1018 			params.visibleFace		= rr::FACETYPE_BACK;
1019 			params.depthWriteMask	= false;
1020 			cases.push_back(params);
1021 
1022 			writeMaskGroup->addChild(new DepthStencilCase(m_context, "depth", "Depth Write Mask", cases));
1023 		}
1024 
1025 		// Stencil write masks.
1026 		{
1027 			static const struct
1028 			{
1029 				rr::FaceType	visibleFace;
1030 				deUint32		frontWriteMask;
1031 				deUint32		backWriteMask;
1032 			} stencilWmaskCases[] =
1033 			{
1034 				{ rr::FACETYPE_FRONT,	~0u,	0u		},
1035 				{ rr::FACETYPE_FRONT,	0u,		~0u		},
1036 				{ rr::FACETYPE_FRONT,	0xfu,	0xf0u	},
1037 				{ rr::FACETYPE_FRONT,	0x2u,	0x4u	},
1038 				{ rr::FACETYPE_BACK,	0u,		~0u		},
1039 				{ rr::FACETYPE_BACK,	~0u,	0u		},
1040 				{ rr::FACETYPE_BACK,	0xf0u,	0xfu	},
1041 				{ rr::FACETYPE_BACK,	0x4u,	0x2u	}
1042 			};
1043 
1044 			DepthStencilParams params;
1045 
1046 			params.depthFunc			= GL_LEQUAL;
1047 			params.depth				= 0.0f;
1048 			params.depthTestEnabled		= true;
1049 			params.depthWriteMask		= true;
1050 			params.stencilTestEnabled	= true;
1051 
1052 			params.stencil[rr::FACETYPE_FRONT].function			= GL_NOTEQUAL;
1053 			params.stencil[rr::FACETYPE_FRONT].reference		= 1;
1054 			params.stencil[rr::FACETYPE_FRONT].stencilFailOp	= GL_INVERT;
1055 			params.stencil[rr::FACETYPE_FRONT].depthFailOp		= GL_INCR;
1056 			params.stencil[rr::FACETYPE_FRONT].depthPassOp		= GL_DECR;
1057 			params.stencil[rr::FACETYPE_FRONT].compareMask		= ~0u;
1058 
1059 			params.stencil[rr::FACETYPE_BACK].function			= GL_ALWAYS;
1060 			params.stencil[rr::FACETYPE_BACK].reference			= 0;
1061 			params.stencil[rr::FACETYPE_BACK].stencilFailOp		= GL_REPLACE;
1062 			params.stencil[rr::FACETYPE_BACK].depthFailOp		= GL_INVERT;
1063 			params.stencil[rr::FACETYPE_BACK].depthPassOp		= GL_INCR;
1064 			params.stencil[rr::FACETYPE_BACK].compareMask		= ~0u;
1065 
1066 			vector<DepthStencilParams> cases;
1067 			for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(stencilWmaskCases); ndx++)
1068 			{
1069 				params.visibleFace								= stencilWmaskCases[ndx].visibleFace;
1070 				params.stencil[rr::FACETYPE_FRONT].writeMask	= stencilWmaskCases[ndx].frontWriteMask;
1071 				params.stencil[rr::FACETYPE_BACK].writeMask		= stencilWmaskCases[ndx].backWriteMask;
1072 				cases.push_back(params);
1073 			}
1074 
1075 			writeMaskGroup->addChild(new DepthStencilCase(m_context, "stencil", "Stencil Write Mask", cases));
1076 		}
1077 
1078 		// Depth & stencil write masks.
1079 		{
1080 			static const struct
1081 			{
1082 				bool			depthWriteMask;
1083 				rr::FaceType	visibleFace;
1084 				deUint32		frontWriteMask;
1085 				deUint32		backWriteMask;
1086 			} depthStencilWmaskCases[] =
1087 			{
1088 				{ false,	rr::FACETYPE_FRONT,		~0u,	0u		},
1089 				{ false,	rr::FACETYPE_FRONT,		0u,		~0u		},
1090 				{ false,	rr::FACETYPE_FRONT,		0xfu,	0xf0u	},
1091 				{ true,		rr::FACETYPE_FRONT,		~0u,	0u		},
1092 				{ true,		rr::FACETYPE_FRONT,		0u,		~0u		},
1093 				{ true,		rr::FACETYPE_FRONT,		0xfu,	0xf0u	},
1094 				{ false,	rr::FACETYPE_BACK,		0u,		~0u		},
1095 				{ false,	rr::FACETYPE_BACK,		~0u,	0u		},
1096 				{ false,	rr::FACETYPE_BACK,		0xf0u,	0xfu	},
1097 				{ true,		rr::FACETYPE_BACK,		0u,		~0u		},
1098 				{ true,		rr::FACETYPE_BACK,		~0u,	0u		},
1099 				{ true,		rr::FACETYPE_BACK,		0xf0u,	0xfu	}
1100 			};
1101 
1102 			DepthStencilParams params;
1103 
1104 			params.depthFunc			= GL_LEQUAL;
1105 			params.depth				= 0.0f;
1106 			params.depthTestEnabled		= true;
1107 			params.depthWriteMask		= true;
1108 			params.stencilTestEnabled	= true;
1109 
1110 			params.stencil[rr::FACETYPE_FRONT].function			= GL_NOTEQUAL;
1111 			params.stencil[rr::FACETYPE_FRONT].reference		= 1;
1112 			params.stencil[rr::FACETYPE_FRONT].stencilFailOp	= GL_INVERT;
1113 			params.stencil[rr::FACETYPE_FRONT].depthFailOp		= GL_INCR;
1114 			params.stencil[rr::FACETYPE_FRONT].depthPassOp		= GL_DECR;
1115 			params.stencil[rr::FACETYPE_FRONT].compareMask		= ~0u;
1116 
1117 			params.stencil[rr::FACETYPE_BACK].function			= GL_ALWAYS;
1118 			params.stencil[rr::FACETYPE_BACK].reference			= 0;
1119 			params.stencil[rr::FACETYPE_BACK].stencilFailOp		= GL_REPLACE;
1120 			params.stencil[rr::FACETYPE_BACK].depthFailOp		= GL_INVERT;
1121 			params.stencil[rr::FACETYPE_BACK].depthPassOp		= GL_INCR;
1122 			params.stencil[rr::FACETYPE_BACK].compareMask		= ~0u;
1123 
1124 			vector<DepthStencilParams> cases;
1125 			for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(depthStencilWmaskCases); ndx++)
1126 			{
1127 				params.depthWriteMask							= depthStencilWmaskCases[ndx].depthWriteMask;
1128 				params.visibleFace								= depthStencilWmaskCases[ndx].visibleFace;
1129 				params.stencil[rr::FACETYPE_FRONT].writeMask	= depthStencilWmaskCases[ndx].frontWriteMask;
1130 				params.stencil[rr::FACETYPE_BACK].writeMask		= depthStencilWmaskCases[ndx].backWriteMask;
1131 				cases.push_back(params);
1132 			}
1133 
1134 			writeMaskGroup->addChild(new DepthStencilCase(m_context, "both", "Depth and Stencil Write Masks", cases));
1135 		}
1136 	}
1137 
1138 	// Randomized cases
1139 	{
1140 		tcu::TestCaseGroup* randomGroup = new tcu::TestCaseGroup(m_testCtx, "random", "Randomized Depth and Stencil Test Cases");
1141 		addChild(randomGroup);
1142 
1143 		for (int caseNdx = 0; caseNdx < NUM_RANDOM_CASES; caseNdx++)
1144 		{
1145 			vector<DepthStencilParams>	subCases	(NUM_RANDOM_SUB_CASES);
1146 			de::Random					rnd			(deInt32Hash(caseNdx) ^ deInt32Hash(m_testCtx.getCommandLine().getBaseSeed()));
1147 
1148 			for (vector<DepthStencilParams>::iterator iter = subCases.begin(); iter != subCases.end(); ++iter)
1149 				randomDepthStencilState(rnd, *iter);
1150 
1151 			randomGroup->addChild(new DepthStencilCase(m_context, de::toString(caseNdx).c_str(), "", subCases));
1152 		}
1153 	}
1154 }
1155 
1156 } // Functional
1157 } // gles2
1158 } // deqp
1159