1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program OpenGL ES 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 OpenGL ES 3plus wrapper context.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "gluES3PlusWrapperContext.hpp"
25 #include "gluRenderContext.hpp"
26 #include "gluRenderConfig.hpp"
27 #include "glwInitFunctions.hpp"
28 #include "glwFunctionLoader.hpp"
29 #include "gluContextFactory.hpp"
30 #include "gluContextInfo.hpp"
31 #include "gluShaderUtil.hpp"
32 #include "deThreadLocal.hpp"
33 #include "deSTLUtil.hpp"
34 #include "deUniquePtr.hpp"
35 #include "glwEnums.hpp"
36 
37 #include <sstream>
38 #include <vector>
39 #include <string>
40 #include <cstring>
41 #include <algorithm>
42 #include <map>
43 
44 namespace glu
45 {
46 
47 namespace es3plus
48 {
49 
50 using std::vector;
51 using std::string;
52 
53 class Context
54 {
55 public:
56 								Context			(const glu::RenderContext& ctx);
57 								~Context		(void);
58 
59 	void						addExtension	(const char* name);
60 
61 	const glw::Functions&		gl;			//!< GL 4.3 core context functions.
62 
63 	// Wrapper state.
64 	string						vendor;
65 	string						version;
66 	string						renderer;
67 	string						shadingLanguageVersion;
68 	string						extensions;
69 	vector<string>				extensionList;
70 	bool						primitiveRestartEnabled;
71 
72 	deUint32					defaultVAO;
73 	bool						defaultVAOBound;
74 
75 	const glu::GLSLVersion		nativeGLSLVersion;
76 };
77 
Context(const glu::RenderContext & ctx)78 Context::Context (const glu::RenderContext& ctx)
79 	: gl						(ctx.getFunctions())
80 	, vendor					("drawElements")
81 	, version					("OpenGL ES 3.1")
82 	, renderer					((const char*)gl.getString(GL_RENDERER))
83 	, shadingLanguageVersion	("OpenGL ES GLSL ES 3.1")
84 	, primitiveRestartEnabled	(false)
85 	, defaultVAO				(0)
86 	, defaultVAOBound			(false)
87 	, nativeGLSLVersion			(glu::getContextTypeGLSLVersion(ctx.getType()))
88 {
89 	const de::UniquePtr<glu::ContextInfo> ctxInfo(glu::ContextInfo::create(ctx));
90 
91 	gl.genVertexArrays(1, &defaultVAO);
92 	if (gl.getError() != GL_NO_ERROR || defaultVAO == 0)
93 		throw tcu::InternalError("Failed to allocate VAO for emulation");
94 
95 	gl.bindVertexArray(defaultVAO);
96 	if (gl.getError() != GL_NO_ERROR)
97 		throw tcu::InternalError("Failed to bind default VAO");
98 	defaultVAOBound = true;
99 
100 	gl.enable(GL_PROGRAM_POINT_SIZE);
101 	gl.getError(); // supress potential errors, feature is not critical
102 
103 	gl.enable(GL_TEXTURE_CUBE_MAP_SEAMLESS);
104 	gl.getError(); // suppress
105 
106 	// Extensions
107 	addExtension("GL_OES_texture_stencil8");
108 	addExtension("GL_OES_sample_shading");
109 	addExtension("GL_OES_sample_variables");
110 	addExtension("GL_OES_shader_multisample_interpolation");
111 	addExtension("GL_OES_shader_image_atomic");
112 	addExtension("GL_OES_texture_storage_multisample_2d_array");
113 
114 	// Enable only if base ctx supports these or compatible GL_NV_blend_equation_advanced ext
115 	if (ctxInfo->isExtensionSupported("GL_NV_blend_equation_advanced") ||
116 		ctxInfo->isExtensionSupported("GL_KHR_blend_equation_advanced"))
117 	{
118 		addExtension("GL_KHR_blend_equation_advanced");
119 	}
120 	if (ctxInfo->isExtensionSupported("GL_NV_blend_equation_advanced_coherent") ||
121 		ctxInfo->isExtensionSupported("GL_KHR_blend_equation_advanced_coherent"))
122 	{
123 		addExtension("GL_KHR_blend_equation_advanced_coherent");
124 	}
125 
126 	addExtension("GL_EXT_shader_io_blocks");
127 	addExtension("GL_EXT_geometry_shader");
128 	addExtension("GL_EXT_geometry_point_size");
129 	addExtension("GL_EXT_tessellation_shader");
130 	addExtension("GL_EXT_tessellation_point_size");
131 	addExtension("GL_EXT_gpu_shader5");
132 	addExtension("GL_KHR_debug");
133 	addExtension("GL_EXT_texture_cube_map_array");
134 	addExtension("GL_EXT_shader_implicit_conversions");
135 	addExtension("GL_EXT_primitive_bounding_box");
136 	addExtension("GL_EXT_texture_sRGB_decode");
137 	addExtension("GL_EXT_texture_border_clamp");
138 	addExtension("GL_EXT_texture_buffer");
139 	addExtension("GL_EXT_draw_buffers_indexed");
140 }
141 
~Context(void)142 Context::~Context (void)
143 {
144 	if (defaultVAO)
145 		gl.deleteVertexArrays(1, &defaultVAO);
146 }
147 
addExtension(const char * name)148 void Context::addExtension (const char* name)
149 {
150 	if (!extensions.empty())
151 		extensions += " ";
152 	extensions += name;
153 
154 	extensionList.push_back(name);
155 }
156 
157 static de::ThreadLocal tls_context;
158 
setCurrentContext(Context * context)159 void setCurrentContext (Context* context)
160 {
161 	tls_context.set(context);
162 }
163 
getCurrentContext(void)164 inline Context* getCurrentContext (void)
165 {
166 	return (Context*)tls_context.get();
167 }
168 
getIntegerv(deUint32 pname,deInt32 * params)169 static GLW_APICALL void GLW_APIENTRY getIntegerv (deUint32 pname, deInt32* params)
170 {
171 	Context* context = getCurrentContext();
172 
173 	if (context)
174 	{
175 		if (pname == GL_NUM_EXTENSIONS && params)
176 			*params = (deInt32)context->extensionList.size();
177 		else
178 			context->gl.getIntegerv(pname, params);
179 	}
180 }
181 
getString(deUint32 name)182 static GLW_APICALL const glw::GLubyte* GLW_APIENTRY getString (deUint32 name)
183 {
184 	Context* context = getCurrentContext();
185 
186 	if (context)
187 	{
188 		switch (name)
189 		{
190 			case GL_VENDOR:						return (const glw::GLubyte*)context->vendor.c_str();
191 			case GL_VERSION:					return (const glw::GLubyte*)context->version.c_str();
192 			case GL_RENDERER:					return (const glw::GLubyte*)context->renderer.c_str();
193 			case GL_SHADING_LANGUAGE_VERSION:	return (const glw::GLubyte*)context->shadingLanguageVersion.c_str();
194 			case GL_EXTENSIONS:					return (const glw::GLubyte*)context->extensions.c_str();
195 			default:							return context->gl.getString(name);
196 		}
197 	}
198 	else
199 		return DE_NULL;
200 }
201 
getStringi(deUint32 name,deUint32 index)202 static GLW_APICALL const glw::GLubyte* GLW_APIENTRY getStringi (deUint32 name, deUint32 index)
203 {
204 	Context* context = getCurrentContext();
205 
206 	if (context)
207 	{
208 		if (name == GL_EXTENSIONS)
209 		{
210 			if ((size_t)index < context->extensionList.size())
211 				return (const glw::GLubyte*)context->extensionList[index].c_str();
212 			else
213 				return context->gl.getStringi(name, ~0u);
214 		}
215 		else
216 			return context->gl.getStringi(name, index);
217 	}
218 	else
219 		return DE_NULL;
220 }
221 
enable(deUint32 cap)222 static GLW_APICALL void GLW_APIENTRY enable (deUint32 cap)
223 {
224 	Context* context = getCurrentContext();
225 
226 	if (context)
227 	{
228 		if (cap == GL_PRIMITIVE_RESTART_FIXED_INDEX)
229 		{
230 			context->primitiveRestartEnabled = true;
231 			// \todo [2013-09-30 pyry] Call to glPrimitiveRestartIndex() is required prior to all draw calls!
232 		}
233 		else
234 			context->gl.enable(cap);
235 	}
236 }
237 
disable(deUint32 cap)238 static GLW_APICALL void GLW_APIENTRY disable (deUint32 cap)
239 {
240 	Context* context = getCurrentContext();
241 
242 	if (context)
243 	{
244 		if (cap == GL_PRIMITIVE_RESTART_FIXED_INDEX)
245 			context->primitiveRestartEnabled = false;
246 		else
247 			context->gl.disable(cap);
248 	}
249 }
250 
bindVertexArray(deUint32 array)251 static GLW_APICALL void GLW_APIENTRY bindVertexArray (deUint32 array)
252 {
253 	Context* context = getCurrentContext();
254 
255 	if (context)
256 	{
257 		context->gl.bindVertexArray(array == 0 ? context->defaultVAO : array);
258 		context->defaultVAOBound = (array == 0);
259 	}
260 }
261 
hint(deUint32 target,deUint32 mode)262 static GLW_APICALL void GLW_APIENTRY hint (deUint32 target, deUint32 mode)
263 {
264 	Context* context = getCurrentContext();
265 
266 	if (context)
267 	{
268 		if (target != GL_GENERATE_MIPMAP_HINT)
269 			context->gl.hint(target, mode);
270 		// \todo [2013-09-30 pyry] Verify mode.
271 	}
272 }
273 
translateShaderSource(deUint32 shaderType,std::ostream & dst,const std::string & src,const std::vector<std::string> & filteredExtensions,GLSLVersion version)274 static void translateShaderSource (deUint32 shaderType, std::ostream& dst, const std::string& src, const std::vector<std::string>& filteredExtensions, GLSLVersion version)
275 {
276 	bool				foundVersion		= false;
277 	std::istringstream	istr				(src);
278 	std::string			line;
279 	int					srcLineNdx			= 1;
280 	bool				preprocessorSection	= true;
281 
282 	while (std::getline(istr, line, '\n'))
283 	{
284 		if (preprocessorSection && !line.empty() && line[0] != '#')
285 		{
286 			preprocessorSection = false;
287 
288 			// ARB_separate_shader_objects requires gl_PerVertex to be explicitly declared
289 			if (shaderType == GL_VERTEX_SHADER)
290 			{
291 				dst << "out gl_PerVertex {\n"
292 					<< "    vec4 gl_Position;\n"
293 					<< "    float gl_PointSize;\n"
294 					<< "    float gl_ClipDistance[];\n"
295 					<< "};\n"
296 					<< "#line " << (srcLineNdx + 1) << "\n";
297 			}
298 			else if (shaderType == GL_TESS_CONTROL_SHADER)
299 			{
300 				dst << "#extension GL_ARB_tessellation_shader : enable\n"
301 					<< "in gl_PerVertex {\n"
302 					<< "	highp vec4 gl_Position;\n"
303 					<< "	highp float gl_PointSize;\n"
304 					<< "} gl_in[gl_MaxPatchVertices];\n"
305 					<< "out gl_PerVertex {\n"
306 					<< "	highp vec4 gl_Position;\n"
307 					<< "	highp float gl_PointSize;\n"
308 					<< "} gl_out[];\n"
309 					<< "#line " << (srcLineNdx + 1) << "\n";
310 			}
311 			else if (shaderType == GL_TESS_EVALUATION_SHADER)
312 			{
313 				dst << "#extension GL_ARB_tessellation_shader : enable\n"
314 					<< "in gl_PerVertex {\n"
315 					<< "	highp vec4 gl_Position;\n"
316 					<< "	highp float gl_PointSize;\n"
317 					<< "} gl_in[gl_MaxPatchVertices];\n"
318 					<< "out gl_PerVertex {\n"
319 					<< "	highp vec4 gl_Position;\n"
320 					<< "	highp float gl_PointSize;\n"
321 					<< "};\n"
322 					<< "#line " << (srcLineNdx + 1) << "\n";
323 			}
324 			else if (shaderType == GL_GEOMETRY_SHADER)
325 			{
326 				dst << "in gl_PerVertex {\n"
327 					<< "	highp vec4 gl_Position;\n"
328 					<< "	highp float gl_PointSize;\n"
329 					<< "} gl_in[];\n"
330 					<< "out gl_PerVertex {\n"
331 					<< "	highp vec4 gl_Position;\n"
332 					<< "	highp float gl_PointSize;\n"
333 					<< "};\n"
334 					<< "#line " << (srcLineNdx + 1) << "\n";
335 			}
336 
337 			// GL_EXT_primitive_bounding_box tessellation no-op fallback
338 			if (shaderType == GL_TESS_CONTROL_SHADER)
339 			{
340 				dst << "#define gl_BoundingBoxEXT _dummy_unused_output_for_primitive_bbox\n"
341 					<< "patch out vec4 _dummy_unused_output_for_primitive_bbox[2];\n"
342 					<< "#line " << (srcLineNdx + 1) << "\n";
343 			}
344 		}
345 
346 		if (line == "#version 310 es")
347 		{
348 			foundVersion = true;
349 			dst << glu::getGLSLVersionDeclaration(version) << "\n";
350 		}
351 		else if (line == "#version 300 es")
352 		{
353 			foundVersion = true;
354 			dst << "#version 330\n";
355 		}
356 		else if (line.substr(0, 10) == "precision ")
357 		{
358 			const size_t	precPos		= 10;
359 			const size_t	precEndPos	= line.find(' ', precPos);
360 			const size_t	endPos		= line.find(';');
361 
362 			if (precEndPos != std::string::npos && endPos != std::string::npos && endPos > precEndPos+1)
363 			{
364 				const size_t		typePos		= precEndPos+1;
365 				const std::string	precision	= line.substr(precPos, precEndPos-precPos);
366 				const std::string	type		= line.substr(typePos, endPos-typePos);
367 				const bool			precOk		= precision == "lowp" || precision == "mediump" || precision == "highp";
368 
369 				if (precOk &&
370 					(type == "image2D" || type == "uimage2D" || type == "iimage2D" ||
371 					 type == "imageCube" || type == "uimageCube" || type == "iimageCube" ||
372 					 type == "image3D" || type == "iimage3D" || type == "uimage3D" ||
373 					 type == "image2DArray" || type == "iimage2DArray" || type == "uimage2DArray" ||
374 					 type == "imageCubeArray" || type == "iimageCubeArray" || type == "uimageCubeArray"))
375 					dst << "// "; // Filter out statement
376 			}
377 
378 			dst << line << "\n";
379 		}
380 		else if (line.substr(0, 11) == "#extension ")
381 		{
382 			const size_t	extNamePos		= 11;
383 			const size_t	extNameEndPos	= line.find_first_of(" :", extNamePos);
384 			const size_t	behaviorPos		= line.find_first_not_of(" :", extNameEndPos);
385 
386 			if (extNameEndPos != std::string::npos && behaviorPos != std::string::npos)
387 			{
388 				const std::string	extName				= line.substr(extNamePos, extNameEndPos-extNamePos);
389 				const std::string	behavior			= line.substr(behaviorPos);
390 				const bool			filteredExtension	= de::contains(filteredExtensions.begin(), filteredExtensions.end(), extName);
391 				const bool			validBehavior		= behavior == "require" || behavior == "enable" || behavior == "warn" || behavior == "disable";
392 
393 				if (filteredExtension && validBehavior)
394 					dst << "// "; // Filter out extension
395 			}
396 			dst << line << "\n";
397 		}
398 		else if (line.substr(0, 21) == "layout(blend_support_")
399 			dst << "// " << line << "\n";
400 		else
401 			dst << line << "\n";
402 
403 		srcLineNdx += 1;
404 	}
405 
406 	DE_ASSERT(foundVersion);
407 	DE_UNREF(foundVersion);
408 }
409 
translateShaderSources(deUint32 shaderType,deInt32 count,const char * const * strings,const int * length,const std::vector<std::string> & filteredExtensions,GLSLVersion version)410 static std::string translateShaderSources (deUint32 shaderType, deInt32 count, const char* const* strings, const int* length, const std::vector<std::string>& filteredExtensions, GLSLVersion version)
411 {
412 	std::ostringstream	srcIn;
413 	std::ostringstream	srcOut;
414 
415 	for (int ndx = 0; ndx < count; ndx++)
416 	{
417 		const int len = length && length[ndx] >= 0 ? length[ndx] : (int)strlen(strings[ndx]);
418 		srcIn << std::string(strings[ndx], strings[ndx] + len);
419 	}
420 
421 	translateShaderSource(shaderType, srcOut, srcIn.str(), filteredExtensions, version);
422 
423 	return srcOut.str();
424 }
425 
shaderSource(deUint32 shader,deInt32 count,const char * const * strings,const int * length)426 static GLW_APICALL void GLW_APIENTRY shaderSource (deUint32 shader, deInt32 count, const char* const* strings, const int* length)
427 {
428 	Context* context = getCurrentContext();
429 
430 	if (context)
431 	{
432 		if (count > 0 && strings)
433 		{
434 			deInt32				shaderType = GL_NONE;
435 			context->gl.getShaderiv(shader, GL_SHADER_TYPE, &shaderType);
436 			{
437 				const std::string	translatedSrc	= translateShaderSources(shaderType, count, strings, length, context->extensionList, context->nativeGLSLVersion);
438 				const char*			srcPtr			= translatedSrc.c_str();
439 				context->gl.shaderSource(shader, 1, &srcPtr, DE_NULL);
440 			}
441 		}
442 		else
443 			context->gl.shaderSource(shader, count, strings, length);
444 	}
445 }
446 
bindFramebuffer(deUint32 target,deUint32 framebuffer)447 static GLW_APICALL void GLW_APIENTRY bindFramebuffer (deUint32 target, deUint32 framebuffer)
448 {
449 	Context* context = getCurrentContext();
450 
451 	if (context)
452 	{
453 		context->gl.bindFramebuffer(target, framebuffer);
454 
455 		// Emulate ES behavior where sRGB conversion is only controlled by color buffer format.
456 		if (target == GL_FRAMEBUFFER || target == GL_DRAW_FRAMEBUFFER || target == GL_READ_FRAMEBUFFER)
457 			((framebuffer != 0) ? context->gl.enable : context->gl.disable)(GL_FRAMEBUFFER_SRGB);
458 	}
459 }
460 
blendBarrierKHR(void)461 static GLW_APICALL void GLW_APIENTRY blendBarrierKHR (void)
462 {
463 	Context* context = getCurrentContext();
464 
465 	if (context)
466 	{
467 		// \todo [2014-03-18 pyry] Use BlendBarrierNV() if supported
468 		context->gl.finish();
469 	}
470 }
471 
createShaderProgramv(deUint32 type,deInt32 count,const char * const * strings)472 static GLW_APICALL deUint32 GLW_APIENTRY createShaderProgramv (deUint32 type, deInt32 count, const char* const* strings)
473 {
474 	Context* context = getCurrentContext();
475 
476 	if (context)
477 	{
478 		if (count > 0 && strings)
479 		{
480 			const std::string	translatedSrc	= translateShaderSources(type, count, strings, DE_NULL, context->extensionList, context->nativeGLSLVersion);
481 			const char*			srcPtr			= translatedSrc.c_str();
482 			return context->gl.createShaderProgramv(type, 1, &srcPtr);
483 		}
484 		else
485 			return context->gl.createShaderProgramv(type, count, strings);
486 	}
487 	return 0;
488 }
489 
dummyPrimitiveBoundingBox(float minX,float minY,float minZ,float minW,float maxX,float maxY,float maxZ,float maxW)490 static GLW_APICALL void GLW_APIENTRY dummyPrimitiveBoundingBox (float minX, float minY, float minZ, float minW, float maxX, float maxY, float maxZ, float maxW)
491 {
492 	// dummy no-op. No-op is a valid implementation. States queries are not emulated.
493 	DE_UNREF(minX);
494 	DE_UNREF(minY);
495 	DE_UNREF(minZ);
496 	DE_UNREF(minW);
497 	DE_UNREF(maxX);
498 	DE_UNREF(maxY);
499 	DE_UNREF(maxZ);
500 	DE_UNREF(maxW);
501 }
502 
initFunctions(glw::Functions * dst,const glw::Functions & src)503 static void initFunctions (glw::Functions* dst, const glw::Functions& src)
504 {
505 	// Functions directly passed to GL context.
506 #include "gluES3PlusWrapperFuncs.inl"
507 
508 	// Wrapped functions.
509 	dst->bindVertexArray		= bindVertexArray;
510 	dst->disable				= disable;
511 	dst->enable					= enable;
512 	dst->getIntegerv			= getIntegerv;
513 	dst->getString				= getString;
514 	dst->getStringi				= getStringi;
515 	dst->hint					= hint;
516 	dst->shaderSource			= shaderSource;
517 	dst->createShaderProgramv	= createShaderProgramv;
518 	dst->bindFramebuffer		= bindFramebuffer;
519 
520 	// Extension functions
521 	{
522 		using std::map;
523 
524 		class ExtFuncLoader : public glw::FunctionLoader
525 		{
526 		public:
527 			ExtFuncLoader (const map<string, glw::GenericFuncType>& extFuncs)
528 				: m_extFuncs(extFuncs)
529 			{
530 			}
531 
532 			glw::GenericFuncType get (const char* name) const
533 			{
534 				map<string, glw::GenericFuncType>::const_iterator pos = m_extFuncs.find(name);
535 				return pos != m_extFuncs.end() ? pos->second : DE_NULL;
536 			}
537 
538 		private:
539 			const map<string, glw::GenericFuncType>& m_extFuncs;
540 		};
541 
542 		map<string, glw::GenericFuncType>	extFuncMap;
543 		const ExtFuncLoader					extFuncLoader	(extFuncMap);
544 
545 		// OES_sample_shading
546 		extFuncMap["glMinSampleShadingOES"]			= (glw::GenericFuncType)src.minSampleShading;
547 
548 		// OES_texture_storage_multisample_2d_array
549 		extFuncMap["glTexStorage3DMultisampleOES"]	= (glw::GenericFuncType)src.texStorage3DMultisample;
550 
551 		// KHR_blend_equation_advanced
552 		extFuncMap["glBlendBarrierKHR"]				= (glw::GenericFuncType)blendBarrierKHR;
553 
554 		// EXT_tessellation_shader
555 		extFuncMap["glPatchParameteriEXT"]			= (glw::GenericFuncType)src.patchParameteri;
556 
557 		// EXT_geometry_shader
558 		extFuncMap["glFramebufferTextureEXT"]		= (glw::GenericFuncType)src.framebufferTexture;
559 
560 		// KHR_debug
561 		extFuncMap["glDebugMessageControlKHR"]		= (glw::GenericFuncType)src.debugMessageControl;
562 		extFuncMap["glDebugMessageInsertKHR"]		= (glw::GenericFuncType)src.debugMessageInsert;
563 		extFuncMap["glDebugMessageCallbackKHR"]		= (glw::GenericFuncType)src.debugMessageCallback;
564 		extFuncMap["glGetDebugMessageLogKHR"]		= (glw::GenericFuncType)src.getDebugMessageLog;
565 		extFuncMap["glGetPointervKHR"] 				= (glw::GenericFuncType)src.getPointerv;
566 		extFuncMap["glPushDebugGroupKHR"]			= (glw::GenericFuncType)src.pushDebugGroup;
567 		extFuncMap["glPopDebugGroupKHR"] 			= (glw::GenericFuncType)src.popDebugGroup;
568 		extFuncMap["glObjectLabelKHR"] 				= (glw::GenericFuncType)src.objectLabel;
569 		extFuncMap["glGetObjectLabelKHR"]			= (glw::GenericFuncType)src.getObjectLabel;
570 		extFuncMap["glObjectPtrLabelKHR"]			= (glw::GenericFuncType)src.objectPtrLabel;
571 		extFuncMap["glGetObjectPtrLabelKHR"]		= (glw::GenericFuncType)src.getObjectPtrLabel;
572 
573 		// GL_EXT_primitive_bounding_box (dummy no-op)
574 		extFuncMap["glPrimitiveBoundingBoxEXT"]		= (glw::GenericFuncType)dummyPrimitiveBoundingBox;
575 
576 		// GL_EXT_texture_border_clamp
577 		extFuncMap["glTexParameterIivEXT"]			= (glw::GenericFuncType)src.texParameterIiv;
578 		extFuncMap["glTexParameterIuivEXT"]			= (glw::GenericFuncType)src.texParameterIuiv;
579 		extFuncMap["glGetTexParameterIivEXT"]		= (glw::GenericFuncType)src.getTexParameterIiv;
580 		extFuncMap["glGetTexParameterIuivEXT"]		= (glw::GenericFuncType)src.getTexParameterIuiv;
581 		extFuncMap["glSamplerParameterIivEXT"]		= (glw::GenericFuncType)src.samplerParameterIiv;
582 		extFuncMap["glSamplerParameterIuivEXT"]		= (glw::GenericFuncType)src.samplerParameterIuiv;
583 		extFuncMap["glGetSamplerParameterIivEXT"]	= (glw::GenericFuncType)src.getSamplerParameterIiv;
584 		extFuncMap["glGetSamplerParameterIuivEXT"]	= (glw::GenericFuncType)src.getSamplerParameterIuiv;
585 
586 		// GL_EXT_texture_buffer
587 		extFuncMap["glTexBufferEXT"]				= (glw::GenericFuncType)src.texBuffer;
588 		extFuncMap["glTexBufferRangeEXT"]			= (glw::GenericFuncType)src.texBufferRange;
589 
590 		// GL_EXT_draw_buffers_indexed
591 		extFuncMap["glEnableiEXT"]					= (glw::GenericFuncType)src.enablei;
592 		extFuncMap["glDisableiEXT"]					= (glw::GenericFuncType)src.disablei;
593 		extFuncMap["glBlendEquationiEXT"]			= (glw::GenericFuncType)src.blendEquationi;
594 		extFuncMap["glBlendEquationSeparateiEXT"]	= (glw::GenericFuncType)src.blendEquationSeparatei;
595 		extFuncMap["glBlendFunciEXT"]				= (glw::GenericFuncType)src.blendFunci;
596 		extFuncMap["glBlendFuncSeparateiEXT"]		= (glw::GenericFuncType)src.blendFuncSeparatei;
597 		extFuncMap["glColorMaskiEXT"]				= (glw::GenericFuncType)src.colorMaski;
598 		extFuncMap["glIsEnablediEXT"]				= (glw::GenericFuncType)src.isEnabledi;
599 
600 		{
601 			int	numExts	= 0;
602 			dst->getIntegerv(GL_NUM_EXTENSIONS, &numExts);
603 
604 			if (numExts > 0)
605 			{
606 				vector<const char*> extStr(numExts);
607 
608 				for (int ndx = 0; ndx < numExts; ndx++)
609 					extStr[ndx] = (const char*)dst->getStringi(GL_EXTENSIONS, ndx);
610 
611 				glw::initExtensionsES(dst, &extFuncLoader, (int)extStr.size(), &extStr[0]);
612 			}
613 		}
614 	}
615 }
616 
617 } // es3plus
618 
ES3PlusWrapperContext(const ContextFactory & factory,const RenderConfig & config,const tcu::CommandLine & cmdLine)619 ES3PlusWrapperContext::ES3PlusWrapperContext (const ContextFactory& factory, const RenderConfig& config, const tcu::CommandLine& cmdLine)
620 	: m_context		(DE_NULL)
621 	, m_wrapperCtx	(DE_NULL)
622 {
623 	// Flags that are valid for both core & es context. Currently only excludes CONTEXT_FORWARD_COMPATIBLE
624 	const ContextFlags validContextFlags = CONTEXT_ROBUST | CONTEXT_DEBUG;
625 
626 	static const ContextType wrappableNativeTypes[] =
627 	{
628 		ContextType(ApiType::core(4,4), config.type.getFlags() & validContextFlags),	// !< higher in the list, preferred
629 		ContextType(ApiType::core(4,3), config.type.getFlags() & validContextFlags),
630 	};
631 
632 	if (config.type.getAPI() != ApiType::es(3,1))
633 		throw tcu::NotSupportedError("Unsupported context type (ES3.1 wrapper supports only ES3.1)");
634 
635 	// try to create any wrappable context
636 
637 	for (int nativeCtxNdx = 0; nativeCtxNdx < DE_LENGTH_OF_ARRAY(wrappableNativeTypes); ++nativeCtxNdx)
638 	{
639 		glu::ContextType nativeContext = wrappableNativeTypes[nativeCtxNdx];
640 
641 		try
642 		{
643 			glu::RenderConfig nativeConfig = config;
644 			nativeConfig.type = nativeContext;
645 
646 			m_context		= factory.createContext(nativeConfig, cmdLine);
647 			m_wrapperCtx	= new es3plus::Context(*m_context);
648 
649 			es3plus::setCurrentContext(m_wrapperCtx);
650 			es3plus::initFunctions(&m_functions, m_context->getFunctions());
651 			break;
652 		}
653 		catch (...)
654 		{
655 			es3plus::setCurrentContext(DE_NULL);
656 
657 			delete m_wrapperCtx;
658 			delete m_context;
659 
660 			m_wrapperCtx = DE_NULL;
661 			m_context = DE_NULL;
662 
663 			// throw only if all tries failed (that is, this was the last potential target)
664 			if (nativeCtxNdx + 1 == DE_LENGTH_OF_ARRAY(wrappableNativeTypes))
665 				throw;
666 			else
667 				continue;
668 		}
669 	}
670 }
671 
~ES3PlusWrapperContext(void)672 ES3PlusWrapperContext::~ES3PlusWrapperContext (void)
673 {
674 	delete m_wrapperCtx;
675 	delete m_context;
676 }
677 
getType(void) const678 ContextType ES3PlusWrapperContext::getType (void) const
679 {
680 	return ContextType(ApiType::es(3,1), m_context->getType().getFlags());
681 }
682 
683 } // glu
684