| /*------------------------------------------------------------------------- |
| * drawElements Quality Program OpenGL ES Utilities |
| * ------------------------------------------------ |
| * |
| * Copyright 2014 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| * |
| *//*! |
| * \file |
| * \brief OpenGL ES rendering context. |
| *//*--------------------------------------------------------------------*/ |
| |
| #include "gluRenderContext.hpp" |
| #include "gluDefs.hpp" |
| #include "gluRenderConfig.hpp" |
| #include "gluES3PlusWrapperContext.hpp" |
| #include "gluFboRenderContext.hpp" |
| #include "gluPlatform.hpp" |
| #include "gluStrUtil.hpp" |
| #include "glwInitFunctions.hpp" |
| #include "glwEnums.hpp" |
| #include "tcuPlatform.hpp" |
| #include "tcuCommandLine.hpp" |
| #include "deStringUtil.hpp" |
| #include "deSTLUtil.hpp" |
| |
| namespace glu |
| { |
| |
| inline bool versionGreaterOrEqual (ApiType a, ApiType b) |
| { |
| return a.getMajorVersion() > b.getMajorVersion() || |
| (a.getMajorVersion() == b.getMajorVersion() && a.getMinorVersion() >= b.getMinorVersion()); |
| } |
| |
| bool contextSupports (ContextType ctxType, ApiType requiredApiType) |
| { |
| // \todo [2014-10-06 pyry] Check exact forward-compatible restrictions. |
| const bool forwardCompatible = (ctxType.getFlags() & CONTEXT_FORWARD_COMPATIBLE) != 0; |
| |
| if (isContextTypeES(ctxType)) |
| { |
| DE_ASSERT(!forwardCompatible); |
| return requiredApiType.getProfile() == PROFILE_ES && |
| versionGreaterOrEqual(ctxType.getAPI(), requiredApiType); |
| } |
| else if (isContextTypeGLCore(ctxType)) |
| { |
| if (forwardCompatible) |
| return ctxType.getAPI() == requiredApiType; |
| else |
| return requiredApiType.getProfile() == PROFILE_CORE && |
| versionGreaterOrEqual(ctxType.getAPI(), requiredApiType); |
| } |
| else if (isContextTypeGLCompatibility(ctxType)) |
| { |
| DE_ASSERT(!forwardCompatible); |
| return (requiredApiType.getProfile() == PROFILE_CORE || requiredApiType.getProfile() == PROFILE_COMPATIBILITY) && |
| versionGreaterOrEqual(ctxType.getAPI(), requiredApiType); |
| } |
| else |
| { |
| DE_ASSERT(false); |
| return false; |
| } |
| } |
| |
| static ContextFlags parseContextFlags (const std::string& flagsStr) |
| { |
| const std::vector<std::string> flagNames = de::splitString(flagsStr, ','); |
| ContextFlags flags = ContextFlags(0); |
| static const struct |
| { |
| const char* name; |
| ContextFlags flag; |
| } s_flagMap[] = |
| { |
| { "debug", CONTEXT_DEBUG }, |
| { "robust", CONTEXT_ROBUST } |
| }; |
| |
| for (std::vector<std::string>::const_iterator flagIter = flagNames.begin(); flagIter != flagNames.end(); ++flagIter) |
| { |
| int ndx; |
| for (ndx = 0; ndx < DE_LENGTH_OF_ARRAY(s_flagMap); ndx++) |
| { |
| if (*flagIter == s_flagMap[ndx].name) |
| { |
| flags = flags | s_flagMap[ndx].flag; |
| break; |
| } |
| } |
| |
| if (ndx == DE_LENGTH_OF_ARRAY(s_flagMap)) |
| { |
| tcu::print("ERROR: Unrecognized GL context flag '%s'\n", flagIter->c_str()); |
| tcu::print("Supported GL context flags:\n"); |
| |
| for (ndx = 0; ndx < DE_LENGTH_OF_ARRAY(s_flagMap); ndx++) |
| tcu::print(" %s\n", s_flagMap[ndx].name); |
| |
| throw tcu::NotSupportedError((std::string("Unknown GL context flag '") + *flagIter + "'").c_str(), DE_NULL, __FILE__, __LINE__); |
| } |
| } |
| |
| return flags; |
| } |
| |
| RenderContext* createDefaultRenderContext (tcu::Platform& platform, const tcu::CommandLine& cmdLine, ApiType apiType) |
| { |
| const ContextFactoryRegistry& registry = platform.getGLPlatform().getContextFactoryRegistry(); |
| RenderConfig config; |
| const char* factoryName = cmdLine.getGLContextType(); |
| const ContextFactory* factory = DE_NULL; |
| ContextFlags ctxFlags = ContextFlags(0); |
| |
| if (registry.empty()) |
| throw tcu::NotSupportedError("OpenGL is not supported", DE_NULL, __FILE__, __LINE__); |
| |
| if (cmdLine.getGLContextFlags()) |
| ctxFlags = parseContextFlags(cmdLine.getGLContextFlags()); |
| |
| config.type = glu::ContextType(apiType, ctxFlags); |
| parseRenderConfig(&config, cmdLine); |
| |
| if (factoryName) |
| { |
| factory = registry.getFactoryByName(factoryName); |
| |
| if (!factory) |
| { |
| tcu::print("ERROR: Unknown or unsupported GL context type '%s'\n", factoryName); |
| tcu::print("Supported GL context types:\n"); |
| |
| for (int factoryNdx = 0; factoryNdx < (int)registry.getFactoryCount(); factoryNdx++) |
| { |
| const ContextFactory* curFactory = registry.getFactoryByIndex(factoryNdx); |
| tcu::print(" %s: %s\n", curFactory->getName(), curFactory->getDescription()); |
| } |
| |
| throw tcu::NotSupportedError((std::string("Unknown GL context type '") + factoryName + "'").c_str(), DE_NULL, __FILE__, __LINE__); |
| } |
| } |
| else |
| factory = registry.getDefaultFactory(); |
| |
| if (cmdLine.getSurfaceType() == tcu::SURFACETYPE_FBO) |
| return new FboRenderContext(*factory, config, cmdLine); |
| else |
| return factory->createContext(config, cmdLine); |
| } |
| |
| static std::vector<std::string> getExtensions (const glw::Functions& gl, ApiType apiType) |
| { |
| using std::vector; |
| using std::string; |
| |
| if (apiType.getProfile() == PROFILE_ES && apiType.getMajorVersion() == 2) |
| { |
| TCU_CHECK(gl.getString); |
| |
| const char* extStr = (const char*)gl.getString(GL_EXTENSIONS); |
| GLU_EXPECT_NO_ERROR(gl.getError(), "glGetString(GL_EXTENSIONS)"); |
| |
| if (extStr) |
| return de::splitString(extStr); |
| else |
| throw tcu::TestError("glGetString(GL_EXTENSIONS) returned null pointer", DE_NULL, __FILE__, __LINE__); |
| } |
| else |
| { |
| int numExtensions = 0; |
| vector<string> extensions; |
| |
| TCU_CHECK(gl.getIntegerv && gl.getStringi); |
| |
| gl.getIntegerv(GL_NUM_EXTENSIONS, &numExtensions); |
| GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv(GL_NUM_EXTENSIONS)"); |
| |
| if (numExtensions > 0) |
| { |
| extensions.resize(numExtensions); |
| |
| for (int ndx = 0; ndx < numExtensions; ndx++) |
| { |
| const char* const ext = (const char*)gl.getStringi(GL_EXTENSIONS, ndx); |
| GLU_EXPECT_NO_ERROR(gl.getError(), "glGetStringi(GL_EXTENSIONS)"); |
| |
| if (ext) |
| extensions[ndx] = ext; |
| else |
| throw tcu::TestError("glGetStringi(GL_EXTENSIONS) returned null pointer", DE_NULL, __FILE__, __LINE__); |
| } |
| |
| } |
| |
| return extensions; |
| } |
| } |
| |
| bool hasExtension (const glw::Functions& gl, ApiType apiType, const std::string& extension) |
| { |
| std::vector<std::string> extensions(getExtensions(gl, apiType)); |
| |
| return de::contains(extensions.begin(), extensions.end(), extension); |
| } |
| |
| void initCoreFunctions (glw::Functions* dst, const glw::FunctionLoader* loader, ApiType apiType) |
| { |
| static const struct |
| { |
| ApiType apiType; |
| void (*initFunc) (glw::Functions* gl, const glw::FunctionLoader* loader); |
| } s_initFuncs[] = |
| { |
| { ApiType::es(2,0), glw::initES20 }, |
| { ApiType::es(3,0), glw::initES30 }, |
| { ApiType::es(3,1), glw::initES31 }, |
| { ApiType::es(3,2), glw::initES32 }, |
| { ApiType::core(3,0), glw::initGL30Core }, |
| { ApiType::core(3,1), glw::initGL31Core }, |
| { ApiType::core(3,2), glw::initGL32Core }, |
| { ApiType::core(3,3), glw::initGL33Core }, |
| { ApiType::core(4,0), glw::initGL40Core }, |
| { ApiType::core(4,1), glw::initGL41Core }, |
| { ApiType::core(4,2), glw::initGL42Core }, |
| { ApiType::core(4,3), glw::initGL43Core }, |
| { ApiType::core(4,4), glw::initGL44Core }, |
| }; |
| |
| for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(s_initFuncs); ndx++) |
| { |
| if (s_initFuncs[ndx].apiType == apiType) |
| { |
| s_initFuncs[ndx].initFunc(dst, loader); |
| return; |
| } |
| } |
| |
| throw tcu::InternalError(std::string("Don't know how to load functions for ") + de::toString(apiType)); |
| } |
| |
| void initExtensionFunctions (glw::Functions* dst, const glw::FunctionLoader* loader, ApiType apiType) |
| { |
| std::vector<std::string> extensions = getExtensions(*dst, apiType); |
| |
| if (!extensions.empty()) |
| { |
| std::vector<const char*> extStr(extensions.size()); |
| |
| for (size_t ndx = 0; ndx < extensions.size(); ndx++) |
| extStr[ndx] = extensions[ndx].c_str(); |
| |
| initExtensionFunctions(dst, loader, apiType, (int)extStr.size(), &extStr[0]); |
| } |
| } |
| |
| void initExtensionFunctions (glw::Functions* dst, const glw::FunctionLoader* loader, ApiType apiType, int numExtensions, const char* const* extensions) |
| { |
| if (apiType.getProfile() == PROFILE_ES) |
| glw::initExtensionsES(dst, loader, numExtensions, extensions); |
| else |
| glw::initExtensionsGL(dst, loader, numExtensions, extensions); |
| } |
| |
| void initFunctions (glw::Functions* dst, const glw::FunctionLoader* loader, ApiType apiType) |
| { |
| initCoreFunctions(dst, loader, apiType); |
| initExtensionFunctions(dst, loader, apiType); |
| } |
| |
| } // glu |