1 /*-------------------------------------------------------------------------
2 * drawElements Quality Program Tester Core
3 * ----------------------------------------
5 * Copyright 2014 The Android Open Source Project
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
11 * http://www.apache.org/licenses/LICENSE-2.0
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.
21 * \brief EGL utilities for interfacing with GL APIs.
22 *//*--------------------------------------------------------------------*/
24 #include "egluGLUtil.hpp"
26 #include "egluUtil.hpp"
27 #include "eglwLibrary.hpp"
28 #include "eglwEnums.hpp"
29 #include "glwEnums.hpp"
40 glw::GLenum getImageGLTarget (EGLenum source)
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;
57 EGLint apiRenderableType (glu::ApiType apiType)
59 switch (apiType.getProfile())
61 case glu::PROFILE_CORE:
62 case glu::PROFILE_COMPATIBILITY:
63 return EGL_OPENGL_BIT;
65 switch (apiType.getMajorVersion())
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"); break;
74 DE_FATAL("Unknown GL API");
80 EGLContext createGLContext (const Library& egl,
83 const glu::ContextType& contextType,
84 eglw::EGLContext sharedContext,
85 glu::ResetNotificationStrategy resetNotificationStrategy)
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;
93 if (glu::isContextTypeES(contextType))
95 api = EGL_OPENGL_ES_API;
97 if (contextType.getMajorVersion() <= 2)
99 attribList.push_back(EGL_CONTEXT_CLIENT_VERSION);
100 attribList.push_back(contextType.getMajorVersion());
104 if (!khrCreateContextSupported)
105 TCU_THROW(NotSupportedError, "EGL_KHR_create_context is required for OpenGL ES 3.0 and newer");
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());
115 DE_ASSERT(glu::isContextTypeGLCore(contextType) || glu::isContextTypeGLCompatibility(contextType));
117 if (!khrCreateContextSupported)
118 TCU_THROW(NotSupportedError, "EGL_KHR_create_context is required for OpenGL context creation");
120 api = EGL_OPENGL_API;
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);
131 if (contextType.getFlags() != glu::ContextFlags(0))
135 if (!khrCreateContextSupported)
136 TCU_THROW(NotSupportedError, "EGL_KHR_create_context is required for creating robust/debug/forward-compatible contexts");
138 if ((contextType.getFlags() & glu::CONTEXT_DEBUG) != 0)
139 flags |= EGL_CONTEXT_OPENGL_DEBUG_BIT_KHR;
141 if ((contextType.getFlags() & glu::CONTEXT_ROBUST) != 0)
143 if (glu::isContextTypeES(contextType))
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");
148 attribList.push_back(EGL_CONTEXT_OPENGL_ROBUST_ACCESS_EXT);
149 attribList.push_back(EGL_TRUE);
152 flags |= EGL_CONTEXT_OPENGL_ROBUST_ACCESS_BIT_KHR;
155 if ((contextType.getFlags() & glu::CONTEXT_NO_ERROR) != 0)
157 if (khrCreateContextNoErrorSupported)
159 attribList.push_back(EGL_CONTEXT_OPENGL_NO_ERROR_KHR);
160 attribList.push_back(EGL_TRUE);
163 throw tcu::NotSupportedError("EGL_KHR_create_context_no_error is required for creating no-error contexts");
166 if ((contextType.getFlags() & glu::CONTEXT_FORWARD_COMPATIBLE) != 0)
168 if (!glu::isContextTypeGLCore(contextType))
169 TCU_THROW(InternalError, "Only OpenGL core contexts can be forward-compatible");
171 flags |= EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE_BIT_KHR;
174 attribList.push_back(EGL_CONTEXT_FLAGS_KHR);
175 attribList.push_back(flags);
177 if (resetNotificationStrategy != glu::RESET_NOTIFICATION_STRATEGY_NOT_SPECIFIED)
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);
184 TCU_THROW(NotSupportedError, "EGL 1.5 or EGL_EXT_create_context_robustness is required for creating robust context");
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);
191 TCU_THROW(InternalError, "Unknown reset notification strategy");
195 attribList.push_back(EGL_NONE);
197 EGLU_CHECK_CALL(egl, bindAPI(api));
198 context = egl.createContext(display, eglConfig, sharedContext, &(attribList[0]));
199 EGLU_CHECK_MSG(egl, "eglCreateContext()");
204 static bool configMatches (const eglw::Library& egl, eglw::EGLDisplay display, eglw::EGLConfig eglConfig, const glu::RenderConfig& renderConfig)
206 // \todo [2014-03-12 pyry] Check other attributes like double-buffer bit.
209 EGLint renderableType = 0;
210 EGLint requiredRenderable = apiRenderableType(renderConfig.type.getAPI());
212 EGLU_CHECK_CALL(egl, getConfigAttrib(display, eglConfig, EGL_RENDERABLE_TYPE, &renderableType));
214 if ((renderableType & requiredRenderable) == 0)
218 if (renderConfig.surfaceType != glu::RenderConfig::SURFACETYPE_DONT_CARE)
220 EGLint surfaceType = 0;
221 EGLint requiredSurface = 0;
223 switch (renderConfig.surfaceType)
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;
232 EGLU_CHECK_CALL(egl, getConfigAttrib(display, eglConfig, EGL_SURFACE_TYPE, &surfaceType));
234 if ((surfaceType & requiredSurface) == 0)
241 int glu::RenderConfig::*field;
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 },
255 for (int attribNdx = 0; attribNdx < DE_LENGTH_OF_ARRAY(s_attribs); attribNdx++)
257 if (renderConfig.*s_attribs[attribNdx].field != glu::RenderConfig::DONT_CARE)
260 EGLU_CHECK_CALL(egl, getConfigAttrib(display, eglConfig, s_attribs[attribNdx].attrib, &value));
261 if (value != renderConfig.*s_attribs[attribNdx].field)
270 EGLConfig chooseConfig (const Library& egl, EGLDisplay display, const glu::RenderConfig& config)
272 const std::vector<EGLConfig> configs = eglu::getConfigs(egl, display);
274 for (vector<EGLConfig>::const_iterator iter = configs.begin(); iter != configs.end(); ++iter)
276 if (configMatches(egl, display, *iter, config))
280 throw tcu::NotSupportedError("Matching EGL config not found", DE_NULL, __FILE__, __LINE__);