• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program OpenGL Utilities
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 Draw call utilities.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "gluDrawUtil.hpp"
25 #include "gluRenderContext.hpp"
26 #include "gluObjectWrapper.hpp"
27 #include "glwFunctions.hpp"
28 #include "glwEnums.hpp"
29 #include "deInt32.h"
30 #include "deMemory.h"
31 
32 #include <vector>
33 #include <set>
34 #include <iterator>
35 
36 namespace glu
37 {
38 namespace
39 {
40 
41 struct VertexAttributeDescriptor
42 {
43 	int							location;
44 	VertexComponentType			componentType;
45 	VertexComponentConversion	convert;
46 	int							numComponents;
47 	int							numElements;
48 	int							stride;				//!< Stride or 0 if using default stride.
49 	const void*					pointer;			//!< Pointer or offset.
50 
VertexAttributeDescriptorglu::__anon608820be0111::VertexAttributeDescriptor51 	VertexAttributeDescriptor (int							location_,
52 							   VertexComponentType			componentType_,
53 							   VertexComponentConversion	convert_,
54 							   int							numComponents_,
55 							   int							numElements_,
56 							   int							stride_,
57 							   const void*					pointer_)
58 		: location		(location_)
59 		, componentType	(componentType_)
60 		, convert		(convert_)
61 		, numComponents	(numComponents_)
62 		, numElements	(numElements_)
63 		, stride		(stride_)
64 		, pointer		(pointer_)
65 	{
66 	}
67 
VertexAttributeDescriptorglu::__anon608820be0111::VertexAttributeDescriptor68 	VertexAttributeDescriptor (void)
69 		: location		(0)
70 		, componentType	(VTX_COMP_TYPE_LAST)
71 		, convert		(VTX_COMP_CONVERT_LAST)
72 		, numComponents	(0)
73 		, numElements	(0)
74 		, stride		(0)
75 		, pointer		(0)
76 	{
77 	}
78 };
79 
80 struct VertexBufferLayout
81 {
82 	int										size;
83 	std::vector<VertexAttributeDescriptor>	attributes;
84 
VertexBufferLayoutglu::__anon608820be0111::VertexBufferLayout85 	VertexBufferLayout (int size_ = 0)
86 		: size(size_)
87 	{
88 	}
89 };
90 
91 struct VertexBufferDescriptor
92 {
93 	deUint32								buffer;
94 	std::vector<VertexAttributeDescriptor>	attributes;
95 
VertexBufferDescriptorglu::__anon608820be0111::VertexBufferDescriptor96 	VertexBufferDescriptor (deUint32 buffer_ = 0)
97 		: buffer(buffer_)
98 	{
99 	}
100 };
101 
102 class VertexBuffer : public Buffer
103 {
104 public:
105 	enum Type
106 	{
107 		TYPE_PLANAR = 0,	//!< Data for each vertex array resides in a separate contiguous block in buffer.
108 		TYPE_STRIDED,		//!< Vertex arrays are interleaved.
109 
110 		TYPE_LAST
111 	};
112 
113 									VertexBuffer		(const RenderContext& context, int numBindings, const VertexArrayBinding* bindings, Type type = TYPE_PLANAR);
114 									~VertexBuffer		(void);
115 
getDescriptor(void) const116 	const VertexBufferDescriptor&	getDescriptor		(void) const { return m_layout; }
117 
118 private:
119 									VertexBuffer		(const VertexBuffer& other);
120 	VertexBuffer&					operator=			(const VertexBuffer& other);
121 
122 	VertexBufferDescriptor			m_layout;
123 };
124 
125 class IndexBuffer : public Buffer
126 {
127 public:
128 									IndexBuffer			(const RenderContext& context, IndexType indexType, int numIndices, const void* indices);
129 									~IndexBuffer		(void);
130 
131 private:
132 									IndexBuffer			(const IndexBuffer& other);
133 	IndexBuffer&					operator=			(const IndexBuffer& other);
134 };
135 
getVtxCompGLType(VertexComponentType type)136 static deUint32 getVtxCompGLType (VertexComponentType type)
137 {
138 	switch (type)
139 	{
140 		case VTX_COMP_UNSIGNED_INT8:	return GL_UNSIGNED_BYTE;
141 		case VTX_COMP_UNSIGNED_INT16:	return GL_UNSIGNED_SHORT;
142 		case VTX_COMP_UNSIGNED_INT32:	return GL_UNSIGNED_INT;
143 		case VTX_COMP_SIGNED_INT8:		return GL_BYTE;
144 		case VTX_COMP_SIGNED_INT16:		return GL_SHORT;
145 		case VTX_COMP_SIGNED_INT32:		return GL_INT;
146 		case VTX_COMP_FIXED:			return GL_FIXED;
147 		case VTX_COMP_HALF_FLOAT:		return GL_HALF_FLOAT;
148 		case VTX_COMP_FLOAT:			return GL_FLOAT;
149 		default:
150 			DE_ASSERT(false);
151 			return GL_NONE;
152 	}
153 }
154 
getVtxCompSize(VertexComponentType type)155 static int getVtxCompSize (VertexComponentType type)
156 {
157 	switch (type)
158 	{
159 		case VTX_COMP_UNSIGNED_INT8:	return 1;
160 		case VTX_COMP_UNSIGNED_INT16:	return 2;
161 		case VTX_COMP_UNSIGNED_INT32:	return 4;
162 		case VTX_COMP_SIGNED_INT8:		return 1;
163 		case VTX_COMP_SIGNED_INT16:		return 2;
164 		case VTX_COMP_SIGNED_INT32:		return 4;
165 		case VTX_COMP_FIXED:			return 4;
166 		case VTX_COMP_HALF_FLOAT:		return 2;
167 		case VTX_COMP_FLOAT:			return 4;
168 		default:
169 			DE_ASSERT(false);
170 			return 0;
171 	}
172 }
173 
getIndexGLType(IndexType type)174 static deUint32 getIndexGLType (IndexType type)
175 {
176 	switch (type)
177 	{
178 		case INDEXTYPE_UINT8:	return GL_UNSIGNED_BYTE;
179 		case INDEXTYPE_UINT16:	return GL_UNSIGNED_SHORT;
180 		case INDEXTYPE_UINT32:	return GL_UNSIGNED_INT;
181 		default:
182 			DE_ASSERT(false);
183 			return 0;
184 	}
185 }
186 
getIndexSize(IndexType type)187 static int getIndexSize (IndexType type)
188 {
189 	switch (type)
190 	{
191 		case INDEXTYPE_UINT8:	return 1;
192 		case INDEXTYPE_UINT16:	return 2;
193 		case INDEXTYPE_UINT32:	return 4;
194 		default:
195 			DE_ASSERT(false);
196 			return 0;
197 	}
198 }
199 
getPrimitiveGLType(PrimitiveType type)200 static deUint32 getPrimitiveGLType (PrimitiveType type)
201 {
202 	switch (type)
203 	{
204 		case PRIMITIVETYPE_TRIANGLES:		return GL_TRIANGLES;
205 		case PRIMITIVETYPE_TRIANGLE_STRIP:	return GL_TRIANGLE_STRIP;
206 		case PRIMITIVETYPE_TRIANGLE_FAN:	return GL_TRIANGLE_FAN;
207 		case PRIMITIVETYPE_LINES:			return GL_LINES;
208 		case PRIMITIVETYPE_LINE_STRIP:		return GL_LINE_STRIP;
209 		case PRIMITIVETYPE_LINE_LOOP:		return GL_LINE_LOOP;
210 		case PRIMITIVETYPE_POINTS:			return GL_POINTS;
211 		case PRIMITIVETYPE_PATCHES:			return GL_PATCHES;
212 		default:
213 			DE_ASSERT(false);
214 			return 0;
215 	}
216 }
217 
218 //! Lower named bindings to locations and eliminate bindings that are not used by program.
219 template<typename InputIter, typename OutputIter>
namedBindingsToProgramLocations(const glw::Functions & gl,deUint32 program,InputIter first,InputIter end,OutputIter out)220 static OutputIter namedBindingsToProgramLocations (const glw::Functions& gl, deUint32 program, InputIter first, InputIter end, OutputIter out)
221 {
222 	for (InputIter cur = first; cur != end; ++cur)
223 	{
224 		const BindingPoint& binding = cur->binding;
225 		if (binding.type == BindingPoint::BPTYPE_NAME)
226 		{
227 			DE_ASSERT(binding.location >= 0);
228 			int location = gl.getAttribLocation(program, binding.name.c_str());
229 			if (location >= 0)
230 			{
231 				// Add binding.location as an offset to accommodate matrices.
232 				*out = VertexArrayBinding(BindingPoint(location + binding.location), cur->pointer);
233 				++out;
234 			}
235 		}
236 		else
237 		{
238 			*out = *cur;
239 			++out;
240 		}
241 	}
242 
243 	return out;
244 }
245 
getMinimumAlignment(const VertexArrayPointer & pointer)246 static deUint32 getMinimumAlignment (const VertexArrayPointer& pointer)
247 {
248 	// \todo [2013-05-07 pyry] What is the actual min?
249 	DE_UNREF(pointer);
250 	return (deUint32)sizeof(float);
251 }
252 
253 template<typename BindingIter>
areVertexArrayLocationsValid(BindingIter first,BindingIter end)254 static bool areVertexArrayLocationsValid (BindingIter first, BindingIter end)
255 {
256 	std::set<int> usedLocations;
257 	for (BindingIter cur = first; cur != end; ++cur)
258 	{
259 		const BindingPoint& binding = cur->binding;
260 
261 		if (binding.type != BindingPoint::BPTYPE_LOCATION)
262 			return false;
263 
264 		if (usedLocations.find(binding.location) != usedLocations.end())
265 			return false;
266 
267 		usedLocations.insert(binding.location);
268 	}
269 
270 	return true;
271 }
272 
273 // \todo [2013-05-08 pyry] Buffer upload should try to match pointers to reduce dataset size.
274 
appendAttributeNonStrided(VertexBufferLayout & layout,const VertexArrayBinding & va)275 static void appendAttributeNonStrided (VertexBufferLayout& layout, const VertexArrayBinding& va)
276 {
277 	const int	offset		= deAlign32(layout.size, getMinimumAlignment(va.pointer));
278 	const int	elementSize	= getVtxCompSize(va.pointer.componentType)*va.pointer.numComponents;
279 	const int	size		= elementSize*va.pointer.numElements;
280 
281 	// Must be assigned to location at this point.
282 	DE_ASSERT(va.binding.type == BindingPoint::BPTYPE_LOCATION);
283 
284 	layout.attributes.push_back(VertexAttributeDescriptor(va.binding.location,
285 														  va.pointer.componentType,
286 														  va.pointer.convert,
287 														  va.pointer.numComponents,
288 														  va.pointer.numElements,
289 														  0, // default stride
290 														  (const void*)(deUintptr)offset));
291 	layout.size = offset+size;
292 }
293 
294 template<typename BindingIter>
computeNonStridedBufferLayout(VertexBufferLayout & layout,BindingIter first,BindingIter end)295 static void computeNonStridedBufferLayout (VertexBufferLayout& layout, BindingIter first, BindingIter end)
296 {
297 	for (BindingIter iter = first; iter != end; ++iter)
298 		appendAttributeNonStrided(layout, *iter);
299 }
300 
copyToLayout(void * dstBasePtr,const VertexAttributeDescriptor & dstVA,const VertexArrayPointer & srcPtr)301 static void copyToLayout (void* dstBasePtr, const VertexAttributeDescriptor& dstVA, const VertexArrayPointer& srcPtr)
302 {
303 	DE_ASSERT(dstVA.componentType	== srcPtr.componentType &&
304 			  dstVA.numComponents	== srcPtr.numComponents &&
305 			  dstVA.numElements		== srcPtr.numElements);
306 
307 	const int	elementSize			= getVtxCompSize(dstVA.componentType)*dstVA.numComponents;
308 	const bool	srcHasCustomStride	= srcPtr.stride != 0 && srcPtr.stride != elementSize;
309 	const bool	dstHasCustomStride	= dstVA.stride != 0 && dstVA.stride != elementSize;
310 
311 	if (srcHasCustomStride || dstHasCustomStride)
312 	{
313 		const int	dstStride	= dstVA.stride != 0 ? dstVA.stride : elementSize;
314 		const int	srcStride	= srcPtr.stride != 0 ? srcPtr.stride : elementSize;
315 
316 		for (int ndx = 0; ndx < dstVA.numElements; ndx++)
317 			deMemcpy((deUint8*)dstBasePtr + (deUintptr)dstVA.pointer + ndx*dstStride, (const deUint8*)srcPtr.data + ndx*srcStride, elementSize);
318 	}
319 	else
320 		deMemcpy((deUint8*)dstBasePtr + (deUintptr)dstVA.pointer, srcPtr.data, elementSize*dstVA.numElements);
321 }
322 
uploadBufferData(const glw::Functions & gl,deUint32 buffer,deUint32 usage,const VertexBufferLayout & layout,const VertexArrayPointer * srcArrays)323 void uploadBufferData (const glw::Functions& gl, deUint32 buffer, deUint32 usage, const VertexBufferLayout& layout, const VertexArrayPointer* srcArrays)
324 {
325 	// Create temporary data buffer for upload.
326 	std::vector<deUint8> localBuf(layout.size);
327 
328 	for (int attrNdx = 0; attrNdx < (int)layout.attributes.size(); ++attrNdx)
329 		copyToLayout(&localBuf[0], layout.attributes[attrNdx], srcArrays[attrNdx]);
330 
331 	gl.bindBuffer(GL_ARRAY_BUFFER, buffer);
332 	gl.bufferData(GL_ARRAY_BUFFER, (int)localBuf.size(), &localBuf[0], usage);
333 	gl.bindBuffer(GL_ARRAY_BUFFER, 0);
334 	GLU_EXPECT_NO_ERROR(gl.getError(), "Uploading buffer data failed");
335 }
336 
337 // VertexBuffer
338 
VertexBuffer(const RenderContext & context,int numBindings,const VertexArrayBinding * bindings,Type type)339 VertexBuffer::VertexBuffer (const RenderContext& context, int numBindings, const VertexArrayBinding* bindings, Type type)
340 	: Buffer(context)
341 {
342 	const glw::Functions&	gl		= context.getFunctions();
343 	const deUint32			usage	= GL_STATIC_DRAW;
344 	VertexBufferLayout		layout;
345 
346 	if (!areVertexArrayLocationsValid(bindings, bindings+numBindings))
347 		throw tcu::TestError("Invalid vertex array locations");
348 
349 	if (type == TYPE_PLANAR)
350 		computeNonStridedBufferLayout(layout, bindings, bindings+numBindings);
351 	else
352 		throw tcu::InternalError("Strided layout is not yet supported");
353 
354 	std::vector<VertexArrayPointer> srcPtrs(numBindings);
355 	for (int ndx = 0; ndx < numBindings; ndx++)
356 		srcPtrs[ndx] = bindings[ndx].pointer;
357 
358 	DE_ASSERT(srcPtrs.size() == layout.attributes.size());
359 	if (!srcPtrs.empty())
360 		uploadBufferData(gl, m_object, usage, layout, &srcPtrs[0]);
361 
362 	// Construct descriptor.
363 	m_layout.buffer		= m_object;
364 	m_layout.attributes	= layout.attributes;
365 }
366 
~VertexBuffer(void)367 VertexBuffer::~VertexBuffer (void)
368 {
369 }
370 
371 // IndexBuffer
372 
IndexBuffer(const RenderContext & context,IndexType indexType,int numIndices,const void * indices)373 IndexBuffer::IndexBuffer (const RenderContext& context, IndexType indexType, int numIndices, const void* indices)
374 	: Buffer(context)
375 {
376 	const glw::Functions&	gl		= context.getFunctions();
377 	const deUint32			usage	= GL_STATIC_DRAW;
378 
379 	gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_object);
380 	gl.bufferData(GL_ELEMENT_ARRAY_BUFFER, numIndices*getIndexSize(indexType), indices, usage);
381 	gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
382 	GLU_EXPECT_NO_ERROR(gl.getError(), "Uploading index data failed");
383 }
384 
~IndexBuffer(void)385 IndexBuffer::~IndexBuffer (void)
386 {
387 }
388 
getUserPointerDescriptor(const VertexArrayBinding & vertexArray)389 static inline VertexAttributeDescriptor getUserPointerDescriptor (const VertexArrayBinding& vertexArray)
390 {
391 	DE_ASSERT(vertexArray.binding.type == BindingPoint::BPTYPE_LOCATION);
392 
393 	return VertexAttributeDescriptor(vertexArray.binding.location,
394 									 vertexArray.pointer.componentType,
395 									 vertexArray.pointer.convert,
396 									 vertexArray.pointer.numComponents,
397 									 vertexArray.pointer.numElements,
398 									 vertexArray.pointer.stride,
399 									 vertexArray.pointer.data);
400 }
401 
402 //! Setup VA according to allocation spec. Assumes that other state (VAO binding, buffer) is set already.
setVertexAttribPointer(const glw::Functions & gl,const VertexAttributeDescriptor & va)403 static void setVertexAttribPointer (const glw::Functions& gl, const VertexAttributeDescriptor& va)
404 {
405 	const bool		isIntType		= de::inRange<int>(va.componentType, VTX_COMP_UNSIGNED_INT8, VTX_COMP_SIGNED_INT32);
406 	const bool		isSpecialType	= de::inRange<int>(va.componentType, VTX_COMP_FIXED, VTX_COMP_FLOAT);
407 	const deUint32	compTypeGL		= getVtxCompGLType(va.componentType);
408 
409 	DE_ASSERT(isIntType != isSpecialType); // Must be either int or special type.
410 	DE_ASSERT(isIntType || va.convert == VTX_COMP_CONVERT_NONE); // Conversion allowed only for special types.
411 	DE_UNREF(isSpecialType);
412 
413 	gl.enableVertexAttribArray(va.location);
414 
415 	if (isIntType && va.convert == VTX_COMP_CONVERT_NONE)
416 		gl.vertexAttribIPointer(va.location, va.numComponents, compTypeGL, va.stride, va.pointer);
417 	else
418 		gl.vertexAttribPointer(va.location, va.numComponents, compTypeGL, va.convert == VTX_COMP_CONVERT_NORMALIZE_TO_FLOAT ? GL_TRUE : GL_FALSE, va.stride, va.pointer);
419 }
420 
421 //! Setup vertex buffer and attributes.
setVertexBufferAttributes(const glw::Functions & gl,const VertexBufferDescriptor & buffer)422 static void setVertexBufferAttributes (const glw::Functions& gl, const VertexBufferDescriptor& buffer)
423 {
424 	gl.bindBuffer(GL_ARRAY_BUFFER, buffer.buffer);
425 
426 	for (std::vector<VertexAttributeDescriptor>::const_iterator vaIter = buffer.attributes.begin(); vaIter != buffer.attributes.end(); ++vaIter)
427 		setVertexAttribPointer(gl, *vaIter);
428 
429 	gl.bindBuffer(GL_ARRAY_BUFFER, 0);
430 }
431 
disableVertexArrays(const glw::Functions & gl,const std::vector<VertexArrayBinding> & bindings)432 static void disableVertexArrays (const glw::Functions& gl, const std::vector<VertexArrayBinding>& bindings)
433 {
434 	for (std::vector<VertexArrayBinding>::const_iterator vaIter = bindings.begin(); vaIter != bindings.end(); ++vaIter)
435 	{
436 		DE_ASSERT(vaIter->binding.type == BindingPoint::BPTYPE_LOCATION);
437 		gl.disableVertexAttribArray(vaIter->binding.location);
438 	}
439 }
440 
441 #if defined(DE_DEBUG)
isProgramActive(const RenderContext & context,deUint32 program)442 static bool isProgramActive (const RenderContext& context, deUint32 program)
443 {
444 	// \todo [2013-05-08 pyry] Is this query broken?
445 /*	deUint32 activeProgram = 0;
446 	context.getFunctions().getIntegerv(GL_ACTIVE_PROGRAM, (int*)&activeProgram);
447 	GLU_EXPECT_NO_ERROR(context.getFunctions().getError(), "oh");
448 	return activeProgram == program;*/
449 	DE_UNREF(context);
450 	DE_UNREF(program);
451 	return true;
452 }
453 
isDrawCallValid(int numVertexArrays,const VertexArrayBinding * vertexArrays,const PrimitiveList & primitives)454 static bool isDrawCallValid (int numVertexArrays, const VertexArrayBinding* vertexArrays, const PrimitiveList& primitives)
455 {
456 	if (numVertexArrays < 0)
457 		return false;
458 
459 	if ((primitives.indexType == INDEXTYPE_LAST) != (primitives.indices == 0))
460 		return false;
461 
462 	if (primitives.numElements < 0)
463 		return false;
464 
465 	if (!primitives.indices)
466 	{
467 		for (int ndx = 0; ndx < numVertexArrays; ndx++)
468 		{
469 			if (primitives.numElements > vertexArrays[ndx].pointer.numElements)
470 				return false;
471 		}
472 	}
473 	// \todo [2013-05-08 pyry] We could walk whole index array and determine index range
474 
475 	return true;
476 }
477 #endif // DE_DEBUG
478 
drawNonIndexed(const glw::Functions & gl,PrimitiveType type,int numElements)479 static inline void drawNonIndexed (const glw::Functions& gl, PrimitiveType type, int numElements)
480 {
481 	deUint32 mode = getPrimitiveGLType(type);
482 	gl.drawArrays(mode, 0, numElements);
483 }
484 
drawIndexed(const glw::Functions & gl,PrimitiveType type,int numElements,IndexType indexType,const void * indexPtr)485 static inline void drawIndexed (const glw::Functions& gl, PrimitiveType type, int numElements, IndexType indexType, const void* indexPtr)
486 {
487 	deUint32	mode		= getPrimitiveGLType(type);
488 	deUint32	indexGLType	= getIndexGLType(indexType);
489 
490 	gl.drawElements(mode, numElements, indexGLType, indexPtr);
491 }
492 
493 } // anonymous
494 
drawFromUserPointers(const RenderContext & context,deUint32 program,int numVertexArrays,const VertexArrayBinding * vertexArrays,const PrimitiveList & primitives,DrawUtilCallback * callback)495 void drawFromUserPointers (const RenderContext& context, deUint32 program, int numVertexArrays, const VertexArrayBinding* vertexArrays, const PrimitiveList& primitives, DrawUtilCallback* callback)
496 {
497 	const glw::Functions&				gl		= context.getFunctions();
498 	std::vector<VertexArrayBinding>		bindingsWithLocations;
499 
500 	DE_ASSERT(isDrawCallValid(numVertexArrays, vertexArrays, primitives));
501 	DE_ASSERT(isProgramActive(context, program));
502 
503 	// Lower bindings to locations.
504 	namedBindingsToProgramLocations(gl, program, vertexArrays, vertexArrays+numVertexArrays, std::inserter(bindingsWithLocations, bindingsWithLocations.begin()));
505 
506 	TCU_CHECK(areVertexArrayLocationsValid(bindingsWithLocations.begin(), bindingsWithLocations.end()));
507 
508 	// Set VA state.
509 	for (std::vector<VertexArrayBinding>::const_iterator vaIter = bindingsWithLocations.begin(); vaIter != bindingsWithLocations.end(); ++vaIter)
510 		setVertexAttribPointer(gl, getUserPointerDescriptor(*vaIter));
511 
512 	if (callback)
513 		callback->beforeDrawCall();
514 
515 	if (primitives.indices)
516 		drawIndexed(gl, primitives.type, primitives.numElements, primitives.indexType, primitives.indices);
517 	else
518 		drawNonIndexed(gl, primitives.type, primitives.numElements);
519 
520 	if (callback)
521 		callback->afterDrawCall();
522 
523 	// Disable attribute arrays or otherwise someone later on might get crash thanks to invalid pointers.
524 	disableVertexArrays(gl, bindingsWithLocations);
525 }
526 
drawFromBuffers(const RenderContext & context,deUint32 program,int numVertexArrays,const VertexArrayBinding * vertexArrays,const PrimitiveList & primitives,DrawUtilCallback * callback)527 void drawFromBuffers (const RenderContext& context, deUint32 program, int numVertexArrays, const VertexArrayBinding* vertexArrays, const PrimitiveList& primitives, DrawUtilCallback* callback)
528 {
529 	const glw::Functions&				gl		= context.getFunctions();
530 	std::vector<VertexArrayBinding>		bindingsWithLocations;
531 
532 	DE_ASSERT(isDrawCallValid(numVertexArrays, vertexArrays, primitives));
533 	DE_ASSERT(isProgramActive(context, program));
534 
535 	// Lower bindings to locations.
536 	namedBindingsToProgramLocations(gl, program, vertexArrays, vertexArrays+numVertexArrays, std::inserter(bindingsWithLocations, bindingsWithLocations.begin()));
537 
538 	TCU_CHECK(areVertexArrayLocationsValid(bindingsWithLocations.begin(), bindingsWithLocations.end()));
539 
540 	// Create buffers for duration of draw call.
541 	{
542 		VertexBuffer vertexBuffer (context, (int)bindingsWithLocations.size(), (bindingsWithLocations.empty()) ? (DE_NULL) : (&bindingsWithLocations[0]));
543 
544 		// Set state.
545 		setVertexBufferAttributes(gl, vertexBuffer.getDescriptor());
546 
547 		if (primitives.indices)
548 		{
549 			IndexBuffer indexBuffer(context, primitives.indexType, primitives.numElements, primitives.indices);
550 
551 			gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, *indexBuffer);
552 
553 			if (callback)
554 				callback->beforeDrawCall();
555 
556 			drawIndexed(gl, primitives.type, primitives.numElements, primitives.indexType, 0);
557 
558 			if (callback)
559 				callback->afterDrawCall();
560 
561 			gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
562 		}
563 		else
564 		{
565 			if (callback)
566 				callback->beforeDrawCall();
567 
568 			drawNonIndexed(gl, primitives.type, primitives.numElements);
569 
570 			if (callback)
571 				callback->afterDrawCall();
572 		}
573 	}
574 
575 	// Disable attribute arrays or otherwise someone later on might get crash thanks to invalid pointers.
576 	for (std::vector<VertexArrayBinding>::const_iterator vaIter = bindingsWithLocations.begin(); vaIter != bindingsWithLocations.end(); ++vaIter)
577 		gl.disableVertexAttribArray(vaIter->binding.location);
578 }
579 
drawFromVAOBuffers(const RenderContext & context,deUint32 program,int numVertexArrays,const VertexArrayBinding * vertexArrays,const PrimitiveList & primitives,DrawUtilCallback * callback)580 void drawFromVAOBuffers (const RenderContext& context, deUint32 program, int numVertexArrays, const VertexArrayBinding* vertexArrays, const PrimitiveList& primitives, DrawUtilCallback* callback)
581 {
582 	const glw::Functions&	gl		= context.getFunctions();
583 	VertexArray				vao		(context);
584 
585 	gl.bindVertexArray(*vao);
586 	drawFromBuffers(context, program, numVertexArrays, vertexArrays, primitives, callback);
587 	gl.bindVertexArray(0);
588 }
589 
draw(const RenderContext & context,deUint32 program,int numVertexArrays,const VertexArrayBinding * vertexArrays,const PrimitiveList & primitives,DrawUtilCallback * callback)590 void draw (const RenderContext& context, deUint32 program, int numVertexArrays, const VertexArrayBinding* vertexArrays, const PrimitiveList& primitives, DrawUtilCallback* callback)
591 {
592 	const glu::ContextType ctxType = context.getType();
593 
594 	if (isContextTypeGLCore(ctxType) || contextSupports(ctxType, ApiType::es(3,1)))
595 		drawFromVAOBuffers(context, program, numVertexArrays, vertexArrays, primitives, callback);
596 	else
597 	{
598 		DE_ASSERT(isContextTypeES(ctxType));
599 		drawFromUserPointers(context, program, numVertexArrays, vertexArrays, primitives, callback);
600 	}
601 }
602 
603 } // glu
604