/*-------------------------------------------------------------------------
* drawElements Quality Program OpenGL ES 3.0 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 State Query tests.
*//*--------------------------------------------------------------------*/
#include "es3fIntegerStateQueryTests.hpp"
#include "es3fApiCase.hpp"
#include "glsStateQueryUtil.hpp"
#include "gluRenderContext.hpp"
#include "gluContextInfo.hpp"
#include "gluStrUtil.hpp"
#include "tcuRenderTarget.hpp"
#include "deRandom.hpp"
#include "glwEnums.hpp"
using namespace glw; // GLint and other GL types
using deqp::gls::StateQueryUtil::StateQueryMemoryWriteGuard;
#ifndef GL_SLUMINANCE_NV
#define GL_SLUMINANCE_NV 0x8C46
#endif
#ifndef GL_SLUMINANCE_ALPHA_NV
#define GL_SLUMINANCE_ALPHA_NV 0x8C44
#endif
#ifndef GL_BGR_NV
#define GL_BGR_NV 0x80E0
#endif
namespace deqp
{
namespace gles3
{
namespace Functional
{
namespace IntegerStateQueryVerifiers
{
// StateVerifier
class StateVerifier : protected glu::CallLogWrapper
{
public:
StateVerifier (const glw::Functions& gl, tcu::TestLog& log, const char* testNamePostfix);
virtual ~StateVerifier (); // make GCC happy
const char* getTestNamePostfix (void) const;
virtual void verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference) = DE_NULL;
virtual void verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3) = DE_NULL;
virtual void verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3) = DE_NULL;
virtual void verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference) = DE_NULL;
virtual void verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference) = DE_NULL;
virtual void verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference) = DE_NULL;
virtual void verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1) = DE_NULL;
virtual void verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength) = DE_NULL;
virtual void verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits) = DE_NULL;
private:
const char* const m_testNamePostfix;
};
StateVerifier::StateVerifier (const glw::Functions& gl, tcu::TestLog& log, const char* testNamePostfix)
: glu::CallLogWrapper (gl, log)
, m_testNamePostfix (testNamePostfix)
{
enableLogging(true);
}
StateVerifier::~StateVerifier ()
{
}
const char* StateVerifier::getTestNamePostfix (void) const
{
return m_testNamePostfix;
}
// GetBooleanVerifier
class GetBooleanVerifier : public StateVerifier
{
public:
GetBooleanVerifier (const glw::Functions& gl, tcu::TestLog& log);
void verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3);
void verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3);
void verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference);
void verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1);
void verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength);
void verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits);
};
GetBooleanVerifier::GetBooleanVerifier (const glw::Functions& gl, tcu::TestLog& log)
: StateVerifier(gl, log, "_getboolean")
{
}
void GetBooleanVerifier::verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLboolean> state;
glGetBooleanv(name, &state);
if (!state.verifyValidity(testCtx))
return;
const GLboolean expectedGLState = reference ? GL_TRUE : GL_FALSE;
if (state != expectedGLState)
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected " << (expectedGLState==GL_TRUE ? "GL_TRUE" : "GL_FALSE") << "; got " << (state == GL_TRUE ? "GL_TRUE" : (state == GL_FALSE ? "GL_FALSE" : "non-boolean")) << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
void GetBooleanVerifier::verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3)
{
verifyInteger4Mask(testCtx, name, reference0, true, reference1, true, reference2, true, reference3, true);
}
void GetBooleanVerifier::verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLboolean[4]> boolVector4;
glGetBooleanv(name, boolVector4);
if (!boolVector4.verifyValidity(testCtx))
return;
const GLboolean referenceAsGLBoolean[] =
{
reference0 ? (GLboolean)GL_TRUE : (GLboolean)GL_FALSE,
reference1 ? (GLboolean)GL_TRUE : (GLboolean)GL_FALSE,
reference2 ? (GLboolean)GL_TRUE : (GLboolean)GL_FALSE,
reference3 ? (GLboolean)GL_TRUE : (GLboolean)GL_FALSE,
};
if ((enableRef0 && (boolVector4[0] != referenceAsGLBoolean[0])) ||
(enableRef1 && (boolVector4[1] != referenceAsGLBoolean[1])) ||
(enableRef2 && (boolVector4[2] != referenceAsGLBoolean[2])) ||
(enableRef3 && (boolVector4[3] != referenceAsGLBoolean[3])))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected "
<< (enableRef0 ? (referenceAsGLBoolean[0] ? "GL_TRUE" : "GL_FALSE") : " - ") << ", "
<< (enableRef1 ? (referenceAsGLBoolean[1] ? "GL_TRUE" : "GL_FALSE") : " - ") << ", "
<< (enableRef2 ? (referenceAsGLBoolean[2] ? "GL_TRUE" : "GL_FALSE") : " - ") << ", "
<< (enableRef3 ? (referenceAsGLBoolean[3] ? "GL_TRUE" : "GL_FALSE") : " - ") << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
void GetBooleanVerifier::verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLboolean> state;
glGetBooleanv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state == GL_TRUE) // state is non-zero, could be greater than reference (correct)
return;
if (state == GL_FALSE) // state is zero
{
if (reference > 0) // and reference is greater than zero?
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
else
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE or GL_FALSE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
void GetBooleanVerifier::verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLboolean> state;
glGetBooleanv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state == GL_TRUE) // state is non-zero, could be greater than reference (correct)
return;
if (state == GL_FALSE) // state is zero
{
if (reference > 0) // and reference is greater than zero?
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
else
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE or GL_FALSE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
void GetBooleanVerifier::verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLboolean> state;
glGetBooleanv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state == GL_TRUE) // state is non-zero, could be less than reference (correct)
return;
if (state == GL_FALSE) // state is zero
{
if (reference < 0) // and reference is less than zero?
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
else
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE or GL_FALSE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
void GetBooleanVerifier::verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLboolean[2]> boolVector;
glGetBooleanv(name, boolVector);
if (!boolVector.verifyValidity(testCtx))
return;
const GLboolean referenceAsGLBoolean[2] =
{
reference0 ? (GLboolean)GL_TRUE : (GLboolean)GL_FALSE,
reference1 ? (GLboolean)GL_TRUE : (GLboolean)GL_FALSE
};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(referenceAsGLBoolean); ++ndx)
{
if (boolVector[ndx] == GL_TRUE) // state is non-zero, could be greater than any integer
{
continue;
}
else if (boolVector[ndx] == GL_FALSE) // state is zero
{
if (referenceAsGLBoolean[ndx] > 0) // and reference is greater than zero?
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
else
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE or GL_FALSE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
}
void GetBooleanVerifier::verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLboolean> state;
glGetBooleanv(name, &state);
if (!state.verifyValidity(testCtx))
return;
for (size_t ndx = 0; ndx < referencesLength; ++ndx)
{
const GLboolean expectedGLState = references[ndx] ? GL_TRUE : GL_FALSE;
if (state == expectedGLState)
return;
}
testCtx.getLog() << TestLog::Message << "// ERROR: got " << (state==GL_TRUE ? "GL_TRUE" : "GL_FALSE") << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
void GetBooleanVerifier::verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits)
{
// if stencilBits == 0, the mask is allowed to be either GL_TRUE or GL_FALSE
// otherwise it must be GL_TRUE
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLboolean> state;
glGetBooleanv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (stencilBits > 0 && state != GL_TRUE)
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected GL_TRUE" << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid boolean value");
}
}
//GetIntegerVerifier
class GetIntegerVerifier : public StateVerifier
{
public:
GetIntegerVerifier (const glw::Functions& gl, tcu::TestLog& log);
void verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3);
void verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3);
void verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference);
void verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1);
void verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength);
void verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits);
};
GetIntegerVerifier::GetIntegerVerifier (const glw::Functions& gl, tcu::TestLog& log)
: StateVerifier(gl, log, "_getinteger")
{
}
void GetIntegerVerifier::verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint> state;
glGetIntegerv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state != reference)
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected " << reference << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetIntegerVerifier::verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3)
{
verifyInteger4Mask(testCtx, name, reference0, true, reference1, true, reference2, true, reference3, true);
}
void GetIntegerVerifier::verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint[4]> intVector4;
glGetIntegerv(name, intVector4);
if (!intVector4.verifyValidity(testCtx))
return;
if ((enableRef0 && (intVector4[0] != reference0)) ||
(enableRef1 && (intVector4[1] != reference1)) ||
(enableRef2 && (intVector4[2] != reference2)) ||
(enableRef3 && (intVector4[3] != reference3)))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected "
<< (enableRef0?"":"(") << reference0 << (enableRef0?"":")") << ", "
<< (enableRef1?"":"(") << reference1 << (enableRef1?"":")") << ", "
<< (enableRef2?"":"(") << reference2 << (enableRef2?"":")") << ", "
<< (enableRef3?"":"(") << reference3 << (enableRef3?"":")") << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetIntegerVerifier::verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint> state;
glGetIntegerv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state < reference)
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << reference << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetIntegerVerifier::verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint> state;
glGetIntegerv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (GLuint(state) < reference)
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << reference << "; got " << GLuint(state) << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetIntegerVerifier::verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint> state;
glGetIntegerv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state > reference)
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected less or equal to " << reference << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetIntegerVerifier::verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint[2]> intVector2;
glGetIntegerv(name, intVector2);
if (!intVector2.verifyValidity(testCtx))
return;
if (intVector2[0] < reference0 || intVector2[1] < reference1)
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << reference0 << ", " << reference1 << "; got " << intVector2[0] << ", " << intVector2[0] << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetIntegerVerifier::verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint> state;
glGetIntegerv(name, &state);
if (!state.verifyValidity(testCtx))
return;
for (size_t ndx = 0; ndx < referencesLength; ++ndx)
{
const GLint expectedGLState = references[ndx];
if (state == expectedGLState)
return;
}
testCtx.getLog() << TestLog::Message << "// ERROR: got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
void GetIntegerVerifier::verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint> state;
glGetIntegerv(name, &state);
if (!state.verifyValidity(testCtx))
return;
const GLint reference = (1 << stencilBits) - 1;
if ((state & reference) != reference) // the least significant stencilBits bits should be on
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected minimum mask of " << reference << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid mask value");
}
}
//GetInteger64Verifier
class GetInteger64Verifier : public StateVerifier
{
public:
GetInteger64Verifier (const glw::Functions& gl, tcu::TestLog& log);
void verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3);
void verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3);
void verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference);
void verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1);
void verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength);
void verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits);
};
GetInteger64Verifier::GetInteger64Verifier (const glw::Functions& gl, tcu::TestLog& log)
: StateVerifier(gl, log, "_getinteger64")
{
}
void GetInteger64Verifier::verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint64> state;
glGetInteger64v(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state != GLint64(reference))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected " << reference << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetInteger64Verifier::verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3)
{
verifyInteger4Mask(testCtx, name, reference0, true, reference1, true, reference2, true, reference3, true);
}
void GetInteger64Verifier::verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint64[4]> intVector4;
glGetInteger64v(name, intVector4);
if (!intVector4.verifyValidity(testCtx))
return;
if ((enableRef0 && (intVector4[0] != GLint64(reference0))) ||
(enableRef1 && (intVector4[1] != GLint64(reference1))) ||
(enableRef2 && (intVector4[2] != GLint64(reference2))) ||
(enableRef3 && (intVector4[3] != GLint64(reference3))))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected "
<< (enableRef0?"":"(") << reference0 << (enableRef0?"":")") << ", "
<< (enableRef1?"":"(") << reference1 << (enableRef1?"":")") << ", "
<< (enableRef2?"":"(") << reference2 << (enableRef2?"":")") << ", "
<< (enableRef3?"":"(") << reference3 << (enableRef3?"":")") << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetInteger64Verifier::verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint64> state;
glGetInteger64v(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state < GLint64(reference))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << GLint64(reference) << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetInteger64Verifier::verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint64> state;
glGetInteger64v(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (GLuint(state) < GLint64(reference))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << GLint64(reference) << "; got " << GLuint(state) << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetInteger64Verifier::verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint64> state;
glGetInteger64v(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state > GLint64(reference))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected less or equal to " << GLint64(reference) << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetInteger64Verifier::verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint64[2]> intVector2;
glGetInteger64v(name, intVector2);
if (!intVector2.verifyValidity(testCtx))
return;
if (intVector2[0] < GLint64(reference0) || intVector2[1] < GLint64(reference1))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << GLint64(reference0) << ", " << GLint64(reference1) << "; got " << intVector2[0] << ", " << intVector2[1] << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
}
void GetInteger64Verifier::verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint64> state;
glGetInteger64v(name, &state);
if (!state.verifyValidity(testCtx))
return;
for (size_t ndx = 0; ndx < referencesLength; ++ndx)
{
const GLint64 expectedGLState = GLint64(references[ndx]);
if (state == expectedGLState)
return;
}
testCtx.getLog() << TestLog::Message << "// ERROR: got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid integer value");
}
void GetInteger64Verifier::verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLint64> state;
glGetInteger64v(name, &state);
if (!state.verifyValidity(testCtx))
return;
const GLint64 reference = (1ULL << stencilBits) - 1;
if ((state & reference) != reference) // the least significant stencilBits bits should be on
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected mimimum mask of " << reference << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid mask value");
}
}
//GetFloatVerifier
class GetFloatVerifier : public StateVerifier
{
public:
GetFloatVerifier (const glw::Functions& gl, tcu::TestLog& log);
void verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3);
void verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3);
void verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference);
void verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference);
void verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1);
void verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength);
void verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits);
};
GetFloatVerifier::GetFloatVerifier (const glw::Functions& gl, tcu::TestLog& log)
: StateVerifier(gl, log, "_getfloat")
{
}
void GetFloatVerifier::verifyInteger (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
const GLfloat referenceAsFloat = GLfloat(reference);
DE_ASSERT(reference == GLint(referenceAsFloat)); // reference integer must have 1:1 mapping to float for this to work. Reference value is always such value in these tests
StateQueryMemoryWriteGuard<GLfloat> state;
glGetFloatv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state != referenceAsFloat)
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected " << referenceAsFloat << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid float value");
}
}
void GetFloatVerifier::verifyInteger4 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1, GLint reference2, GLint reference3)
{
verifyInteger4Mask(testCtx, name, reference0, true, reference1, true, reference2, true, reference3, true);
}
void GetFloatVerifier::verifyInteger4Mask (tcu::TestContext& testCtx, GLenum name, GLint reference0, bool enableRef0, GLint reference1, bool enableRef1, GLint reference2, bool enableRef2, GLint reference3, bool enableRef3)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLfloat[4]> floatVector4;
glGetFloatv(name, floatVector4);
if (!floatVector4.verifyValidity(testCtx))
return;
if ((enableRef0 && (floatVector4[0] != GLfloat(reference0))) ||
(enableRef1 && (floatVector4[1] != GLfloat(reference1))) ||
(enableRef2 && (floatVector4[2] != GLfloat(reference2))) ||
(enableRef3 && (floatVector4[3] != GLfloat(reference3))))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected "
<< (enableRef0?"":"(") << GLfloat(reference0) << (enableRef0?"":")") << ", "
<< (enableRef1?"":"(") << GLfloat(reference1) << (enableRef1?"":")") << ", "
<< (enableRef2?"":"(") << GLfloat(reference2) << (enableRef2?"":")") << ", "
<< (enableRef3?"":"(") << GLfloat(reference3) << (enableRef3?"":")") << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid float value");
}
}
void GetFloatVerifier::verifyIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLfloat> state;
glGetFloatv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state < GLfloat(reference))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << GLfloat(reference) << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid float value");
}
}
void GetFloatVerifier::verifyUnsignedIntegerGreaterOrEqual (tcu::TestContext& testCtx, GLenum name, GLuint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLfloat> state;
glGetFloatv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state < GLfloat(reference))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << GLfloat(reference) << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid float value");
}
}
void GetFloatVerifier::verifyIntegerLessOrEqual (tcu::TestContext& testCtx, GLenum name, GLint reference)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLfloat> state;
glGetFloatv(name, &state);
if (!state.verifyValidity(testCtx))
return;
if (state > GLfloat(reference))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected less or equal to " << GLfloat(reference) << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid float value");
}
}
void GetFloatVerifier::verifyIntegerGreaterOrEqual2 (tcu::TestContext& testCtx, GLenum name, GLint reference0, GLint reference1)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLfloat[2]> floatVector2;
glGetFloatv(name, floatVector2);
if (!floatVector2.verifyValidity(testCtx))
return;
if (floatVector2[0] < GLfloat(reference0) || floatVector2[1] < GLfloat(reference1))
{
testCtx.getLog() << TestLog::Message << "// ERROR: expected greater or equal to " << GLfloat(reference0) << ", " << GLfloat(reference1) << "; got " << floatVector2[0] << ", " << floatVector2[1] << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid float value");
}
}
void GetFloatVerifier::verifyIntegerAnyOf (tcu::TestContext& testCtx, GLenum name, const GLint references[], size_t referencesLength)
{
using tcu::TestLog;
StateQueryMemoryWriteGuard<GLfloat> state;
glGetFloatv(name, &state);
if (!state.verifyValidity(testCtx))
return;
for (size_t ndx = 0; ndx < referencesLength; ++ndx)
{
const GLfloat expectedGLState = GLfloat(references[ndx]);
DE_ASSERT(references[ndx] == GLint(expectedGLState)); // reference integer must have 1:1 mapping to float for this to work. Reference value is always such value in these tests
if (state == expectedGLState)
return;
}
testCtx.getLog() << TestLog::Message << "// ERROR: got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Got invalid float value");
}
void GetFloatVerifier::verifyStencilMaskInitial (tcu::TestContext& testCtx, GLenum name, int stencilBits)
{
// checking the mask bits with float doesn't make much sense because of conversion errors
// just verify that the value is greater or equal to the minimum value
const GLint reference = (1 << stencilBits) - 1;
verifyIntegerGreaterOrEqual(testCtx, name, reference);
}
} // IntegerStateQueryVerifiers
namespace
{
using namespace IntegerStateQueryVerifiers;
using namespace deqp::gls::StateQueryUtil;
class ConstantMinimumValueTestCase : public ApiCase
{
public:
ConstantMinimumValueTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum targetName, GLint minValue)
: ApiCase (context, name, description)
, m_targetName (targetName)
, m_minValue (minValue)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyUnsignedIntegerGreaterOrEqual(m_testCtx, m_targetName, m_minValue);
expectError(GL_NO_ERROR);
}
private:
GLenum m_targetName;
GLint m_minValue;
StateVerifier* m_verifier;
};
class ConstantMaximumValueTestCase : public ApiCase
{
public:
ConstantMaximumValueTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum targetName, GLint minValue)
: ApiCase (context, name, description)
, m_targetName (targetName)
, m_minValue (minValue)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyIntegerLessOrEqual(m_testCtx, m_targetName, m_minValue);
expectError(GL_NO_ERROR);
}
private:
GLenum m_targetName;
GLint m_minValue;
StateVerifier* m_verifier;
};
class SampleBuffersTestCase : public ApiCase
{
public:
SampleBuffersTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
const int expectedSampleBuffers = (m_context.getRenderTarget().getNumSamples() > 1) ? 1 : 0;
m_log << tcu::TestLog::Message << "Sample count is " << (m_context.getRenderTarget().getNumSamples()) << ", expecting GL_SAMPLE_BUFFERS to be " << expectedSampleBuffers << tcu::TestLog::EndMessage;
m_verifier->verifyInteger(m_testCtx, GL_SAMPLE_BUFFERS, expectedSampleBuffers);
expectError(GL_NO_ERROR);
}
private:
StateVerifier* m_verifier;
};
class SamplesTestCase : public ApiCase
{
public:
SamplesTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
// MSAA?
if (m_context.getRenderTarget().getNumSamples() > 1)
{
m_log << tcu::TestLog::Message << "Sample count is " << (m_context.getRenderTarget().getNumSamples()) << tcu::TestLog::EndMessage;
m_verifier->verifyInteger(m_testCtx, GL_SAMPLES, m_context.getRenderTarget().getNumSamples());
expectError(GL_NO_ERROR);
}
else
{
const glw::GLint validSamples[] = {0, 1};
m_log << tcu::TestLog::Message << "Expecting GL_SAMPLES to be 0 or 1" << tcu::TestLog::EndMessage;
m_verifier->verifyIntegerAnyOf(m_testCtx, GL_SAMPLES, validSamples, DE_LENGTH_OF_ARRAY(validSamples));
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class HintTestCase : public ApiCase
{
public:
HintTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum targetName)
: ApiCase (context, name, description)
, m_targetName (targetName)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, m_targetName, GL_DONT_CARE);
expectError(GL_NO_ERROR);
glHint(m_targetName, GL_NICEST);
m_verifier->verifyInteger(m_testCtx, m_targetName, GL_NICEST);
expectError(GL_NO_ERROR);
glHint(m_targetName, GL_FASTEST);
m_verifier->verifyInteger(m_testCtx, m_targetName, GL_FASTEST);
expectError(GL_NO_ERROR);
glHint(m_targetName, GL_DONT_CARE);
m_verifier->verifyInteger(m_testCtx, m_targetName, GL_DONT_CARE);
expectError(GL_NO_ERROR);
}
private:
GLenum m_targetName;
StateVerifier* m_verifier;
};
class DepthFuncTestCase : public ApiCase
{
public:
DepthFuncTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, GL_DEPTH_FUNC, GL_LESS);
expectError(GL_NO_ERROR);
const GLenum depthFunctions[] = {GL_NEVER, GL_ALWAYS, GL_LESS, GL_LEQUAL, GL_EQUAL, GL_GREATER, GL_GEQUAL, GL_NOTEQUAL};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(depthFunctions); ndx++)
{
glDepthFunc(depthFunctions[ndx]);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_DEPTH_FUNC, depthFunctions[ndx]);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class CullFaceTestCase : public ApiCase
{
public:
CullFaceTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, GL_CULL_FACE_MODE, GL_BACK);
expectError(GL_NO_ERROR);
const GLenum cullFaces[] = {GL_FRONT, GL_BACK, GL_FRONT_AND_BACK};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(cullFaces); ndx++)
{
glCullFace(cullFaces[ndx]);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_CULL_FACE_MODE, cullFaces[ndx]);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class FrontFaceTestCase : public ApiCase
{
public:
FrontFaceTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, GL_FRONT_FACE, GL_CCW);
expectError(GL_NO_ERROR);
const GLenum frontFaces[] = {GL_CW, GL_CCW};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(frontFaces); ndx++)
{
glFrontFace(frontFaces[ndx]);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_FRONT_FACE, frontFaces[ndx]);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class ViewPortTestCase : public ApiCase
{
public:
ViewPortTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
de::Random rnd(0xabcdef);
GLint maxViewportDimensions[2] = {0};
GLfloat viewportBoundsRange[2] = {0.0f};
GLboolean hasViewportArray = false;
glGetIntegerv(GL_MAX_VIEWPORT_DIMS, maxViewportDimensions);
hasViewportArray = m_context.getContextInfo().isExtensionSupported("GL_OES_viewport_array") ||
m_context.getContextInfo().isExtensionSupported("GL_NV_viewport_array");
if (hasViewportArray)
{
glGetFloatv(GL_VIEWPORT_BOUNDS_RANGE, viewportBoundsRange);
}
// verify initial value of first two values
m_verifier->verifyInteger4(m_testCtx, GL_VIEWPORT, 0, 0, m_context.getRenderTarget().getWidth(), m_context.getRenderTarget().getHeight());
expectError(GL_NO_ERROR);
const int numIterations = 120;
for (int i = 0; i < numIterations; ++i)
{
GLint x = rnd.getInt(-64000, 64000);
GLint y = rnd.getInt(-64000, 64000);
GLsizei width = rnd.getInt(0, maxViewportDimensions[0]);
GLsizei height = rnd.getInt(0, maxViewportDimensions[1]);
glViewport(x, y, width, height);
if (hasViewportArray)
{
m_verifier->verifyInteger4(m_testCtx, GL_VIEWPORT,
de::clamp(x, deFloorFloatToInt32(viewportBoundsRange[0]), deFloorFloatToInt32(viewportBoundsRange[1])),
de::clamp(y, deFloorFloatToInt32(viewportBoundsRange[0]), deFloorFloatToInt32(viewportBoundsRange[1])),
width, height);
}
else
{
m_verifier->verifyInteger4(m_testCtx, GL_VIEWPORT, x, y, width, height);
}
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class ScissorBoxTestCase : public ApiCase
{
public:
ScissorBoxTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
de::Random rnd(0xabcdef);
// verify initial value of first two values
m_verifier->verifyInteger4Mask(m_testCtx, GL_SCISSOR_BOX, 0, true, 0, true, 0, false, 0, false);
expectError(GL_NO_ERROR);
const int numIterations = 120;
for (int i = 0; i < numIterations; ++i)
{
GLint left = rnd.getInt(-64000, 64000);
GLint bottom = rnd.getInt(-64000, 64000);
GLsizei width = rnd.getInt(0, 64000);
GLsizei height = rnd.getInt(0, 64000);
glScissor(left, bottom, width, height);
m_verifier->verifyInteger4(m_testCtx, GL_SCISSOR_BOX, left, bottom, width, height);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class MaxViewportDimsTestCase : public ApiCase
{
public:
MaxViewportDimsTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyIntegerGreaterOrEqual2(m_testCtx, GL_MAX_VIEWPORT_DIMS, m_context.getRenderTarget().getWidth(), m_context.getRenderTarget().getHeight());
expectError(GL_NO_ERROR);
}
private:
StateVerifier* m_verifier;
};
class StencilRefTestCase : public ApiCase
{
public:
StencilRefTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, m_testTargetName, 0);
expectError(GL_NO_ERROR);
const int stencilBits = m_context.getRenderTarget().getStencilBits();
for (int stencilBit = 0; stencilBit < stencilBits; ++stencilBit)
{
const int ref = 1 << stencilBit;
glStencilFunc(GL_ALWAYS, ref, 0); // mask should not affect the REF
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, ref);
expectError(GL_NO_ERROR);
glStencilFunc(GL_ALWAYS, ref, ref);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, ref);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
};
class StencilRefSeparateTestCase : public ApiCase
{
public:
StencilRefSeparateTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, GLenum stencilFuncTargetFace)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
, m_stencilFuncTargetFace (stencilFuncTargetFace)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, m_testTargetName, 0);
expectError(GL_NO_ERROR);
const int stencilBits = m_context.getRenderTarget().getStencilBits();
for (int stencilBit = 0; stencilBit < stencilBits; ++stencilBit)
{
const int ref = 1 << stencilBit;
glStencilFuncSeparate(m_stencilFuncTargetFace, GL_ALWAYS, ref, 0);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, ref);
expectError(GL_NO_ERROR);
glStencilFuncSeparate(m_stencilFuncTargetFace, GL_ALWAYS, ref, ref);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, ref);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
GLenum m_stencilFuncTargetFace;
};
class StencilOpTestCase : public ApiCase
{
public:
StencilOpTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum stencilOpName)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_stencilOpName (stencilOpName)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, m_stencilOpName, GL_KEEP);
expectError(GL_NO_ERROR);
const GLenum stencilOpValues[] = {GL_KEEP, GL_ZERO, GL_REPLACE, GL_INCR, GL_DECR, GL_INVERT, GL_INCR_WRAP, GL_DECR_WRAP};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(stencilOpValues); ++ndx)
{
SetStencilOp(stencilOpValues[ndx]);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_stencilOpName, stencilOpValues[ndx]);
expectError(GL_NO_ERROR);
}
}
protected:
virtual void SetStencilOp (GLenum stencilOpValue)
{
switch (m_stencilOpName)
{
case GL_STENCIL_FAIL:
case GL_STENCIL_BACK_FAIL:
glStencilOp(stencilOpValue, GL_KEEP, GL_KEEP);
break;
case GL_STENCIL_PASS_DEPTH_FAIL:
case GL_STENCIL_BACK_PASS_DEPTH_FAIL:
glStencilOp(GL_KEEP, stencilOpValue, GL_KEEP);
break;
case GL_STENCIL_PASS_DEPTH_PASS:
case GL_STENCIL_BACK_PASS_DEPTH_PASS:
glStencilOp(GL_KEEP, GL_KEEP, stencilOpValue);
break;
default:
DE_ASSERT(false && "should not happen");
break;
}
}
StateVerifier* m_verifier;
GLenum m_stencilOpName;
};
class StencilOpSeparateTestCase : public StencilOpTestCase
{
public:
StencilOpSeparateTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum stencilOpName, GLenum stencilOpFace)
: StencilOpTestCase (context, verifier, name, description, stencilOpName)
, m_stencilOpFace (stencilOpFace)
{
}
private:
void SetStencilOp (GLenum stencilOpValue)
{
switch (m_stencilOpName)
{
case GL_STENCIL_FAIL:
case GL_STENCIL_BACK_FAIL:
glStencilOpSeparate(m_stencilOpFace, stencilOpValue, GL_KEEP, GL_KEEP);
break;
case GL_STENCIL_PASS_DEPTH_FAIL:
case GL_STENCIL_BACK_PASS_DEPTH_FAIL:
glStencilOpSeparate(m_stencilOpFace, GL_KEEP, stencilOpValue, GL_KEEP);
break;
case GL_STENCIL_PASS_DEPTH_PASS:
case GL_STENCIL_BACK_PASS_DEPTH_PASS:
glStencilOpSeparate(m_stencilOpFace, GL_KEEP, GL_KEEP, stencilOpValue);
break;
default:
DE_ASSERT(false && "should not happen");
break;
}
}
GLenum m_stencilOpFace;
};
class StencilFuncTestCase : public ApiCase
{
public:
StencilFuncTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, GL_STENCIL_FUNC, GL_ALWAYS);
expectError(GL_NO_ERROR);
const GLenum stencilfuncValues[] = {GL_NEVER, GL_ALWAYS, GL_LESS, GL_LEQUAL, GL_EQUAL, GL_GEQUAL, GL_GREATER, GL_NOTEQUAL};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(stencilfuncValues); ++ndx)
{
glStencilFunc(stencilfuncValues[ndx], 0, 0);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_STENCIL_FUNC, stencilfuncValues[ndx]);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_STENCIL_BACK_FUNC, stencilfuncValues[ndx]);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class StencilFuncSeparateTestCase : public ApiCase
{
public:
StencilFuncSeparateTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum stencilFuncName, GLenum stencilFuncFace)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_stencilFuncName (stencilFuncName)
, m_stencilFuncFace (stencilFuncFace)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, m_stencilFuncName, GL_ALWAYS);
expectError(GL_NO_ERROR);
const GLenum stencilfuncValues[] = {GL_NEVER, GL_ALWAYS, GL_LESS, GL_LEQUAL, GL_EQUAL, GL_GEQUAL, GL_GREATER, GL_NOTEQUAL};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(stencilfuncValues); ++ndx)
{
glStencilFuncSeparate(m_stencilFuncFace, stencilfuncValues[ndx], 0, 0);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_stencilFuncName, stencilfuncValues[ndx]);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_stencilFuncName;
GLenum m_stencilFuncFace;
};
class StencilMaskTestCase : public ApiCase
{
public:
StencilMaskTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
{
}
void test (void)
{
const int stencilBits = m_context.getRenderTarget().getStencilBits();
m_verifier->verifyStencilMaskInitial(m_testCtx, m_testTargetName, stencilBits);
expectError(GL_NO_ERROR);
for (int stencilBit = 0; stencilBit < stencilBits; ++stencilBit)
{
const int mask = 1 << stencilBit;
glStencilFunc(GL_ALWAYS, 0, mask);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, mask);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
};
class StencilMaskSeparateTestCase : public ApiCase
{
public:
StencilMaskSeparateTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, GLenum stencilFuncTargetFace)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
, m_stencilFuncTargetFace (stencilFuncTargetFace)
{
}
void test (void)
{
const int stencilBits = m_context.getRenderTarget().getStencilBits();
m_verifier->verifyStencilMaskInitial(m_testCtx, m_testTargetName, stencilBits);
expectError(GL_NO_ERROR);
for (int stencilBit = 0; stencilBit < stencilBits; ++stencilBit)
{
const int mask = 1 << stencilBit;
glStencilFuncSeparate(m_stencilFuncTargetFace, GL_ALWAYS, 0, mask);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, mask);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
GLenum m_stencilFuncTargetFace;
};
class StencilWriteMaskTestCase : public ApiCase
{
public:
StencilWriteMaskTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
{
}
void test (void)
{
const int stencilBits = m_context.getRenderTarget().getStencilBits();
for (int stencilBit = 0; stencilBit < stencilBits; ++stencilBit)
{
const int mask = 1 << stencilBit;
glStencilMask(mask);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, mask);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
};
class StencilWriteMaskSeparateTestCase : public ApiCase
{
public:
StencilWriteMaskSeparateTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, GLenum stencilTargetFace)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
, m_stencilTargetFace (stencilTargetFace)
{
}
void test (void)
{
const int stencilBits = m_context.getRenderTarget().getStencilBits();
for (int stencilBit = 0; stencilBit < stencilBits; ++stencilBit)
{
const int mask = 1 << stencilBit;
glStencilMaskSeparate(m_stencilTargetFace, mask);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, mask);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
GLenum m_stencilTargetFace;
};
class PixelStoreTestCase : public ApiCase
{
public:
PixelStoreTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, int initialValue)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
, m_initialValue (initialValue)
{
}
void test (void)
{
de::Random rnd(0xabcdef);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, m_initialValue);
expectError(GL_NO_ERROR);
const int numIterations = 120;
for (int i = 0; i < numIterations; ++i)
{
const int referenceValue = rnd.getInt(0, 64000);
glPixelStorei(m_testTargetName, referenceValue);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, referenceValue);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
int m_initialValue;
};
class PixelStoreAlignTestCase : public ApiCase
{
public:
PixelStoreAlignTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, m_testTargetName, 4);
expectError(GL_NO_ERROR);
const int alignments[] = {1, 2, 4, 8};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(alignments); ++ndx)
{
const int referenceValue = alignments[ndx];
glPixelStorei(m_testTargetName, referenceValue);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, referenceValue);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
};
class BlendFuncTestCase : public ApiCase
{
public:
BlendFuncTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, int initialValue)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
, m_initialValue (initialValue)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, m_testTargetName, m_initialValue);
expectError(GL_NO_ERROR);
const GLenum blendFuncValues[] =
{
GL_ZERO, GL_ONE, GL_SRC_COLOR, GL_ONE_MINUS_SRC_COLOR, GL_DST_COLOR, GL_ONE_MINUS_DST_COLOR,
GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA, GL_DST_ALPHA, GL_ONE_MINUS_DST_ALPHA, GL_CONSTANT_COLOR,
GL_ONE_MINUS_CONSTANT_COLOR, GL_CONSTANT_ALPHA, GL_ONE_MINUS_CONSTANT_ALPHA,
GL_SRC_ALPHA_SATURATE
};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(blendFuncValues); ++ndx)
{
const GLenum referenceValue = blendFuncValues[ndx];
SetBlendFunc(referenceValue);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, referenceValue);
expectError(GL_NO_ERROR);
}
}
protected:
virtual void SetBlendFunc (GLenum func)
{
switch (m_testTargetName)
{
case GL_BLEND_SRC_RGB:
case GL_BLEND_SRC_ALPHA:
glBlendFunc(func, GL_ZERO);
break;
case GL_BLEND_DST_RGB:
case GL_BLEND_DST_ALPHA:
glBlendFunc(GL_ZERO, func);
break;
default:
DE_ASSERT(false && "should not happen");
break;
}
}
StateVerifier* m_verifier;
GLenum m_testTargetName;
int m_initialValue;
};
class BlendFuncSeparateTestCase : public BlendFuncTestCase
{
public:
BlendFuncSeparateTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, int initialValue)
: BlendFuncTestCase (context, verifier, name, description, testTargetName, initialValue)
{
}
void SetBlendFunc (GLenum func)
{
switch (m_testTargetName)
{
case GL_BLEND_SRC_RGB:
glBlendFuncSeparate(func, GL_ZERO, GL_ZERO, GL_ZERO);
break;
case GL_BLEND_DST_RGB:
glBlendFuncSeparate(GL_ZERO, func, GL_ZERO, GL_ZERO);
break;
case GL_BLEND_SRC_ALPHA:
glBlendFuncSeparate(GL_ZERO, GL_ZERO, func, GL_ZERO);
break;
case GL_BLEND_DST_ALPHA:
glBlendFuncSeparate(GL_ZERO, GL_ZERO, GL_ZERO, func);
break;
default:
DE_ASSERT(false && "should not happen");
break;
}
}
};
class BlendEquationTestCase : public ApiCase
{
public:
BlendEquationTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, int initialValue)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
, m_initialValue (initialValue)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, m_testTargetName, m_initialValue);
expectError(GL_NO_ERROR);
const GLenum blendFuncValues[] =
{
GL_FUNC_ADD, GL_FUNC_SUBTRACT, GL_FUNC_REVERSE_SUBTRACT, GL_MIN, GL_MAX
};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(blendFuncValues); ++ndx)
{
const GLenum referenceValue = blendFuncValues[ndx];
SetBlendEquation(referenceValue);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, m_testTargetName, referenceValue);
expectError(GL_NO_ERROR);
}
}
protected:
virtual void SetBlendEquation (GLenum equation)
{
glBlendEquation(equation);
}
StateVerifier* m_verifier;
GLenum m_testTargetName;
int m_initialValue;
};
class BlendEquationSeparateTestCase : public BlendEquationTestCase
{
public:
BlendEquationSeparateTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, int initialValue)
: BlendEquationTestCase (context, verifier, name, description, testTargetName, initialValue)
{
}
protected:
void SetBlendEquation (GLenum equation)
{
switch (m_testTargetName)
{
case GL_BLEND_EQUATION_RGB:
glBlendEquationSeparate(equation, GL_FUNC_ADD);
break;
case GL_BLEND_EQUATION_ALPHA:
glBlendEquationSeparate(GL_FUNC_ADD, equation);
break;
default:
DE_ASSERT(false && "should not happen");
break;
}
}
};
class ImplementationArrayTestCase : public ApiCase
{
public:
ImplementationArrayTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description, GLenum testTargetName, GLenum testTargetLengthTargetName, int minValue)
: ApiCase (context, name, description)
, m_verifier (verifier)
, m_testTargetName (testTargetName)
, m_testTargetLengthTargetName (testTargetLengthTargetName)
, m_minValue (minValue)
{
}
void test (void)
{
m_verifier->verifyIntegerGreaterOrEqual(m_testCtx, m_testTargetLengthTargetName, m_minValue);
expectError(GL_NO_ERROR);
GLint targetArrayLength = 0;
glGetIntegerv(m_testTargetLengthTargetName, &targetArrayLength);
expectError(GL_NO_ERROR);
if (targetArrayLength)
{
std::vector<GLint> queryResult;
queryResult.resize(targetArrayLength, 0);
glGetIntegerv(m_testTargetName, &queryResult[0]);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
GLenum m_testTargetName;
GLenum m_testTargetLengthTargetName;
int m_minValue;
};
class BindingTest : public TestCase
{
public:
BindingTest (Context& context,
const char* name,
const char* desc,
QueryType type);
IterateResult iterate (void);
virtual void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const = 0;
protected:
const QueryType m_type;
};
BindingTest::BindingTest (Context& context,
const char* name,
const char* desc,
QueryType type)
: TestCase (context, name, desc)
, m_type (type)
{
}
BindingTest::IterateResult BindingTest::iterate (void)
{
glu::CallLogWrapper gl (m_context.getRenderContext().getFunctions(), m_context.getTestContext().getLog());
tcu::ResultCollector result (m_context.getTestContext().getLog(), " // ERROR: ");
gl.enableLogging(true);
test(gl, result);
result.setTestContextResult(m_testCtx);
return STOP;
}
class TransformFeedbackBindingTestCase : public BindingTest
{
public:
TransformFeedbackBindingTestCase (Context& context, QueryType type, const char* name)
: BindingTest(context, name, "GL_TRANSFORM_FEEDBACK_BINDING", type)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
static const char* transformFeedbackTestVertSource =
"#version 300 es\n"
"void main (void)\n"
"{\n"
" gl_Position = vec4(0.0);\n"
"}\n\0";
static const char* transformFeedbackTestFragSource =
"#version 300 es\n"
"layout(location = 0) out mediump vec4 fragColor;"
"void main (void)\n"
"{\n"
" fragColor = vec4(0.0);\n"
"}\n\0";
GLuint shaderVert;
GLuint shaderFrag;
GLuint shaderProg;
GLuint transformfeedback = 0;
GLuint feedbackBufferId = 0;
{
const tcu::ScopedLogSection section(gl.getLog(), "Initial", "Initial");
verifyStateInteger(result, gl, GL_TRANSFORM_FEEDBACK_BINDING, 0, m_type);
}
gl.glGenTransformFeedbacks(1, &transformfeedback);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenTransformFeedbacks");
{
const tcu::ScopedLogSection section(gl.getLog(), "VertexShader", "Vertex Shader");
GLint compileStatus = -1;
shaderVert = gl.glCreateShader(GL_VERTEX_SHADER);
gl.glShaderSource(shaderVert, 1, &transformFeedbackTestVertSource, DE_NULL);
gl.glCompileShader(shaderVert);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glCompileShader");
gl.glGetShaderiv(shaderVert, GL_COMPILE_STATUS, &compileStatus);
if (compileStatus != GL_TRUE)
result.fail("expected GL_TRUE");
}
{
const tcu::ScopedLogSection section(gl.getLog(), "FragmentShader", "Fragment Shader");
GLint compileStatus = -1;
shaderFrag = gl.glCreateShader(GL_FRAGMENT_SHADER);
gl.glShaderSource(shaderFrag, 1, &transformFeedbackTestFragSource, DE_NULL);
gl.glCompileShader(shaderFrag);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glCompileShader");
gl.glGetShaderiv(shaderFrag, GL_COMPILE_STATUS, &compileStatus);
if (compileStatus != GL_TRUE)
result.fail("expected GL_TRUE");
}
{
const tcu::ScopedLogSection section(gl.getLog(), "Program", "Create and bind program");
const char* transform_feedback_outputs = "gl_Position";
GLint linkStatus = -1;
shaderProg = gl.glCreateProgram();
gl.glAttachShader(shaderProg, shaderVert);
gl.glAttachShader(shaderProg, shaderFrag);
gl.glTransformFeedbackVaryings(shaderProg, 1, &transform_feedback_outputs, GL_INTERLEAVED_ATTRIBS);
gl.glLinkProgram(shaderProg);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glLinkProgram");
gl.glGetProgramiv(shaderProg, GL_LINK_STATUS, &linkStatus);
if (linkStatus != GL_TRUE)
result.fail("expected GL_TRUE");
}
gl.glBindTransformFeedback(GL_TRANSFORM_FEEDBACK, transformfeedback);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glBindTransformFeedback");
gl.glGenBuffers(1, &feedbackBufferId);
gl.glBindBuffer(GL_TRANSFORM_FEEDBACK_BUFFER, feedbackBufferId);
gl.glBufferData(GL_TRANSFORM_FEEDBACK_BUFFER, 16, NULL, GL_DYNAMIC_READ);
gl.glBindBufferBase(GL_TRANSFORM_FEEDBACK_BUFFER, 0, feedbackBufferId);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "bind buffers");
gl.glUseProgram(shaderProg);
verifyStateInteger(result, gl, GL_TRANSFORM_FEEDBACK_BINDING, transformfeedback, m_type);
gl.glUseProgram(0);
gl.glBindTransformFeedback(GL_TRANSFORM_FEEDBACK, 0);
gl.glDeleteTransformFeedbacks(1, &transformfeedback);
verifyStateInteger(result, gl, GL_TRANSFORM_FEEDBACK_BINDING, 0, m_type);
gl.glDeleteBuffers(1, &feedbackBufferId);
gl.glDeleteShader(shaderVert);
gl.glDeleteShader(shaderFrag);
gl.glDeleteProgram(shaderProg);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDeleteProgram");
}
};
class CurrentProgramBindingTestCase : public BindingTest
{
public:
CurrentProgramBindingTestCase (Context& context, QueryType type, const char* name, const char* description)
: BindingTest(context, name, description, type)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
static const char* testVertSource =
"#version 300 es\n"
"void main (void)\n"
"{\n"
" gl_Position = vec4(0.0);\n"
"}\n\0";
static const char* testFragSource =
"#version 300 es\n"
"layout(location = 0) out mediump vec4 fragColor;"
"void main (void)\n"
"{\n"
" fragColor = vec4(0.0);\n"
"}\n\0";
GLuint shaderVert;
GLuint shaderFrag;
GLuint shaderProg;
{
const tcu::ScopedLogSection section(gl.getLog(), "Initial", "Initial");
verifyStateInteger(result, gl, GL_CURRENT_PROGRAM, 0, m_type);
}
{
const tcu::ScopedLogSection section(gl.getLog(), "VertexShader", "Vertex Shader");
GLint compileStatus = -1;
shaderVert = gl.glCreateShader(GL_VERTEX_SHADER);
gl.glShaderSource(shaderVert, 1, &testVertSource, DE_NULL);
gl.glCompileShader(shaderVert);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glCompileShader");
gl.glGetShaderiv(shaderVert, GL_COMPILE_STATUS, &compileStatus);
if (compileStatus != GL_TRUE)
result.fail("expected GL_TRUE");
}
{
const tcu::ScopedLogSection section(gl.getLog(), "FragmentShader", "Fragment Shader");
GLint compileStatus = -1;
shaderFrag = gl.glCreateShader(GL_FRAGMENT_SHADER);
gl.glShaderSource(shaderFrag, 1, &testFragSource, DE_NULL);
gl.glCompileShader(shaderFrag);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glCompileShader");
gl.glGetShaderiv(shaderFrag, GL_COMPILE_STATUS, &compileStatus);
if (compileStatus != GL_TRUE)
result.fail("expected GL_TRUE");
}
{
const tcu::ScopedLogSection section(gl.getLog(), "Program", "Create and bind program");
GLint linkStatus = -1;
shaderProg = gl.glCreateProgram();
gl.glAttachShader(shaderProg, shaderVert);
gl.glAttachShader(shaderProg, shaderFrag);
gl.glLinkProgram(shaderProg);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glLinkProgram");
gl.glGetProgramiv(shaderProg, GL_LINK_STATUS, &linkStatus);
if (linkStatus != GL_TRUE)
result.fail("expected GL_TRUE");
gl.glUseProgram(shaderProg);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glUseProgram");
verifyStateInteger(result, gl, GL_CURRENT_PROGRAM, shaderProg, m_type);
}
{
const tcu::ScopedLogSection section(gl.getLog(), "Delete", "Delete program while in use");
gl.glDeleteShader(shaderVert);
gl.glDeleteShader(shaderFrag);
gl.glDeleteProgram(shaderProg);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDeleteProgram");
verifyStateInteger(result, gl, GL_CURRENT_PROGRAM, shaderProg, m_type);
}
{
const tcu::ScopedLogSection section(gl.getLog(), "Unbind", "Unbind program");
gl.glUseProgram(0);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glUseProgram");
verifyStateInteger(result, gl, GL_CURRENT_PROGRAM, 0, m_type);
}
}
};
class VertexArrayBindingTestCase : public BindingTest
{
public:
VertexArrayBindingTestCase (Context& context, QueryType type, const char* name, const char* description)
: BindingTest(context, name, description, type)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
verifyStateInteger(result, gl, GL_VERTEX_ARRAY_BINDING, 0, m_type);
GLuint vertexArrayObject = 0;
gl.glGenVertexArrays(1, &vertexArrayObject);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenVertexArrays");
gl.glBindVertexArray(vertexArrayObject);
verifyStateInteger(result, gl, GL_VERTEX_ARRAY_BINDING, vertexArrayObject, m_type);
gl.glDeleteVertexArrays(1, &vertexArrayObject);
verifyStateInteger(result, gl, GL_VERTEX_ARRAY_BINDING, 0, m_type);
}
};
class BufferBindingTestCase : public BindingTest
{
public:
BufferBindingTestCase (Context& context, QueryType type, const char* name, const char* description, GLenum bufferBindingName, GLenum bufferType)
: BindingTest (context, name, description, type)
, m_bufferBindingName (bufferBindingName)
, m_bufferType (bufferType)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
verifyStateInteger(result, gl, m_bufferBindingName, 0, m_type);
GLuint bufferObject = 0;
gl.glGenBuffers(1, &bufferObject);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenBuffers");
gl.glBindBuffer(m_bufferType, bufferObject);
verifyStateInteger(result, gl, m_bufferBindingName, bufferObject, m_type);
gl.glDeleteBuffers(1, &bufferObject);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDeleteBuffers");
verifyStateInteger(result, gl, m_bufferBindingName, 0, m_type);
}
private:
const GLenum m_bufferBindingName;
const GLenum m_bufferType;
};
class ElementArrayBufferBindingTestCase : public BindingTest
{
public:
ElementArrayBufferBindingTestCase (Context& context, QueryType type, const char* name)
: BindingTest(context, name, "GL_ELEMENT_ARRAY_BUFFER_BINDING", type)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
// Test with default VAO
{
const tcu::ScopedLogSection section(gl.getLog(), "DefaultVAO", "Test with default VAO");
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, 0, m_type);
GLuint bufferObject = 0;
gl.glGenBuffers(1, &bufferObject);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenBuffers");
gl.glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, bufferObject);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, bufferObject, m_type);
gl.glDeleteBuffers(1, &bufferObject);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, 0, m_type);
}
// Test with multiple VAOs
{
const tcu::ScopedLogSection section(gl.getLog(), "WithVAO", "Test with VAO");
GLuint vaos[2] = {0};
GLuint buffers[2] = {0};
gl.glGenVertexArrays(2, vaos);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenVertexArrays");
gl.glGenBuffers(2, buffers);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenBuffers");
// initial
gl.glBindVertexArray(vaos[0]);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, 0, m_type);
// after setting
gl.glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, buffers[0]);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, buffers[0], m_type);
// initial of vao 2
gl.glBindVertexArray(vaos[1]);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, 0, m_type);
// after setting to 2
gl.glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, buffers[1]);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, buffers[1], m_type);
// vao 1 still has buffer 1 bound?
gl.glBindVertexArray(vaos[0]);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, buffers[0], m_type);
// deleting clears from bound vaos ...
gl.glDeleteBuffers(2, buffers);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, 0, m_type);
// ... but does not from non-bound vaos?
gl.glBindVertexArray(vaos[1]);
verifyStateInteger(result, gl, GL_ELEMENT_ARRAY_BUFFER_BINDING, buffers[1], m_type);
gl.glDeleteVertexArrays(2, vaos);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDeleteVertexArrays");
}
}
};
class StencilClearValueTestCase : public ApiCase
{
public:
StencilClearValueTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, GL_STENCIL_CLEAR_VALUE, 0);
expectError(GL_NO_ERROR);
const int stencilBits = m_context.getRenderTarget().getStencilBits();
for (int stencilBit = 0; stencilBit < stencilBits; ++stencilBit)
{
const int ref = 1 << stencilBit;
glClearStencil(ref); // mask should not affect the REF
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_STENCIL_CLEAR_VALUE, ref);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class ActiveTextureTestCase : public ApiCase
{
public:
ActiveTextureTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
m_verifier->verifyInteger(m_testCtx, GL_ACTIVE_TEXTURE, GL_TEXTURE0);
expectError(GL_NO_ERROR);
GLint textureUnits = 0;
glGetIntegerv(GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS, &textureUnits);
expectError(GL_NO_ERROR);
for (int ndx = 0; ndx < textureUnits; ++ndx)
{
glActiveTexture(GL_TEXTURE0 + ndx);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_ACTIVE_TEXTURE, GL_TEXTURE0 + ndx);
expectError(GL_NO_ERROR);
}
}
private:
StateVerifier* m_verifier;
};
class RenderbufferBindingTestCase : public BindingTest
{
public:
RenderbufferBindingTestCase (Context& context, QueryType type, const char* name, const char* description)
: BindingTest(context, name, description, type)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
verifyStateInteger(result, gl, GL_RENDERBUFFER_BINDING, 0, m_type);
GLuint renderBuffer = 0;
gl.glGenRenderbuffers(1, &renderBuffer);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenRenderbuffers");
gl.glBindRenderbuffer(GL_RENDERBUFFER, renderBuffer);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glBindRenderbuffer");
verifyStateInteger(result, gl, GL_RENDERBUFFER_BINDING, renderBuffer, m_type);
gl.glDeleteRenderbuffers(1, &renderBuffer);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDeleteRenderbuffers");
verifyStateInteger(result, gl, GL_RENDERBUFFER_BINDING, 0, m_type);
}
};
class SamplerObjectBindingTestCase : public BindingTest
{
public:
SamplerObjectBindingTestCase (Context& context, QueryType type, const char* name, const char* description)
: BindingTest(context, name, description, type)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
verifyStateInteger(result, gl, GL_SAMPLER_BINDING, 0, m_type);
{
const tcu::ScopedLogSection section(gl.getLog(), "SingleUnit", "Single unit");
GLuint sampler = 0;
gl.glGenSamplers(1, &sampler);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenSamplers");
gl.glBindSampler(0, sampler);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glBindSampler");
verifyStateInteger(result, gl, GL_SAMPLER_BINDING, sampler, m_type);
gl.glDeleteSamplers(1, &sampler);
verifyStateInteger(result, gl, GL_SAMPLER_BINDING, 0, m_type);
}
{
const tcu::ScopedLogSection section(gl.getLog(), "MultipleUnits", "Multiple units");
GLuint samplerA = 0;
GLuint samplerB = 0;
gl.glGenSamplers(1, &samplerA);
gl.glGenSamplers(1, &samplerB);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenSamplers");
gl.glBindSampler(1, samplerA);
gl.glBindSampler(2, samplerB);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glBindSampler");
verifyStateInteger(result, gl, GL_SAMPLER_BINDING, 0, m_type);
gl.glActiveTexture(GL_TEXTURE1);
verifyStateInteger(result, gl, GL_SAMPLER_BINDING, samplerA, m_type);
gl.glActiveTexture(GL_TEXTURE2);
verifyStateInteger(result, gl, GL_SAMPLER_BINDING, samplerB, m_type);
gl.glDeleteSamplers(1, &samplerB);
gl.glDeleteSamplers(1, &samplerA);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDeleteSamplers");
}
}
};
class TextureBindingTestCase : public BindingTest
{
public:
TextureBindingTestCase (Context& context, QueryType type, const char* name, const char* description, GLenum testBindingName, GLenum textureType)
: BindingTest (context, name, description, type)
, m_testBindingName (testBindingName)
, m_textureType (textureType)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
verifyStateInteger(result, gl, m_testBindingName, 0, m_type);
GLuint texture = 0;
gl.glGenTextures(1, &texture);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenTextures");
gl.glBindTexture(m_textureType, texture);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glBindTexture");
verifyStateInteger(result, gl, m_testBindingName, texture, m_type);
gl.glDeleteTextures(1, &texture);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDeleteTextures");
verifyStateInteger(result, gl, m_testBindingName, 0, m_type);
}
private:
const GLenum m_testBindingName;
const GLenum m_textureType;
};
class FrameBufferBindingTestCase : public BindingTest
{
public:
FrameBufferBindingTestCase (Context& context, QueryType type, const char* name, const char* description)
: BindingTest(context, name, description, type)
{
}
void test (glu::CallLogWrapper& gl, tcu::ResultCollector& result) const
{
verifyStateInteger(result, gl, GL_DRAW_FRAMEBUFFER_BINDING, 0, m_type);
verifyStateInteger(result, gl, GL_FRAMEBUFFER_BINDING, 0, m_type);
verifyStateInteger(result, gl, GL_READ_FRAMEBUFFER_BINDING, 0, m_type);
GLuint framebufferId = 0;
gl.glGenFramebuffers(1, &framebufferId);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glGenFramebuffers");
gl.glBindFramebuffer(GL_FRAMEBUFFER, framebufferId);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "bind GL_FRAMEBUFFER");
verifyStateInteger(result, gl, GL_DRAW_FRAMEBUFFER_BINDING, framebufferId, m_type);
verifyStateInteger(result, gl, GL_FRAMEBUFFER_BINDING, framebufferId, m_type);
verifyStateInteger(result, gl, GL_READ_FRAMEBUFFER_BINDING, framebufferId, m_type);
gl.glBindFramebuffer(GL_FRAMEBUFFER, 0);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "unbind GL_FRAMEBUFFER");
verifyStateInteger(result, gl, GL_DRAW_FRAMEBUFFER_BINDING, 0, m_type);
verifyStateInteger(result, gl, GL_FRAMEBUFFER_BINDING, 0, m_type);
verifyStateInteger(result, gl, GL_READ_FRAMEBUFFER_BINDING, 0, m_type);
gl.glBindFramebuffer(GL_READ_FRAMEBUFFER, framebufferId);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "bind GL_READ_FRAMEBUFFER");
verifyStateInteger(result, gl, GL_DRAW_FRAMEBUFFER_BINDING, 0, m_type);
verifyStateInteger(result, gl, GL_FRAMEBUFFER_BINDING, 0, m_type);
verifyStateInteger(result, gl, GL_READ_FRAMEBUFFER_BINDING, framebufferId, m_type);
gl.glBindFramebuffer(GL_DRAW_FRAMEBUFFER, framebufferId);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "bind GL_DRAW_FRAMEBUFFER");
verifyStateInteger(result, gl, GL_DRAW_FRAMEBUFFER_BINDING, framebufferId, m_type);
verifyStateInteger(result, gl, GL_FRAMEBUFFER_BINDING, framebufferId, m_type);
verifyStateInteger(result, gl, GL_READ_FRAMEBUFFER_BINDING, framebufferId, m_type);
gl.glDeleteFramebuffers(1, &framebufferId);
GLS_COLLECT_GL_ERROR(result, gl.glGetError(), "glDeleteFramebuffers");
verifyStateInteger(result, gl, GL_DRAW_FRAMEBUFFER_BINDING, 0, m_type);
verifyStateInteger(result, gl, GL_FRAMEBUFFER_BINDING, 0, m_type);
verifyStateInteger(result, gl, GL_READ_FRAMEBUFFER_BINDING, 0, m_type);
}
};
class ImplementationColorReadTestCase : public ApiCase
{
public:
ImplementationColorReadTestCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
const GLint defaultColorTypes[] =
{
GL_UNSIGNED_BYTE, GL_BYTE, GL_UNSIGNED_SHORT, GL_SHORT,
GL_UNSIGNED_INT, GL_INT, GL_HALF_FLOAT, GL_FLOAT, GL_UNSIGNED_SHORT_5_6_5,
GL_UNSIGNED_SHORT_4_4_4_4, GL_UNSIGNED_SHORT_5_5_5_1,
GL_UNSIGNED_INT_2_10_10_10_REV, GL_UNSIGNED_INT_10F_11F_11F_REV
};
const GLint defaultColorFormats[] =
{
GL_RGBA, GL_RGBA_INTEGER, GL_RGB, GL_RGB_INTEGER,
GL_RG, GL_RG_INTEGER, GL_RED, GL_RED_INTEGER
};
std::vector<GLint> validColorTypes;
std::vector<GLint> validColorFormats;
// Defined by the spec
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(defaultColorTypes); ++ndx)
validColorTypes.push_back(defaultColorTypes[ndx]);
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(defaultColorFormats); ++ndx)
validColorFormats.push_back(defaultColorFormats[ndx]);
// Extensions
if (m_context.getContextInfo().isExtensionSupported("GL_EXT_texture_format_BGRA8888") ||
m_context.getContextInfo().isExtensionSupported("GL_APPLE_texture_format_BGRA8888"))
validColorFormats.push_back(GL_BGRA);
if (m_context.getContextInfo().isExtensionSupported("GL_EXT_read_format_bgra"))
{
validColorFormats.push_back(GL_BGRA);
validColorTypes.push_back(GL_UNSIGNED_SHORT_4_4_4_4_REV);
validColorTypes.push_back(GL_UNSIGNED_SHORT_1_5_5_5_REV);
}
if (m_context.getContextInfo().isExtensionSupported("GL_IMG_read_format"))
{
validColorFormats.push_back(GL_BGRA);
validColorTypes.push_back(GL_UNSIGNED_SHORT_4_4_4_4_REV);
}
if (m_context.getContextInfo().isExtensionSupported("GL_NV_sRGB_formats"))
{
validColorFormats.push_back(GL_SLUMINANCE_NV);
validColorFormats.push_back(GL_SLUMINANCE_ALPHA_NV);
}
if (m_context.getContextInfo().isExtensionSupported("GL_NV_bgr"))
{
validColorFormats.push_back(GL_BGR_NV);
}
m_verifier->verifyIntegerAnyOf(m_testCtx, GL_IMPLEMENTATION_COLOR_READ_TYPE, &validColorTypes[0], validColorTypes.size());
m_verifier->verifyIntegerAnyOf(m_testCtx, GL_IMPLEMENTATION_COLOR_READ_FORMAT, &validColorFormats[0], validColorFormats.size());
expectError(GL_NO_ERROR);
}
private:
StateVerifier* m_verifier;
};
class ReadBufferCase : public ApiCase
{
public:
ReadBufferCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
const GLint validInitialValues[] = {GL_BACK, GL_NONE};
m_verifier->verifyIntegerAnyOf(m_testCtx, GL_READ_BUFFER, validInitialValues, DE_LENGTH_OF_ARRAY(validInitialValues));
expectError(GL_NO_ERROR);
glReadBuffer(GL_NONE);
m_verifier->verifyInteger(m_testCtx, GL_READ_BUFFER, GL_NONE);
expectError(GL_NO_ERROR);
glReadBuffer(GL_BACK);
m_verifier->verifyInteger(m_testCtx, GL_READ_BUFFER, GL_BACK);
expectError(GL_NO_ERROR);
// test GL_READ_BUFFER with framebuffers
GLuint framebufferId = 0;
glGenFramebuffers(1, &framebufferId);
expectError(GL_NO_ERROR);
GLuint renderbuffer_id = 0;
glGenRenderbuffers(1, &renderbuffer_id);
expectError(GL_NO_ERROR);
glBindRenderbuffer(GL_RENDERBUFFER, renderbuffer_id);
expectError(GL_NO_ERROR);
glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 128, 128);
expectError(GL_NO_ERROR);
glBindFramebuffer(GL_READ_FRAMEBUFFER, framebufferId);
expectError(GL_NO_ERROR);
glFramebufferRenderbuffer(GL_READ_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, renderbuffer_id);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_READ_BUFFER, GL_COLOR_ATTACHMENT0);
glDeleteFramebuffers(1, &framebufferId);
glDeleteRenderbuffers(1, &renderbuffer_id);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_READ_BUFFER, GL_BACK);
expectError(GL_NO_ERROR);
}
private:
StateVerifier* m_verifier;
};
class DrawBufferCase : public ApiCase
{
public:
DrawBufferCase (Context& context, StateVerifier* verifier, const char* name, const char* description)
: ApiCase (context, name, description)
, m_verifier (verifier)
{
}
void test (void)
{
const GLint validInitialValues[] = {GL_BACK, GL_NONE};
m_verifier->verifyIntegerAnyOf(m_testCtx, GL_DRAW_BUFFER0, validInitialValues, DE_LENGTH_OF_ARRAY(validInitialValues));
expectError(GL_NO_ERROR);
GLenum bufs = GL_NONE;
glDrawBuffers(1, &bufs);
m_verifier->verifyInteger(m_testCtx, GL_DRAW_BUFFER0, GL_NONE);
expectError(GL_NO_ERROR);
bufs = GL_BACK;
glDrawBuffers(1, &bufs);
m_verifier->verifyInteger(m_testCtx, GL_DRAW_BUFFER0, GL_BACK);
expectError(GL_NO_ERROR);
// test GL_DRAW_BUFFER with framebuffers
GLuint framebufferId = 0;
glGenFramebuffers(1, &framebufferId);
expectError(GL_NO_ERROR);
GLuint renderbuffer_ids[2] = {0};
glGenRenderbuffers(2, renderbuffer_ids);
expectError(GL_NO_ERROR);
glBindRenderbuffer(GL_RENDERBUFFER, renderbuffer_ids[0]);
expectError(GL_NO_ERROR);
glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 128, 128);
expectError(GL_NO_ERROR);
glBindRenderbuffer(GL_RENDERBUFFER, renderbuffer_ids[1]);
expectError(GL_NO_ERROR);
glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 128, 128);
expectError(GL_NO_ERROR);
glBindFramebuffer(GL_DRAW_FRAMEBUFFER, framebufferId);
expectError(GL_NO_ERROR);
glFramebufferRenderbuffer(GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, renderbuffer_ids[0]);
expectError(GL_NO_ERROR);
glFramebufferRenderbuffer(GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, GL_RENDERBUFFER, renderbuffer_ids[1]);
expectError(GL_NO_ERROR);
// only the initial state the draw buffer for fragment color zero is defined
m_verifier->verifyInteger(m_testCtx, GL_DRAW_BUFFER0, GL_COLOR_ATTACHMENT0);
GLenum bufTargets[2] = {GL_NONE, GL_COLOR_ATTACHMENT1};
glDrawBuffers(2, bufTargets);
m_verifier->verifyInteger(m_testCtx, GL_DRAW_BUFFER0, GL_NONE);
m_verifier->verifyInteger(m_testCtx, GL_DRAW_BUFFER1, GL_COLOR_ATTACHMENT1);
glDeleteFramebuffers(1, &framebufferId);
glDeleteRenderbuffers(2, renderbuffer_ids);
expectError(GL_NO_ERROR);
m_verifier->verifyInteger(m_testCtx, GL_DRAW_BUFFER0, GL_BACK);
expectError(GL_NO_ERROR);
}
private:
StateVerifier* m_verifier;
};
static const char* getQueryTypeSuffix (QueryType type)
{
switch (type)
{
case QUERY_BOOLEAN: return "_getboolean";
case QUERY_INTEGER: return "_getinteger";
case QUERY_INTEGER64: return "_getinteger64";
case QUERY_FLOAT: return "_getfloat";
default:
DE_ASSERT(DE_FALSE);
return DE_NULL;
}
}
#define FOR_EACH_VERIFIER(VERIFIERS, CODE_BLOCK) \
for (int _verifierNdx = 0; _verifierNdx < DE_LENGTH_OF_ARRAY(VERIFIERS); _verifierNdx++) \
{ \
StateVerifier* verifier = (VERIFIERS)[_verifierNdx]; \
CODE_BLOCK; \
}
#define FOR_EACH_QUERYTYPE(QUERYTYPES, CODE_BLOCK) \
for (int _queryTypeNdx = 0; _queryTypeNdx < DE_LENGTH_OF_ARRAY(QUERYTYPES); _queryTypeNdx++) \
{ \
const QueryType queryType = (QUERYTYPES)[_queryTypeNdx]; \
CODE_BLOCK; \
}
} // anonymous
IntegerStateQueryTests::IntegerStateQueryTests (Context& context)
: TestCaseGroup (context, "integers", "Integer Values")
, m_verifierBoolean (DE_NULL)
, m_verifierInteger (DE_NULL)
, m_verifierInteger64 (DE_NULL)
, m_verifierFloat (DE_NULL)
{
}
IntegerStateQueryTests::~IntegerStateQueryTests (void)
{
deinit();
}
void IntegerStateQueryTests::init (void)
{
static const QueryType queryTypes[] =
{
QUERY_BOOLEAN,
QUERY_INTEGER,
QUERY_INTEGER64,
QUERY_FLOAT,
};
DE_ASSERT(m_verifierBoolean == DE_NULL);
DE_ASSERT(m_verifierInteger == DE_NULL);
DE_ASSERT(m_verifierInteger64 == DE_NULL);
DE_ASSERT(m_verifierFloat == DE_NULL);
m_verifierBoolean = new GetBooleanVerifier (m_context.getRenderContext().getFunctions(), m_context.getTestContext().getLog());
m_verifierInteger = new GetIntegerVerifier (m_context.getRenderContext().getFunctions(), m_context.getTestContext().getLog());
m_verifierInteger64 = new GetInteger64Verifier (m_context.getRenderContext().getFunctions(), m_context.getTestContext().getLog());
m_verifierFloat = new GetFloatVerifier (m_context.getRenderContext().getFunctions(), m_context.getTestContext().getLog());
const struct LimitedStateInteger
{
const char* name;
const char* description;
GLenum targetName;
GLint value;
} implementationMinLimits[] =
{
{ "subpixel_bits", "SUBPIXEL_BITS has minimum value of 4", GL_SUBPIXEL_BITS, 4 },
{ "max_3d_texture_size", "MAX_3D_TEXTURE_SIZE has minimum value of 256", GL_MAX_3D_TEXTURE_SIZE, 256 },
{ "max_texture_size", "MAX_TEXTURE_SIZE has minimum value of 2048", GL_MAX_TEXTURE_SIZE, 2048},
{ "max_array_texture_layers", "MAX_ARRAY_TEXTURE_LAYERS has minimum value of 256", GL_MAX_ARRAY_TEXTURE_LAYERS, 256 },
{ "max_cube_map_texture_size", "MAX_CUBE_MAP_TEXTURE_SIZE has minimum value of 2048", GL_MAX_CUBE_MAP_TEXTURE_SIZE, 2048},
{ "max_renderbuffer_size", "MAX_RENDERBUFFER_SIZE has minimum value of 2048", GL_MAX_RENDERBUFFER_SIZE, 2048},
{ "max_draw_buffers", "MAX_DRAW_BUFFERS has minimum value of 4", GL_MAX_DRAW_BUFFERS, 4 },
{ "max_color_attachments", "MAX_COLOR_ATTACHMENTS has minimum value of 4", GL_MAX_COLOR_ATTACHMENTS, 4 },
{ "max_elements_indices", "MAX_ELEMENTS_INDICES has minimum value of 0", GL_MAX_ELEMENTS_INDICES, 0 },
{ "max_elements_vertices", "MAX_ELEMENTS_VERTICES has minimum value of 0", GL_MAX_ELEMENTS_VERTICES, 0 },
{ "num_extensions", "NUM_EXTENSIONS has minimum value of 0", GL_NUM_EXTENSIONS, 0 },
{ "major_version", "MAJOR_VERSION has minimum value of 3", GL_MAJOR_VERSION, 3 },
{ "minor_version", "MINOR_VERSION has minimum value of 0", GL_MINOR_VERSION, 0 },
{ "max_vertex_attribs", "MAX_VERTEX_ATTRIBS has minimum value of 16", GL_MAX_VERTEX_ATTRIBS, 16 },
{ "max_vertex_uniform_components", "MAX_VERTEX_UNIFORM_COMPONENTS has minimum value of 1024", GL_MAX_VERTEX_UNIFORM_COMPONENTS, 1024},
{ "max_vertex_uniform_vectors", "MAX_VERTEX_UNIFORM_VECTORS has minimum value of 256", GL_MAX_VERTEX_UNIFORM_VECTORS, 256 },
{ "max_vertex_uniform_blocks", "MAX_VERTEX_UNIFORM_BLOCKS has minimum value of 12", GL_MAX_VERTEX_UNIFORM_BLOCKS, 12 },
{ "max_vertex_output_components", "MAX_VERTEX_OUTPUT_COMPONENTS has minimum value of 64", GL_MAX_VERTEX_OUTPUT_COMPONENTS, 64 },
{ "max_vertex_texture_image_units", "MAX_VERTEX_TEXTURE_IMAGE_UNITS has minimum value of 16", GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS, 16 },
{ "max_fragment_uniform_components", "MAX_FRAGMENT_UNIFORM_COMPONENTS has minimum value of 896", GL_MAX_FRAGMENT_UNIFORM_COMPONENTS, 896 },
{ "max_fragment_uniform_vectors", "MAX_FRAGMENT_UNIFORM_VECTORS has minimum value of 224", GL_MAX_FRAGMENT_UNIFORM_VECTORS, 224 },
{ "max_fragment_uniform_blocks", "MAX_FRAGMENT_UNIFORM_BLOCKS has minimum value of 12", GL_MAX_FRAGMENT_UNIFORM_BLOCKS, 12 },
{ "max_fragment_input_components", "MAX_FRAGMENT_INPUT_COMPONENTS has minimum value of 60", GL_MAX_FRAGMENT_INPUT_COMPONENTS, 60 },
{ "max_texture_image_units", "MAX_TEXTURE_IMAGE_UNITS has minimum value of 16", GL_MAX_TEXTURE_IMAGE_UNITS, 16 },
{ "max_program_texel_offset", "MAX_PROGRAM_TEXEL_OFFSET has minimum value of 7", GL_MAX_PROGRAM_TEXEL_OFFSET, 7 },
{ "max_uniform_buffer_bindings", "MAX_UNIFORM_BUFFER_BINDINGS has minimum value of 24", GL_MAX_UNIFORM_BUFFER_BINDINGS, 24 },
{ "max_combined_uniform_blocks", "MAX_COMBINED_UNIFORM_BLOCKS has minimum value of 24", GL_MAX_COMBINED_UNIFORM_BLOCKS, 24 },
{ "max_varying_components", "MAX_VARYING_COMPONENTS has minimum value of 60", GL_MAX_VARYING_COMPONENTS, 60 },
{ "max_varying_vectors", "MAX_VARYING_VECTORS has minimum value of 15", GL_MAX_VARYING_VECTORS, 15 },
{ "max_combined_texture_image_units", "MAX_COMBINED_TEXTURE_IMAGE_UNITS has minimum value of 32", GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS, 32 },
{ "max_transform_feedback_interleaved_components", "MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS has minimum value of 64", GL_MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS, 64 },
{ "max_transform_feedback_separate_attribs", "MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS has minimum value of 4", GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS, 4 },
{ "max_transform_feedback_separate_components", "MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS has minimum value of 4", GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS, 4 },
{ "max_samples", "MAX_SAMPLES has minimum value of 4", GL_MAX_SAMPLES, 4 },
{ "red_bits", "RED_BITS has minimum value of 0", GL_RED_BITS, 0 },
{ "green_bits", "GREEN_BITS has minimum value of 0", GL_GREEN_BITS, 0 },
{ "blue_bits", "BLUE_BITS has minimum value of 0", GL_BLUE_BITS, 0 },
{ "alpha_bits", "ALPHA_BITS has minimum value of 0", GL_ALPHA_BITS, 0 },
{ "depth_bits", "DEPTH_BITS has minimum value of 0", GL_DEPTH_BITS, 0 },
{ "stencil_bits", "STENCIL_BITS has minimum value of 0", GL_STENCIL_BITS, 0 },
};
const LimitedStateInteger implementationMaxLimits[] =
{
{ "min_program_texel_offset", "MIN_PROGRAM_TEXEL_OFFSET has maximum value of -8", GL_MIN_PROGRAM_TEXEL_OFFSET, -8 },
{ "uniform_buffer_offset_alignment", "UNIFORM_BUFFER_OFFSET_ALIGNMENT has minimum value of 1", GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT, 256 },
};
// \note implementation defined limits have their own tests so just check the conversions to boolean, int64 and float
StateVerifier* implementationLimitVerifiers[] = {m_verifierBoolean, m_verifierInteger64, m_verifierFloat};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(implementationMinLimits); testNdx++)
FOR_EACH_VERIFIER(implementationLimitVerifiers, addChild(new ConstantMinimumValueTestCase(m_context, verifier, (std::string(implementationMinLimits[testNdx].name) + verifier->getTestNamePostfix()).c_str(), implementationMinLimits[testNdx].description, implementationMinLimits[testNdx].targetName, implementationMinLimits[testNdx].value)));
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(implementationMaxLimits); testNdx++)
FOR_EACH_VERIFIER(implementationLimitVerifiers, addChild(new ConstantMaximumValueTestCase(m_context, verifier, (std::string(implementationMaxLimits[testNdx].name) + verifier->getTestNamePostfix()).c_str(), implementationMaxLimits[testNdx].description, implementationMaxLimits[testNdx].targetName, implementationMaxLimits[testNdx].value)));
StateVerifier* normalVerifiers[] = {m_verifierBoolean, m_verifierInteger, m_verifierInteger64, m_verifierFloat};
FOR_EACH_VERIFIER(implementationLimitVerifiers, addChild(new SampleBuffersTestCase (m_context, verifier, (std::string("sample_buffers") + verifier->getTestNamePostfix()).c_str(), "SAMPLE_BUFFERS")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new SamplesTestCase (m_context, verifier, (std::string("samples") + verifier->getTestNamePostfix()).c_str(), "SAMPLES")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new HintTestCase (m_context, verifier, (std::string("generate_mipmap_hint") + verifier->getTestNamePostfix()).c_str(), "GENERATE_MIPMAP_HINT", GL_GENERATE_MIPMAP_HINT)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new HintTestCase (m_context, verifier, (std::string("fragment_shader_derivative_hint") + verifier->getTestNamePostfix()).c_str(), "FRAGMENT_SHADER_DERIVATIVE_HINT", GL_FRAGMENT_SHADER_DERIVATIVE_HINT)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new DepthFuncTestCase (m_context, verifier, (std::string("depth_func") + verifier->getTestNamePostfix()).c_str(), "DEPTH_FUNC")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new CullFaceTestCase (m_context, verifier, (std::string("cull_face_mode") + verifier->getTestNamePostfix()).c_str(), "CULL_FACE_MODE")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new FrontFaceTestCase (m_context, verifier, (std::string("front_face_mode") + verifier->getTestNamePostfix()).c_str(), "FRONT_FACE")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new ViewPortTestCase (m_context, verifier, (std::string("viewport") + verifier->getTestNamePostfix()).c_str(), "VIEWPORT")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new ScissorBoxTestCase (m_context, verifier, (std::string("scissor_box") + verifier->getTestNamePostfix()).c_str(), "SCISSOR_BOX")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new MaxViewportDimsTestCase (m_context, verifier, (std::string("max_viewport_dims") + verifier->getTestNamePostfix()).c_str(), "MAX_VIEWPORT_DIMS")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilRefTestCase (m_context, verifier, (std::string("stencil_ref") + verifier->getTestNamePostfix()).c_str(), "STENCIL_REF", GL_STENCIL_REF)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilRefTestCase (m_context, verifier, (std::string("stencil_back_ref") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_REF", GL_STENCIL_BACK_REF)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilRefSeparateTestCase (m_context, verifier, (std::string("stencil_ref_separate") + verifier->getTestNamePostfix()).c_str(), "STENCIL_REF (separate)", GL_STENCIL_REF, GL_FRONT)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilRefSeparateTestCase (m_context, verifier, (std::string("stencil_ref_separate_both") + verifier->getTestNamePostfix()).c_str(), "STENCIL_REF (separate)", GL_STENCIL_REF, GL_FRONT_AND_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilRefSeparateTestCase (m_context, verifier, (std::string("stencil_back_ref_separate") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_REF (separate)", GL_STENCIL_BACK_REF, GL_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilRefSeparateTestCase (m_context, verifier, (std::string("stencil_back_ref_separate_both") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_REF (separate)", GL_STENCIL_BACK_REF, GL_FRONT_AND_BACK)));
const struct NamedStencilOp
{
const char* name;
const char* frontDescription;
GLenum frontTarget;
const char* backDescription;
GLenum backTarget;
} stencilOps[] =
{
{ "fail", "STENCIL_FAIL", GL_STENCIL_FAIL, "STENCIL_BACK_FAIL", GL_STENCIL_BACK_FAIL },
{ "depth_fail", "STENCIL_PASS_DEPTH_FAIL", GL_STENCIL_PASS_DEPTH_FAIL, "STENCIL_BACK_PASS_DEPTH_FAIL", GL_STENCIL_BACK_PASS_DEPTH_FAIL },
{ "depth_pass", "STENCIL_PASS_DEPTH_PASS", GL_STENCIL_PASS_DEPTH_PASS, "STENCIL_BACK_PASS_DEPTH_PASS", GL_STENCIL_BACK_PASS_DEPTH_PASS }
};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(stencilOps); testNdx++)
{
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilOpTestCase (m_context, verifier, (std::string("stencil_") + stencilOps[testNdx].name + verifier->getTestNamePostfix()).c_str(), stencilOps[testNdx].frontDescription, stencilOps[testNdx].frontTarget)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilOpTestCase (m_context, verifier, (std::string("stencil_back_") + stencilOps[testNdx].name + verifier->getTestNamePostfix()).c_str(), stencilOps[testNdx].backDescription, stencilOps[testNdx].backTarget)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilOpSeparateTestCase (m_context, verifier, (std::string("stencil_") + stencilOps[testNdx].name + "_separate_both" + verifier->getTestNamePostfix()).c_str(), stencilOps[testNdx].frontDescription, stencilOps[testNdx].frontTarget, GL_FRONT_AND_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilOpSeparateTestCase (m_context, verifier, (std::string("stencil_back_") + stencilOps[testNdx].name + "_separate_both" + verifier->getTestNamePostfix()).c_str(), stencilOps[testNdx].backDescription, stencilOps[testNdx].backTarget, GL_FRONT_AND_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilOpSeparateTestCase (m_context, verifier, (std::string("stencil_") + stencilOps[testNdx].name + "_separate" + verifier->getTestNamePostfix()).c_str(), stencilOps[testNdx].frontDescription, stencilOps[testNdx].frontTarget, GL_FRONT)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilOpSeparateTestCase (m_context, verifier, (std::string("stencil_back_") + stencilOps[testNdx].name + "_separate" + verifier->getTestNamePostfix()).c_str(), stencilOps[testNdx].backDescription, stencilOps[testNdx].backTarget, GL_BACK)));
}
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilFuncTestCase (m_context, verifier, (std::string("stencil_func") + verifier->getTestNamePostfix()).c_str(), "STENCIL_FUNC")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilFuncSeparateTestCase (m_context, verifier, (std::string("stencil_func_separate") + verifier->getTestNamePostfix()).c_str(), "STENCIL_FUNC (separate)", GL_STENCIL_FUNC, GL_FRONT)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilFuncSeparateTestCase (m_context, verifier, (std::string("stencil_func_separate_both") + verifier->getTestNamePostfix()).c_str(), "STENCIL_FUNC (separate)", GL_STENCIL_FUNC, GL_FRONT_AND_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilFuncSeparateTestCase (m_context, verifier, (std::string("stencil_back_func_separate") + verifier->getTestNamePostfix()).c_str(), "STENCIL_FUNC (separate)", GL_STENCIL_BACK_FUNC, GL_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilFuncSeparateTestCase (m_context, verifier, (std::string("stencil_back_func_separate_both") + verifier->getTestNamePostfix()).c_str(), "STENCIL_FUNC (separate)", GL_STENCIL_BACK_FUNC, GL_FRONT_AND_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilMaskTestCase (m_context, verifier, (std::string("stencil_value_mask") + verifier->getTestNamePostfix()).c_str(), "STENCIL_VALUE_MASK", GL_STENCIL_VALUE_MASK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilMaskTestCase (m_context, verifier, (std::string("stencil_back_value_mask") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_VALUE_MASK", GL_STENCIL_BACK_VALUE_MASK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilMaskSeparateTestCase (m_context, verifier, (std::string("stencil_value_mask_separate") + verifier->getTestNamePostfix()).c_str(), "STENCIL_VALUE_MASK (separate)", GL_STENCIL_VALUE_MASK, GL_FRONT)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilMaskSeparateTestCase (m_context, verifier, (std::string("stencil_value_mask_separate_both") + verifier->getTestNamePostfix()).c_str(), "STENCIL_VALUE_MASK (separate)", GL_STENCIL_VALUE_MASK, GL_FRONT_AND_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilMaskSeparateTestCase (m_context, verifier, (std::string("stencil_back_value_mask_separate") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_VALUE_MASK (separate)", GL_STENCIL_BACK_VALUE_MASK, GL_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilMaskSeparateTestCase (m_context, verifier, (std::string("stencil_back_value_mask_separate_both") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_VALUE_MASK (separate)", GL_STENCIL_BACK_VALUE_MASK, GL_FRONT_AND_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilWriteMaskTestCase (m_context, verifier, (std::string("stencil_writemask") + verifier->getTestNamePostfix()).c_str(), "STENCIL_WRITEMASK", GL_STENCIL_WRITEMASK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilWriteMaskTestCase (m_context, verifier, (std::string("stencil_back_writemask") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_WRITEMASK", GL_STENCIL_BACK_WRITEMASK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilWriteMaskSeparateTestCase (m_context, verifier, (std::string("stencil_writemask_separate") + verifier->getTestNamePostfix()).c_str(), "STENCIL_WRITEMASK (separate)", GL_STENCIL_WRITEMASK, GL_FRONT)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilWriteMaskSeparateTestCase (m_context, verifier, (std::string("stencil_writemask_separate_both") + verifier->getTestNamePostfix()).c_str(), "STENCIL_WRITEMASK (separate)", GL_STENCIL_WRITEMASK, GL_FRONT_AND_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilWriteMaskSeparateTestCase (m_context, verifier, (std::string("stencil_back_writemask_separate") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_WRITEMASK (separate)", GL_STENCIL_BACK_WRITEMASK, GL_BACK)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilWriteMaskSeparateTestCase (m_context, verifier, (std::string("stencil_back_writemask_separate_both") + verifier->getTestNamePostfix()).c_str(), "STENCIL_BACK_WRITEMASK (separate)", GL_STENCIL_BACK_WRITEMASK, GL_FRONT_AND_BACK)));
const struct PixelStoreState
{
const char* name;
const char* description;
GLenum target;
int initialValue;
} pixelStoreStates[] =
{
{ "unpack_image_height","UNPACK_IMAGE_HEIGHT", GL_UNPACK_IMAGE_HEIGHT, 0 },
{ "unpack_skip_images", "UNPACK_SKIP_IMAGES", GL_UNPACK_SKIP_IMAGES, 0 },
{ "unpack_row_length", "UNPACK_ROW_LENGTH", GL_UNPACK_ROW_LENGTH, 0 },
{ "unpack_skip_rows", "UNPACK_SKIP_ROWS", GL_UNPACK_SKIP_ROWS, 0 },
{ "unpack_skip_pixels", "UNPACK_SKIP_PIXELS", GL_UNPACK_SKIP_PIXELS, 0 },
{ "pack_row_length", "PACK_ROW_LENGTH", GL_PACK_ROW_LENGTH, 0 },
{ "pack_skip_rows", "PACK_SKIP_ROWS", GL_PACK_SKIP_ROWS, 0 },
{ "pack_skip_pixels", "PACK_SKIP_PIXELS", GL_PACK_SKIP_PIXELS, 0 }
};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(pixelStoreStates); testNdx++)
{
FOR_EACH_VERIFIER(normalVerifiers, addChild(new PixelStoreTestCase(m_context, verifier, (std::string(pixelStoreStates[testNdx].name) + verifier->getTestNamePostfix()).c_str(), pixelStoreStates[testNdx].description, pixelStoreStates[testNdx].target, pixelStoreStates[testNdx].initialValue)));
}
FOR_EACH_VERIFIER(normalVerifiers, addChild(new PixelStoreAlignTestCase(m_context, verifier, (std::string("unpack_alignment") + verifier->getTestNamePostfix()).c_str(), "UNPACK_ALIGNMENT", GL_UNPACK_ALIGNMENT)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new PixelStoreAlignTestCase(m_context, verifier, (std::string("pack_alignment") + verifier->getTestNamePostfix()).c_str(), "PACK_ALIGNMENT", GL_PACK_ALIGNMENT)));
const struct BlendColorState
{
const char* name;
const char* description;
GLenum target;
int initialValue;
} blendColorStates[] =
{
{ "blend_src_rgb", "BLEND_SRC_RGB", GL_BLEND_SRC_RGB, GL_ONE },
{ "blend_src_alpha", "BLEND_SRC_ALPHA", GL_BLEND_SRC_ALPHA, GL_ONE },
{ "blend_dst_rgb", "BLEND_DST_RGB", GL_BLEND_DST_RGB, GL_ZERO },
{ "blend_dst_alpha", "BLEND_DST_ALPHA", GL_BLEND_DST_ALPHA, GL_ZERO }
};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(blendColorStates); testNdx++)
{
FOR_EACH_VERIFIER(normalVerifiers, addChild(new BlendFuncTestCase (m_context, verifier, (std::string(blendColorStates[testNdx].name) + verifier->getTestNamePostfix()).c_str(), blendColorStates[testNdx].description, blendColorStates[testNdx].target, blendColorStates[testNdx].initialValue)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new BlendFuncSeparateTestCase (m_context, verifier, (std::string(blendColorStates[testNdx].name) + "_separate" + verifier->getTestNamePostfix()).c_str(), blendColorStates[testNdx].description, blendColorStates[testNdx].target, blendColorStates[testNdx].initialValue)));
}
const struct BlendEquationState
{
const char* name;
const char* description;
GLenum target;
int initialValue;
} blendEquationStates[] =
{
{ "blend_equation_rgb", "BLEND_EQUATION_RGB", GL_BLEND_EQUATION_RGB, GL_FUNC_ADD },
{ "blend_equation_alpha", "BLEND_EQUATION_ALPHA", GL_BLEND_EQUATION_ALPHA, GL_FUNC_ADD }
};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(blendEquationStates); testNdx++)
{
FOR_EACH_VERIFIER(normalVerifiers, addChild(new BlendEquationTestCase (m_context, verifier, (std::string(blendEquationStates[testNdx].name) + + verifier->getTestNamePostfix()).c_str(), blendEquationStates[testNdx].description, blendEquationStates[testNdx].target, blendEquationStates[testNdx].initialValue)));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new BlendEquationSeparateTestCase (m_context, verifier, (std::string(blendEquationStates[testNdx].name) + "_separate" + verifier->getTestNamePostfix()).c_str(), blendEquationStates[testNdx].description, blendEquationStates[testNdx].target, blendEquationStates[testNdx].initialValue)));
}
const struct ImplementationArrayReturningState
{
const char* name;
const char* description;
GLenum target;
GLenum targetLengthTarget;
int minLength;
} implementationArrayReturningStates[] =
{
{ "compressed_texture_formats", "COMPRESSED_TEXTURE_FORMATS", GL_COMPRESSED_TEXTURE_FORMATS, GL_NUM_COMPRESSED_TEXTURE_FORMATS, 10 },
{ "program_binary_formats", "PROGRAM_BINARY_FORMATS", GL_PROGRAM_BINARY_FORMATS, GL_NUM_PROGRAM_BINARY_FORMATS, 0 },
{ "shader_binary_formats", "SHADER_BINARY_FORMATS", GL_SHADER_BINARY_FORMATS, GL_NUM_SHADER_BINARY_FORMATS, 0 }
};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(implementationArrayReturningStates); testNdx++)
{
FOR_EACH_VERIFIER(normalVerifiers, addChild(new ImplementationArrayTestCase(m_context, verifier, (std::string(implementationArrayReturningStates[testNdx].name) + verifier->getTestNamePostfix()).c_str(), implementationArrayReturningStates[testNdx].description, implementationArrayReturningStates[testNdx].target, implementationArrayReturningStates[testNdx].targetLengthTarget, implementationArrayReturningStates[testNdx].minLength)));
}
const struct BufferBindingState
{
const char* name;
const char* description;
GLenum target;
GLenum type;
} bufferBindingStates[] =
{
{ "array_buffer_binding", "ARRAY_BUFFER_BINDING", GL_ARRAY_BUFFER_BINDING, GL_ARRAY_BUFFER },
{ "uniform_buffer_binding", "UNIFORM_BUFFER_BINDING", GL_UNIFORM_BUFFER_BINDING, GL_UNIFORM_BUFFER },
{ "pixel_pack_buffer_binding", "PIXEL_PACK_BUFFER_BINDING", GL_PIXEL_PACK_BUFFER_BINDING, GL_PIXEL_PACK_BUFFER },
{ "pixel_unpack_buffer_binding", "PIXEL_UNPACK_BUFFER_BINDING", GL_PIXEL_UNPACK_BUFFER_BINDING, GL_PIXEL_UNPACK_BUFFER },
{ "transform_feedback_buffer_binding", "TRANSFORM_FEEDBACK_BUFFER_BINDING", GL_TRANSFORM_FEEDBACK_BUFFER_BINDING, GL_TRANSFORM_FEEDBACK_BUFFER},
{ "copy_read_buffer_binding", "COPY_READ_BUFFER_BINDING", GL_COPY_READ_BUFFER_BINDING, GL_COPY_READ_BUFFER },
{ "copy_write_buffer_binding", "COPY_WRITE_BUFFER_BINDING", GL_COPY_WRITE_BUFFER_BINDING, GL_COPY_WRITE_BUFFER }
};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(bufferBindingStates); testNdx++)
{
FOR_EACH_QUERYTYPE(queryTypes, addChild(new BufferBindingTestCase(m_context, queryType, (std::string(bufferBindingStates[testNdx].name) + getQueryTypeSuffix(queryType)).c_str(), bufferBindingStates[testNdx].description, bufferBindingStates[testNdx].target, bufferBindingStates[testNdx].type)));
}
FOR_EACH_QUERYTYPE(queryTypes, addChild(new ElementArrayBufferBindingTestCase (m_context, queryType, (std::string("element_array_buffer_binding") + getQueryTypeSuffix(queryType)).c_str())));
FOR_EACH_QUERYTYPE(queryTypes, addChild(new TransformFeedbackBindingTestCase (m_context, queryType, (std::string("transform_feedback_binding") + getQueryTypeSuffix(queryType)).c_str())));
FOR_EACH_QUERYTYPE(queryTypes, addChild(new CurrentProgramBindingTestCase (m_context, queryType, (std::string("current_program_binding") + getQueryTypeSuffix(queryType)).c_str(), "CURRENT_PROGRAM")));
FOR_EACH_QUERYTYPE(queryTypes, addChild(new VertexArrayBindingTestCase (m_context, queryType, (std::string("vertex_array_binding") + getQueryTypeSuffix(queryType)).c_str(), "VERTEX_ARRAY_BINDING")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new StencilClearValueTestCase (m_context, verifier, (std::string("stencil_clear_value") + verifier->getTestNamePostfix()).c_str(), "STENCIL_CLEAR_VALUE")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new ActiveTextureTestCase (m_context, verifier, (std::string("active_texture") + verifier->getTestNamePostfix()).c_str(), "ACTIVE_TEXTURE")));
FOR_EACH_QUERYTYPE(queryTypes, addChild(new RenderbufferBindingTestCase (m_context, queryType, (std::string("renderbuffer_binding") + getQueryTypeSuffix(queryType)).c_str(), "RENDERBUFFER_BINDING")));
FOR_EACH_QUERYTYPE(queryTypes, addChild(new SamplerObjectBindingTestCase (m_context, queryType, (std::string("sampler_binding") + getQueryTypeSuffix(queryType)).c_str(), "SAMPLER_BINDING")));
const struct TextureBinding
{
const char* name;
const char* description;
GLenum target;
GLenum type;
} textureBindings[] =
{
{ "texture_binding_2d", "TEXTURE_BINDING_2D", GL_TEXTURE_BINDING_2D, GL_TEXTURE_2D },
{ "texture_binding_3d", "TEXTURE_BINDING_3D", GL_TEXTURE_BINDING_3D, GL_TEXTURE_3D },
{ "texture_binding_2d_array", "TEXTURE_BINDING_2D_ARRAY", GL_TEXTURE_BINDING_2D_ARRAY, GL_TEXTURE_2D_ARRAY },
{ "texture_binding_cube_map", "TEXTURE_BINDING_CUBE_MAP", GL_TEXTURE_BINDING_CUBE_MAP, GL_TEXTURE_CUBE_MAP }
};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(textureBindings); testNdx++)
{
FOR_EACH_QUERYTYPE(queryTypes, addChild(new TextureBindingTestCase(m_context, queryType, (std::string(textureBindings[testNdx].name) + getQueryTypeSuffix(queryType)).c_str(), textureBindings[testNdx].description, textureBindings[testNdx].target, textureBindings[testNdx].type)));
}
FOR_EACH_QUERYTYPE(queryTypes, addChild(new FrameBufferBindingTestCase (m_context, queryType, (std::string("framebuffer_binding") + getQueryTypeSuffix(queryType)).c_str(), "DRAW_FRAMEBUFFER_BINDING and READ_FRAMEBUFFER_BINDING")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new ImplementationColorReadTestCase (m_context, verifier, (std::string("implementation_color_read") + verifier->getTestNamePostfix()).c_str(), "IMPLEMENTATION_COLOR_READ_TYPE and IMPLEMENTATION_COLOR_READ_FORMAT")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new ReadBufferCase (m_context, verifier, (std::string("read_buffer") + verifier->getTestNamePostfix()).c_str(), "READ_BUFFER")));
FOR_EACH_VERIFIER(normalVerifiers, addChild(new DrawBufferCase (m_context, verifier, (std::string("draw_buffer") + verifier->getTestNamePostfix()).c_str(), "DRAW_BUFFER")));
}
void IntegerStateQueryTests::deinit (void)
{
if (m_verifierBoolean)
{
delete m_verifierBoolean;
m_verifierBoolean = DE_NULL;
}
if (m_verifierInteger)
{
delete m_verifierInteger;
m_verifierInteger = DE_NULL;
}
if (m_verifierInteger64)
{
delete m_verifierInteger64;
m_verifierInteger64 = DE_NULL;
}
if (m_verifierFloat)
{
delete m_verifierFloat;
m_verifierFloat = DE_NULL;
}
this->TestCaseGroup::deinit();
}
} // Functional
} // gles3
} // deqp