306 lines
6.9 KiB
C++
306 lines
6.9 KiB
C++
/*-------------------------------------------------------------------------
|
|
* 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
|