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 Buffer test utilities.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es2fBufferTestUtil.hpp"
25 #include "tcuRandomValueIterator.hpp"
26 #include "tcuSurface.hpp"
27 #include "tcuImageCompare.hpp"
28 #include "tcuVector.hpp"
29 #include "tcuFormatUtil.hpp"
30 #include "tcuTextureUtil.hpp"
31 #include "tcuRenderTarget.hpp"
32 #include "gluPixelTransfer.hpp"
33 #include "gluRenderContext.hpp"
34 #include "gluStrUtil.hpp"
35 #include "gluShaderProgram.hpp"
36 #include "deMemory.h"
37 #include "deStringUtil.hpp"
38 
39 #include <algorithm>
40 
41 #include "glwEnums.hpp"
42 #include "glwFunctions.hpp"
43 
44 namespace deqp
45 {
46 namespace gles2
47 {
48 namespace Functional
49 {
50 namespace BufferTestUtil
51 {
52 
53 enum
54 {
55 	VERIFY_QUAD_SIZE					= 8,		//!< Quad size in VertexArrayVerifier
56 	MAX_LINES_PER_INDEX_ARRAY_DRAW		= 128,		//!< Maximum number of lines per one draw in IndexArrayVerifier
57 	INDEX_ARRAY_DRAW_VIEWPORT_WIDTH		= 128,
58 	INDEX_ARRAY_DRAW_VIEWPORT_HEIGHT	= 128
59 };
60 
61 static const bool LOG_VERIFIER_CALLS = false; //! \note Especially array verifier generates a lot of calls.
62 
63 using tcu::TestLog;
64 using std::vector;
65 using std::string;
66 using std::set;
67 
68 // Helper functions.
69 
fillWithRandomBytes(deUint8 * ptr,int numBytes,deUint32 seed)70 void fillWithRandomBytes (deUint8* ptr, int numBytes, deUint32 seed)
71 {
72 	std::copy(tcu::RandomValueIterator<deUint8>::begin(seed, numBytes), tcu::RandomValueIterator<deUint8>::end(), ptr);
73 }
74 
compareByteArrays(tcu::TestLog & log,const deUint8 * resPtr,const deUint8 * refPtr,int numBytes)75 bool compareByteArrays (tcu::TestLog& log, const deUint8* resPtr, const deUint8* refPtr, int numBytes)
76 {
77 	bool			isOk			= true;
78 	const int		maxSpanLen		= 8;
79 	const int		maxDiffSpans	= 4;
80 	int				numDiffSpans	= 0;
81 	int				diffSpanStart	= -1;
82 	int				ndx				= 0;
83 
84 	log << TestLog::Section("Verify", "Verification result");
85 
86 	for (;ndx < numBytes; ndx++)
87 	{
88 		if (resPtr[ndx] != refPtr[ndx])
89 		{
90 			if (diffSpanStart < 0)
91 				diffSpanStart = ndx;
92 
93 			isOk = false;
94 		}
95 		else if (diffSpanStart >= 0)
96 		{
97 			if (numDiffSpans < maxDiffSpans)
98 			{
99 				int len			= ndx-diffSpanStart;
100 				int	printLen	= de::min(len, maxSpanLen);
101 
102 				log << TestLog::Message << len << " byte difference at offset " << diffSpanStart << "\n"
103 										<< "  expected "	<< tcu::formatArray(tcu::Format::HexIterator<deUint8>(refPtr+diffSpanStart), tcu::Format::HexIterator<deUint8>(refPtr+diffSpanStart+printLen)) << "\n"
104 										<< "  got "			<< tcu::formatArray(tcu::Format::HexIterator<deUint8>(resPtr+diffSpanStart), tcu::Format::HexIterator<deUint8>(resPtr+diffSpanStart+printLen))
105 					<< TestLog::EndMessage;
106 			}
107 			else
108 				log << TestLog::Message << "(output too long, truncated)" << TestLog::EndMessage;
109 
110 			numDiffSpans	+= 1;
111 			diffSpanStart	 = -1;
112 		}
113 	}
114 
115 	if (diffSpanStart >= 0)
116 	{
117 		if (numDiffSpans < maxDiffSpans)
118 		{
119 				int len			= ndx-diffSpanStart;
120 				int	printLen	= de::min(len, maxSpanLen);
121 
122 				log << TestLog::Message << len << " byte difference at offset " << diffSpanStart << "\n"
123 										<< "  expected "	<< tcu::formatArray(tcu::Format::HexIterator<deUint8>(refPtr+diffSpanStart), tcu::Format::HexIterator<deUint8>(refPtr+diffSpanStart+printLen)) << "\n"
124 										<< "  got "			<< tcu::formatArray(tcu::Format::HexIterator<deUint8>(resPtr+diffSpanStart), tcu::Format::HexIterator<deUint8>(resPtr+diffSpanStart+printLen))
125 					<< TestLog::EndMessage;
126 		}
127 		else
128 			log << TestLog::Message << "(output too long, truncated)" << TestLog::EndMessage;
129 	}
130 
131 	log << TestLog::Message << (isOk ? "Verification passed." : "Verification FAILED!") << TestLog::EndMessage;
132 	log << TestLog::EndSection;
133 
134 	return isOk;
135 }
136 
getBufferTargetName(deUint32 target)137 const char* getBufferTargetName (deUint32 target)
138 {
139 	switch (target)
140 	{
141 		case GL_ARRAY_BUFFER:				return "array";
142 		case GL_ELEMENT_ARRAY_BUFFER:		return "element_array";
143 		default:
144 			DE_ASSERT(false);
145 			return DE_NULL;
146 	}
147 }
148 
getUsageHintName(deUint32 hint)149 const char* getUsageHintName (deUint32 hint)
150 {
151 	switch (hint)
152 	{
153 		case GL_STREAM_DRAW:	return "stream_draw";
154 		case GL_STATIC_DRAW:	return "static_draw";
155 		case GL_DYNAMIC_DRAW:	return "dynamic_draw";
156 		default:
157 			DE_ASSERT(false);
158 			return DE_NULL;
159 	}
160 }
161 
162 // BufferCase
163 
BufferCase(Context & context,const char * name,const char * description)164 BufferCase::BufferCase (Context& context, const char* name, const char* description)
165 	: TestCase			(context, name, description)
166 	, CallLogWrapper	(context.getRenderContext().getFunctions(), m_context.getTestContext().getLog())
167 {
168 }
169 
~BufferCase(void)170 BufferCase::~BufferCase (void)
171 {
172 	enableLogging(false);
173 	BufferCase::deinit();
174 }
175 
init(void)176 void BufferCase::init (void)
177 {
178 	enableLogging(true);
179 }
180 
deinit(void)181 void BufferCase::deinit (void)
182 {
183 	for (set<deUint32>::const_iterator bufIter = m_allocatedBuffers.begin(); bufIter != m_allocatedBuffers.end(); bufIter++)
184 		glDeleteBuffers(1, &(*bufIter));
185 }
186 
genBuffer(void)187 deUint32 BufferCase::genBuffer (void)
188 {
189 	deUint32 buf = 0;
190 	glGenBuffers(1, &buf);
191 	if (buf != 0)
192 	{
193 		try
194 		{
195 			m_allocatedBuffers.insert(buf);
196 		}
197 		catch (const std::exception&)
198 		{
199 			glDeleteBuffers(1, &buf);
200 			throw;
201 		}
202 	}
203 	return buf;
204 }
205 
deleteBuffer(deUint32 buffer)206 void BufferCase::deleteBuffer (deUint32 buffer)
207 {
208 	glDeleteBuffers(1, &buffer);
209 	m_allocatedBuffers.erase(buffer);
210 }
211 
checkError(void)212 void BufferCase::checkError (void)
213 {
214 	glw::GLenum err = glGetError();
215 	if (err != GL_NO_ERROR)
216 		throw tcu::TestError(string("Got ") + glu::getErrorStr(err).toString());
217 }
218 
219 // ReferenceBuffer
220 
setSize(int numBytes)221 void ReferenceBuffer::setSize (int numBytes)
222 {
223 	m_data.resize(numBytes);
224 }
225 
setData(int numBytes,const deUint8 * bytes)226 void ReferenceBuffer::setData (int numBytes, const deUint8* bytes)
227 {
228 	m_data.resize(numBytes);
229 	std::copy(bytes, bytes+numBytes, m_data.begin());
230 }
231 
setSubData(int offset,int numBytes,const deUint8 * bytes)232 void ReferenceBuffer::setSubData (int offset, int numBytes, const deUint8* bytes)
233 {
234 	DE_ASSERT(de::inBounds(offset, 0, (int)m_data.size()) && de::inRange(offset+numBytes, offset, (int)m_data.size()));
235 	std::copy(bytes, bytes+numBytes, m_data.begin()+offset);
236 }
237 
238 // BufferVerifierBase
239 
BufferVerifierBase(Context & context)240 BufferVerifierBase::BufferVerifierBase (Context& context)
241 	: CallLogWrapper	(context.getRenderContext().getFunctions(), context.getTestContext().getLog())
242 	, m_context			(context)
243 {
244 	enableLogging(LOG_VERIFIER_CALLS);
245 }
246 
247 // BufferVerifier
248 
BufferVerifier(Context & context,VerifyType verifyType)249 BufferVerifier::BufferVerifier (Context& context, VerifyType verifyType)
250 	: m_verifier(DE_NULL)
251 {
252 	switch (verifyType)
253 	{
254 		case VERIFY_AS_VERTEX_ARRAY:	m_verifier = new VertexArrayVerifier(context);	break;
255 		case VERIFY_AS_INDEX_ARRAY:		m_verifier = new IndexArrayVerifier	(context);	break;
256 		default:
257 			TCU_FAIL("Unsupported verifier");
258 	}
259 }
260 
~BufferVerifier(void)261 BufferVerifier::~BufferVerifier (void)
262 {
263 	delete m_verifier;
264 }
265 
verify(deUint32 buffer,const deUint8 * reference,int offset,int numBytes)266 bool BufferVerifier::verify (deUint32 buffer, const deUint8* reference, int offset, int numBytes)
267 {
268 	DE_ASSERT(numBytes >= getMinSize());
269 	DE_ASSERT(offset%getAlignment() == 0);
270 	DE_ASSERT((offset+numBytes)%getAlignment() == 0);
271 	return m_verifier->verify(buffer, reference, offset, numBytes);
272 }
273 
274 // VertexArrayVerifier
275 
VertexArrayVerifier(Context & context)276 VertexArrayVerifier::VertexArrayVerifier (Context& context)
277 	: BufferVerifierBase	(context)
278 	, m_program				(DE_NULL)
279 	, m_posLoc				(0)
280 	, m_byteVecLoc			(0)
281 {
282 	m_program = new glu::ShaderProgram(m_context.getRenderContext(), glu::makeVtxFragSources(
283 		"attribute highp vec2 a_position;\n"
284 		"attribute mediump vec3 a_byteVec;\n"
285 		"varying mediump vec3 v_byteVec;\n"
286 		"void main (void)\n"
287 		"{\n"
288 		"	gl_Position = vec4(a_position, 0.0, 1.0);\n"
289 		"	v_byteVec = a_byteVec;\n"
290 		"}\n",
291 
292 		"varying mediump vec3 v_byteVec;\n"
293 		"void main (void)\n"
294 		"{\n"
295 		"	gl_FragColor = vec4(v_byteVec, 1.0);\n"
296 		"}\n"));
297 
298 	if (!m_program->isOk())
299 	{
300 		m_context.getTestContext().getLog() << *m_program;
301 		delete m_program;
302 		TCU_FAIL("Compile failed");
303 	}
304 
305 	const glw::Functions& funcs = context.getRenderContext().getFunctions();
306 	m_posLoc		= funcs.getAttribLocation(m_program->getProgram(), "a_position");
307 	m_byteVecLoc	= funcs.getAttribLocation(m_program->getProgram(), "a_byteVec");
308 }
309 
~VertexArrayVerifier(void)310 VertexArrayVerifier::~VertexArrayVerifier (void)
311 {
312 	delete m_program;
313 }
314 
computePositions(vector<tcu::Vec2> & positions,int gridSizeX,int gridSizeY)315 static void computePositions (vector<tcu::Vec2>& positions, int gridSizeX, int gridSizeY)
316 {
317 	positions.resize(gridSizeX*gridSizeY*4);
318 
319 	for (int y = 0; y < gridSizeY; y++)
320 	for (int x = 0; x < gridSizeX; x++)
321 	{
322 		float	sx0			= (x+0) / (float)gridSizeX;
323 		float	sy0			= (y+0) / (float)gridSizeY;
324 		float	sx1			= (x+1) / (float)gridSizeX;
325 		float	sy1			= (y+1) / (float)gridSizeY;
326 		float	fx0			= 2.0f * sx0 - 1.0f;
327 		float	fy0			= 2.0f * sy0 - 1.0f;
328 		float	fx1			= 2.0f * sx1 - 1.0f;
329 		float	fy1			= 2.0f * sy1 - 1.0f;
330 		int		baseNdx		= (y * gridSizeX + x)*4;
331 
332 		positions[baseNdx+0] = tcu::Vec2(fx0, fy0);
333 		positions[baseNdx+1] = tcu::Vec2(fx0, fy1);
334 		positions[baseNdx+2] = tcu::Vec2(fx1, fy0);
335 		positions[baseNdx+3] = tcu::Vec2(fx1, fy1);
336 	}
337 }
338 
computeIndices(vector<deUint16> & indices,int gridSizeX,int gridSizeY)339 static void computeIndices (vector<deUint16>& indices, int gridSizeX, int gridSizeY)
340 {
341 	indices.resize(3 * 2 * gridSizeX * gridSizeY);
342 
343 	for (int quadNdx = 0; quadNdx < gridSizeX*gridSizeY; quadNdx++)
344 	{
345 		int v00 = quadNdx*4 + 0;
346 		int v01 = quadNdx*4 + 1;
347 		int v10 = quadNdx*4 + 2;
348 		int v11 = quadNdx*4 + 3;
349 
350 		DE_ASSERT(v11 < (1<<16));
351 
352 		indices[quadNdx*6 + 0] = (deUint16)v10;
353 		indices[quadNdx*6 + 1] = (deUint16)v00;
354 		indices[quadNdx*6 + 2] = (deUint16)v01;
355 
356 		indices[quadNdx*6 + 3] = (deUint16)v10;
357 		indices[quadNdx*6 + 4] = (deUint16)v01;
358 		indices[quadNdx*6 + 5] = (deUint16)v11;
359 	}
360 }
361 
fetchVtxColor(const deUint8 * ptr,int vtxNdx)362 static inline tcu::Vec4 fetchVtxColor (const deUint8* ptr, int vtxNdx)
363 {
364 	return tcu::RGBA(*(ptr + vtxNdx*3 + 0),
365 					 *(ptr + vtxNdx*3 + 1),
366 					 *(ptr + vtxNdx*3 + 2),
367 					 255).toVec();
368 }
369 
renderQuadGridReference(tcu::Surface & dst,int numQuads,int rowLength,const deUint8 * inPtr)370 static void renderQuadGridReference (tcu::Surface& dst, int numQuads, int rowLength, const deUint8* inPtr)
371 {
372 	using tcu::Vec4;
373 
374 	dst.setSize(rowLength*VERIFY_QUAD_SIZE, (numQuads/rowLength + (numQuads%rowLength != 0 ? 1 : 0))*VERIFY_QUAD_SIZE);
375 
376 	tcu::PixelBufferAccess dstAccess = dst.getAccess();
377 	tcu::clear(dstAccess, tcu::IVec4(0, 0, 0, 0xff));
378 
379 	for (int quadNdx = 0; quadNdx < numQuads; quadNdx++)
380 	{
381 		int		x0		= (quadNdx%rowLength)*VERIFY_QUAD_SIZE;
382 		int		y0		= (quadNdx/rowLength)*VERIFY_QUAD_SIZE;
383 		Vec4	v00		= fetchVtxColor(inPtr, quadNdx*4 + 0);
384 		Vec4	v10		= fetchVtxColor(inPtr, quadNdx*4 + 1);
385 		Vec4	v01		= fetchVtxColor(inPtr, quadNdx*4 + 2);
386 		Vec4	v11		= fetchVtxColor(inPtr, quadNdx*4 + 3);
387 
388 		for (int y = 0; y < VERIFY_QUAD_SIZE; y++)
389 		for (int x = 0; x < VERIFY_QUAD_SIZE; x++)
390 		{
391 			float		fx		= (float)(x+0.5f) / (float)VERIFY_QUAD_SIZE;
392 			float		fy		= (float)(y+0.5f) / (float)VERIFY_QUAD_SIZE;
393 
394 			bool		tri		= fx + fy <= 1.0f;
395 			float		tx		= tri ? fx : (1.0f-fx);
396 			float		ty		= tri ? fy : (1.0f-fy);
397 			const Vec4&	t0		= tri ? v00 : v11;
398 			const Vec4&	t1		= tri ? v01 : v10;
399 			const Vec4&	t2		= tri ? v10 : v01;
400 			Vec4		color	= t0 + (t1-t0)*tx + (t2-t0)*ty;
401 
402 			dstAccess.setPixel(color, x0+x, y0+y);
403 		}
404 	}
405 }
406 
verify(deUint32 buffer,const deUint8 * refPtr,int offset,int numBytes)407 bool VertexArrayVerifier::verify (deUint32 buffer, const deUint8* refPtr, int offset, int numBytes)
408 {
409 	const tcu::RenderTarget&	renderTarget		= m_context.getRenderContext().getRenderTarget();
410 	const int					numBytesInVtx		= 3;
411 	const int					numBytesInQuad		= numBytesInVtx*4;
412 	int							maxQuadsX			= de::min(128, renderTarget.getWidth()	/ VERIFY_QUAD_SIZE);
413 	int							maxQuadsY			= de::min(128, renderTarget.getHeight()	/ VERIFY_QUAD_SIZE);
414 	int							maxQuadsPerBatch	= maxQuadsX*maxQuadsY;
415 	int							numVerified			= 0;
416 	deUint32					program				= m_program->getProgram();
417 	tcu::RGBA					threshold			= renderTarget.getPixelFormat().getColorThreshold() + tcu::RGBA(3,3,3,3);
418 	bool						isOk				= true;
419 
420 	vector<tcu::Vec2>			positions;
421 	vector<deUint16>			indices;
422 
423 	tcu::Surface				rendered;
424 	tcu::Surface				reference;
425 
426 	DE_ASSERT(numBytes >= numBytesInQuad); // Can't render full quad with smaller buffers.
427 
428 	computePositions(positions, maxQuadsX, maxQuadsY);
429 	computeIndices(indices, maxQuadsX, maxQuadsY);
430 
431 	// Reset buffer bindings.
432 	glBindBuffer				(GL_ELEMENT_ARRAY_BUFFER,	0);
433 	glBindBuffer				(GL_ARRAY_BUFFER,			0);
434 
435 	// Setup rendering state.
436 	glViewport					(0, 0, maxQuadsX*VERIFY_QUAD_SIZE, maxQuadsY*VERIFY_QUAD_SIZE);
437 	glClearColor				(0.0f, 0.0f, 0.0f, 1.0f);
438 	glUseProgram				(program);
439 	glEnableVertexAttribArray	(m_posLoc);
440 	glVertexAttribPointer		(m_posLoc, 2, GL_FLOAT, GL_FALSE, 0, &positions[0]);
441 	glEnableVertexAttribArray	(m_byteVecLoc);
442 	glBindBuffer				(GL_ARRAY_BUFFER, buffer);
443 
444 	while (numVerified < numBytes)
445 	{
446 		int		numRemaining		= numBytes-numVerified;
447 		bool	isLeftoverBatch		= numRemaining < numBytesInQuad;
448 		int		numBytesToVerify	= isLeftoverBatch ? numBytesInQuad				: de::min(maxQuadsPerBatch*numBytesInQuad, numRemaining - numRemaining%numBytesInQuad);
449 		int		curOffset			= isLeftoverBatch ? (numBytes-numBytesInQuad)	: numVerified;
450 		int		numQuads			= numBytesToVerify/numBytesInQuad;
451 		int		numCols				= de::min(maxQuadsX, numQuads);
452 		int		numRows				= numQuads/maxQuadsX + (numQuads%maxQuadsX != 0 ? 1 : 0);
453 		string	imageSetDesc		= string("Bytes ") + de::toString(offset+curOffset) + " to " + de::toString(offset+curOffset+numBytesToVerify-1);
454 
455 		DE_ASSERT(numBytesToVerify > 0 && numBytesToVerify%numBytesInQuad == 0);
456 		DE_ASSERT(de::inBounds(curOffset, 0, numBytes));
457 		DE_ASSERT(de::inRange(curOffset+numBytesToVerify, curOffset, numBytes));
458 
459 		// Render batch.
460 		glClear					(GL_COLOR_BUFFER_BIT);
461 		glVertexAttribPointer	(m_byteVecLoc, 3, GL_UNSIGNED_BYTE, GL_TRUE, 0, (const glw::GLvoid*)(deUintptr)(offset + curOffset));
462 		glDrawElements			(GL_TRIANGLES, numQuads*6, GL_UNSIGNED_SHORT, &indices[0]);
463 
464 		renderQuadGridReference(reference,  numQuads, numCols, refPtr + offset + curOffset);
465 
466 		rendered.setSize(numCols*VERIFY_QUAD_SIZE, numRows*VERIFY_QUAD_SIZE);
467 		glu::readPixels(m_context.getRenderContext(), 0, 0, rendered.getAccess());
468 
469 		if (!tcu::pixelThresholdCompare(m_context.getTestContext().getLog(), "RenderResult", imageSetDesc.c_str(), reference, rendered, threshold, tcu::COMPARE_LOG_RESULT))
470 		{
471 			isOk = false;
472 			break;
473 		}
474 
475 		numVerified += isLeftoverBatch ? numRemaining : numBytesToVerify;
476 	}
477 
478 	glDisableVertexAttribArray	(m_posLoc);
479 	glDisableVertexAttribArray	(m_byteVecLoc);
480 
481 	return isOk;
482 }
483 
484 // IndexArrayVerifier
485 
IndexArrayVerifier(Context & context)486 IndexArrayVerifier::IndexArrayVerifier (Context& context)
487 	: BufferVerifierBase	(context)
488 	, m_program				(DE_NULL)
489 	, m_posLoc				(0)
490 	, m_colorLoc			(0)
491 {
492 	m_program = new glu::ShaderProgram(m_context.getRenderContext(), glu::makeVtxFragSources(
493 		"attribute highp vec2 a_position;\n"
494 		"attribute mediump vec3 a_color;\n"
495 		"varying mediump vec3 v_color;\n"
496 		"void main (void)\n"
497 		"{\n"
498 		"	gl_Position = vec4(a_position, 0.0, 1.0);\n"
499 		"	v_color = a_color;\n"
500 		"}\n",
501 
502 		"varying mediump vec3 v_color;\n"
503 		"void main (void)\n"
504 		"{\n"
505 		"	gl_FragColor = vec4(v_color, 1.0);\n"
506 		"}\n"));
507 
508 	if (!m_program->isOk())
509 	{
510 		m_context.getTestContext().getLog() << *m_program;
511 		delete m_program;
512 		TCU_FAIL("Compile failed");
513 	}
514 
515 	const glw::Functions& funcs = context.getRenderContext().getFunctions();
516 	m_posLoc	= funcs.getAttribLocation(m_program->getProgram(), "a_position");
517 	m_colorLoc	= funcs.getAttribLocation(m_program->getProgram(), "a_color");
518 }
519 
~IndexArrayVerifier(void)520 IndexArrayVerifier::~IndexArrayVerifier (void)
521 {
522 	delete m_program;
523 }
524 
computeIndexVerifierPositions(std::vector<tcu::Vec2> & dst)525 static void computeIndexVerifierPositions (std::vector<tcu::Vec2>& dst)
526 {
527 	const int	numPosX		= 16;
528 	const int	numPosY		= 16;
529 
530 	dst.resize(numPosX*numPosY);
531 
532 	for (int y = 0; y < numPosY; y++)
533 	{
534 		for (int x = 0; x < numPosX; x++)
535 		{
536 			float	xf	= float(x) / float(numPosX-1);
537 			float	yf	= float(y) / float(numPosY-1);
538 
539 			dst[y*numPosX + x] = tcu::Vec2(2.0f*xf - 1.0f, 2.0f*yf - 1.0f);
540 		}
541 	}
542 }
543 
computeIndexVerifierColors(std::vector<tcu::Vec3> & dst)544 static void computeIndexVerifierColors (std::vector<tcu::Vec3>& dst)
545 {
546 	const int	numColors	= 256;
547 	const float	minVal		= 0.1f;
548 	const float maxVal		= 0.5f;
549 	de::Random	rnd			(0xabc231);
550 
551 	dst.resize(numColors);
552 
553 	for (std::vector<tcu::Vec3>::iterator i = dst.begin(); i != dst.end(); ++i)
554 	{
555 		i->x()	= rnd.getFloat(minVal, maxVal);
556 		i->y()	= rnd.getFloat(minVal, maxVal);
557 		i->z()	= rnd.getFloat(minVal, maxVal);
558 	}
559 }
560 
561 template<typename T>
execVertexFetch(T * dst,const T * src,const deUint8 * indices,int numIndices)562 static void execVertexFetch (T* dst, const T* src, const deUint8* indices, int numIndices)
563 {
564 	for (int i = 0; i < numIndices; ++i)
565 		dst[i] = src[indices[i]];
566 }
567 
verify(deUint32 buffer,const deUint8 * refPtr,int offset,int numBytes)568 bool IndexArrayVerifier::verify (deUint32 buffer, const deUint8* refPtr, int offset, int numBytes)
569 {
570 	const tcu::RenderTarget&	renderTarget		= m_context.getRenderContext().getRenderTarget();
571 	const int					viewportW			= de::min<int>(INDEX_ARRAY_DRAW_VIEWPORT_WIDTH, renderTarget.getWidth());
572 	const int					viewportH			= de::min<int>(INDEX_ARRAY_DRAW_VIEWPORT_HEIGHT, renderTarget.getHeight());
573 	const int					minBytesPerBatch	= 2;
574 	const tcu::RGBA				threshold			(0,0,0,0);
575 
576 	std::vector<tcu::Vec2>		positions;
577 	std::vector<tcu::Vec3>		colors;
578 
579 	std::vector<tcu::Vec2>		fetchedPos			(MAX_LINES_PER_INDEX_ARRAY_DRAW+1);
580 	std::vector<tcu::Vec3>		fetchedColor		(MAX_LINES_PER_INDEX_ARRAY_DRAW+1);
581 
582 	tcu::Surface				indexBufferImg		(viewportW, viewportH);
583 	tcu::Surface				referenceImg		(viewportW, viewportH);
584 
585 	int							numVerified			= 0;
586 	bool						isOk				= true;
587 
588 	DE_STATIC_ASSERT(sizeof(tcu::Vec2) == sizeof(float)*2);
589 	DE_STATIC_ASSERT(sizeof(tcu::Vec3) == sizeof(float)*3);
590 
591 	computeIndexVerifierPositions(positions);
592 	computeIndexVerifierColors(colors);
593 
594 	// Reset buffer bindings.
595 	glBindBuffer				(GL_ARRAY_BUFFER,			0);
596 	glBindBuffer				(GL_ELEMENT_ARRAY_BUFFER,	buffer);
597 
598 	// Setup rendering state.
599 	glViewport					(0, 0, viewportW, viewportH);
600 	glClearColor				(0.0f, 0.0f, 0.0f, 1.0f);
601 	glUseProgram				(m_program->getProgram());
602 	glEnableVertexAttribArray	(m_posLoc);
603 	glEnableVertexAttribArray	(m_colorLoc);
604 	glEnable					(GL_BLEND);
605 	glBlendFunc					(GL_ONE, GL_ONE);
606 	glBlendEquation				(GL_FUNC_ADD);
607 
608 	while (numVerified < numBytes)
609 	{
610 		int		numRemaining		= numBytes-numVerified;
611 		bool	isLeftoverBatch		= numRemaining < minBytesPerBatch;
612 		int		numBytesToVerify	= isLeftoverBatch ? minBytesPerBatch			: de::min(MAX_LINES_PER_INDEX_ARRAY_DRAW+1, numRemaining);
613 		int		curOffset			= isLeftoverBatch ? (numBytes-minBytesPerBatch)	: numVerified;
614 		string	imageSetDesc		= string("Bytes ") + de::toString(offset+curOffset) + " to " + de::toString(offset+curOffset+numBytesToVerify-1);
615 
616 		// Step 1: Render using index buffer.
617 		glClear					(GL_COLOR_BUFFER_BIT);
618 		glVertexAttribPointer	(m_posLoc, 2, GL_FLOAT, GL_FALSE, 0, &positions[0]);
619 		glVertexAttribPointer	(m_colorLoc, 3, GL_FLOAT, GL_FALSE, 0, &colors[0]);
620 		glDrawElements			(GL_LINE_STRIP, numBytesToVerify, GL_UNSIGNED_BYTE, (void*)(deUintptr)(offset+curOffset));
621 		glu::readPixels			(m_context.getRenderContext(), 0, 0, indexBufferImg.getAccess());
622 
623 		// Step 2: Do manual fetch and render without index buffer.
624 		execVertexFetch(&fetchedPos[0], &positions[0], refPtr+offset+curOffset, numBytesToVerify);
625 		execVertexFetch(&fetchedColor[0], &colors[0], refPtr+offset+curOffset, numBytesToVerify);
626 
627 		glClear					(GL_COLOR_BUFFER_BIT);
628 		glVertexAttribPointer	(m_posLoc, 2, GL_FLOAT, GL_FALSE, 0, &fetchedPos[0]);
629 		glVertexAttribPointer	(m_colorLoc, 3, GL_FLOAT, GL_FALSE, 0, &fetchedColor[0]);
630 		glDrawArrays			(GL_LINE_STRIP, 0, numBytesToVerify);
631 		glu::readPixels			(m_context.getRenderContext(), 0, 0, referenceImg.getAccess());
632 
633 		if (!tcu::pixelThresholdCompare(m_context.getTestContext().getLog(), "RenderResult", imageSetDesc.c_str(), referenceImg, indexBufferImg, threshold, tcu::COMPARE_LOG_RESULT))
634 		{
635 			isOk = false;
636 			break;
637 		}
638 
639 		numVerified += isLeftoverBatch ? numRemaining : numBytesToVerify;
640 	}
641 
642 	return isOk;
643 }
644 
645 } // BufferTestUtil
646 } // Functional
647 } // gles2
648 } // deqp
649