blob: 5a9f21dd1fb3a1814cca1d0b8e5612cc1e083a6e [file] [log] [blame] [edit]
/*-------------------------------------------------------------------------
* 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 multi context tests
*//*--------------------------------------------------------------------*/
#include "teglMultiContextTests.hpp"
#include "egluUtil.hpp"
#include "egluUnique.hpp"
#include "egluStrUtil.hpp"
#include "egluConfigFilter.hpp"
#include "eglwLibrary.hpp"
#include "eglwEnums.hpp"
#include "gluDefs.hpp"
#include "glwFunctions.hpp"
#include "glwEnums.hpp"
#include "tcuResultCollector.hpp"
#include "tcuTestLog.hpp"
#include "deRandom.hpp"
#include <vector>
namespace deqp
{
namespace egl
{
namespace
{
using tcu::TestLog;
class MultiContextTest : public TestCase
{
public:
enum Use
{
USE_NONE = 0,
USE_MAKECURRENT,
USE_CLEAR,
USE_LAST
};
enum Sharing
{
SHARING_NONE = 0,
SHARING_SHARED,
SHARING_LAST
};
MultiContextTest(EglTestContext &eglTestCtx, Sharing sharing, Use use, const char *name, const char *description);
IterateResult iterate(void);
private:
const Sharing m_sharing;
const Use m_use;
};
MultiContextTest::MultiContextTest(EglTestContext &eglTestCtx, Sharing sharing, Use use, const char *name,
const char *description)
: TestCase(eglTestCtx, name, description)
, m_sharing(sharing)
, m_use(use)
{
}
bool isES2Renderable(const eglu::CandidateConfig &c)
{
return (c.get(EGL_RENDERABLE_TYPE) & EGL_OPENGL_ES2_BIT) == EGL_OPENGL_ES2_BIT;
}
bool supportsPBuffer(const eglu::CandidateConfig &c)
{
return (c.get(EGL_SURFACE_TYPE) & EGL_PBUFFER_BIT) == EGL_PBUFFER_BIT;
}
eglw::EGLConfig getConfig(const eglw::Library &egl, eglw::EGLDisplay display)
{
eglu::FilterList filters;
filters << isES2Renderable;
filters << supportsPBuffer;
return eglu::chooseSingleConfig(egl, display, filters);
}
tcu::TestCase::IterateResult MultiContextTest::iterate(void)
{
const uint32_t seed = m_sharing == SHARING_SHARED ? 2498541716u : 8825414;
const size_t maxContextCount = 128;
const size_t minContextCount = 32;
const eglw::EGLint attribList[] = {EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE};
const eglw::EGLint pbufferAttribList[] = {EGL_WIDTH, 64, EGL_HEIGHT, 64, EGL_NONE};
TestLog &log = m_testCtx.getLog();
tcu::ResultCollector resultCollector(log);
de::Random rng(seed);
const eglw::Library &egl = m_eglTestCtx.getLibrary();
const eglu::UniqueDisplay display(egl, eglu::getAndInitDisplay(m_eglTestCtx.getNativeDisplay()));
const eglw::EGLConfig config = getConfig(egl, *display);
const eglu::UniqueSurface surface(
egl, *display,
m_use != USE_NONE ? egl.createPbufferSurface(*display, config, pbufferAttribList) : EGL_NO_SURFACE);
EGLU_CHECK_MSG(egl, "Failed to create pbuffer.");
std::vector<eglw::EGLContext> contexts;
glw::Functions gl;
contexts.reserve(maxContextCount);
log << TestLog::Message << "Trying to create " << maxContextCount
<< (m_sharing == SHARING_SHARED ? " shared " : " ") << "contexts." << TestLog::EndMessage;
log << TestLog::Message << "Requiring that at least " << minContextCount << " contexts can be created."
<< TestLog::EndMessage;
if (m_use == USE_CLEAR)
m_eglTestCtx.initGLFunctions(&gl, glu::ApiType::es(2, 0));
EGLU_CHECK_CALL(egl, bindAPI(EGL_OPENGL_ES_API));
try
{
for (size_t contextCount = 0; contextCount < maxContextCount; contextCount++)
{
const eglw::EGLContext sharedContext =
(m_sharing == SHARING_SHARED && contextCount > 0 ? contexts[rng.getUint32() % (uint32_t)contextCount] :
EGL_NO_CONTEXT);
const eglw::EGLContext context = egl.createContext(*display, config, sharedContext, attribList);
const eglw::EGLint error = egl.getError();
if (context == EGL_NO_CONTEXT || error != EGL_SUCCESS)
{
log << TestLog::Message << "Got error after creating " << contextCount << " contexts."
<< TestLog::EndMessage;
if (error == EGL_BAD_ALLOC)
{
if (contextCount < minContextCount)
resultCollector.fail("Couldn't create the minimum number of contexts required.");
else
log << TestLog::Message << "Got EGL_BAD_ALLOC." << TestLog::EndMessage;
}
else
resultCollector.fail("eglCreateContext() produced error that is not EGL_BAD_ALLOC: " +
eglu::getErrorStr(error).toString());
if (context != EGL_NO_CONTEXT)
resultCollector.fail("eglCreateContext() produced error, but context is not EGL_NO_CONTEXT");
break;
}
else
{
contexts.push_back(context);
if (m_use == USE_MAKECURRENT || m_use == USE_CLEAR)
{
const eglw::EGLBoolean result = egl.makeCurrent(*display, *surface, *surface, context);
const eglw::EGLint makeCurrentError = egl.getError();
if (!result || makeCurrentError != EGL_SUCCESS)
{
log << TestLog::Message << "Failed to make " << (contextCount + 1)
<< "th context current: " << eglu::getErrorStr(makeCurrentError) << TestLog::EndMessage;
resultCollector.fail("Failed to make context current");
break;
}
else if (m_use == USE_CLEAR)
{
gl.clearColor(0.25f, 0.75f, 0.50f, 1.00f);
gl.clear(GL_COLOR_BUFFER_BIT);
gl.finish();
GLU_CHECK_GLW_MSG(gl, "Failed to clear color.");
}
}
}
}
for (size_t contextNdx = 0; contextNdx < contexts.size(); contextNdx++)
{
EGLU_CHECK_CALL(egl, destroyContext(*display, contexts[contextNdx]));
contexts[contextNdx] = EGL_NO_CONTEXT;
}
if (m_use == USE_MAKECURRENT || m_use == USE_CLEAR)
EGLU_CHECK_CALL(egl, makeCurrent(*display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT));
}
catch (...)
{
for (size_t contextNdx = 0; contextNdx < contexts.size(); contextNdx++)
{
if (contexts[contextNdx] != EGL_NO_CONTEXT)
EGLU_CHECK_CALL(egl, destroyContext(*display, contexts[contextNdx]));
}
if (m_use == USE_MAKECURRENT || m_use == USE_CLEAR)
EGLU_CHECK_CALL(egl, makeCurrent(*display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT));
throw;
}
resultCollector.setTestContextResult(m_testCtx);
return STOP;
}
} // namespace
TestCaseGroup *createMultiContextTests(EglTestContext &eglTestCtx)
{
de::MovePtr<TestCaseGroup> group(new TestCaseGroup(eglTestCtx, "multicontext", "EGL multi context tests."));
group->addChild(new MultiContextTest(eglTestCtx, MultiContextTest::SHARING_NONE, MultiContextTest::USE_NONE,
"non_shared", "Create multiple non-shared contexts."));
group->addChild(new MultiContextTest(eglTestCtx, MultiContextTest::SHARING_SHARED, MultiContextTest::USE_NONE,
"shared", "Create multiple shared contexts."));
group->addChild(new MultiContextTest(eglTestCtx, MultiContextTest::SHARING_NONE, MultiContextTest::USE_MAKECURRENT,
"non_shared_make_current", "Create multiple non-shared contexts."));
group->addChild(new MultiContextTest(eglTestCtx, MultiContextTest::SHARING_SHARED,
MultiContextTest::USE_MAKECURRENT, "shared_make_current",
"Create multiple shared contexts."));
group->addChild(new MultiContextTest(eglTestCtx, MultiContextTest::SHARING_NONE, MultiContextTest::USE_CLEAR,
"non_shared_clear", "Create multiple non-shared contexts."));
group->addChild(new MultiContextTest(eglTestCtx, MultiContextTest::SHARING_SHARED, MultiContextTest::USE_CLEAR,
"shared_clear", "Create multiple shared contexts."));
return group.release();
}
} // namespace egl
} // namespace deqp