/*-------------------------------------------------------------------------
* 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_KHR_surfaceless_context extension tests
*//*--------------------------------------------------------------------*/
#include "teglSurfacelessContextTests.hpp"
#include "teglSimpleConfigCase.hpp"
#include "egluStrUtil.hpp"
#include "tcuTestLog.hpp"
#include <EGL/eglext.h>
#include <string>
#include <vector>
#include <algorithm>
#if !defined(EGL_OPENGL_ES3_BIT_KHR)
# define EGL_OPENGL_ES3_BIT_KHR 0x0040
#endif
#if !defined(EGL_CONTEXT_MAJOR_VERSION_KHR)
# define EGL_CONTEXT_MAJOR_VERSION_KHR EGL_CONTEXT_CLIENT_VERSION
#endif
using std::vector;
using std::string;
using tcu::TestLog;
namespace deqp
{
namespace egl
{
namespace
{
class SurfacelessContextCase : public SimpleConfigCase
{
public:
SurfacelessContextCase (EglTestContext& eglTestCtx, const char* name, const char* description, const vector<EGLint>& configIds);
~SurfacelessContextCase (void);
void executeForConfig (tcu::egl::Display& display, EGLConfig config);
};
SurfacelessContextCase::SurfacelessContextCase (EglTestContext& eglTestCtx, const char* name, const char* description, const vector<EGLint>& configIds)
: SimpleConfigCase(eglTestCtx, name, description, configIds)
{
}
SurfacelessContextCase::~SurfacelessContextCase (void)
{
}
void SurfacelessContextCase::executeForConfig (tcu::egl::Display& display, EGLConfig config)
{
TestLog& log = m_testCtx.getLog();
const EGLint id = display.getConfigAttrib(config, EGL_CONFIG_ID);
const EGLint apiBits = display.getConfigAttrib(config, EGL_RENDERABLE_TYPE);
static const EGLint es1Attrs[] = { EGL_CONTEXT_CLIENT_VERSION, 1, EGL_NONE };
static const EGLint es2Attrs[] = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE };
static const EGLint es3Attrs[] = { EGL_CONTEXT_MAJOR_VERSION_KHR, 3, EGL_NONE };
static const struct
{
const char* name;
EGLenum api;
EGLint apiBit;
const EGLint* ctxAttrs;
} apis[] =
{
{ "OpenGL", EGL_OPENGL_API, EGL_OPENGL_BIT, DE_NULL },
{ "OpenGL ES 1", EGL_OPENGL_ES_API, EGL_OPENGL_ES_BIT, es1Attrs },
{ "OpenGL ES 2", EGL_OPENGL_ES_API, EGL_OPENGL_ES2_BIT, es2Attrs },
{ "OpenGL ES 3", EGL_OPENGL_ES_API, EGL_OPENGL_ES3_BIT_KHR, es3Attrs },
{ "OpenVG", EGL_OPENVG_API, EGL_OPENVG_BIT, DE_NULL }
};
{
vector<string> extensions;
display.getExtensions(extensions);
if (std::find(extensions.begin(), extensions.end(), string("EGL_KHR_surfaceless_context")) == extensions.end())
throw tcu::NotSupportedError("EGL_KHR_surfaceless_context not supported", "", __FILE__, __LINE__);
}
for (int apiNdx = 0; apiNdx < (int)DE_LENGTH_OF_ARRAY(apis); apiNdx++)
{
if ((apiBits & apis[apiNdx].apiBit) == 0)
continue; // Not supported API
log << TestLog::Message << "Creating " << apis[apiNdx].name << " context with config ID " << id << TestLog::EndMessage;
TCU_CHECK_EGL();
TCU_CHECK_EGL_CALL(eglBindAPI(apis[apiNdx].api));
EGLContext context = eglCreateContext(display.getEGLDisplay(), config, EGL_NO_CONTEXT, apis[apiNdx].ctxAttrs);
TCU_CHECK_EGL();
if (!eglMakeCurrent(display.getEGLDisplay(), EGL_NO_SURFACE, EGL_NO_SURFACE, context))
{
const EGLenum err = eglGetError();
if (err == EGL_BAD_MATCH)
{
log << TestLog::Message << " eglMakeCurrent() failed with EGL_BAD_MATCH. Context doesn't support surfaceless mode." << TestLog::EndMessage;
TCU_CHECK_EGL_CALL(eglDestroyContext(display.getEGLDisplay(), context));
continue;
}
else
{
log << TestLog::Message << " Fail, context: " << tcu::toHex(context) << ", error: " << eglu::getErrorName(err) << TestLog::EndMessage;
m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Failed to make context current");
continue;
}
}
TCU_CHECK_EGL_CALL(eglMakeCurrent(display.getEGLDisplay(), EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT));
// Destroy
TCU_CHECK_EGL_CALL(eglDestroyContext(display.getEGLDisplay(), context));
log << TestLog::Message << " Pass" << TestLog::EndMessage;
}
}
} // anonymous
SurfacelessContextTests::SurfacelessContextTests (EglTestContext& eglTestCtx)
: TestCaseGroup (eglTestCtx, "surfaceless_context", "EGL_KHR_surfaceless_context extension tests")
{
}
void SurfacelessContextTests::init (void)
{
vector<NamedConfigIdSet> configIdSets;
eglu::FilterList filters;
NamedConfigIdSet::getDefaultSets(configIdSets, m_eglTestCtx.getConfigs(), filters);
for (vector<NamedConfigIdSet>::const_iterator i = configIdSets.begin(); i != configIdSets.end(); i++)
addChild(new SurfacelessContextCase(m_eglTestCtx, i->getName(), i->getDescription(), i->getConfigIds()));
}
} // egl
} // deqp