/*------------------------------------------------------------------------- * drawElements Quality Program EGL Module * --------------------------------------- * * Copyright 2016 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 thread clean up tests *//*--------------------------------------------------------------------*/ #include "teglThreadCleanUpTests.hpp" #include "egluUtil.hpp" #include "egluUnique.hpp" #include "egluConfigFilter.hpp" #include "eglwLibrary.hpp" #include "eglwEnums.hpp" #include "tcuMaybe.hpp" #include "tcuTestLog.hpp" #include "deThread.hpp" namespace deqp { namespace egl { namespace { using namespace eglw; using tcu::TestLog; bool isES2Renderable (const eglu::CandidateConfig& c) { return (c.get(EGL_RENDERABLE_TYPE) & EGL_OPENGL_ES2_BIT) == EGL_OPENGL_ES2_BIT; } bool isPBuffer (const eglu::CandidateConfig& c) { return (c.surfaceType() & EGL_PBUFFER_BIT) == EGL_PBUFFER_BIT; } class Thread : public de::Thread { public: Thread (const Library& egl, EGLDisplay display, EGLSurface surface, EGLContext context, EGLConfig config, tcu::Maybe<eglu::Error>& error) : m_egl (egl) , m_display (display) , m_surface (surface) , m_context (context) , m_config (config) , m_error (error) { } void testContext (EGLContext context) { if (m_surface != EGL_NO_SURFACE) { EGLU_CHECK_MSG(m_egl, "eglCreateContext"); m_egl.makeCurrent(m_display, m_surface, m_surface, context); EGLU_CHECK_MSG(m_egl, "eglMakeCurrent"); } else { const EGLint attribs[] = { EGL_WIDTH, 32, EGL_HEIGHT, 32, EGL_NONE }; const eglu::UniqueSurface surface (m_egl, m_display, m_egl.createPbufferSurface(m_display, m_config, attribs)); EGLU_CHECK_MSG(m_egl, "eglCreateContext"); m_egl.makeCurrent(m_display, *surface, *surface, context); EGLU_CHECK_MSG(m_egl, "eglMakeCurrent"); } } void run (void) { try { const EGLint attribList[] = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE }; m_egl.bindAPI(EGL_OPENGL_ES_API); if (m_context == EGL_NO_CONTEXT) { const eglu::UniqueContext context (m_egl, m_display, m_egl.createContext(m_display, m_config, EGL_NO_CONTEXT, attribList)); testContext(*context); } else { testContext(m_context); } } catch (const eglu::Error& error) { m_error = error; } m_egl.releaseThread(); } private: const Library& m_egl; const EGLDisplay m_display; const EGLSurface m_surface; const EGLContext m_context; const EGLConfig m_config; tcu::Maybe<eglu::Error>& m_error; }; class ThreadCleanUpTest : public TestCase { public: enum ContextType { CONTEXTTYPE_SINGLE = 0, CONTEXTTYPE_MULTI }; enum SurfaceType { SURFACETYPE_SINGLE = 0, SURFACETYPE_MULTI }; static std::string testCaseName (ContextType contextType, SurfaceType surfaceType) { std::string name; if (contextType == CONTEXTTYPE_SINGLE) name += "single_context_"; else name += "multi_context_"; if (surfaceType ==SURFACETYPE_SINGLE) name += "single_surface"; else name += "multi_surface"; return name; } ThreadCleanUpTest (EglTestContext& eglTestCtx, ContextType contextType, SurfaceType surfaceType) : TestCase (eglTestCtx, testCaseName(contextType, surfaceType).c_str(), "Simple thread context clean up test") , m_contextType (contextType) , m_surfaceType (surfaceType) , m_iterCount (250) , m_iterNdx (0) , m_display (EGL_NO_DISPLAY) , m_config (0) , m_surface (EGL_NO_SURFACE) , m_context (EGL_NO_CONTEXT) { } ~ThreadCleanUpTest (void) { deinit(); } void init (void) { const Library& egl = m_eglTestCtx.getLibrary(); m_display = eglu::getAndInitDisplay(m_eglTestCtx.getNativeDisplay()); { eglu::FilterList filters; filters << isES2Renderable << isPBuffer; m_config = eglu::chooseSingleConfig(egl, m_display, filters); } if (m_contextType == CONTEXTTYPE_SINGLE) { const EGLint attribList[] = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE }; egl.bindAPI(EGL_OPENGL_ES_API); m_context = egl.createContext(m_display, m_config, EGL_NO_CONTEXT, attribList); EGLU_CHECK_MSG(egl, "Failed to create context"); } if (m_surfaceType == SURFACETYPE_SINGLE) { const EGLint attribs[] = { EGL_WIDTH, 32, EGL_HEIGHT, 32, EGL_NONE }; m_surface = egl.createPbufferSurface(m_display, m_config, attribs); EGLU_CHECK_MSG(egl, "Failed to create surface"); } } void deinit (void) { const Library& egl = m_eglTestCtx.getLibrary(); if (m_surface != EGL_NO_SURFACE) { egl.destroySurface(m_display, m_surface); m_surface = EGL_NO_SURFACE; } if (m_context != EGL_NO_CONTEXT) { egl.destroyContext(m_display, m_context); m_context = EGL_NO_CONTEXT; } if (m_display != EGL_NO_DISPLAY) { egl.terminate(m_display); m_display = EGL_NO_DISPLAY; } } IterateResult iterate (void) { if (m_iterNdx < m_iterCount) { tcu::Maybe<eglu::Error> error; Thread thread (m_eglTestCtx.getLibrary(), m_display, m_surface, m_context, m_config, error); thread.start(); thread.join(); if (error) { m_testCtx.getLog() << TestLog::Message << "Failed. Got error: " << error->getMessage() << TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, error->getMessage()); return STOP; } m_iterNdx++; return CONTINUE; } else { m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); return STOP; } } private: const ContextType m_contextType; const SurfaceType m_surfaceType; const size_t m_iterCount; size_t m_iterNdx; EGLDisplay m_display; EGLConfig m_config; EGLSurface m_surface; EGLContext m_context; }; } // anonymous TestCaseGroup* createThreadCleanUpTest (EglTestContext& eglTestCtx) { de::MovePtr<TestCaseGroup> group (new TestCaseGroup(eglTestCtx, "thread_cleanup", "Thread cleanup tests")); group->addChild(new ThreadCleanUpTest(eglTestCtx, ThreadCleanUpTest::CONTEXTTYPE_SINGLE, ThreadCleanUpTest::SURFACETYPE_SINGLE)); group->addChild(new ThreadCleanUpTest(eglTestCtx, ThreadCleanUpTest::CONTEXTTYPE_MULTI, ThreadCleanUpTest::SURFACETYPE_SINGLE)); group->addChild(new ThreadCleanUpTest(eglTestCtx, ThreadCleanUpTest::CONTEXTTYPE_SINGLE, ThreadCleanUpTest::SURFACETYPE_MULTI)); group->addChild(new ThreadCleanUpTest(eglTestCtx, ThreadCleanUpTest::CONTEXTTYPE_MULTI, ThreadCleanUpTest::SURFACETYPE_MULTI)); return group.release(); } } // egl } // deqp