/*-------------------------------------------------------------------------
* 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 Pixel Buffer Object tests
*//*--------------------------------------------------------------------*/
#include "es3fPixelBufferObjectTests.hpp"
#include "tcuTexture.hpp"
#include "tcuTextureUtil.hpp"
#include "tcuImageCompare.hpp"
#include "tcuTestLog.hpp"
#include "tcuRenderTarget.hpp"
#include "gluTextureUtil.hpp"
#include "gluPixelTransfer.hpp"
#include "gluShaderProgram.hpp"
#include "deRandom.hpp"
#include "deString.h"
#include <string>
#include <sstream>
#include "glw.h"
using std::string;
using std::stringstream;
namespace deqp
{
namespace gles3
{
namespace Functional
{
namespace
{
class ReadPixelsTest : public TestCase
{
public:
struct TestSpec
{
enum FramebufferType
{
FRAMEBUFFERTYPE_NATIVE = 0,
FRAMEBUFFERTYPE_RENDERBUFFER
};
string name;
string description;
bool useColorClear;
bool renderTriangles;
FramebufferType framebufferType;
GLenum renderbufferFormat;
};
ReadPixelsTest (Context& context, const TestSpec& spec);
~ReadPixelsTest (void);
void init (void);
void deinit (void);
IterateResult iterate (void);
private:
void renderTriangle (const tcu::Vec3& a, const tcu::Vec3& b, const tcu::Vec3& c);
void clearColor (float r, float g, float b, float a);
de::Random m_random;
tcu::TestLog& m_log;
glu::ShaderProgram* m_program;
TestSpec::FramebufferType m_framebuffeType;
GLenum m_renderbufferFormat;
tcu::TextureChannelClass m_texChannelClass;
bool m_useColorClears;
bool m_renderTriangles;
GLfloat m_colorScale;
};
ReadPixelsTest::ReadPixelsTest (Context& context, const TestSpec& spec)
: TestCase (context, spec.name.c_str(), spec.description.c_str())
, m_random (deStringHash(spec.name.c_str()))
, m_log (m_testCtx.getLog())
, m_program (NULL)
, m_framebuffeType (spec.framebufferType)
, m_renderbufferFormat (spec.renderbufferFormat)
, m_texChannelClass (tcu::TEXTURECHANNELCLASS_LAST)
, m_useColorClears (spec.useColorClear)
, m_renderTriangles (spec.renderTriangles)
, m_colorScale (1.0f)
{
if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_NATIVE)
{
m_colorScale = 1.0f;
}
else if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_RENDERBUFFER)
{
m_texChannelClass = tcu::getTextureChannelClass(glu::mapGLInternalFormat(spec.renderbufferFormat).type);
switch (m_texChannelClass)
{
case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
m_colorScale = 1.0f;
break;
case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
m_colorScale = 100.0f;
break;
case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
m_colorScale = 100.0f;
break;
case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
m_colorScale = 100.0f;
break;
default:
DE_ASSERT(false);
}
}
else
DE_ASSERT(false);
}
ReadPixelsTest::~ReadPixelsTest (void)
{
}
void ReadPixelsTest::init (void)
{
// Check extensions
if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_RENDERBUFFER)
{
bool supported = false;
if (m_renderbufferFormat == GL_RGBA16F
|| m_renderbufferFormat == GL_RG16F)
{
std::istringstream extensions(std::string((const char*)glGetString(GL_EXTENSIONS)));
std::string extension;
while (std::getline(extensions, extension, ' '))
{
if (extension=="GL_EXT_color_buffer_half_float")
{
supported = true;
break;
}
if (extension=="GL_EXT_color_buffer_float")
{
supported = true;
break;
}
}
}
else if (m_renderbufferFormat == GL_RGBA32F
|| m_renderbufferFormat == GL_RG32F
|| m_renderbufferFormat == GL_R11F_G11F_B10F)
{
std::istringstream extensions(std::string((const char*)glGetString(GL_EXTENSIONS)));
std::string extension;
while (std::getline(extensions, extension, ' '))
{
if (extension=="GL_EXT_color_buffer_float")
{
supported = true;
break;
}
}
}
else
supported = true;
if (!supported)
throw tcu::NotSupportedError("Renderbuffer format not supported", "", __FILE__, __LINE__);
}
std::string outtype = "";
if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_NATIVE)
outtype = "vec4";
else if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_RENDERBUFFER)
{
switch (m_texChannelClass)
{
case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
outtype = "vec4";
break;
case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
outtype = "ivec4";
break;
case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
outtype = "uvec4";
break;
case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
outtype = "vec4";
break;
default:
DE_ASSERT(false);
}
}
else
DE_ASSERT(false);
const char* vertexShaderSource =
"#version 300 es\n"
"in mediump vec3 a_position;\n"
"in mediump vec4 a_color;\n"
"uniform mediump float u_colorScale;\n"
"out mediump vec4 v_color;\n"
"void main(void)\n"
"{\n"
"\tgl_Position = vec4(a_position, 1.0);\n"
"\tv_color = u_colorScale * a_color;\n"
"}";
stringstream fragmentShaderSource;
fragmentShaderSource <<
"#version 300 es\n"
"in mediump vec4 v_color;\n";
fragmentShaderSource << "layout (location = 0) out mediump " << outtype << " o_color;\n"
"void main(void)\n"
"{\n"
"\to_color = " << outtype << "(v_color);\n"
"}";
m_program = new glu::ShaderProgram(m_context.getRenderContext(), glu::makeVtxFragSources(vertexShaderSource, fragmentShaderSource.str()));
if (!m_program->isOk())
{
m_log << *m_program;
TCU_FAIL("Failed to compile shader");
}
}
void ReadPixelsTest::deinit (void)
{
if (m_program)
delete m_program;
m_program = NULL;
}
void ReadPixelsTest::renderTriangle (const tcu::Vec3& a, const tcu::Vec3& b, const tcu::Vec3& c)
{
float positions[3*3];
positions[0] = a.x();
positions[1] = a.y();
positions[2] = a.z();
positions[3] = b.x();
positions[4] = b.y();
positions[5] = b.z();
positions[6] = c.x();
positions[7] = c.y();
positions[8] = c.z();
float colors[] = {
1.0f, 0.0f, 0.0f, 1.0f,
0.0f, 1.0f, 0.0f, 1.0f,
0.0f, 0.0f, 1.0f, 1.0f
};
GLU_CHECK_CALL(glUseProgram(m_program->getProgram()));
GLuint coordLoc = (GLuint)-1;
GLuint colorLoc = (GLuint)-1;
GLuint colorScaleLoc = (GLuint)-1;
colorScaleLoc = glGetUniformLocation(m_program->getProgram(), "u_colorScale");
TCU_CHECK(colorScaleLoc != (GLuint)-1);
GLU_CHECK_CALL(glUniform1f(colorScaleLoc, m_colorScale));
coordLoc = glGetAttribLocation(m_program->getProgram(), "a_position");
TCU_CHECK(coordLoc != (GLuint)-1);
colorLoc = glGetAttribLocation(m_program->getProgram(), "a_color");
TCU_CHECK(colorLoc != (GLuint)-1);
GLU_CHECK_CALL(glEnableVertexAttribArray(colorLoc));
GLU_CHECK_CALL(glEnableVertexAttribArray(coordLoc));
GLU_CHECK_CALL(glVertexAttribPointer(coordLoc, 3, GL_FLOAT, GL_FALSE, 0, positions));
GLU_CHECK_CALL(glVertexAttribPointer(colorLoc, 4, GL_FLOAT, GL_FALSE, 0, colors));
GLU_CHECK_CALL(glDrawArrays(GL_TRIANGLES, 0, 3));
GLU_CHECK_CALL(glDisableVertexAttribArray(colorLoc));
GLU_CHECK_CALL(glDisableVertexAttribArray(coordLoc));
}
void ReadPixelsTest::clearColor (float r, float g, float b, float a)
{
if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_NATIVE)
{
GLU_CHECK_CALL(glClearColor(r, g, b, a));
GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
}
else if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_RENDERBUFFER)
{
switch (m_texChannelClass)
{
case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
{
GLU_CHECK_CALL(glClearColor(r, g, b, a));
GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT));
break;
}
case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
{
GLint color[4] = { (GLint)r, (GLint)g, (GLint)b, (GLint)a };
GLU_CHECK_CALL(glClearBufferiv(GL_COLOR, 0, color));
break;
}
case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
{
GLuint color[4] = { (GLuint)r, (GLuint)g, (GLuint)b, (GLuint)a };
GLU_CHECK_CALL(glClearBufferuiv(GL_COLOR, 0, color));
break;
}
case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
{
GLfloat color[4] = { (GLfloat)r, (GLfloat)g, (GLfloat)b, (GLfloat)a };
GLU_CHECK_CALL(glClearBufferfv(GL_COLOR, 0, color));
break;
}
default:
DE_ASSERT(false);
}
}
else
DE_ASSERT(false);
}
TestCase::IterateResult ReadPixelsTest::iterate(void)
{
int width = m_context.getRenderTarget().getWidth();
int height = m_context.getRenderTarget().getHeight();
GLuint framebuffer = 0;
GLuint renderbuffer = 0;
switch (m_framebuffeType)
{
case TestSpec::FRAMEBUFFERTYPE_NATIVE:
GLU_CHECK_CALL(glBindFramebuffer(GL_FRAMEBUFFER, framebuffer));
break;
case TestSpec::FRAMEBUFFERTYPE_RENDERBUFFER:
{
GLU_CHECK_CALL(glGenFramebuffers(1, &framebuffer));
GLU_CHECK_CALL(glGenRenderbuffers(1, &renderbuffer));
GLU_CHECK_CALL(glBindRenderbuffer(GL_RENDERBUFFER, renderbuffer));
GLU_CHECK_CALL(glRenderbufferStorage(GL_RENDERBUFFER, m_renderbufferFormat, width, height));
GLU_CHECK_CALL(glBindFramebuffer(GL_FRAMEBUFFER, framebuffer));
GLU_CHECK_CALL(glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, renderbuffer));
break;
}
default:
DE_ASSERT(false);
}
clearColor(m_colorScale * 0.4f, m_colorScale * 1.0f, m_colorScale * 0.5f, m_colorScale * 1.0f);
if (m_useColorClears)
{
const int maxClearCount = 10;
const int minClearCount = 6;
const int minClearSize = 15;
int clearCount = m_random.getInt(minClearCount, maxClearCount);
for (int clearNdx = 0; clearNdx < clearCount; clearNdx++)
{
int clearX = m_random.getInt(0, width - minClearSize);
int clearY = m_random.getInt(0, height - minClearSize);
int clearWidth = m_random.getInt(minClearSize, width - clearX);
int clearHeight = m_random.getInt(minClearSize, height - clearY);
float clearRed = m_colorScale * m_random.getFloat();
float clearGreen = m_colorScale * m_random.getFloat();
float clearBlue = m_colorScale * m_random.getFloat();
float clearAlpha = m_colorScale * (0.5f + 0.5f * m_random.getFloat());
GLU_CHECK_CALL(glEnable(GL_SCISSOR_TEST));
GLU_CHECK_CALL(glScissor(clearX, clearY, clearWidth, clearHeight));
clearColor(clearRed, clearGreen, clearBlue, clearAlpha);
}
GLU_CHECK_CALL(glDisable(GL_SCISSOR_TEST));
}
if (m_renderTriangles)
{
const int minTriangleCount = 4;
const int maxTriangleCount = 10;
int triangleCount = m_random.getInt(minTriangleCount, maxTriangleCount);
for (int triangleNdx = 0; triangleNdx < triangleCount; triangleNdx++)
{
float x1 = 2.0f * m_random.getFloat() - 1.0f;
float y1 = 2.0f * m_random.getFloat() - 1.0f;
float z1 = 2.0f * m_random.getFloat() - 1.0f;
float x2 = 2.0f * m_random.getFloat() - 1.0f;
float y2 = 2.0f * m_random.getFloat() - 1.0f;
float z2 = 2.0f * m_random.getFloat() - 1.0f;
float x3 = 2.0f * m_random.getFloat() - 1.0f;
float y3 = 2.0f * m_random.getFloat() - 1.0f;
float z3 = 2.0f * m_random.getFloat() - 1.0f;
renderTriangle(tcu::Vec3(x1, y1, z1), tcu::Vec3(x2, y2, z2), tcu::Vec3(x3, y3, z3));
}
}
tcu::TextureFormat readFormat;
GLenum readPixelsFormat;
GLenum readPixelsType;
bool floatCompare;
if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_NATIVE)
{
readFormat = glu::mapGLTransferFormat(GL_RGBA, GL_UNSIGNED_BYTE);
readPixelsFormat = GL_RGBA;
readPixelsType = GL_UNSIGNED_BYTE;
floatCompare = false;
}
else if (m_framebuffeType == TestSpec::FRAMEBUFFERTYPE_RENDERBUFFER)
{
switch (m_texChannelClass)
{
case tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT:
readFormat = glu::mapGLTransferFormat(GL_RGBA, GL_UNSIGNED_BYTE);
readPixelsFormat = GL_RGBA;
readPixelsType = GL_UNSIGNED_BYTE;
floatCompare = true;
break;
case tcu::TEXTURECHANNELCLASS_SIGNED_INTEGER:
readFormat = glu::mapGLTransferFormat(GL_RGBA_INTEGER, GL_INT);
readPixelsFormat = GL_RGBA_INTEGER;
readPixelsType = GL_INT;
floatCompare = false;
break;
case tcu::TEXTURECHANNELCLASS_UNSIGNED_INTEGER:
readFormat = glu::mapGLTransferFormat(GL_RGBA_INTEGER, GL_UNSIGNED_INT);
readPixelsFormat = GL_RGBA_INTEGER;
readPixelsType = GL_UNSIGNED_INT;
floatCompare = false;
break;
case tcu::TEXTURECHANNELCLASS_FLOATING_POINT:
readFormat = glu::mapGLTransferFormat(GL_RGBA, GL_FLOAT);
readPixelsFormat = GL_RGBA;
readPixelsType = GL_FLOAT;
floatCompare = true;
break;
default:
DE_ASSERT(false);
// Silence warnings
readFormat = glu::mapGLTransferFormat(GL_RGBA, GL_FLOAT);
readPixelsFormat = GL_RGBA;
readPixelsType = GL_FLOAT;
floatCompare = true;
}
}
else
{
// Silence warnings
readFormat = glu::mapGLTransferFormat(GL_RGBA, GL_FLOAT);
readPixelsFormat = GL_RGBA;
readPixelsType = GL_FLOAT;
floatCompare = true;
DE_ASSERT(false);
}
tcu::Texture2D readRefrence (readFormat, width, height);
const int readDataSize = readRefrence.getWidth() * readRefrence.getHeight() * readFormat.getPixelSize();
readRefrence.allocLevel(0);
GLuint pixelBuffer = (GLuint)-1;
GLU_CHECK_CALL(glGenBuffers(1, &pixelBuffer));
GLU_CHECK_CALL(glBindBuffer(GL_PIXEL_PACK_BUFFER, pixelBuffer));
GLU_CHECK_CALL(glBufferData(GL_PIXEL_PACK_BUFFER, readDataSize, NULL, GL_STREAM_READ));
GLU_CHECK_CALL(glReadPixels(0, 0, width, height, readPixelsFormat, readPixelsType, 0));
const deUint8* bufferData = (const deUint8*)glMapBufferRange(GL_PIXEL_PACK_BUFFER, 0, readDataSize, GL_MAP_READ_BIT);
GLU_CHECK_MSG("glMapBufferRange() failed");
tcu::ConstPixelBufferAccess readResult(readFormat, width, height, 1, bufferData);
GLU_CHECK_CALL(glBindBuffer(GL_PIXEL_PACK_BUFFER, 0));
GLU_CHECK_CALL(glReadPixels(0, 0, width, height, readPixelsFormat, readPixelsType, readRefrence.getLevel(0).getDataPtr()));
if (framebuffer)
GLU_CHECK_CALL(glDeleteFramebuffers(1, &framebuffer));
if (renderbuffer)
GLU_CHECK_CALL(glDeleteRenderbuffers(1, &renderbuffer));
bool isOk = false;
if (floatCompare)
{
const tcu::IVec4 formatBitDepths = tcu::getTextureFormatBitDepth(readFormat);
const float redThreshold = 2.0f / (float)(1 << deMin32(m_context.getRenderTarget().getPixelFormat().redBits, formatBitDepths.x()));
const float greenThreshold = 2.0f / (float)(1 << deMin32(m_context.getRenderTarget().getPixelFormat().greenBits, formatBitDepths.y()));
const float blueThreshold = 2.0f / (float)(1 << deMin32(m_context.getRenderTarget().getPixelFormat().blueBits, formatBitDepths.z()));
const float alphaThreshold = 2.0f / (float)(1 << deMin32(m_context.getRenderTarget().getPixelFormat().alphaBits, formatBitDepths.w()));
isOk = tcu::floatThresholdCompare(m_log, "Result comparision", "Result of read pixels to memory compared with result of read pixels to buffer", readRefrence.getLevel(0), readResult, tcu::Vec4(redThreshold, greenThreshold, blueThreshold, alphaThreshold), tcu::COMPARE_LOG_RESULT);
}
else
{
isOk = tcu::intThresholdCompare(m_log, "Result comparision", "Result of read pixels to memory compared with result of read pixels to buffer", readRefrence.getLevel(0), readResult, tcu::UVec4(0, 0, 0, 0), tcu::COMPARE_LOG_RESULT);
}
GLU_CHECK_CALL(glBindBuffer(GL_PIXEL_PACK_BUFFER, pixelBuffer));
GLU_CHECK_CALL(glUnmapBuffer(GL_PIXEL_PACK_BUFFER));
GLU_CHECK_CALL(glDeleteBuffers(1, &pixelBuffer));
if (isOk)
{
m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
return STOP;
}
else
{
m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
return STOP;
}
}
} // anonymous
PixelBufferObjectTests::PixelBufferObjectTests (Context& context)
: TestCaseGroup (context, "pbo", "Pixel buffer objects tests")
{
}
PixelBufferObjectTests::~PixelBufferObjectTests (void)
{
}
void PixelBufferObjectTests::init (void)
{
TestCaseGroup* nativeFramebufferGroup = new TestCaseGroup(m_context, "native", "Tests with reading from native framebuffer");
ReadPixelsTest::TestSpec nativeFramebufferTests[] = {
{
"clears",
"Simple read pixels test with color clears",
true,
false,
ReadPixelsTest::TestSpec::FRAMEBUFFERTYPE_NATIVE,
GL_NONE
},
{
"triangles",
"Simple read pixels test rendering triangles",
false,
true,
ReadPixelsTest::TestSpec::FRAMEBUFFERTYPE_NATIVE,
GL_NONE
}
};
for (int testNdx = 0; testNdx < DE_LENGTH_OF_ARRAY(nativeFramebufferTests); testNdx++)
{
nativeFramebufferGroup->addChild(new ReadPixelsTest(m_context, nativeFramebufferTests[testNdx]));
}
addChild(nativeFramebufferGroup);
TestCaseGroup* renderbufferGroup = new TestCaseGroup(m_context, "renderbuffer", "Tests with reading from renderbuffer");
GLenum renderbufferFormats[] = {
GL_RGBA8,
GL_RGBA8I,
GL_RGBA8UI,
GL_RGBA16F,
GL_RGBA16I,
GL_RGBA16UI,
GL_RGBA32F,
GL_RGBA32I,
GL_RGBA32UI,
GL_SRGB8_ALPHA8,
GL_RGB10_A2,
GL_RGB10_A2UI,
GL_RGBA4,
GL_RGB5_A1,
GL_RGB8,
GL_RGB565,
GL_R11F_G11F_B10F,
GL_RG8,
GL_RG8I,
GL_RG8UI,
GL_RG16F,
GL_RG16I,
GL_RG16UI,
GL_RG32F,
GL_RG32I,
GL_RG32UI
};
const char* renderbufferFormatsStr[] = {
"rgba8",
"rgba8i",
"rgba8ui",
"rgba16f",
"rgba16i",
"rgba16ui",
"rgba32f",
"rgba32i",
"rgba32ui",
"srgb8_alpha8",
"rgb10_a2",
"rgb10_a2ui",
"rgba4",
"rgb5_a1",
"rgb8",
"rgb565",
"r11f_g11f_b10f",
"rg8",
"rg8i",
"rg8ui",
"rg16f",
"rg16i",
"rg16ui",
"rg32f",
"rg32i",
"rg32ui"
};
DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(renderbufferFormatsStr) == DE_LENGTH_OF_ARRAY(renderbufferFormats));
for (int formatNdx = 0; formatNdx < DE_LENGTH_OF_ARRAY(renderbufferFormats); formatNdx++)
{
for (int trianglesClears = 0; trianglesClears < 2; trianglesClears++)
{
ReadPixelsTest::TestSpec testSpec;
testSpec.name = string(renderbufferFormatsStr [formatNdx]) + "_" + (trianglesClears == 0 ? "triangles" : "clears"),
testSpec.description = testSpec.name;
testSpec.useColorClear = trianglesClears == 1,
testSpec.renderTriangles = trianglesClears == 0,
testSpec.framebufferType = ReadPixelsTest::TestSpec::FRAMEBUFFERTYPE_RENDERBUFFER,
testSpec.renderbufferFormat = renderbufferFormats[formatNdx];
renderbufferGroup->addChild(new ReadPixelsTest(m_context, testSpec));
}
}
addChild(renderbufferGroup);
}
} // Functional
} // gles3
} // deqp