| /*------------------------------------------------------------------------- |
| * drawElements Quality Program Tester Core |
| * ---------------------------------------- |
| * |
| * 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 EGL utilities |
| *//*--------------------------------------------------------------------*/ |
| |
| #include "egluUtil.hpp" |
| #include "egluDefs.hpp" |
| #include "egluNativeDisplay.hpp" |
| #include "tcuCommandLine.hpp" |
| #include "deSTLUtil.hpp" |
| |
| #include <algorithm> |
| #include <sstream> |
| |
| using std::string; |
| using std::vector; |
| |
| #if !defined(EGL_EXT_platform_base) |
| # define EGL_EXT_platform_base 1 |
| typedef EGLDisplay (EGLAPIENTRYP PFNEGLGETPLATFORMDISPLAYEXTPROC) (EGLenum platform, void *native_display, const EGLint *attrib_list); |
| typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC) (EGLDisplay dpy, EGLConfig config, void *native_window, const EGLint *attrib_list); |
| typedef EGLSurface (EGLAPIENTRYP PFNEGLCREATEPLATFORMPIXMAPSURFACEEXTPROC) (EGLDisplay dpy, EGLConfig config, void *native_pixmap, const EGLint *attrib_list); |
| #endif // EGL_EXT_platform_base |
| |
| namespace eglu |
| { |
| |
| vector<string> getPlatformExtensions (void) |
| { |
| const char* const extensionStr = eglQueryString(EGL_NO_DISPLAY, EGL_EXTENSIONS); |
| const EGLint result = eglGetError(); |
| |
| if (result == EGL_SUCCESS) |
| { |
| std::istringstream stream (extensionStr); |
| string currentExtension; |
| vector<string> extensions; |
| |
| while (std::getline(stream, currentExtension, ' ')) |
| extensions.push_back(currentExtension); |
| |
| return extensions; |
| } |
| else if (result != EGL_BAD_DISPLAY) |
| throw Error(result, "eglQueryString(EGL_NO_DISPLAY, EGL_EXTENSIONS)", DE_NULL, __FILE__, __LINE__); |
| else |
| return vector<string>(); |
| } |
| |
| vector<string> getClientExtensions (EGLDisplay display) |
| { |
| const char* const extensionStr = eglQueryString(display, EGL_EXTENSIONS); |
| const EGLint result = eglGetError(); |
| |
| if (result == EGL_SUCCESS) |
| { |
| std::istringstream stream (extensionStr); |
| string currentExtension; |
| vector<string> extensions; |
| |
| while (std::getline(stream, currentExtension, ' ')) |
| extensions.push_back(currentExtension); |
| |
| return extensions; |
| } |
| else |
| throw Error(result, "eglQueryString(display, EGL_EXTENSIONS)", DE_NULL, __FILE__, __LINE__); |
| } |
| |
| vector<EGLConfig> getConfigs (EGLDisplay display) |
| { |
| vector<EGLConfig> configs; |
| EGLint configCount = 0; |
| EGLU_CHECK_CALL(eglGetConfigs(display, DE_NULL, 0, &configCount)); |
| |
| if (configCount > 0) |
| { |
| configs.resize(configCount); |
| EGLU_CHECK_CALL(eglGetConfigs(display, &(configs[0]), (EGLint)configs.size(), &configCount)); |
| } |
| |
| return configs; |
| } |
| |
| vector<EGLConfig> chooseConfig (EGLDisplay display, const AttribMap& attribs) |
| { |
| vector<EGLint> attribList; |
| |
| for (AttribMap::const_iterator it = attribs.begin(); it != attribs.end(); ++it) |
| { |
| attribList.push_back(it->first); |
| attribList.push_back(it->second); |
| } |
| |
| attribList.push_back(EGL_NONE); |
| |
| { |
| EGLint numConfigs = 0; |
| EGLU_CHECK_CALL(eglChooseConfig(display, &attribList.front(), DE_NULL, 0, &numConfigs)); |
| |
| { |
| vector<EGLConfig> configs(numConfigs); |
| |
| if (numConfigs > 0) |
| EGLU_CHECK_CALL(eglChooseConfig(display, &attribList.front(), &configs.front(), numConfigs, &numConfigs)); |
| |
| return configs; |
| } |
| } |
| } |
| |
| EGLConfig chooseSingleConfig (EGLDisplay display, const AttribMap& attribs) |
| { |
| vector<EGLConfig> configs (chooseConfig(display, attribs)); |
| if (configs.empty()) |
| TCU_THROW(NotSupportedError, "No suitable EGL configuration found"); |
| |
| return configs.front(); |
| } |
| |
| EGLint getConfigAttribInt (EGLDisplay display, EGLConfig config, EGLint attrib) |
| { |
| EGLint value = 0; |
| EGLU_CHECK_CALL(eglGetConfigAttrib(display, config, attrib, &value)); |
| return value; |
| } |
| |
| EGLint querySurfaceInt (EGLDisplay display, EGLSurface surface, EGLint attrib) |
| { |
| EGLint value = 0; |
| EGLU_CHECK_CALL(eglQuerySurface(display, surface, attrib, &value)); |
| return value; |
| } |
| |
| tcu::IVec2 getSurfaceSize (EGLDisplay display, EGLSurface surface) |
| { |
| const EGLint width = querySurfaceInt(display, surface, EGL_WIDTH); |
| const EGLint height = querySurfaceInt(display, surface, EGL_HEIGHT); |
| return tcu::IVec2(width, height); |
| } |
| |
| tcu::IVec2 getSurfaceResolution (EGLDisplay display, EGLSurface surface) |
| { |
| const EGLint hRes = querySurfaceInt(display, surface, EGL_HORIZONTAL_RESOLUTION); |
| const EGLint vRes = querySurfaceInt(display, surface, EGL_VERTICAL_RESOLUTION); |
| |
| if (hRes == EGL_UNKNOWN || vRes == EGL_UNKNOWN) |
| TCU_THROW(NotSupportedError, "Surface doesn't support pixel density queries"); |
| return tcu::IVec2(hRes, vRes); |
| } |
| |
| //! Get EGLdisplay using eglGetDisplay() or eglGetPlatformDisplayEXT() |
| EGLDisplay getDisplay (NativeDisplay& nativeDisplay) |
| { |
| const bool supportsLegacyGetDisplay = (nativeDisplay.getCapabilities() & NativeDisplay::CAPABILITY_GET_DISPLAY_LEGACY) != 0; |
| const bool supportsPlatformGetDisplay = (nativeDisplay.getCapabilities() & NativeDisplay::CAPABILITY_GET_DISPLAY_PLATFORM) != 0; |
| bool usePlatformExt = false; |
| EGLDisplay display = EGL_NO_DISPLAY; |
| |
| TCU_CHECK_INTERNAL(supportsLegacyGetDisplay || supportsPlatformGetDisplay); |
| |
| if (supportsPlatformGetDisplay) |
| { |
| const vector<string> platformExts = getPlatformExtensions(); |
| usePlatformExt = de::contains(platformExts.begin(), platformExts.end(), string("EGL_EXT_platform_base")) && |
| de::contains(platformExts.begin(), platformExts.end(), string(nativeDisplay.getPlatformExtensionName())); |
| } |
| |
| if (usePlatformExt) |
| { |
| const PFNEGLGETPLATFORMDISPLAYEXTPROC getPlatformDisplay = (PFNEGLGETPLATFORMDISPLAYEXTPROC)eglGetProcAddress("eglGetPlatformDisplayEXT"); |
| |
| EGLU_CHECK_MSG("eglGetProcAddress()"); |
| TCU_CHECK(getPlatformDisplay); |
| |
| display = getPlatformDisplay(nativeDisplay.getPlatformType(), nativeDisplay.getPlatformNative(), DE_NULL); |
| EGLU_CHECK_MSG("eglGetPlatformDisplayEXT()"); |
| TCU_CHECK(display != EGL_NO_DISPLAY); |
| } |
| else if (supportsLegacyGetDisplay) |
| { |
| display = eglGetDisplay(nativeDisplay.getLegacyNative()); |
| EGLU_CHECK_MSG("eglGetDisplay()"); |
| TCU_CHECK(display != EGL_NO_DISPLAY); |
| } |
| else |
| throw tcu::InternalError("No supported way to get EGL display", DE_NULL, __FILE__, __LINE__); |
| |
| DE_ASSERT(display != EGL_NO_DISPLAY); |
| return display; |
| } |
| |
| //! Create EGL window surface using eglCreateWindowSurface() or eglCreatePlatformWindowSurfaceEXT() |
| EGLSurface createWindowSurface (NativeDisplay& nativeDisplay, NativeWindow& window, EGLDisplay display, EGLConfig config, const EGLAttrib* attribList) |
| { |
| const bool supportsLegacyCreate = (window.getCapabilities() & NativeWindow::CAPABILITY_CREATE_SURFACE_LEGACY) != 0; |
| const bool supportsPlatformCreate = (window.getCapabilities() & NativeWindow::CAPABILITY_CREATE_SURFACE_PLATFORM) != 0; |
| bool usePlatformExt = false; |
| EGLSurface surface = EGL_NO_SURFACE; |
| |
| TCU_CHECK_INTERNAL(supportsLegacyCreate || supportsPlatformCreate); |
| |
| if (supportsPlatformCreate) |
| { |
| const vector<string> platformExts = getPlatformExtensions(); |
| usePlatformExt = de::contains(platformExts.begin(), platformExts.end(), string("EGL_EXT_platform_base")) && |
| de::contains(platformExts.begin(), platformExts.end(), string(nativeDisplay.getPlatformExtensionName())); |
| } |
| |
| // \todo [2014-03-13 pyry] EGL 1.5 core support |
| if (usePlatformExt) |
| { |
| const PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC createPlatformWindowSurface = (PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC)eglGetProcAddress("eglCreatePlatformWindowSurfaceEXT"); |
| const vector<EGLint> legacyAttribs = toLegacyAttribList(attribList); |
| |
| EGLU_CHECK_MSG("eglGetProcAddress()"); |
| TCU_CHECK(createPlatformWindowSurface); |
| |
| surface = createPlatformWindowSurface(display, config, window.getPlatformNative(), &legacyAttribs[0]); |
| EGLU_CHECK_MSG("eglCreatePlatformWindowSurfaceEXT()"); |
| TCU_CHECK(surface != EGL_NO_SURFACE); |
| } |
| else if (supportsLegacyCreate) |
| { |
| const vector<EGLint> legacyAttribs = toLegacyAttribList(attribList); |
| surface = eglCreateWindowSurface(display, config, window.getLegacyNative(), &legacyAttribs[0]); |
| EGLU_CHECK_MSG("eglCreateWindowSurface()"); |
| TCU_CHECK(surface != EGL_NO_SURFACE); |
| } |
| else |
| throw tcu::InternalError("No supported way to create EGL window surface", DE_NULL, __FILE__, __LINE__); |
| |
| DE_ASSERT(surface != EGL_NO_SURFACE); |
| return surface; |
| } |
| |
| //! Create EGL pixmap surface using eglCreatePixmapSurface() or eglCreatePlatformPixmapSurfaceEXT() |
| EGLSurface createPixmapSurface (NativeDisplay& nativeDisplay, NativePixmap& pixmap, EGLDisplay display, EGLConfig config, const EGLAttrib* attribList) |
| { |
| const bool supportsLegacyCreate = (pixmap.getCapabilities() & NativePixmap::CAPABILITY_CREATE_SURFACE_LEGACY) != 0; |
| const bool supportsPlatformCreate = (pixmap.getCapabilities() & NativePixmap::CAPABILITY_CREATE_SURFACE_PLATFORM) != 0; |
| bool usePlatformExt = false; |
| EGLSurface surface = EGL_NO_SURFACE; |
| |
| TCU_CHECK_INTERNAL(supportsLegacyCreate || supportsPlatformCreate); |
| |
| if (supportsPlatformCreate) |
| { |
| const vector<string> platformExts = getPlatformExtensions(); |
| usePlatformExt = de::contains(platformExts.begin(), platformExts.end(), string("EGL_EXT_platform_base")) && |
| de::contains(platformExts.begin(), platformExts.end(), string(nativeDisplay.getPlatformExtensionName())); |
| } |
| |
| if (usePlatformExt) |
| { |
| const PFNEGLCREATEPLATFORMPIXMAPSURFACEEXTPROC createPlatformPixmapSurface = (PFNEGLCREATEPLATFORMPIXMAPSURFACEEXTPROC)eglGetProcAddress("eglCreatePlatformPixmapSurfaceEXT"); |
| const vector<EGLint> legacyAttribs = toLegacyAttribList(attribList); |
| |
| EGLU_CHECK_MSG("eglGetProcAddress()"); |
| TCU_CHECK(createPlatformPixmapSurface); |
| |
| surface = createPlatformPixmapSurface(display, config, pixmap.getPlatformNative(), &legacyAttribs[0]); |
| EGLU_CHECK_MSG("eglCreatePlatformPixmapSurfaceEXT()"); |
| TCU_CHECK(surface != EGL_NO_SURFACE); |
| } |
| else if (supportsLegacyCreate) |
| { |
| const vector<EGLint> legacyAttribs = toLegacyAttribList(attribList); |
| surface = eglCreatePixmapSurface(display, config, pixmap.getLegacyNative(), &legacyAttribs[0]); |
| EGLU_CHECK_MSG("eglCreatePixmapSurface()"); |
| TCU_CHECK(surface != EGL_NO_SURFACE); |
| } |
| else |
| throw tcu::InternalError("No supported way to create EGL pixmap surface", DE_NULL, __FILE__, __LINE__); |
| |
| DE_ASSERT(surface != EGL_NO_SURFACE); |
| return surface; |
| } |
| |
| static WindowParams::Visibility getWindowVisibility (tcu::WindowVisibility visibility) |
| { |
| switch (visibility) |
| { |
| case tcu::WINDOWVISIBILITY_WINDOWED: return WindowParams::VISIBILITY_VISIBLE; |
| case tcu::WINDOWVISIBILITY_FULLSCREEN: return WindowParams::VISIBILITY_FULLSCREEN; |
| case tcu::WINDOWVISIBILITY_HIDDEN: return WindowParams::VISIBILITY_HIDDEN; |
| |
| default: |
| DE_ASSERT(false); |
| return WindowParams::VISIBILITY_DONT_CARE; |
| } |
| } |
| |
| WindowParams::Visibility parseWindowVisibility (const tcu::CommandLine& commandLine) |
| { |
| return getWindowVisibility(commandLine.getVisibility()); |
| } |
| |
| vector<EGLint> toLegacyAttribList (const EGLAttrib* attribs) |
| { |
| const deUint64 attribMask = 0xffffffffull; //!< Max bits that can be used |
| vector<EGLint> legacyAttribs; |
| |
| if (attribs) |
| { |
| for (const EGLAttrib* attrib = attribs; *attrib != EGL_NONE; attrib += 2) |
| { |
| if ((attrib[0] & ~attribMask) || (attrib[1] & ~attribMask)) |
| throw tcu::InternalError("Failed to translate EGLAttrib to EGLint", DE_NULL, __FILE__, __LINE__); |
| |
| legacyAttribs.push_back((EGLint)attrib[0]); |
| legacyAttribs.push_back((EGLint)attrib[1]); |
| } |
| } |
| |
| legacyAttribs.push_back(EGL_NONE); |
| |
| return legacyAttribs; |
| } |
| |
| } // eglu |