/*-------------------------------------------------------------------------
* 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 Rbo state query tests.
*//*--------------------------------------------------------------------*/
#include "es3fRboStateQueryTests.hpp"
#include "glsStateQueryUtil.hpp"
#include "es3fApiCase.hpp"
#include "gluRenderContext.hpp"
#include "glwEnums.hpp"
#include "glwFunctions.hpp"
#include "deRandom.hpp"
#include "deMath.h"
using namespace glw; // GLint and other GL types
using deqp::gls::StateQueryUtil::StateQueryMemoryWriteGuard;
namespace deqp
{
namespace gles3
{
namespace Functional
{
namespace
{
void checkRenderbufferComponentSize (tcu::TestContext& testCtx, glu::CallLogWrapper& gl, int r, int g, int b, int a, int d, int s)
{
using tcu::TestLog;
const int referenceSizes[] = {r, g, b, a, d, s};
const GLenum paramNames[] =
{
GL_RENDERBUFFER_RED_SIZE,
GL_RENDERBUFFER_GREEN_SIZE,
GL_RENDERBUFFER_BLUE_SIZE,
GL_RENDERBUFFER_ALPHA_SIZE,
GL_RENDERBUFFER_DEPTH_SIZE,
GL_RENDERBUFFER_STENCIL_SIZE
};
DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(referenceSizes) == DE_LENGTH_OF_ARRAY(paramNames));
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(referenceSizes); ++ndx)
{
if (referenceSizes[ndx] == -1)
continue;
StateQueryMemoryWriteGuard<GLint> state;
gl.glGetRenderbufferParameteriv(GL_RENDERBUFFER, paramNames[ndx], &state);
if (!state.verifyValidity(testCtx))
return;
if (state < referenceSizes[ndx])
{
testCtx.getLog() << TestLog::Message << "// ERROR: Expected greater or equal to " << referenceSizes[ndx] << "; got " << state << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "got invalid value");
}
}
}
void checkIntEquals (tcu::TestContext& testCtx, GLint got, GLint expected)
{
using tcu::TestLog;
if (got != expected)
{
testCtx.getLog() << TestLog::Message << "// ERROR: Expected " << expected << "; got " << got << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "got invalid value");
}
}
void checkIntGreaterOrEqual (tcu::TestContext& testCtx, GLint got, GLint expected)
{
using tcu::TestLog;
if (got < expected)
{
testCtx.getLog() << TestLog::Message << "// ERROR: Expected greater or equal to " << expected << "; got " << got << TestLog::EndMessage;
if (testCtx.getTestResult() == QP_TEST_RESULT_PASS)
testCtx.setTestResult(QP_TEST_RESULT_FAIL, "got invalid value");
}
}
void checkRenderbufferParam (tcu::TestContext& testCtx, glu::CallLogWrapper& gl, GLenum pname, GLenum reference)
{
StateQueryMemoryWriteGuard<GLint> state;
gl.glGetRenderbufferParameteriv(GL_RENDERBUFFER, pname, &state);
if (state.verifyValidity(testCtx))
checkIntEquals(testCtx, state, reference);
}
void checkRenderbufferParamGreaterOrEqual (tcu::TestContext& testCtx, glu::CallLogWrapper& gl, GLenum pname, GLenum reference)
{
StateQueryMemoryWriteGuard<GLint> state;
gl.glGetRenderbufferParameteriv(GL_RENDERBUFFER, pname, &state);
if (state.verifyValidity(testCtx))
checkIntGreaterOrEqual(testCtx, state, reference);
}
class RboSizeCase : public ApiCase
{
public:
RboSizeCase (Context& context, const char* name, const char* description)
: ApiCase(context, name, description)
{
}
void test (void)
{
de::Random rnd(0xabcdef);
GLuint renderbufferID = 0;
glGenRenderbuffers(1, &renderbufferID);
glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
expectError(GL_NO_ERROR);
checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_WIDTH, 0);
checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_HEIGHT, 0);
expectError(GL_NO_ERROR);
const int numIterations = 60;
for (int i = 0; i < numIterations; ++i)
{
const GLint w = rnd.getInt(0, 128);
const GLint h = rnd.getInt(0, 128);
glRenderbufferStorage(GL_RENDERBUFFER, GL_RGB8, w, h);
checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_WIDTH, w);
checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_HEIGHT, h);
}
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, 0);
glDeleteRenderbuffers(1, &renderbufferID);
}
};
class RboInternalFormatCase : public ApiCase
{
public:
RboInternalFormatCase (Context& context, const char* name, const char* description)
: ApiCase(context, name, description)
{
}
void test (void)
{
GLuint renderbufferID = 0;
glGenRenderbuffers(1, &renderbufferID);
glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
expectError(GL_NO_ERROR);
checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_INTERNAL_FORMAT, GL_RGBA4);
expectError(GL_NO_ERROR);
const GLenum requiredColorformats[] =
{
GL_R8, GL_RG8, GL_RGB8, GL_RGB565, GL_RGBA4, GL_RGB5_A1, GL_RGBA8, GL_RGB10_A2,
GL_RGB10_A2UI, GL_SRGB8_ALPHA8, GL_R8I, GL_R8UI, GL_R16I, GL_R16UI, GL_R32I, GL_R32UI,
GL_RG8I, GL_RG8UI, GL_RG16I, GL_RG16UI, GL_RG32I, GL_RG32UI, GL_RGBA8I, GL_RGBA8UI,
GL_RGBA16I, GL_RGBA16UI, GL_RGBA32I, GL_RGBA32UI
};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(requiredColorformats); ++ndx)
{
glRenderbufferStorage(GL_RENDERBUFFER, requiredColorformats[ndx], 128, 128);
expectError(GL_NO_ERROR);
checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_INTERNAL_FORMAT, requiredColorformats[ndx]);
}
glDeleteRenderbuffers(1, &renderbufferID);
}
};
class RboComponentSizeColorCase : public ApiCase
{
public:
RboComponentSizeColorCase (Context& context, const char* name, const char* description)
: ApiCase(context, name, description)
{
}
void test (void)
{
GLuint renderbufferID = 0;
glGenRenderbuffers(1, &renderbufferID);
glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
expectError(GL_NO_ERROR);
checkRenderbufferComponentSize(m_testCtx, *this, 0, 0, 0, 0, 0, 0);
expectError(GL_NO_ERROR);
const struct ColorFormat
{
GLenum internalFormat;
int bitsR, bitsG, bitsB, bitsA;
} requiredColorFormats[] =
{
{ GL_R8, 8, 0, 0, 0 },
{ GL_RG8, 8, 8, 0, 0 },
{ GL_RGB8, 8, 8, 8, 0 },
{ GL_RGB565, 5, 6, 5, 0 },
{ GL_RGBA4, 4, 4, 4, 4 },
{ GL_RGB5_A1, 5, 5, 5, 1 },
{ GL_RGBA8, 8, 8, 8, 8 },
{ GL_RGB10_A2, 10, 10, 10, 2 },
{ GL_RGB10_A2UI, 10, 10, 10, 2 },
{ GL_SRGB8_ALPHA8, 8, 8, 8, 8 },
{ GL_R8I, 8, 0, 0, 0 },
{ GL_R8UI, 8, 0, 0, 0 },
{ GL_R16I, 16, 0, 0, 0 },
{ GL_R16UI, 16, 0, 0, 0 },
{ GL_R32I, 32, 0, 0, 0 },
{ GL_R32UI, 32, 0, 0, 0 },
{ GL_RG8I, 8, 8, 0, 0 },
{ GL_RG8UI, 8, 8, 0, 0 },
{ GL_RG16I, 16, 16, 0, 0 },
{ GL_RG16UI, 16, 16, 0, 0 },
{ GL_RG32I, 32, 32, 0, 0 },
{ GL_RG32UI, 32, 32, 0, 0 },
{ GL_RGBA8I, 8, 8, 8, 8 },
{ GL_RGBA8UI, 8, 8, 8, 8 },
{ GL_RGBA16I, 16, 16, 16, 16 },
{ GL_RGBA16UI, 16, 16, 16, 16 },
{ GL_RGBA32I, 32, 32, 32, 32 },
{ GL_RGBA32UI, 32, 32, 32, 32 }
};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(requiredColorFormats); ++ndx)
{
glRenderbufferStorage(GL_RENDERBUFFER, requiredColorFormats[ndx].internalFormat, 128, 128);
expectError(GL_NO_ERROR);
checkRenderbufferComponentSize(m_testCtx, *this, requiredColorFormats[ndx].bitsR, requiredColorFormats[ndx].bitsG, requiredColorFormats[ndx].bitsB, requiredColorFormats[ndx].bitsA, -1, -1);
}
glDeleteRenderbuffers(1, &renderbufferID);
}
};
class RboComponentSizeDepthCase : public ApiCase
{
public:
RboComponentSizeDepthCase (Context& context, const char* name, const char* description)
: ApiCase(context, name, description)
{
}
void test (void)
{
using tcu::TestLog;
GLuint renderbufferID = 0;
glGenRenderbuffers(1, &renderbufferID);
glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
expectError(GL_NO_ERROR);
const struct DepthFormat
{
GLenum internalFormat;
int dbits;
int sbits;
} requiredDepthFormats[] =
{
{ GL_DEPTH_COMPONENT16, 16, 0 },
{ GL_DEPTH_COMPONENT24, 24, 0 },
{ GL_DEPTH_COMPONENT32F, 32, 0 },
{ GL_DEPTH24_STENCIL8, 24, 8 },
{ GL_DEPTH32F_STENCIL8, 32, 8 },
};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(requiredDepthFormats); ++ndx)
{
glRenderbufferStorage(GL_RENDERBUFFER, requiredDepthFormats[ndx].internalFormat, 128, 128);
expectError(GL_NO_ERROR);
checkRenderbufferComponentSize(m_testCtx, *this, -1, -1, -1, -1, requiredDepthFormats[ndx].dbits, requiredDepthFormats[ndx].sbits);
}
// STENCIL_INDEX8 is required, in that case sBits >= 8
{
glRenderbufferStorage(GL_RENDERBUFFER, GL_STENCIL_INDEX8, 128, 128);
expectError(GL_NO_ERROR);
StateQueryMemoryWriteGuard<GLint> state;
glGetRenderbufferParameteriv(GL_RENDERBUFFER, GL_RENDERBUFFER_STENCIL_SIZE, &state);
if (state.verifyValidity(m_testCtx) && state < 8)
{
m_testCtx.getLog() << TestLog::Message << "// ERROR: Expected greater or equal to 8; got " << state << TestLog::EndMessage;
if (m_testCtx.getTestResult() == QP_TEST_RESULT_PASS)
m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "got invalid value");
}
}
glDeleteRenderbuffers(1, &renderbufferID);
}
};
class RboSamplesCase : public ApiCase
{
public:
RboSamplesCase (Context& context, const char* name, const char* description)
: ApiCase(context, name, description)
{
}
void test (void)
{
de::Random rnd(0xabcdef);
GLuint renderbufferID = 0;
glGenRenderbuffers(1, &renderbufferID);
glBindRenderbuffer(GL_RENDERBUFFER, renderbufferID);
expectError(GL_NO_ERROR);
checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_SAMPLES, 0);
expectError(GL_NO_ERROR);
StateQueryMemoryWriteGuard<GLint> max_samples;
glGetIntegerv(GL_MAX_SAMPLES, &max_samples);
if (!max_samples.verifyValidity(m_testCtx))
return;
// 0 samples is a special case
{
glRenderbufferStorageMultisample(GL_RENDERBUFFER, 0, GL_RGBA8, 128, 128);
expectError(GL_NO_ERROR);
checkRenderbufferParam(m_testCtx, *this, GL_RENDERBUFFER_SAMPLES, 0);
}
// test [1, n] samples
for (int samples = 1; samples <= max_samples; ++samples)
{
glRenderbufferStorageMultisample(GL_RENDERBUFFER, samples, GL_RGBA8, 128, 128);
expectError(GL_NO_ERROR);
checkRenderbufferParamGreaterOrEqual(m_testCtx, *this, GL_RENDERBUFFER_SAMPLES, samples);
}
glDeleteRenderbuffers(1, &renderbufferID);
}
};
} // anonymous
RboStateQueryTests::RboStateQueryTests (Context& context)
: TestCaseGroup(context, "rbo", "Rbo State Query tests")
{
}
void RboStateQueryTests::init (void)
{
addChild(new RboSizeCase (m_context, "renderbuffer_size", "RENDERBUFFER_WIDTH and RENDERBUFFER_HEIGHT"));
addChild(new RboInternalFormatCase (m_context, "renderbuffer_internal_format", "RENDERBUFFER_INTERNAL_FORMAT"));
addChild(new RboComponentSizeColorCase (m_context, "renderbuffer_component_size_color", "RENDERBUFFER_x_SIZE"));
addChild(new RboComponentSizeDepthCase (m_context, "renderbuffer_component_size_depth", "RENDERBUFFER_x_SIZE"));
addChild(new RboSamplesCase (m_context, "renderbuffer_samples", "RENDERBUFFER_SAMPLES"));
}
} // Functional
} // gles3
} // deqp