/*------------------------------------------------------------------------- * drawElements Quality Program EGL Module * --------------------------------------- * * 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_EXT_client_extensions tests *//*--------------------------------------------------------------------*/ #include "teglClientExtensionTests.hpp" #include "tcuTestLog.hpp" #include "egluUtil.hpp" #include "eglwLibrary.hpp" #include "eglwEnums.hpp" #include "deStringUtil.hpp" #include "deSTLUtil.hpp" #include <vector> #include <set> #include <string> #include <sstream> using std::string; using std::vector; using std::set; using tcu::TestLog; using namespace eglw; namespace deqp { namespace egl { namespace { static const char* const s_displayExtensionList[] = { "EGL_KHR_config_attribs", "EGL_KHR_lock_surface", "EGL_KHR_image", "EGL_KHR_vg_parent_image", "EGL_KHR_gl_texture_2D_image", "EGL_KHR_gl_texture_cubemap_image", "EGL_KHR_gl_texture_3D_image", "EGL_KHR_gl_renderbuffer_image", "EGL_KHR_reusable_sync", "EGL_KHR_image_base", "EGL_KHR_image_pixmap", "EGL_IMG_context_priority", "EGL_KHR_lock_surface2", "EGL_NV_coverage_sample", "EGL_NV_depth_nonlinear", "EGL_NV_sync", "EGL_KHR_fence_sync", "EGL_HI_clientpixmap", "EGL_HI_colorformats", "EGL_MESA_drm_image", "EGL_NV_post_sub_buffer", "EGL_ANGLE_query_surface_pointer", "EGL_ANGLE_surface_d3d_texture_2d_share_handle", "EGL_NV_coverage_sample_resolve", // "EGL_NV_system_time", \todo [mika] Unclear which one this is "EGL_KHR_stream", "EGL_KHR_stream_consumer_gltexture", "EGL_KHR_stream_producer_eglsurface", "EGL_KHR_stream_producer_aldatalocator", "EGL_KHR_stream_fifo", "EGL_EXT_create_context_robustness", "EGL_ANGLE_d3d_share_handle_client_buffer", "EGL_KHR_create_context", "EGL_KHR_surfaceless_context", "EGL_KHR_stream_cross_process_fd", "EGL_EXT_multiview_window", "EGL_KHR_wait_sync", "EGL_NV_post_convert_rounding", "EGL_NV_native_query", "EGL_NV_3dvision_surface", "EGL_ANDROID_framebuffer_target", "EGL_ANDROID_blob_cache", "EGL_ANDROID_image_native_buffer", "EGL_ANDROID_native_fence_sync", "EGL_ANDROID_recordable", "EGL_EXT_buffer_age", "EGL_EXT_image_dma_buf_import", "EGL_ARM_pixmap_multisample_discard", "EGL_EXT_swap_buffers_with_damage", "EGL_NV_stream_sync", "EGL_KHR_cl_event", "EGL_KHR_get_all_proc_addresses" }; static const char* const s_clientExtensionList[] = { "EGL_EXT_platform_base", "EGL_EXT_client_extensions", "EGL_EXT_platform_x11", "EGL_KHR_client_get_all_proc_addresses", "EGL_MESA_platform_gbm", "EGL_EXT_platform_wayland" }; class BaseTest : public TestCase { public: BaseTest (EglTestContext& eglTestCtx); IterateResult iterate (void); }; BaseTest::BaseTest (EglTestContext& eglTestCtx) : TestCase(eglTestCtx, "base", "Basic tests for EGL_EXT_client_extensions") { } TestCase::IterateResult BaseTest::iterate (void) { const Library& egl = m_eglTestCtx.getLibrary(); const char* const clientExtesionsStr = egl.queryString(EGL_NO_DISPLAY, EGL_EXTENSIONS); const EGLint eglError = egl.getError(); if (eglError == EGL_BAD_DISPLAY) TCU_THROW(NotSupportedError, "EGL_EXT_client_extensions not supported"); else if (eglError != EGL_SUCCESS) throw eglu::Error(eglError, "eglQueryString()", DE_NULL, __FILE__, __LINE__); TCU_CHECK(clientExtesionsStr); { bool found = false; std::istringstream stream (clientExtesionsStr); string extension; while (std::getline(stream, extension, ' ')) { if (extension == "EGL_EXT_client_extensions") { found = true; break; } } if (found) m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); else { m_testCtx.getLog() << TestLog::Message << "eglQueryString(EGL_NO_DISPLAY, EGL_EXTENSIONS) didn't fail, but extension string doesn't contain EGL_EXT_client_extensions" <<TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail"); } } return STOP; } class CheckExtensionsTest : public TestCase { public: CheckExtensionsTest (EglTestContext& eglTestCtx); IterateResult iterate (void); }; CheckExtensionsTest::CheckExtensionsTest (EglTestContext& eglTestCtx) : TestCase(eglTestCtx, "extensions", "Check that returned extensions are client or display extensions") { } TestCase::IterateResult CheckExtensionsTest::iterate (void) { const Library& egl = m_eglTestCtx.getLibrary(); bool isOk = true; const char* const clientExtensionsStr = egl.queryString(EGL_NO_DISPLAY, EGL_EXTENSIONS); const EGLint eglQueryError = egl.getError(); set<string> knownClientExtensions (s_clientExtensionList, s_clientExtensionList + DE_LENGTH_OF_ARRAY(s_clientExtensionList)); set<string> knownDisplayExtensions (s_displayExtensionList, s_displayExtensionList + DE_LENGTH_OF_ARRAY(s_displayExtensionList)); vector<string> displayExtensions; vector<string> clientExtensions; if (eglQueryError == EGL_BAD_DISPLAY) TCU_THROW(NotSupportedError, "EGL_EXT_client_extensions not supported"); else if (eglQueryError != EGL_SUCCESS) throw eglu::Error(eglQueryError, "eglQueryString()", DE_NULL, __FILE__, __LINE__); TCU_CHECK(clientExtensionsStr); clientExtensions = de::splitString(clientExtensionsStr, ' '); { EGLDisplay display = eglu::getAndInitDisplay(m_eglTestCtx.getNativeDisplay()); displayExtensions = de::splitString(egl.queryString(display, EGL_EXTENSIONS), ' '); egl.terminate(display); } for (int extNdx = 0; extNdx < (int)clientExtensions.size(); extNdx++) { if (knownDisplayExtensions.find(clientExtensions[extNdx]) != knownDisplayExtensions.end()) { m_testCtx.getLog() << TestLog::Message << "'" << clientExtensions[extNdx] << "' is not client extension" << TestLog::EndMessage; isOk = false; } } for (int extNdx = 0; extNdx < (int)displayExtensions.size(); extNdx++) { if (knownClientExtensions.find(displayExtensions[extNdx]) != knownClientExtensions.end()) { m_testCtx.getLog() << TestLog::Message << "'" << displayExtensions[extNdx] << "' is not display extension" << TestLog::EndMessage; isOk = false; } } if (isOk) m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); else m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail"); return STOP; } class DisjointTest : public TestCase { public: DisjointTest (EglTestContext& eglTestCtx); IterateResult iterate (void); }; DisjointTest::DisjointTest (EglTestContext& eglTestCtx) : TestCase(eglTestCtx, "disjoint", "Check that client and display extensions are disjoint") { } TestCase::IterateResult DisjointTest::iterate (void) { const Library& egl = m_eglTestCtx.getLibrary(); const char* const clientExtensionsStr = egl.queryString(EGL_NO_DISPLAY, EGL_EXTENSIONS); const EGLint eglQueryError = egl.getError(); if (eglQueryError == EGL_BAD_DISPLAY) TCU_THROW(NotSupportedError, "EGL_EXT_client_extensions not supported"); else if (eglQueryError != EGL_SUCCESS) throw eglu::Error(eglQueryError, "eglQueryString()", DE_NULL, __FILE__, __LINE__); vector<string> displayExtensions; vector<string> clientExtensions; clientExtensions = de::splitString(clientExtensionsStr, ' '); { EGLDisplay display = eglu::getAndInitDisplay(m_eglTestCtx.getNativeDisplay()); displayExtensions = de::splitString(egl.queryString(display, EGL_EXTENSIONS), ' '); egl.terminate(display); } // Log client extensions { const tcu::ScopedLogSection section(m_testCtx.getLog(), "Client extensions", "Client extensions"); for (int extNdx = 0; extNdx < (int)clientExtensions.size(); extNdx++) m_testCtx.getLog() << TestLog::Message << clientExtensions[extNdx] << TestLog::EndMessage; } // Log display extensions { const tcu::ScopedLogSection section(m_testCtx.getLog(), "Display extensions", "Display extensions"); for (int extNdx = 0; extNdx < (int)displayExtensions.size(); extNdx++) m_testCtx.getLog() << TestLog::Message << displayExtensions[extNdx] << TestLog::EndMessage; } // Check that sets are disjoint { set<string> commonExtensionSet; const set<string> clientExtensionSet(clientExtensions.begin(), clientExtensions.end()); const set<string> displayExtensionSet(displayExtensions.begin(), displayExtensions.end()); for (set<string>::const_iterator iter = clientExtensionSet.begin(); iter != clientExtensionSet.end(); ++iter) { if (displayExtensionSet.find(*iter) != displayExtensionSet.end()) commonExtensionSet.insert(*iter); } for (set<string>::const_iterator iter = commonExtensionSet.begin(); iter != commonExtensionSet.end(); ++iter) m_testCtx.getLog() << TestLog::Message << "Extension '" << *iter << "' exists in client and display extension sets." << TestLog::EndMessage; if (commonExtensionSet.empty()) m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); else { m_testCtx.getLog() << TestLog::Message << "Extension sets are not disjoint" << TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail"); } } return STOP; } } // anonymous ClientExtensionTests::ClientExtensionTests (EglTestContext& eglTestCtx) : TestCaseGroup(eglTestCtx, "client_extensions", "Test for EGL_EXT_client_extensions") { } void ClientExtensionTests::init (void) { addChild(new BaseTest(m_eglTestCtx)); addChild(new DisjointTest(m_eglTestCtx)); addChild(new CheckExtensionsTest(m_eglTestCtx)); } } // egl } // deqp