| /*------------------------------------------------------------------------- |
| * 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 |