/*------------------------------------------------------------------------- * drawElements Quality Program EGL Module * --------------------------------------- * * Copyright 2017 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 Test KHR_wide_color *//*--------------------------------------------------------------------*/ #include "teglWideColorTests.hpp" #include "tcuImageCompare.hpp" #include "tcuTestLog.hpp" #include "tcuSurface.hpp" #include "tcuTextureUtil.hpp" #include "egluNativeWindow.hpp" #include "egluStrUtil.hpp" #include "egluUtil.hpp" #include "egluConfigFilter.hpp" #include "eglwLibrary.hpp" #include "eglwEnums.hpp" #include "gluDefs.hpp" #include "gluRenderContext.hpp" #include "gluShaderProgram.hpp" #include "glw.h" #include "glwDefs.hpp" #include "glwEnums.hpp" #include "glwFunctions.hpp" #include "deMath.h" #include "deRandom.hpp" #include "deString.h" #include "deStringUtil.hpp" #include <string> #include <vector> #include <sstream> using std::string; using std::vector; using glw::GLubyte; using glw::GLfloat; using tcu::IVec2; using namespace eglw; namespace deqp { namespace egl { namespace { typedef tcu::Vec4 Color; class GLES2Renderer; class ReferenceRenderer; class WideColorTests : public TestCaseGroup { public: WideColorTests (EglTestContext& eglTestCtx); void init (void); private: WideColorTests (const WideColorTests&); WideColorTests& operator= (const WideColorTests&); }; class WideColorTest : public TestCase { public: enum DrawType { DRAWTYPE_GLES2_CLEAR, DRAWTYPE_GLES2_RENDER }; WideColorTest (EglTestContext& eglTestCtx, const char* name, const char* description); ~WideColorTest (void); void init (void); void deinit (void); void checkPixelFloatSupport (void); void checkColorSpaceSupport (void); void checkDisplayP3Support (void); void checkDisplayP3PassthroughSupport (void); void check1010102Support (void); void checkFP16Support (void); void checkSCRGBSupport (void); void checkSCRGBLinearSupport (void); void checkbt2020linear (void); void checkbt2020pq (void); void checkSMPTE2086 (void); void checkCTA861_3 (void); protected: void initEGLSurface (EGLConfig config); void initEGLContext (EGLConfig config); EGLDisplay m_eglDisplay; glw::Functions m_gl; }; struct ColoredRect { public: ColoredRect (const IVec2& bottomLeft_, const IVec2& topRight_, const Color& color_); IVec2 bottomLeft; IVec2 topRight; Color color; }; ColoredRect::ColoredRect (const IVec2& bottomLeft_, const IVec2& topRight_, const Color& color_) : bottomLeft (bottomLeft_) , topRight (topRight_) , color (color_) { } void clearColorScreen (const glw::Functions& gl, const Color& clearColor) { gl.clearColor(clearColor.x(), clearColor.y(), clearColor.z(), clearColor.w()); gl.clear(GL_COLOR_BUFFER_BIT); } float windowToDeviceCoordinates (int x, int length) { return (2.0f * float(x) / float(length)) - 1.0f; } class GLES2Renderer { public: GLES2Renderer (const glw::Functions& gl, int width, int height); ~GLES2Renderer (void); void render (const ColoredRect& coloredRect) const; private: GLES2Renderer (const GLES2Renderer&); GLES2Renderer& operator= (const GLES2Renderer&); const glw::Functions& m_gl; glu::ShaderProgram m_glProgram; glw::GLuint m_coordLoc; glw::GLuint m_colorLoc; glw::GLuint m_bufWidth; glw::GLuint m_bufHeight; }; // generate sources for vertex and fragment buffer glu::ProgramSources getSources (void) { const char* const vertexShaderSource = "attribute mediump vec2 a_pos;\n" "attribute mediump vec4 a_color;\n" "varying mediump vec4 v_color;\n" "void main(void)\n" "{\n" "\tv_color = a_color;\n" "\tgl_Position = vec4(a_pos, 0.0, 1.0);\n" "}"; const char* const fragmentShaderSource = "varying mediump vec4 v_color;\n" "void main(void)\n" "{\n" "\tgl_FragColor = v_color;\n" "}"; return glu::makeVtxFragSources(vertexShaderSource, fragmentShaderSource); } GLES2Renderer::GLES2Renderer (const glw::Functions& gl, int width, int height) : m_gl (gl) , m_glProgram (gl, getSources()) , m_coordLoc ((glw::GLuint)-1) , m_colorLoc ((glw::GLuint)-1) , m_bufWidth (width) , m_bufHeight (height) { m_colorLoc = m_gl.getAttribLocation(m_glProgram.getProgram(), "a_color"); m_coordLoc = m_gl.getAttribLocation(m_glProgram.getProgram(), "a_pos"); GLU_EXPECT_NO_ERROR(m_gl.getError(), "Failed to get attribute locations"); } GLES2Renderer::~GLES2Renderer (void) { } void GLES2Renderer::render (const struct ColoredRect &coloredRect) const { const float x1 = windowToDeviceCoordinates(coloredRect.bottomLeft.x(), m_bufWidth); const float y1 = windowToDeviceCoordinates(coloredRect.bottomLeft.y(), m_bufHeight); const float x2 = windowToDeviceCoordinates(coloredRect.topRight.x(), m_bufWidth); const float y2 = windowToDeviceCoordinates(coloredRect.topRight.y(), m_bufHeight); const glw::GLfloat coords[] = { x1, y1, 0.0f, 1.0f, x1, y2, 0.0f, 1.0f, x2, y2, 0.0f, 1.0f, x2, y2, 0.0f, 1.0f, x2, y1, 0.0f, 1.0f, x1, y1, 0.0f, 1.0f }; const glw::GLfloat colors[] = { coloredRect.color.x(), coloredRect.color.y(), coloredRect.color.z(), coloredRect.color.w(), coloredRect.color.x(), coloredRect.color.y(), coloredRect.color.z(), coloredRect.color.w(), coloredRect.color.x(), coloredRect.color.y(), coloredRect.color.z(), coloredRect.color.w(), coloredRect.color.x(), coloredRect.color.y(), coloredRect.color.z(), coloredRect.color.w(), coloredRect.color.x(), coloredRect.color.y(), coloredRect.color.z(), coloredRect.color.w(), coloredRect.color.x(), coloredRect.color.y(), coloredRect.color.z(), coloredRect.color.w(), }; m_gl.useProgram(m_glProgram.getProgram()); GLU_EXPECT_NO_ERROR(m_gl.getError(), "glUseProgram() failed"); m_gl.enableVertexAttribArray(m_coordLoc); m_gl.enableVertexAttribArray(m_colorLoc); GLU_EXPECT_NO_ERROR(m_gl.getError(), "Failed to enable attributes"); m_gl.vertexAttribPointer(m_coordLoc, 4, GL_FLOAT, GL_FALSE, 0, coords); m_gl.vertexAttribPointer(m_colorLoc, 4, GL_FLOAT, GL_TRUE, 0, colors); GLU_EXPECT_NO_ERROR(m_gl.getError(), "Failed to set attribute pointers"); m_gl.drawArrays(GL_TRIANGLES, 0, DE_LENGTH_OF_ARRAY(coords)/4); GLU_EXPECT_NO_ERROR(m_gl.getError(), "glDrawArrays(), failed"); m_gl.disableVertexAttribArray(m_coordLoc); m_gl.disableVertexAttribArray(m_colorLoc); GLU_EXPECT_NO_ERROR(m_gl.getError(), "Failed to disable attributes"); m_gl.useProgram(0); GLU_EXPECT_NO_ERROR(m_gl.getError(), "glUseProgram() failed"); } class ReferenceRenderer { public: ReferenceRenderer (void); private: ReferenceRenderer (const ReferenceRenderer&); ReferenceRenderer& operator= (const ReferenceRenderer&); }; WideColorTest::WideColorTest (EglTestContext& eglTestCtx, const char* name, const char* description) : TestCase (eglTestCtx, name, description) , m_eglDisplay (EGL_NO_DISPLAY) { } WideColorTest::~WideColorTest (void) { deinit(); } void WideColorTest::init (void) { m_eglDisplay = eglu::getAndInitDisplay(m_eglTestCtx.getNativeDisplay()); m_eglTestCtx.initGLFunctions(&m_gl, glu::ApiType::es(2,0)); } void WideColorTest::checkPixelFloatSupport (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_pixel_format_float")) TCU_THROW(NotSupportedError, "EGL_EXT_pixel_format_float is not supported"); } void WideColorTest::checkColorSpaceSupport (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_KHR_gl_colorspace")) TCU_THROW(NotSupportedError, "EGL_KHR_gl_colorspace is not supported"); } void WideColorTest::checkDisplayP3Support (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_gl_colorspace_display_p3")) TCU_THROW(NotSupportedError, "EGL_EXT_gl_colorspace_display_p3 is not supported"); } void WideColorTest::checkDisplayP3PassthroughSupport (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_gl_colorspace_display_p3_passthrough")) TCU_THROW(NotSupportedError, "EGL_EXT_gl_colorspace_display_p3_passthrough is not supported"); } void WideColorTest::checkSCRGBSupport (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_gl_colorspace_scrgb")) TCU_THROW(NotSupportedError, "EGL_EXT_gl_colorspace_scrgb is not supported"); } void WideColorTest::checkSCRGBLinearSupport (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_gl_colorspace_scrgb_linear")) TCU_THROW(NotSupportedError, "EGL_EXT_gl_colorspace_scrgb_linear is not supported"); } void WideColorTest::checkbt2020linear (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_gl_colorspace_bt2020_linear")) TCU_THROW(NotSupportedError, "EGL_EXT_gl_colorspace_bt2020_linear is not supported"); } void WideColorTest::checkbt2020pq (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_gl_colorspace_bt2020_pq")) TCU_THROW(NotSupportedError, "EGL_EXT_gl_colorspace_bt2020_pq is not supported"); } void WideColorTest::checkSMPTE2086 (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_surface_SMPTE2086_metadata")) TCU_THROW(NotSupportedError, "EGL_EXT_surface_SMPTE2086_metadata is not supported"); } void WideColorTest::checkCTA861_3 (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (!eglu::hasExtension(egl, m_eglDisplay, "EGL_EXT_surface_CTA861_3_metadata")) TCU_THROW(NotSupportedError, "EGL_EXT_surface_CTA861_3_metadata is not supported"); } void WideColorTest::check1010102Support (void) { const Library& egl = m_eglTestCtx.getLibrary(); tcu::TestLog& log = m_testCtx.getLog(); const EGLint attribList[] = { EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 10, EGL_GREEN_SIZE, 10, EGL_BLUE_SIZE, 10, EGL_ALPHA_SIZE, 2, EGL_NONE, EGL_NONE }; EGLint numConfigs = 0; EGLConfig config; // Query from EGL implementation EGLU_CHECK_CALL(egl, chooseConfig(m_eglDisplay, &attribList[0], DE_NULL, 0, &numConfigs)); if (numConfigs <= 0) { log << tcu::TestLog::Message << "No configs returned." << tcu::TestLog::EndMessage; TCU_THROW(NotSupportedError, "10:10:10:2 pixel format is not supported"); } log << tcu::TestLog::Message << numConfigs << " configs returned" << tcu::TestLog::EndMessage; EGLU_CHECK_CALL(egl, chooseConfig(m_eglDisplay, &attribList[0], &config, 1, &numConfigs)); if (numConfigs > 1) { log << tcu::TestLog::Message << "Fail, more configs returned than requested." << tcu::TestLog::EndMessage; TCU_FAIL("Too many configs returned"); } EGLint components[4]; EGLU_CHECK_CALL(egl, getConfigAttrib(m_eglDisplay, config, EGL_RED_SIZE, &components[0])); EGLU_CHECK_CALL(egl, getConfigAttrib(m_eglDisplay, config, EGL_GREEN_SIZE, &components[1])); EGLU_CHECK_CALL(egl, getConfigAttrib(m_eglDisplay, config, EGL_BLUE_SIZE, &components[2])); EGLU_CHECK_CALL(egl, getConfigAttrib(m_eglDisplay, config, EGL_ALPHA_SIZE, &components[3])); TCU_CHECK_MSG(components[0] == 10, "Missing 10bit deep red channel"); TCU_CHECK_MSG(components[1] == 10, "Missing 10bit deep green channel"); TCU_CHECK_MSG(components[2] == 10, "Missing 10bit deep blue channel"); TCU_CHECK_MSG(components[3] == 2, "Missing 2bit deep alpha channel"); } void WideColorTest::checkFP16Support (void) { const Library& egl = m_eglTestCtx.getLibrary(); tcu::TestLog& log = m_testCtx.getLog(); EGLint numConfigs = 0; EGLConfig config; const EGLint attribList[] = { EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 16, EGL_GREEN_SIZE, 16, EGL_BLUE_SIZE, 16, EGL_ALPHA_SIZE, 16, EGL_COLOR_COMPONENT_TYPE_EXT, EGL_COLOR_COMPONENT_TYPE_FLOAT_EXT, EGL_NONE, EGL_NONE }; // Query from EGL implementation EGLU_CHECK_CALL(egl, chooseConfig(m_eglDisplay, &attribList[0], DE_NULL, 0, &numConfigs)); if (numConfigs <= 0) { log << tcu::TestLog::Message << "No configs returned." << tcu::TestLog::EndMessage; TCU_THROW(NotSupportedError, "16:16:16:16 pixel format is not supported"); } log << tcu::TestLog::Message << numConfigs << " configs returned" << tcu::TestLog::EndMessage; EGLBoolean success = egl.chooseConfig(m_eglDisplay, &attribList[0], &config, 1, &numConfigs); if (success != EGL_TRUE) { log << tcu::TestLog::Message << "Fail, eglChooseConfig returned an error." << tcu::TestLog::EndMessage; TCU_FAIL("eglChooseConfig failed"); } if (numConfigs > 1) { log << tcu::TestLog::Message << "Fail, more configs returned than requested." << tcu::TestLog::EndMessage; TCU_FAIL("Too many configs returned"); } EGLint components[4]; success = egl.getConfigAttrib(m_eglDisplay, config, EGL_RED_SIZE, &components[0]); TCU_CHECK_MSG(success == EGL_TRUE, "eglGetConfigAttrib failed"); EGLU_CHECK(egl); success = egl.getConfigAttrib(m_eglDisplay, config, EGL_GREEN_SIZE, &components[1]); TCU_CHECK_MSG(success == EGL_TRUE, "eglGetConfigAttrib failed"); EGLU_CHECK(egl); success = egl.getConfigAttrib(m_eglDisplay, config, EGL_BLUE_SIZE, &components[2]); TCU_CHECK_MSG(success == EGL_TRUE, "eglGetConfigAttrib failed"); EGLU_CHECK(egl); success = egl.getConfigAttrib(m_eglDisplay, config, EGL_ALPHA_SIZE, &components[3]); TCU_CHECK_MSG(success == EGL_TRUE, "eglGetConfigAttrib failed"); EGLU_CHECK(egl); TCU_CHECK_MSG(components[0] == 16, "Missing 16bit deep red channel"); TCU_CHECK_MSG(components[1] == 16, "Missing 16bit deep green channel"); TCU_CHECK_MSG(components[2] == 16, "Missing 16bit deep blue channel"); TCU_CHECK_MSG(components[3] == 16, "Missing 16bit deep alpha channel"); } void WideColorTest::deinit (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (m_eglDisplay != EGL_NO_DISPLAY) { egl.terminate(m_eglDisplay); m_eglDisplay = EGL_NO_DISPLAY; } } class WideColorFP16Test : public WideColorTest { public: WideColorFP16Test (EglTestContext& eglTestCtx, const char* name, const char* description); void init (void); void executeTest (void); IterateResult iterate (void); }; WideColorFP16Test::WideColorFP16Test (EglTestContext& eglTestCtx, const char* name, const char* description) : WideColorTest(eglTestCtx, name, description) { } void WideColorFP16Test::executeTest (void) { checkPixelFloatSupport(); checkFP16Support(); } TestCase::IterateResult WideColorFP16Test::iterate (void) { m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); executeTest(); return STOP; } void WideColorFP16Test::init (void) { WideColorTest::init(); } class WideColor1010102Test : public WideColorTest { public: WideColor1010102Test (EglTestContext& eglTestCtx, const char* name, const char* description); void executeTest (void); IterateResult iterate (void); }; WideColor1010102Test::WideColor1010102Test (EglTestContext& eglTestCtx, const char* name, const char* description) : WideColorTest(eglTestCtx, name, description) { } void WideColor1010102Test::executeTest (void) { check1010102Support(); } TestCase::IterateResult WideColor1010102Test::iterate (void) { m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); executeTest(); return STOP; } struct Iteration { float start; float increment; int iterationCount; Iteration(float s, float i, int c) : start(s), increment(i), iterationCount(c) {} }; class WideColorSurfaceTest : public WideColorTest { public: WideColorSurfaceTest (EglTestContext& eglTestCtx, const char* name, const char* description, const EGLint* attribList, EGLint colorSpace, const std::vector<Iteration>& iterations); void init (void); void executeTest (void); IterateResult iterate (void); void addTestAttributes (const EGLint* attributes); protected: void readPixels (const glw::Functions& gl, float* dataPtr); void readPixels (const glw::Functions& gl, deUint32* dataPtr); void readPixels (const glw::Functions& gl, deUint8* dataPtr); deUint32 expectedUint10 (float reference); deUint32 expectedUint2 (float reference); deUint8 expectedUint8 (float reference); deUint8 expectedAlpha8 (float reference); bool checkWithThreshold8 (deUint8 value, deUint8 reference, deUint8 threshold = 1); bool checkWithThreshold10 (deUint32 value, deUint32 reference, deUint32 threshold = 1); bool checkWithThresholdFloat (float value, float reference, float threshold); void doClearTest (EGLSurface surface); void testPixels (float reference, float increment); void testFramebufferColorEncoding (); void writeEglConfig (EGLConfig config); private: std::vector<EGLint> m_attribList; std::vector<EGLint> m_testAttribList; EGLConfig m_eglConfig; EGLint m_surfaceType; EGLint m_componentType; EGLint m_requestedRedSize; EGLint m_redSize; EGLint m_alphaSize; EGLint m_colorSpace; const std::vector<struct Iteration> m_iterations; std::stringstream m_debugLog; }; WideColorSurfaceTest::WideColorSurfaceTest (EglTestContext& eglTestCtx, const char* name, const char* description, const EGLint* attribList, EGLint colorSpace, const std::vector<struct Iteration>& iterations) : WideColorTest (eglTestCtx, name, description) , m_surfaceType (0) , m_componentType (EGL_COLOR_COMPONENT_TYPE_FIXED_EXT) , m_requestedRedSize (0) , m_redSize (0) , m_alphaSize (0) , m_colorSpace (colorSpace) , m_iterations (iterations) { deUint32 idx = 0; while (attribList[idx] != EGL_NONE) { if (attribList[idx] == EGL_COLOR_COMPONENT_TYPE_EXT) { m_componentType = attribList[idx + 1]; } else if (attribList[idx] == EGL_SURFACE_TYPE) { m_surfaceType = attribList[idx+1]; } else if (attribList[idx] == EGL_RED_SIZE) { m_requestedRedSize = attribList[idx + 1]; } m_attribList.push_back(attribList[idx++]); m_attribList.push_back(attribList[idx++]); } m_attribList.push_back(EGL_NONE); } void WideColorSurfaceTest::addTestAttributes(const EGLint *attributes) { deUint32 idx = 0; if (attributes == DE_NULL) return; while (attributes[idx] != EGL_NONE) { m_testAttribList.push_back(attributes[idx++]); m_testAttribList.push_back(attributes[idx++]); } } void WideColorSurfaceTest::init (void) { const Library& egl = m_eglTestCtx.getLibrary(); tcu::TestLog& log = m_testCtx.getLog(); WideColorTest::init(); // Only check for pixel format required for this specific run // If not available, check will abort test with "NotSupported" switch (m_requestedRedSize) { case 10: check1010102Support(); break; case 16: checkPixelFloatSupport(); checkFP16Support(); break; } if (m_colorSpace != EGL_NONE && !eglu::hasExtension(egl, m_eglDisplay, "EGL_KHR_gl_colorspace")) TCU_THROW(NotSupportedError, "EGL_KHR_gl_colorspace is not supported"); switch (m_colorSpace) { case EGL_GL_COLORSPACE_SRGB_KHR: checkColorSpaceSupport(); break; case EGL_GL_COLORSPACE_DISPLAY_P3_EXT: checkDisplayP3Support(); break; case EGL_GL_COLORSPACE_DISPLAY_P3_PASSTHROUGH_EXT: checkDisplayP3PassthroughSupport(); break; case EGL_GL_COLORSPACE_SCRGB_EXT: checkSCRGBSupport(); break; case EGL_GL_COLORSPACE_SCRGB_LINEAR_EXT: checkSCRGBLinearSupport(); break; case EGL_GL_COLORSPACE_BT2020_LINEAR_EXT: checkbt2020linear(); break; case EGL_GL_COLORSPACE_BT2020_PQ_EXT: checkbt2020pq(); break; default: break; } EGLint numConfigs = 0; // Query from EGL implementation EGLU_CHECK_CALL(egl, chooseConfig(m_eglDisplay, &m_attribList[0], DE_NULL, 0, &numConfigs)); if (numConfigs <= 0) { log << tcu::TestLog::Message << "No configs returned." << tcu::TestLog::EndMessage; TCU_THROW(NotSupportedError, "No configs available with the requested attributes"); } log << tcu::TestLog::Message << numConfigs << " configs returned" << tcu::TestLog::EndMessage; EGLBoolean success = egl.chooseConfig(m_eglDisplay, &m_attribList[0], &m_eglConfig, 1, &numConfigs); if (success != EGL_TRUE) { log << tcu::TestLog::Message << "Fail, eglChooseConfig returned an error." << tcu::TestLog::EndMessage; TCU_FAIL("eglChooseConfig failed"); } if (numConfigs > 1) { log << tcu::TestLog::Message << "Fail, more configs returned than requested." << tcu::TestLog::EndMessage; TCU_FAIL("Too many configs returned"); } m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); m_redSize = eglu::getConfigAttribInt(egl, m_eglDisplay, m_eglConfig, EGL_RED_SIZE); m_alphaSize = eglu::getConfigAttribInt(egl, m_eglDisplay, m_eglConfig, EGL_ALPHA_SIZE); writeEglConfig(m_eglConfig); } void WideColorSurfaceTest::readPixels (const glw::Functions& gl, float* dataPtr) { gl.readPixels(0, 0, 1, 1, GL_RGBA, GL_FLOAT, dataPtr); GLU_EXPECT_NO_ERROR(m_gl.getError(), "glReadPixels with floats"); } void WideColorSurfaceTest::readPixels (const glw::Functions& gl, deUint32 *dataPtr) { gl.readPixels(0, 0, 1, 1, GL_RGBA, GL_UNSIGNED_INT_2_10_10_10_REV, dataPtr); GLU_EXPECT_NO_ERROR(m_gl.getError(), "glReadPixels with RGBA_1010102 (32bits)"); } void WideColorSurfaceTest::readPixels (const glw::Functions& gl, deUint8 *dataPtr) { gl.readPixels(0, 0, 1, 1, GL_RGBA, GL_UNSIGNED_BYTE, dataPtr); GLU_EXPECT_NO_ERROR(m_gl.getError(), "glReadPixels with RGBA_8888 (8 bit components)"); } void WideColorSurfaceTest::writeEglConfig (EGLConfig config) { const Library& egl = m_eglTestCtx.getLibrary(); tcu::TestLog& log = m_testCtx.getLog(); qpEglConfigInfo info; EGLint val = 0; info.bufferSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_BUFFER_SIZE); info.redSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_RED_SIZE); info.greenSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_GREEN_SIZE); info.blueSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_BLUE_SIZE); info.luminanceSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_LUMINANCE_SIZE); info.alphaSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_ALPHA_SIZE); info.alphaMaskSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_ALPHA_MASK_SIZE); val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_BIND_TO_TEXTURE_RGB); info.bindToTextureRGB = val == EGL_TRUE ? true : false; val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_BIND_TO_TEXTURE_RGBA); info.bindToTextureRGBA = val == EGL_TRUE ? true : false; val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_COLOR_BUFFER_TYPE); std::string colorBufferType = de::toString(eglu::getColorBufferTypeStr(val)); info.colorBufferType = colorBufferType.c_str(); val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_CONFIG_CAVEAT); std::string caveat = de::toString(eglu::getConfigCaveatStr(val)); info.configCaveat = caveat.c_str(); info.configID = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_CONFIG_ID); val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_CONFORMANT); std::string conformant = de::toString(eglu::getAPIBitsStr(val)); info.conformant = conformant.c_str(); info.depthSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_DEPTH_SIZE); info.level = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_LEVEL); info.maxPBufferWidth = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_MAX_PBUFFER_WIDTH); info.maxPBufferHeight = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_MAX_PBUFFER_HEIGHT); info.maxPBufferPixels = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_MAX_PBUFFER_PIXELS); info.maxSwapInterval = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_MAX_SWAP_INTERVAL); info.minSwapInterval = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_MIN_SWAP_INTERVAL); val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_NATIVE_RENDERABLE); info.nativeRenderable = val == EGL_TRUE ? true : false; val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_RENDERABLE_TYPE); std::string renderableTypes = de::toString(eglu::getAPIBitsStr(val)); info.renderableType = renderableTypes.c_str(); info.sampleBuffers = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_SAMPLE_BUFFERS); info.samples = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_SAMPLES); info.stencilSize = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_STENCIL_SIZE); val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_SURFACE_TYPE); std::string surfaceTypes = de::toString(eglu::getSurfaceBitsStr(val)); info.surfaceTypes = surfaceTypes.c_str(); val = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_TRANSPARENT_TYPE); std::string transparentType = de::toString(eglu::getTransparentTypeStr(val)); info.transparentType = transparentType.c_str(); info.transparentRedValue = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_TRANSPARENT_RED_VALUE); info.transparentGreenValue = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_TRANSPARENT_GREEN_VALUE); info.transparentBlueValue = eglu::getConfigAttribInt(egl, m_eglDisplay, config, EGL_TRANSPARENT_BLUE_VALUE); log.writeEglConfig(&info); } deUint32 WideColorSurfaceTest::expectedUint10 (float reference) { deUint32 expected; if (reference < 0.0) { expected = 0; } else if (reference > 1.0) { expected = 1023; } else { expected = static_cast<deUint32>(deRound(reference * 1023.0)); } return expected; } deUint32 WideColorSurfaceTest::expectedUint2 (float reference) { deUint32 expected; if (reference < 0.0) { expected = 0; } else if (reference > 1.0) { expected = 3; } else { expected = static_cast<deUint32>(deRound(reference * 3.0)); } return expected; } deUint8 WideColorSurfaceTest::expectedUint8 (float reference) { deUint8 expected; if (reference < 0.0) { expected = 0; } else if (reference >= 1.0) { expected = 255; } else { // Apply sRGB transfer function when colorspace is sRGB or Display P3 and // pixel component size is 8 bits (which is why we are here in expectedUint8). if (m_colorSpace == EGL_GL_COLORSPACE_SRGB_KHR || m_colorSpace == EGL_GL_COLORSPACE_DISPLAY_P3_EXT) { float srgbReference; if (reference <= 0.0031308) { srgbReference = 12.92f * reference; } else { float powRef = deFloatPow(reference, (1.0f/2.4f)); srgbReference = (1.055f * powRef) - 0.055f; } expected = static_cast<deUint8>(deRound(srgbReference * 255.0)); } else { expected = static_cast<deUint8>(deRound(reference * 255.0)); } } return expected; } deUint8 WideColorSurfaceTest::expectedAlpha8 (float reference) { deUint8 expected; if (m_alphaSize == 0) { // Surfaces without alpha are read back as opaque. expected = 255; } else if (reference < 0.0) { expected = 0; } else if (reference >= 1.0) { expected = 255; } else { // The sRGB transfer function is not applied to alpha expected = static_cast<deUint8>(deRound(reference * 255.0)); } return expected; } // Return true for value out of range (fail) bool WideColorSurfaceTest::checkWithThreshold8(deUint8 value, deUint8 reference, deUint8 threshold) { const deUint8 low = reference >= threshold ? static_cast<deUint8>(reference - threshold) : 0; const deUint8 high = reference <= (255 - threshold) ? static_cast<deUint8>(reference + threshold) : 255; return !((value >= low) && (value <= high)); } bool WideColorSurfaceTest::checkWithThreshold10(deUint32 value, deUint32 reference, deUint32 threshold) { const deUint32 low = reference >= threshold ? reference - threshold : 0; const deUint32 high = reference <= (1023 - threshold) ? reference + threshold : 1023; return !((value >= low) && (value <= high)); } bool WideColorSurfaceTest::checkWithThresholdFloat(float value, float reference, float threshold) { const float low = reference - threshold; const float high = reference + threshold; return !((value >= low) && (value <= high)); } void WideColorSurfaceTest::testPixels (float reference, float increment) { tcu::TestLog& log = m_testCtx.getLog(); if (m_componentType == EGL_COLOR_COMPONENT_TYPE_FLOAT_EXT) { float pixels[16]; const float expected[4] = { reference, reference + increment, reference - increment, reference + 2 * increment }; readPixels(m_gl, pixels); if (checkWithThresholdFloat(pixels[0], expected[0], increment) || checkWithThresholdFloat(pixels[1], expected[1], increment) || checkWithThresholdFloat(pixels[2], expected[2], increment) || checkWithThresholdFloat(pixels[3], expected[3], increment)) { if (m_debugLog.str().size() > 0) { log << tcu::TestLog::Message << "Prior passing tests\n" << m_debugLog.str() << tcu::TestLog::EndMessage; m_debugLog.str(""); } log << tcu::TestLog::Message << "Image comparison failed: " << "reference = " << reference << ", expected = " << expected[0] << ":" << expected[1] << ":" << expected[2] << ":" << expected[3] << ", result = " << pixels[0] << ":" << pixels[1] << ":" << pixels[2] << ":" << pixels[3] << tcu::TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Color test failed"); } else { // Pixel matches expected value m_debugLog << "Image comparison passed: " << "reference = " << reference << ", result = " << pixels[0] << ":" << pixels[1] << ":" << pixels[2] << ":" << pixels[3] << "\n"; } } else if (m_redSize > 8) { deUint32 buffer[16]; readPixels(m_gl, buffer); deUint32 pixels[4]; deUint32 expected[4]; pixels[0] = buffer[0] & 0x3ff; pixels[1] = (buffer[0] >> 10) & 0x3ff; pixels[2] = (buffer[0] >> 20) & 0x3ff; pixels[3] = (buffer[0] >> 30) & 0x3; expected[0] = expectedUint10(reference); expected[1] = expectedUint10(reference + increment); expected[2] = expectedUint10(reference - increment); expected[3] = expectedUint2(reference + 2 * increment); if (checkWithThreshold10(pixels[0], expected[0]) || checkWithThreshold10(pixels[1], expected[1]) || checkWithThreshold10(pixels[2], expected[2]) || checkWithThreshold10(pixels[3], expected[3])) { if (m_debugLog.str().size() > 0) { log << tcu::TestLog::Message << "Prior passing tests\n" << m_debugLog.str() << tcu::TestLog::EndMessage; m_debugLog.str(""); } log << tcu::TestLog::Message << "Image comparison failed: " << "reference = " << reference << ", expected = " << static_cast<deUint32>(expected[0]) << ":" << static_cast<deUint32>(expected[1]) << ":" << static_cast<deUint32>(expected[2]) << ":" << static_cast<deUint32>(expected[3]) << ", result = " << static_cast<deUint32>(pixels[0]) << ":" << static_cast<deUint32>(pixels[1]) << ":" << static_cast<deUint32>(pixels[2]) << ":" << static_cast<deUint32>(pixels[3]) << tcu::TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Color test failed"); } else { // Pixel matches expected value m_debugLog << "Image comparison passed: " << "reference = " << reference << ", result = " << static_cast<deUint32>(pixels[0]) << ":" << static_cast<deUint32>(pixels[1]) << ":" << static_cast<deUint32>(pixels[2]) << ":" << static_cast<deUint32>(pixels[3]) << "\n"; } } else { deUint8 pixels[16]; deUint8 expected[4]; readPixels(m_gl, pixels); expected[0] = expectedUint8(reference); expected[1] = expectedUint8(reference + increment); expected[2] = expectedUint8(reference - increment); expected[3] = expectedAlpha8(reference + 2 * increment); if (checkWithThreshold8(pixels[0], expected[0]) || checkWithThreshold8(pixels[1], expected[1]) || checkWithThreshold8(pixels[2], expected[2]) || checkWithThreshold8(pixels[3], expected[3])) { if (m_debugLog.str().size() > 0) { log << tcu::TestLog::Message << "Prior passing tests\n" << m_debugLog.str() << tcu::TestLog::EndMessage; m_debugLog.str(""); } log << tcu::TestLog::Message << "Image comparison failed: " << "reference = " << reference << ", expected = " << static_cast<deUint32>(expected[0]) << ":" << static_cast<deUint32>(expected[1]) << ":" << static_cast<deUint32>(expected[2]) << ":" << static_cast<deUint32>(expected[3]) << ", result = " << static_cast<deUint32>(pixels[0]) << ":" << static_cast<deUint32>(pixels[1]) << ":" << static_cast<deUint32>(pixels[2]) << ":" << static_cast<deUint32>(pixels[3]) << tcu::TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Color test failed"); } else { // Pixel matches expected value m_debugLog << "Image comparison passed: " << "reference = " << reference << ", result = " << static_cast<deUint32>(pixels[0]) << ":" << static_cast<deUint32>(pixels[1]) << ":" << static_cast<deUint32>(pixels[2]) << ":" << static_cast<deUint32>(pixels[3]) << "\n"; } } } void WideColorSurfaceTest::testFramebufferColorEncoding() { GLint framebufferColorEncoding; m_gl.getFramebufferAttachmentParameteriv(GL_FRAMEBUFFER, GL_BACK, GL_FRAMEBUFFER_ATTACHMENT_COLOR_ENCODING, &framebufferColorEncoding); GLU_EXPECT_NO_ERROR(m_gl.getError(), "Get framebuffer color encoding"); bool correct = true; if (m_colorSpace == EGL_GL_COLORSPACE_SRGB_KHR || m_colorSpace == EGL_GL_COLORSPACE_DISPLAY_P3_EXT) { if (m_redSize == 8) { correct = framebufferColorEncoding == GL_SRGB; } else if (m_redSize == 16) { correct = framebufferColorEncoding == GL_LINEAR; } } else { correct = framebufferColorEncoding == GL_LINEAR; } if (!correct) { m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Framebuffer color encoding is wrong"); } } void WideColorSurfaceTest::doClearTest (EGLSurface surface) { tcu::TestLog& log = m_testCtx.getLog(); const Library& egl = m_eglTestCtx.getLibrary(); const EGLint attribList[] = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE }; EGLContext eglContext = egl.createContext(m_eglDisplay, m_eglConfig, EGL_NO_CONTEXT, attribList); EGLU_CHECK_MSG(egl, "eglCreateContext"); egl.makeCurrent(m_eglDisplay, surface, surface, eglContext); EGLU_CHECK_MSG(egl, "eglMakeCurrent"); { // put gles2Renderer inside it's own scope so that it's cleaned // up before we hit the destroyContext const GLES2Renderer gles2Renderer(m_gl, 128, 128); std::vector<Iteration>::const_iterator it; // declare an Iterator to a vector of strings log << tcu::TestLog::Message << "m_iterations.count = " << m_iterations.size() << tcu::TestLog::EndMessage; for(it = m_iterations.begin() ; it < m_iterations.end(); it++) { float reference = it->start; log << tcu::TestLog::Message << "start = " << it->start << tcu::TestLog::EndMessage; log << tcu::TestLog::Message << "increment = " << it->increment << tcu::TestLog::EndMessage; log << tcu::TestLog::Message << "count = " << it->iterationCount << tcu::TestLog::EndMessage; m_debugLog.str(""); for (int iterationCount = 0; iterationCount < it->iterationCount; iterationCount++) { const Color clearColor(reference, reference + it->increment, reference - it->increment, reference + 2 * it->increment); clearColorScreen(m_gl, clearColor); GLU_EXPECT_NO_ERROR(m_gl.getError(), "Clear to test value"); testPixels(reference, it->increment); // reset buffer contents so that we know render below did something const Color clearColor2(1.0f - reference, 1.0f, 1.0f, 1.0f); clearColorScreen(m_gl, clearColor2); GLU_EXPECT_NO_ERROR(m_gl.getError(), "Clear to 1.0f - reference value"); const ColoredRect coloredRect (IVec2(0, 0), IVec2(1, 1), clearColor); gles2Renderer.render(coloredRect); testPixels(reference, it->increment); reference += it->increment; // Detect compatible GLES context by querying GL_MAJOR_VERSION. // This query does not exist on GLES2 so succeeding query implies GLES3+ context. glw::GLint majorVersion = 0; m_gl.getIntegerv(GL_MAJOR_VERSION, &majorVersion); if (m_gl.getError() == GL_NO_ERROR) { // This device is ES3 compatible, so do some additional testing testFramebufferColorEncoding(); } } EGLU_CHECK_CALL(egl, swapBuffers(m_eglDisplay, surface)); } } // disconnect surface & context so they can be destroyed when // this function exits. EGLU_CHECK_CALL(egl, makeCurrent(m_eglDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT)); egl.destroyContext(m_eglDisplay, eglContext); } void WideColorSurfaceTest::executeTest (void) { tcu::TestLog& log = m_testCtx.getLog(); const Library& egl = m_eglTestCtx.getLibrary(); const eglu::NativeDisplayFactory& displayFactory = m_eglTestCtx.getNativeDisplayFactory(); eglu::NativeDisplay& nativeDisplay = m_eglTestCtx.getNativeDisplay(); egl.bindAPI(EGL_OPENGL_ES_API); if (m_surfaceType & EGL_PBUFFER_BIT) { log << tcu::TestLog::Message << "Test Pbuffer" << tcu::TestLog::EndMessage; std::vector<EGLint> attribs; attribs.push_back(EGL_WIDTH); attribs.push_back(128); attribs.push_back(EGL_HEIGHT); attribs.push_back(128); if (m_colorSpace != EGL_NONE) { attribs.push_back(EGL_GL_COLORSPACE_KHR); attribs.push_back(m_colorSpace); } attribs.push_back(EGL_NONE); attribs.push_back(EGL_NONE); const EGLSurface surface = egl.createPbufferSurface(m_eglDisplay, m_eglConfig, attribs.data()); if ((surface == EGL_NO_SURFACE) && (egl.getError() == EGL_BAD_MATCH)) { TCU_THROW(NotSupportedError, "Colorspace is not supported with this format"); } TCU_CHECK(surface != EGL_NO_SURFACE); EGLU_CHECK_MSG(egl, "eglCreatePbufferSurface()"); doClearTest(surface); egl.destroySurface(m_eglDisplay, surface); EGLU_CHECK_MSG(egl, "eglDestroySurface()"); } else if (m_surfaceType & EGL_WINDOW_BIT) { log << tcu::TestLog::Message << "Test Window" << tcu::TestLog::EndMessage; const eglu::NativeWindowFactory& windowFactory = eglu::selectNativeWindowFactory(displayFactory, m_testCtx.getCommandLine()); de::UniquePtr<eglu::NativeWindow> window (windowFactory.createWindow(&nativeDisplay, m_eglDisplay, m_eglConfig, DE_NULL, eglu::WindowParams(128, 128, eglu::parseWindowVisibility(m_testCtx.getCommandLine())))); std::vector<EGLAttrib> attribs; if (m_colorSpace != EGL_NONE) { attribs.push_back(EGL_GL_COLORSPACE_KHR); attribs.push_back(m_colorSpace); } attribs.push_back(EGL_NONE); attribs.push_back(EGL_NONE); EGLSurface surface; try { surface = eglu::createWindowSurface(nativeDisplay, *window, m_eglDisplay, m_eglConfig, attribs.data()); } catch (const eglu::Error& error) { if (error.getError() == EGL_BAD_MATCH) TCU_THROW(NotSupportedError, "createWindowSurface is not supported for this config"); throw; } TCU_CHECK(surface != EGL_NO_SURFACE); EGLU_CHECK_MSG(egl, "eglCreateWindowSurface()"); doClearTest(surface); if (m_testAttribList.size() > 0) { for (deUint32 i = 0; i < m_testAttribList.size(); i +=2) { if (!egl.surfaceAttrib(m_eglDisplay, surface, m_testAttribList[i], m_testAttribList[i+1])) { // Implementation can return EGL_BAD_PARAMETER if given value is not supported. EGLint error = egl.getError(); if (error != EGL_BAD_PARAMETER) TCU_FAIL("Unable to set HDR metadata on surface"); log << tcu::TestLog::Message << "Warning: Metadata value " << m_testAttribList[i+1] << " for attrib 0x" << std::hex << m_testAttribList[i] << std::dec << " not supported by the implementation." << tcu::TestLog::EndMessage; m_testAttribList[i+1] = EGL_BAD_PARAMETER; } } for (deUint32 i = 0; i < m_testAttribList.size(); i +=2) { // Skip unsupported values. if (m_testAttribList[i+1] == EGL_BAD_PARAMETER) continue; EGLint value; egl.querySurface(m_eglDisplay, surface, m_testAttribList[i], &value); TCU_CHECK(value == m_testAttribList[i+1]); } } egl.destroySurface(m_eglDisplay, surface); EGLU_CHECK_MSG(egl, "eglDestroySurface()"); } else if (m_surfaceType & EGL_PIXMAP_BIT) { log << tcu::TestLog::Message << "Test Pixmap" << tcu::TestLog::EndMessage; const eglu::NativePixmapFactory& pixmapFactory = eglu::selectNativePixmapFactory(displayFactory, m_testCtx.getCommandLine()); de::UniquePtr<eglu::NativePixmap> pixmap (pixmapFactory.createPixmap(&nativeDisplay, m_eglDisplay, m_eglConfig, DE_NULL, 128, 128)); const EGLSurface surface = eglu::createPixmapSurface(nativeDisplay, *pixmap, m_eglDisplay, m_eglConfig, DE_NULL); TCU_CHECK(surface != EGL_NO_SURFACE); EGLU_CHECK_MSG(egl, "eglCreatePixmapSurface()"); doClearTest(surface); egl.destroySurface(m_eglDisplay, surface); EGLU_CHECK_MSG(egl, "eglDestroySurface()"); } else TCU_FAIL("No valid surface types supported in config"); } TestCase::IterateResult WideColorSurfaceTest::iterate (void) { m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); executeTest(); return STOP; } } // anonymous WideColorTests::WideColorTests (EglTestContext& eglTestCtx) : TestCaseGroup(eglTestCtx, "wide_color", "Wide Color tests") { } void WideColorTests::init (void) { addChild(new WideColorFP16Test(m_eglTestCtx, "fp16", "Verify that FP16 pixel format is present")); addChild(new WideColor1010102Test(m_eglTestCtx, "1010102", "Check if 1010102 pixel format is present")); // This is an increment FP16 can do between -1.0 to 1.0 const float fp16Increment1 = deFloatPow(2.0, -11.0); // This is an increment FP16 can do between 1.0 to 2.0 const float fp16Increment2 = deFloatPow(2.0, -10.0); std::vector<Iteration> iterations; // -0.333251953125f ~ -1/3 as seen in FP16 // Negative values will be 0 on read with fixed point pixel formats iterations.push_back(Iteration(-0.333251953125f, fp16Increment1, 10)); // test crossing 0 iterations.push_back(Iteration(-fp16Increment1 * 5.0f, fp16Increment1, 10)); // test crossing 1.0 // Values > 1.0 will be truncated to 1.0 with fixed point pixel formats iterations.push_back(Iteration(1.0f - fp16Increment2 * 5.0f, fp16Increment2, 10)); const EGLint windowAttribListFP16[] = { EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 16, EGL_GREEN_SIZE, 16, EGL_BLUE_SIZE, 16, EGL_ALPHA_SIZE, 16, EGL_COLOR_COMPONENT_TYPE_EXT, EGL_COLOR_COMPONENT_TYPE_FLOAT_EXT, EGL_NONE, EGL_NONE }; addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_fp16_default_colorspace", "FP16 window surface has FP16 pixels in it", windowAttribListFP16, EGL_NONE, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_fp16_colorspace_srgb", "FP16 window surface, explicit sRGB colorspace", windowAttribListFP16, EGL_GL_COLORSPACE_SRGB_KHR, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_fp16_colorspace_p3", "FP16 window surface, explicit Display-P3 colorspace", windowAttribListFP16, EGL_GL_COLORSPACE_DISPLAY_P3_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_fp16_colorspace_p3_passthrough", "FP16 window surface, explicit Display-P3 colorspace", windowAttribListFP16, EGL_GL_COLORSPACE_DISPLAY_P3_PASSTHROUGH_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_fp16_colorspace_scrgb", "FP16 window surface, explicit scRGB colorspace", windowAttribListFP16, EGL_GL_COLORSPACE_SCRGB_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_fp16_colorspace_scrgb_linear", "FP16 window surface, explicit scRGB linear colorspace", windowAttribListFP16, EGL_GL_COLORSPACE_SCRGB_LINEAR_EXT, iterations)); const EGLint pbufferAttribListFP16[] = { EGL_SURFACE_TYPE, EGL_PBUFFER_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 16, EGL_GREEN_SIZE, 16, EGL_BLUE_SIZE, 16, EGL_ALPHA_SIZE, 16, EGL_COLOR_COMPONENT_TYPE_EXT, EGL_COLOR_COMPONENT_TYPE_FLOAT_EXT, EGL_NONE, EGL_NONE }; addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_fp16_default_colorspace", "FP16 pbuffer surface has FP16 pixels in it", pbufferAttribListFP16, EGL_NONE, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_fp16_colorspace_srgb", "FP16 pbuffer surface, explicit sRGB colorspace", pbufferAttribListFP16, EGL_GL_COLORSPACE_SRGB_KHR, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_fp16_colorspace_p3", "FP16 pbuffer surface, explicit Display-P3 colorspace", pbufferAttribListFP16, EGL_GL_COLORSPACE_DISPLAY_P3_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_fp16_colorspace_p3_passthrough", "FP16 pbuffer surface, explicit Display-P3 colorspace", pbufferAttribListFP16, EGL_GL_COLORSPACE_DISPLAY_P3_PASSTHROUGH_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_fp16_colorspace_scrgb", "FP16 pbuffer surface, explicit scRGB colorspace", pbufferAttribListFP16, EGL_GL_COLORSPACE_SCRGB_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_fp16_colorspace_scrgb_linear", "FP16 pbuffer surface, explicit scRGB linear colorspace", pbufferAttribListFP16, EGL_GL_COLORSPACE_SCRGB_LINEAR_EXT, iterations)); const EGLint windowAttribList1010102[] = { EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 10, EGL_GREEN_SIZE, 10, EGL_BLUE_SIZE, 10, EGL_ALPHA_SIZE, 2, EGL_NONE, EGL_NONE }; addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_1010102_colorspace_default", "1010102 Window surface, default (sRGB) colorspace", windowAttribList1010102, EGL_NONE, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_1010102_colorspace_srgb", "1010102 Window surface, explicit sRGB colorspace", windowAttribList1010102, EGL_GL_COLORSPACE_SRGB_KHR, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_1010102_colorspace_p3", "1010102 Window surface, explicit Display-P3 colorspace", windowAttribList1010102, EGL_GL_COLORSPACE_DISPLAY_P3_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_1010102_colorspace_p3_passthrough", "1010102 Window surface, explicit Display-P3 colorspace", windowAttribList1010102, EGL_GL_COLORSPACE_DISPLAY_P3_PASSTHROUGH_EXT, iterations)); const EGLint pbufferAttribList1010102[] = { EGL_SURFACE_TYPE, EGL_PBUFFER_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 10, EGL_GREEN_SIZE, 10, EGL_BLUE_SIZE, 10, EGL_ALPHA_SIZE, 2, EGL_NONE, EGL_NONE }; addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_1010102_colorspace_default", "1010102 pbuffer surface, default (sRGB) colorspace", pbufferAttribList1010102, EGL_NONE, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_1010102_colorspace_srgb", "1010102 pbuffer surface, explicit sRGB colorspace", pbufferAttribList1010102, EGL_GL_COLORSPACE_SRGB_KHR, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_1010102_colorspace_p3", "1010102 pbuffer surface, explicit Display-P3 colorspace", pbufferAttribList1010102, EGL_GL_COLORSPACE_DISPLAY_P3_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_1010102_colorspace_p3_passthrough", "1010102 pbuffer surface, explicit Display-P3 colorspace", pbufferAttribList1010102, EGL_GL_COLORSPACE_DISPLAY_P3_PASSTHROUGH_EXT, iterations)); const EGLint windowAttribList8888[] = { EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_BLUE_SIZE, 8, EGL_ALPHA_SIZE, 8, EGL_NONE, EGL_NONE }; addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_8888_colorspace_default", "8888 window surface, default (sRGB) colorspace", windowAttribList8888, EGL_NONE, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_8888_colorspace_srgb", "8888 window surface, explicit sRGB colorspace", windowAttribList8888, EGL_GL_COLORSPACE_SRGB_KHR, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_8888_colorspace_p3", "8888 window surface, explicit Display-P3 colorspace", windowAttribList8888, EGL_GL_COLORSPACE_DISPLAY_P3_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_8888_colorspace_p3_passthrough", "8888 window surface, explicit Display-P3 colorspace", windowAttribList8888, EGL_GL_COLORSPACE_DISPLAY_P3_PASSTHROUGH_EXT, iterations)); const EGLint pbufferAttribList8888[] = { EGL_SURFACE_TYPE, EGL_PBUFFER_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_BLUE_SIZE, 8, EGL_ALPHA_SIZE, 8, EGL_NONE, EGL_NONE }; addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_8888_colorspace_default", "8888 pbuffer surface, default (sRGB) colorspace", pbufferAttribList8888, EGL_NONE, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_8888_colorspace_srgb", "8888 pbuffer surface, explicit sRGB colorspace", pbufferAttribList8888, EGL_GL_COLORSPACE_SRGB_KHR, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_8888_colorspace_p3", "8888 pbuffer surface, explicit Display-P3 colorspace", pbufferAttribList8888, EGL_GL_COLORSPACE_DISPLAY_P3_EXT, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_8888_colorspace_p3_passthrough", "8888 pbuffer surface, explicit Display-P3 colorspace", pbufferAttribList8888, EGL_GL_COLORSPACE_DISPLAY_P3_PASSTHROUGH_EXT, iterations)); const EGLint windowAttribList888[] = { EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_BLUE_SIZE, 8, EGL_NONE, EGL_NONE }; addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_888_colorspace_default", "888 window surface, default (sRGB) colorspace", windowAttribList888, EGL_NONE, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_888_colorspace_srgb", "888 window surface, explicit sRGB colorspace", windowAttribList888, EGL_GL_COLORSPACE_SRGB_KHR, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "window_888_colorspace_p3", "888 window surface, explicit Display-P3 colorspace", windowAttribList888, EGL_GL_COLORSPACE_DISPLAY_P3_EXT, iterations)); const EGLint pbufferAttribList888[] = { EGL_SURFACE_TYPE, EGL_PBUFFER_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_BLUE_SIZE, 8, EGL_NONE, EGL_NONE }; addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_888_colorspace_default", "888 pbuffer surface, default (sRGB) colorspace", pbufferAttribList888, EGL_NONE, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_888_colorspace_srgb", "888 pbuffer surface, explicit sRGB colorspace", pbufferAttribList888, EGL_GL_COLORSPACE_SRGB_KHR, iterations)); addChild(new WideColorSurfaceTest(m_eglTestCtx, "pbuffer_888_colorspace_p3", "888 pbuffer surface, explicit Display-P3 colorspace", pbufferAttribList888, EGL_GL_COLORSPACE_DISPLAY_P3_EXT, iterations)); } TestCaseGroup* createWideColorTests (EglTestContext& eglTestCtx) { return new WideColorTests(eglTestCtx); } class Smpte2086ColorTest : public WideColorTest { public: Smpte2086ColorTest (EglTestContext& eglTestCtx, const char* name, const char* description); void executeTest (void); IterateResult iterate (void); }; Smpte2086ColorTest::Smpte2086ColorTest (EglTestContext& eglTestCtx, const char* name, const char* description) : WideColorTest(eglTestCtx, name, description) { } #define METADATA_SCALE(x) (static_cast<EGLint>(x * EGL_METADATA_SCALING_EXT)) void Smpte2086ColorTest::executeTest (void) { tcu::TestLog& log = m_testCtx.getLog(); const Library& egl = m_eglTestCtx.getLibrary(); egl.bindAPI(EGL_OPENGL_ES_API); log << tcu::TestLog::Message << "Test SMPTE 2086 Metadata on Window" << tcu::TestLog::EndMessage; checkSMPTE2086(); // This is an increment FP16 can do between -1.0 to 1.0 const float fp16Increment1 = deFloatPow(2.0, -11.0); // This is an increment FP16 can do between 1.0 to 2.0 const float fp16Increment2 = deFloatPow(2.0, -10.0); std::vector<Iteration> int8888Iterations; // -0.333251953125f ~ -1/3 as seen in fp16 // Negative values will be 0 on read with fixed point pixel formats int8888Iterations.push_back(Iteration(-0.333251953125f, fp16Increment1, 10)); // test crossing 0 int8888Iterations.push_back(Iteration(-fp16Increment1 * 5.0f, fp16Increment1, 10)); // test crossing 1.0 // Values > 1.0 will be truncated to 1.0 with fixed point pixel formats int8888Iterations.push_back(Iteration(1.0f - fp16Increment2 * 5.0f, fp16Increment2, 10)); const EGLint windowAttribList8888[] = { EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_BLUE_SIZE, 8, EGL_ALPHA_SIZE, 8, EGL_NONE, EGL_NONE }; WideColorSurfaceTest testObj(m_eglTestCtx, "window_8888_colorspace_default", "8888 window surface, default (sRGB) colorspace", windowAttribList8888, EGL_NONE, int8888Iterations); const EGLint testAttrs[] = { EGL_SMPTE2086_DISPLAY_PRIMARY_RX_EXT, METADATA_SCALE(0.680), EGL_SMPTE2086_DISPLAY_PRIMARY_RY_EXT, METADATA_SCALE(0.320), EGL_SMPTE2086_DISPLAY_PRIMARY_GX_EXT, METADATA_SCALE(0.265), EGL_SMPTE2086_DISPLAY_PRIMARY_GY_EXT, METADATA_SCALE(0.690), EGL_SMPTE2086_DISPLAY_PRIMARY_BX_EXT, METADATA_SCALE(0.440), EGL_SMPTE2086_DISPLAY_PRIMARY_BY_EXT, METADATA_SCALE(0.320), EGL_SMPTE2086_WHITE_POINT_X_EXT, METADATA_SCALE(0.2200), EGL_SMPTE2086_WHITE_POINT_Y_EXT, METADATA_SCALE(0.2578), EGL_SMPTE2086_MAX_LUMINANCE_EXT, METADATA_SCALE(1.31), EGL_SMPTE2086_MIN_LUMINANCE_EXT, METADATA_SCALE(0.123), EGL_NONE }; testObj.addTestAttributes(testAttrs); testObj.init(); testObj.executeTest(); } TestCase::IterateResult Smpte2086ColorTest::iterate (void) { m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); executeTest(); return STOP; } class Cta8613ColorTest : public WideColorTest { public: Cta8613ColorTest (EglTestContext& eglTestCtx, const char* name, const char* description); void executeTest (void); IterateResult iterate (void); }; Cta8613ColorTest::Cta8613ColorTest (EglTestContext& eglTestCtx, const char* name, const char* description) : WideColorTest(eglTestCtx, name, description) { } #define METADATA_SCALE(x) (static_cast<EGLint>(x * EGL_METADATA_SCALING_EXT)) void Cta8613ColorTest::executeTest (void) { tcu::TestLog& log = m_testCtx.getLog(); const Library& egl = m_eglTestCtx.getLibrary(); egl.bindAPI(EGL_OPENGL_ES_API); log << tcu::TestLog::Message << "Test CTA 861.3 Metadata on Window" << tcu::TestLog::EndMessage; checkCTA861_3(); // This is an increment FP16 can do between -1.0 to 1.0 const float fp16Increment1 = deFloatPow(2.0, -11.0); // This is an increment FP16 can do between 1.0 to 2.0 const float fp16Increment2 = deFloatPow(2.0, -10.0); std::vector<Iteration> int8888Iterations; // -0.333251953125f ~ -1/3 as seen in fp16 // Negative values will be 0 on read with fixed point pixel formats int8888Iterations.push_back(Iteration(-0.333251953125f, fp16Increment1, 10)); // test crossing 0 int8888Iterations.push_back(Iteration(-fp16Increment1 * 5.0f, fp16Increment1, 10)); // test crossing 1.0 // Values > 1.0 will be truncated to 1.0 with fixed point pixel formats int8888Iterations.push_back(Iteration(1.0f - fp16Increment2 * 5.0f, fp16Increment2, 10)); const EGLint windowAttribList8888[] = { EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_RED_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_BLUE_SIZE, 8, EGL_ALPHA_SIZE, 8, EGL_NONE, EGL_NONE }; WideColorSurfaceTest testObj(m_eglTestCtx, "window_8888_colorspace_default", "8888 window surface, default (sRGB) colorspace", windowAttribList8888, EGL_NONE, int8888Iterations); const EGLint testAttrs[] = { EGL_CTA861_3_MAX_CONTENT_LIGHT_LEVEL_EXT, METADATA_SCALE(1.31), EGL_CTA861_3_MAX_FRAME_AVERAGE_LEVEL_EXT, METADATA_SCALE(0.6), EGL_NONE }; testObj.addTestAttributes(testAttrs); testObj.init(); testObj.executeTest(); } TestCase::IterateResult Cta8613ColorTest::iterate (void) { m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); executeTest(); return STOP; } class HdrColorTests : public TestCaseGroup { public: HdrColorTests (EglTestContext& eglTestCtx); void init (void); private: HdrColorTests (const HdrColorTests&); HdrColorTests& operator= (const HdrColorTests&); }; HdrColorTests::HdrColorTests (EglTestContext& eglTestCtx) : TestCaseGroup(eglTestCtx, "hdr_metadata", "HDR Metadata tests") { } void HdrColorTests::init (void) { addChild(new Smpte2086ColorTest(m_eglTestCtx, "smpte2086", "Verify that SMPTE 2086 extension exists")); addChild(new Cta8613ColorTest(m_eglTestCtx, "cta861_3", "Verify that CTA 861.3 extension exists")); } TestCaseGroup* createHdrColorTests (EglTestContext& eglTestCtx) { return new HdrColorTests(eglTestCtx); } } // egl } // deqp