1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program Tester Core
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 EGL utilities for interfacing with GL APIs.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "egluGLUtil.hpp"
25 
26 #include "egluUtil.hpp"
27 #include "eglwLibrary.hpp"
28 #include "eglwEnums.hpp"
29 #include "glwEnums.hpp"
30 
31 #include <vector>
32 
33 using std::vector;
34 
35 namespace eglu
36 {
37 
38 using namespace eglw;
39 
getImageGLTarget(EGLenum source)40 glw::GLenum getImageGLTarget (EGLenum source)
41 {
42 	switch (source)
43 	{
44 		case EGL_GL_TEXTURE_2D_KHR:						return GL_TEXTURE_2D;
45 		case EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_X_KHR:	return GL_TEXTURE_CUBE_MAP_POSITIVE_X;
46 		case EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_Y_KHR:	return GL_TEXTURE_CUBE_MAP_POSITIVE_Y;
47 		case EGL_GL_TEXTURE_CUBE_MAP_POSITIVE_Z_KHR:	return GL_TEXTURE_CUBE_MAP_POSITIVE_Z;
48 		case EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_X_KHR:	return GL_TEXTURE_CUBE_MAP_NEGATIVE_X;
49 		case EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_Y_KHR:	return GL_TEXTURE_CUBE_MAP_NEGATIVE_Y;
50 		case EGL_GL_TEXTURE_CUBE_MAP_NEGATIVE_Z_KHR:	return GL_TEXTURE_CUBE_MAP_NEGATIVE_Z;
51 		case EGL_GL_TEXTURE_3D_KHR:						return GL_TEXTURE_3D;
52 		case EGL_GL_RENDERBUFFER_KHR:					return GL_RENDERBUFFER;
53 		default:	DE_FATAL("Impossible");				return GL_NONE;
54 	}
55 }
56 
apiRenderableType(glu::ApiType apiType)57 EGLint apiRenderableType (glu::ApiType apiType)
58 {
59 	switch (apiType.getProfile())
60 	{
61 		case glu::PROFILE_CORE:
62 		case glu::PROFILE_COMPATIBILITY:
63 			return EGL_OPENGL_BIT;
64 		case glu::PROFILE_ES:
65 			switch (apiType.getMajorVersion())
66 			{
67 				case 1:		return EGL_OPENGL_ES_BIT;
68 				case 2:		return EGL_OPENGL_ES2_BIT;
69 				case 3:		return EGL_OPENGL_ES3_BIT_KHR;
70 				default:	DE_FATAL("Unknown OpenGL ES version");
71 			}
72 			break;
73 		default:
74 			DE_FATAL("Unknown GL API");
75 	}
76 
77 	return 0;
78 }
79 
createGLContext(const Library & egl,EGLDisplay display,EGLContext eglConfig,const glu::ContextType & contextType,eglw::EGLContext sharedContext,glu::ResetNotificationStrategy resetNotificationStrategy)80 EGLContext createGLContext (const Library&					egl,
81 							EGLDisplay						display,
82 							EGLContext						eglConfig,
83 							const glu::ContextType&			contextType,
84 							eglw::EGLContext				sharedContext,
85 							glu::ResetNotificationStrategy	resetNotificationStrategy)
86 {
87 	const bool			khrCreateContextSupported			= hasExtension(egl, display, "EGL_KHR_create_context");
88 	const bool			khrCreateContextNoErrorSupported	= hasExtension(egl, display, "EGL_KHR_create_context_no_error");
89 	EGLContext			context								= EGL_NO_CONTEXT;
90 	EGLenum				api									= EGL_NONE;
91 	vector<EGLint>		attribList;
92 
93 	if (glu::isContextTypeES(contextType))
94 	{
95 		api = EGL_OPENGL_ES_API;
96 
97 		if (contextType.getMajorVersion() <= 2)
98 		{
99 			attribList.push_back(EGL_CONTEXT_CLIENT_VERSION);
100 			attribList.push_back(contextType.getMajorVersion());
101 		}
102 		else
103 		{
104 			if (!khrCreateContextSupported)
105 				TCU_THROW(NotSupportedError, "EGL_KHR_create_context is required for OpenGL ES 3.0 and newer");
106 
107 			attribList.push_back(EGL_CONTEXT_MAJOR_VERSION_KHR);
108 			attribList.push_back(contextType.getMajorVersion());
109 			attribList.push_back(EGL_CONTEXT_MINOR_VERSION_KHR);
110 			attribList.push_back(contextType.getMinorVersion());
111 		}
112 	}
113 	else
114 	{
115 		DE_ASSERT(glu::isContextTypeGLCore(contextType) || glu::isContextTypeGLCompatibility(contextType));
116 
117 		if (!khrCreateContextSupported)
118 			TCU_THROW(NotSupportedError, "EGL_KHR_create_context is required for OpenGL context creation");
119 
120 		api = EGL_OPENGL_API;
121 
122 		attribList.push_back(EGL_CONTEXT_MAJOR_VERSION_KHR);
123 		attribList.push_back(contextType.getMajorVersion());
124 		attribList.push_back(EGL_CONTEXT_MINOR_VERSION_KHR);
125 		attribList.push_back(contextType.getMinorVersion());
126 		attribList.push_back(EGL_CONTEXT_OPENGL_PROFILE_MASK_KHR);
127 		attribList.push_back(glu::isContextTypeGLCore(contextType) ? EGL_CONTEXT_OPENGL_CORE_PROFILE_BIT_KHR
128 																   : EGL_CONTEXT_OPENGL_COMPATIBILITY_PROFILE_BIT_KHR);
129 	}
130 
131 	if (contextType.getFlags() != glu::ContextFlags(0))
132 	{
133 		EGLint flags = 0;
134 
135 		if (!khrCreateContextSupported)
136 			TCU_THROW(NotSupportedError, "EGL_KHR_create_context is required for creating robust/debug/forward-compatible contexts");
137 
138 		if ((contextType.getFlags() & glu::CONTEXT_DEBUG) != 0)
139 			flags |= EGL_CONTEXT_OPENGL_DEBUG_BIT_KHR;
140 
141 		if ((contextType.getFlags() & glu::CONTEXT_ROBUST) != 0)
142 		{
143 			if (glu::isContextTypeES(contextType))
144 			{
145 				if (!hasExtension(egl, display, "EGL_EXT_create_context_robustness") && (getVersion(egl, display) < Version(1, 5)))
146 					TCU_THROW(NotSupportedError, "EGL_EXT_create_context_robustness is required for creating robust context");
147 
148 				attribList.push_back(EGL_CONTEXT_OPENGL_ROBUST_ACCESS_EXT);
149 				attribList.push_back(EGL_TRUE);
150 			}
151 			else
152 				flags |= EGL_CONTEXT_OPENGL_ROBUST_ACCESS_BIT_KHR;
153 		}
154 
155 		if ((contextType.getFlags() & glu::CONTEXT_NO_ERROR) != 0)
156 		{
157 			if (khrCreateContextNoErrorSupported)
158 			{
159 				attribList.push_back(EGL_CONTEXT_OPENGL_NO_ERROR_KHR);
160 				attribList.push_back(EGL_TRUE);
161 			}
162 			else
163 				throw tcu::NotSupportedError("EGL_KHR_create_context_no_error is required for creating no-error contexts");
164 		}
165 
166 		if ((contextType.getFlags() & glu::CONTEXT_FORWARD_COMPATIBLE) != 0)
167 		{
168 			if (!glu::isContextTypeGLCore(contextType))
169 				TCU_THROW(InternalError, "Only OpenGL core contexts can be forward-compatible");
170 
171 			flags |= EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE_BIT_KHR;
172 		}
173 
174 		attribList.push_back(EGL_CONTEXT_FLAGS_KHR);
175 		attribList.push_back(flags);
176 
177 		if (resetNotificationStrategy != glu::RESET_NOTIFICATION_STRATEGY_NOT_SPECIFIED)
178 		{
179 			if (getVersion(egl, display) >= Version(1, 5) || glu::isContextTypeGLCore(contextType) || glu::isContextTypeGLCompatibility(contextType))
180 				attribList.push_back(EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_KHR);
181 			else if (hasExtension(egl, display, "EGL_EXT_create_context_robustness"))
182 				attribList.push_back(EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_EXT);
183 			else
184 				TCU_THROW(NotSupportedError, "EGL 1.5 or EGL_EXT_create_context_robustness is required for creating robust context");
185 
186 			if (resetNotificationStrategy == glu::RESET_NOTIFICATION_STRATEGY_NO_RESET_NOTIFICATION)
187 				attribList.push_back(EGL_NO_RESET_NOTIFICATION_KHR);
188 			else if (resetNotificationStrategy == glu::RESET_NOTIFICATION_STRATEGY_LOSE_CONTEXT_ON_RESET)
189 				attribList.push_back(EGL_LOSE_CONTEXT_ON_RESET_KHR);
190 			else
191 				TCU_THROW(InternalError, "Unknown reset notification strategy");
192 		}
193 	}
194 
195 	attribList.push_back(EGL_NONE);
196 
197 	EGLU_CHECK_CALL(egl, bindAPI(api));
198 	context = egl.createContext(display, eglConfig, sharedContext, &(attribList[0]));
199 	EGLU_CHECK_MSG(egl, "eglCreateContext()");
200 
201 	return context;
202 }
203 
configMatches(const eglw::Library & egl,eglw::EGLDisplay display,eglw::EGLConfig eglConfig,const glu::RenderConfig & renderConfig)204 static bool configMatches (const eglw::Library& egl, eglw::EGLDisplay display, eglw::EGLConfig eglConfig, const glu::RenderConfig& renderConfig)
205 {
206 	// \todo [2014-03-12 pyry] Check other attributes like double-buffer bit.
207 
208 	{
209 		EGLint		renderableType		= 0;
210 		EGLint		requiredRenderable	= apiRenderableType(renderConfig.type.getAPI());
211 
212 		EGLU_CHECK_CALL(egl, getConfigAttrib(display, eglConfig, EGL_RENDERABLE_TYPE, &renderableType));
213 
214 		if ((renderableType & requiredRenderable) == 0)
215 			return false;
216 	}
217 
218 	if (renderConfig.surfaceType != glu::RenderConfig::SURFACETYPE_DONT_CARE)
219 	{
220 		EGLint		surfaceType		= 0;
221 		EGLint		requiredSurface	= 0;
222 
223 		switch (renderConfig.surfaceType)
224 		{
225 			case glu::RenderConfig::SURFACETYPE_WINDOW:				requiredSurface = EGL_WINDOW_BIT;	break;
226 			case glu::RenderConfig::SURFACETYPE_OFFSCREEN_NATIVE:	requiredSurface = EGL_PIXMAP_BIT;	break;
227 			case glu::RenderConfig::SURFACETYPE_OFFSCREEN_GENERIC:	requiredSurface = EGL_PBUFFER_BIT;	break;
228 			default:
229 				DE_ASSERT(false);
230 		}
231 
232 		EGLU_CHECK_CALL(egl, getConfigAttrib(display, eglConfig, EGL_SURFACE_TYPE, &surfaceType));
233 
234 		if ((surfaceType & requiredSurface) == 0)
235 			return false;
236 	}
237 
238 	{
239 		static const struct
240 		{
241 			int	glu::RenderConfig::*field;
242 			EGLint attrib;
243 		} s_attribs[] =
244 		{
245 			{ &glu::RenderConfig::id,			EGL_CONFIG_ID		},
246 			{ &glu::RenderConfig::redBits,		EGL_RED_SIZE		},
247 			{ &glu::RenderConfig::greenBits,	EGL_GREEN_SIZE		},
248 			{ &glu::RenderConfig::blueBits,		EGL_BLUE_SIZE		},
249 			{ &glu::RenderConfig::alphaBits,	EGL_ALPHA_SIZE		},
250 			{ &glu::RenderConfig::depthBits,	EGL_DEPTH_SIZE		},
251 			{ &glu::RenderConfig::stencilBits,	EGL_STENCIL_SIZE	},
252 			{ &glu::RenderConfig::numSamples,	EGL_SAMPLES			},
253 		};
254 
255 		for (int attribNdx = 0; attribNdx < DE_LENGTH_OF_ARRAY(s_attribs); attribNdx++)
256 		{
257 			if (renderConfig.*s_attribs[attribNdx].field != glu::RenderConfig::DONT_CARE)
258 			{
259 				EGLint value = 0;
260 				EGLU_CHECK_CALL(egl, getConfigAttrib(display, eglConfig, s_attribs[attribNdx].attrib, &value));
261 				if (value != renderConfig.*s_attribs[attribNdx].field)
262 					return false;
263 			}
264 		}
265 	}
266 
267 	return true;
268 }
269 
chooseConfig(const Library & egl,EGLDisplay display,const glu::RenderConfig & config)270 EGLConfig chooseConfig (const Library& egl, EGLDisplay display, const glu::RenderConfig& config)
271 {
272 	const std::vector<EGLConfig> configs = eglu::getConfigs(egl, display);
273 
274 	for (vector<EGLConfig>::const_iterator iter = configs.begin(); iter != configs.end(); ++iter)
275 	{
276 		if (configMatches(egl, display, *iter, config))
277 			return *iter;
278 	}
279 
280 	throw tcu::NotSupportedError("Matching EGL config not found", DE_NULL, __FILE__, __LINE__);
281 }
282 
283 }
284