/*------------------------------------------------------------------------- * drawElements Quality Program OpenGL (ES) 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 Buffer test utilities. *//*--------------------------------------------------------------------*/ #include "glsBufferTestUtil.hpp" #include "tcuRandomValueIterator.hpp" #include "tcuSurface.hpp" #include "tcuImageCompare.hpp" #include "tcuVector.hpp" #include "tcuFormatUtil.hpp" #include "tcuTextureUtil.hpp" #include "tcuRenderTarget.hpp" #include "tcuTestLog.hpp" #include "gluPixelTransfer.hpp" #include "gluRenderContext.hpp" #include "gluStrUtil.hpp" #include "gluShaderProgram.hpp" #include "deMemory.h" #include "deStringUtil.hpp" #include "deArrayUtil.hpp" #include <algorithm> #include "glwEnums.hpp" #include "glwFunctions.hpp" namespace deqp { namespace gls { namespace BufferTestUtil { enum { VERIFY_QUAD_SIZE = 8, //!< Quad size in VertexArrayVerifier MAX_LINES_PER_INDEX_ARRAY_DRAW = 128, //!< Maximum number of lines per one draw in IndexArrayVerifier INDEX_ARRAY_DRAW_VIEWPORT_WIDTH = 128, INDEX_ARRAY_DRAW_VIEWPORT_HEIGHT = 128 }; using tcu::TestLog; using std::vector; using std::string; using std::set; // Helper functions. void fillWithRandomBytes (deUint8* ptr, int numBytes, deUint32 seed) { std::copy(tcu::RandomValueIterator<deUint8>::begin(seed, numBytes), tcu::RandomValueIterator<deUint8>::end(), ptr); } bool compareByteArrays (tcu::TestLog& log, const deUint8* resPtr, const deUint8* refPtr, int numBytes) { bool isOk = true; const int maxSpanLen = 8; const int maxDiffSpans = 4; int numDiffSpans = 0; int diffSpanStart = -1; int ndx = 0; log << TestLog::Section("Verify", "Verification result"); for (;ndx < numBytes; ndx++) { if (resPtr[ndx] != refPtr[ndx]) { if (diffSpanStart < 0) diffSpanStart = ndx; isOk = false; } else if (diffSpanStart >= 0) { if (numDiffSpans < maxDiffSpans) { int len = ndx-diffSpanStart; int printLen = de::min(len, maxSpanLen); log << TestLog::Message << len << " byte difference at offset " << diffSpanStart << "\n" << " expected " << tcu::formatArray(tcu::Format::HexIterator<deUint8>(refPtr+diffSpanStart), tcu::Format::HexIterator<deUint8>(refPtr+diffSpanStart+printLen)) << "\n" << " got " << tcu::formatArray(tcu::Format::HexIterator<deUint8>(resPtr+diffSpanStart), tcu::Format::HexIterator<deUint8>(resPtr+diffSpanStart+printLen)) << TestLog::EndMessage; } else log << TestLog::Message << "(output too long, truncated)" << TestLog::EndMessage; numDiffSpans += 1; diffSpanStart = -1; } } if (diffSpanStart >= 0) { if (numDiffSpans < maxDiffSpans) { int len = ndx-diffSpanStart; int printLen = de::min(len, maxSpanLen); log << TestLog::Message << len << " byte difference at offset " << diffSpanStart << "\n" << " expected " << tcu::formatArray(tcu::Format::HexIterator<deUint8>(refPtr+diffSpanStart), tcu::Format::HexIterator<deUint8>(refPtr+diffSpanStart+printLen)) << "\n" << " got " << tcu::formatArray(tcu::Format::HexIterator<deUint8>(resPtr+diffSpanStart), tcu::Format::HexIterator<deUint8>(resPtr+diffSpanStart+printLen)) << TestLog::EndMessage; } else log << TestLog::Message << "(output too long, truncated)" << TestLog::EndMessage; } log << TestLog::Message << (isOk ? "Verification passed." : "Verification FAILED!") << TestLog::EndMessage; log << TestLog::EndSection; return isOk; } const char* getBufferTargetName (deUint32 target) { switch (target) { case GL_ARRAY_BUFFER: return "array"; case GL_COPY_READ_BUFFER: return "copy_read"; case GL_COPY_WRITE_BUFFER: return "copy_write"; case GL_ELEMENT_ARRAY_BUFFER: return "element_array"; case GL_PIXEL_PACK_BUFFER: return "pixel_pack"; case GL_PIXEL_UNPACK_BUFFER: return "pixel_unpack"; case GL_TEXTURE_BUFFER: return "texture"; case GL_TRANSFORM_FEEDBACK_BUFFER: return "transform_feedback"; case GL_UNIFORM_BUFFER: return "uniform"; default: DE_ASSERT(false); return DE_NULL; } } const char* getUsageHintName (deUint32 hint) { switch (hint) { case GL_STREAM_DRAW: return "stream_draw"; case GL_STREAM_READ: return "stream_read"; case GL_STREAM_COPY: return "stream_copy"; case GL_STATIC_DRAW: return "static_draw"; case GL_STATIC_READ: return "static_read"; case GL_STATIC_COPY: return "static_copy"; case GL_DYNAMIC_DRAW: return "dynamic_draw"; case GL_DYNAMIC_READ: return "dynamic_read"; case GL_DYNAMIC_COPY: return "dynamic_copy"; default: DE_ASSERT(false); return DE_NULL; } } // BufferCase BufferCase::BufferCase (tcu::TestContext& testCtx, glu::RenderContext& renderCtx, const char* name, const char* description) : TestCase (testCtx, name, description) , CallLogWrapper (renderCtx.getFunctions(), testCtx.getLog()) , m_renderCtx (renderCtx) { } BufferCase::~BufferCase (void) { enableLogging(false); BufferCase::deinit(); } void BufferCase::init (void) { enableLogging(true); } void BufferCase::deinit (void) { for (set<deUint32>::const_iterator bufIter = m_allocatedBuffers.begin(); bufIter != m_allocatedBuffers.end(); bufIter++) glDeleteBuffers(1, &(*bufIter)); } deUint32 BufferCase::genBuffer (void) { deUint32 buf = 0; glGenBuffers(1, &buf); if (buf != 0) { try { m_allocatedBuffers.insert(buf); } catch (const std::exception&) { glDeleteBuffers(1, &buf); throw; } } return buf; } void BufferCase::deleteBuffer (deUint32 buffer) { glDeleteBuffers(1, &buffer); m_allocatedBuffers.erase(buffer); } void BufferCase::checkError (void) { glw::GLenum err = glGetError(); if (err != GL_NO_ERROR) throw tcu::TestError(string("Got ") + glu::getErrorStr(err).toString()); } // ReferenceBuffer void ReferenceBuffer::setSize (int numBytes) { m_data.resize(numBytes); } void ReferenceBuffer::setData (int numBytes, const deUint8* bytes) { m_data.resize(numBytes); std::copy(bytes, bytes+numBytes, m_data.begin()); } void ReferenceBuffer::setSubData (int offset, int numBytes, const deUint8* bytes) { DE_ASSERT(de::inBounds(offset, 0, (int)m_data.size()) && de::inRange(offset+numBytes, offset, (int)m_data.size())); std::copy(bytes, bytes+numBytes, m_data.begin()+offset); } // BufferWriterBase BufferWriterBase::BufferWriterBase (glu::RenderContext& renderCtx, tcu::TestLog& log) : CallLogWrapper (renderCtx.getFunctions(), log) , m_renderCtx (renderCtx) { enableLogging(true); } void BufferWriterBase::write (deUint32 buffer, int offset, int numBytes, const deUint8* bytes, deUint32 targetHint) { DE_UNREF(targetHint); write(buffer, offset, numBytes, bytes); } // BufferWriter BufferWriter::BufferWriter (glu::RenderContext& renderCtx, tcu::TestLog& log, WriteType writeType) : m_writer(DE_NULL) { switch (writeType) { case WRITE_BUFFER_SUB_DATA: m_writer = new BufferSubDataWriter (renderCtx, log); break; case WRITE_BUFFER_WRITE_MAP: m_writer = new BufferWriteMapWriter (renderCtx, log); break; default: TCU_FAIL("Unsupported writer"); } } BufferWriter::~BufferWriter (void) { delete m_writer; } void BufferWriter::write (deUint32 buffer, int offset, int numBytes, const deUint8* bytes) { DE_ASSERT(numBytes >= getMinSize()); DE_ASSERT(offset%getAlignment() == 0); DE_ASSERT((offset+numBytes)%getAlignment() == 0); return m_writer->write(buffer, offset, numBytes, bytes); } void BufferWriter::write (deUint32 buffer, int offset, int numBytes, const deUint8* bytes, deUint32 targetHint) { DE_ASSERT(numBytes >= getMinSize()); DE_ASSERT(offset%getAlignment() == 0); DE_ASSERT((offset+numBytes)%getAlignment() == 0); return m_writer->write(buffer, offset, numBytes, bytes, targetHint); } // BufferSubDataWriter void BufferSubDataWriter::write (deUint32 buffer, int offset, int numBytes, const deUint8* bytes) { write(buffer, offset, numBytes, bytes, GL_ARRAY_BUFFER); } void BufferSubDataWriter::write (deUint32 buffer, int offset, int numBytes, const deUint8* bytes, deUint32 target) { glBindBuffer(target, buffer); glBufferSubData(target, offset, numBytes, bytes); glBindBuffer(target, 0); GLU_CHECK(); } // BufferWriteMapWriter void BufferWriteMapWriter::write (deUint32 buffer, int offset, int numBytes, const deUint8* bytes) { write(buffer, offset, numBytes, bytes, GL_ARRAY_BUFFER); } void BufferWriteMapWriter::write (deUint32 buffer, int offset, int numBytes, const deUint8* bytes, deUint32 target) { glBindBuffer(target, buffer); void* ptr = glMapBufferRange(target, offset, numBytes, GL_MAP_WRITE_BIT); GLU_CHECK_MSG("glMapBufferRange"); deMemcpy(ptr, bytes, numBytes); glUnmapBuffer(target); glBindBuffer(target, 0); GLU_CHECK(); } // BufferVerifierBase BufferVerifierBase::BufferVerifierBase (glu::RenderContext& renderCtx, tcu::TestLog& log) : CallLogWrapper (renderCtx.getFunctions(), log) , m_renderCtx (renderCtx) , m_log (log) { enableLogging(true); } bool BufferVerifierBase::verify (deUint32 buffer, const deUint8* reference, int offset, int numBytes, deUint32 targetHint) { DE_UNREF(targetHint); return verify(buffer, reference, offset, numBytes); } // BufferVerifier BufferVerifier::BufferVerifier (glu::RenderContext& renderCtx, tcu::TestLog& log, VerifyType verifyType) : m_verifier(DE_NULL) { switch (verifyType) { case VERIFY_AS_VERTEX_ARRAY: m_verifier = new VertexArrayVerifier(renderCtx, log); break; case VERIFY_AS_INDEX_ARRAY: m_verifier = new IndexArrayVerifier (renderCtx, log); break; case VERIFY_BUFFER_READ_MAP: m_verifier = new BufferMapVerifier (renderCtx, log); break; default: TCU_FAIL("Unsupported verifier"); } } BufferVerifier::~BufferVerifier (void) { delete m_verifier; } bool BufferVerifier::verify (deUint32 buffer, const deUint8* reference, int offset, int numBytes) { DE_ASSERT(numBytes >= getMinSize()); DE_ASSERT(offset%getAlignment() == 0); DE_ASSERT((offset+numBytes)%getAlignment() == 0); return m_verifier->verify(buffer, reference, offset, numBytes); } bool BufferVerifier::verify (deUint32 buffer, const deUint8* reference, int offset, int numBytes, deUint32 targetHint) { DE_ASSERT(numBytes >= getMinSize()); DE_ASSERT(offset%getAlignment() == 0); DE_ASSERT((offset+numBytes)%getAlignment() == 0); return m_verifier->verify(buffer, reference, offset, numBytes, targetHint); } // BufferMapVerifier bool BufferMapVerifier::verify (deUint32 buffer, const deUint8* reference, int offset, int numBytes) { return verify(buffer, reference, offset, numBytes, GL_ARRAY_BUFFER); } bool BufferMapVerifier::verify (deUint32 buffer, const deUint8* reference, int offset, int numBytes, deUint32 target) { const deUint8* mapPtr = DE_NULL; bool isOk = false; glBindBuffer(target, buffer); mapPtr = (const deUint8*)glMapBufferRange(target, offset, numBytes, GL_MAP_READ_BIT); GLU_CHECK_MSG("glMapBufferRange"); TCU_CHECK(mapPtr); isOk = compareByteArrays(m_log, mapPtr, reference+offset, numBytes); glUnmapBuffer(target); GLU_CHECK_MSG("glUnmapBuffer"); glBindBuffer(target, 0); return isOk; } // VertexArrayVerifier VertexArrayVerifier::VertexArrayVerifier (glu::RenderContext& renderCtx, tcu::TestLog& log) : BufferVerifierBase (renderCtx, log) , m_program (DE_NULL) , m_posLoc (0) , m_byteVecLoc (0) , m_vao (0) { const glu::ContextType ctxType = renderCtx.getType(); const glu::GLSLVersion glslVersion = glu::isContextTypeES(ctxType) ? glu::GLSL_VERSION_300_ES : glu::GLSL_VERSION_330; DE_ASSERT(glu::isGLSLVersionSupported(ctxType, glslVersion)); m_program = new glu::ShaderProgram(m_renderCtx, glu::makeVtxFragSources( string(glu::getGLSLVersionDeclaration(glslVersion)) + "\n" "in highp vec2 a_position;\n" "in mediump vec3 a_byteVec;\n" "out mediump vec3 v_byteVec;\n" "void main (void)\n" "{\n" " gl_Position = vec4(a_position, 0.0, 1.0);\n" " v_byteVec = a_byteVec;\n" "}\n", string(glu::getGLSLVersionDeclaration(glslVersion)) + "\n" "in mediump vec3 v_byteVec;\n" "layout(location = 0) out mediump vec4 o_color;\n" "void main (void)\n" "{\n" " o_color = vec4(v_byteVec, 1.0);\n" "}\n")); if (!m_program->isOk()) { m_log << *m_program; delete m_program; TCU_FAIL("Compile failed"); } const glw::Functions& gl = m_renderCtx.getFunctions(); m_posLoc = gl.getAttribLocation(m_program->getProgram(), "a_position"); m_byteVecLoc = gl.getAttribLocation(m_program->getProgram(), "a_byteVec"); gl.genVertexArrays(1, &m_vao); gl.genBuffers(1, &m_positionBuf); gl.genBuffers(1, &m_indexBuf); GLU_EXPECT_NO_ERROR(gl.getError(), "Initialization failed"); } VertexArrayVerifier::~VertexArrayVerifier (void) { const glw::Functions& gl = m_renderCtx.getFunctions(); if (m_vao) gl.deleteVertexArrays(1, &m_vao); if (m_positionBuf) gl.deleteBuffers(1, &m_positionBuf); if (m_indexBuf) gl.deleteBuffers(1, &m_indexBuf); delete m_program; } static void computePositions (vector<tcu::Vec2>& positions, int gridSizeX, int gridSizeY) { positions.resize(gridSizeX*gridSizeY*4); for (int y = 0; y < gridSizeY; y++) for (int x = 0; x < gridSizeX; x++) { float sx0 = (float)(x+0) / (float)gridSizeX; float sy0 = (float)(y+0) / (float)gridSizeY; float sx1 = (float)(x+1) / (float)gridSizeX; float sy1 = (float)(y+1) / (float)gridSizeY; float fx0 = 2.0f * sx0 - 1.0f; float fy0 = 2.0f * sy0 - 1.0f; float fx1 = 2.0f * sx1 - 1.0f; float fy1 = 2.0f * sy1 - 1.0f; int baseNdx = (y * gridSizeX + x)*4; positions[baseNdx+0] = tcu::Vec2(fx0, fy0); positions[baseNdx+1] = tcu::Vec2(fx0, fy1); positions[baseNdx+2] = tcu::Vec2(fx1, fy0); positions[baseNdx+3] = tcu::Vec2(fx1, fy1); } } static void computeIndices (vector<deUint16>& indices, int gridSizeX, int gridSizeY) { indices.resize(3 * 2 * gridSizeX * gridSizeY); for (int quadNdx = 0; quadNdx < gridSizeX*gridSizeY; quadNdx++) { int v00 = quadNdx*4 + 0; int v01 = quadNdx*4 + 1; int v10 = quadNdx*4 + 2; int v11 = quadNdx*4 + 3; DE_ASSERT(v11 < (1<<16)); indices[quadNdx*6 + 0] = (deUint16)v10; indices[quadNdx*6 + 1] = (deUint16)v00; indices[quadNdx*6 + 2] = (deUint16)v01; indices[quadNdx*6 + 3] = (deUint16)v10; indices[quadNdx*6 + 4] = (deUint16)v01; indices[quadNdx*6 + 5] = (deUint16)v11; } } static inline tcu::Vec4 fetchVtxColor (const deUint8* ptr, int vtxNdx) { return tcu::RGBA(*(ptr + vtxNdx*3 + 0), *(ptr + vtxNdx*3 + 1), *(ptr + vtxNdx*3 + 2), 255).toVec(); } static void renderQuadGridReference (tcu::Surface& dst, int numQuads, int rowLength, const deUint8* inPtr) { using tcu::Vec4; dst.setSize(rowLength*VERIFY_QUAD_SIZE, (numQuads/rowLength + (numQuads%rowLength != 0 ? 1 : 0))*VERIFY_QUAD_SIZE); tcu::PixelBufferAccess dstAccess = dst.getAccess(); tcu::clear(dstAccess, tcu::IVec4(0, 0, 0, 0xff)); for (int quadNdx = 0; quadNdx < numQuads; quadNdx++) { int x0 = (quadNdx%rowLength)*VERIFY_QUAD_SIZE; int y0 = (quadNdx/rowLength)*VERIFY_QUAD_SIZE; Vec4 v00 = fetchVtxColor(inPtr, quadNdx*4 + 0); Vec4 v10 = fetchVtxColor(inPtr, quadNdx*4 + 1); Vec4 v01 = fetchVtxColor(inPtr, quadNdx*4 + 2); Vec4 v11 = fetchVtxColor(inPtr, quadNdx*4 + 3); for (int y = 0; y < VERIFY_QUAD_SIZE; y++) for (int x = 0; x < VERIFY_QUAD_SIZE; x++) { float fx = ((float)x+0.5f) / (float)VERIFY_QUAD_SIZE; float fy = ((float)y+0.5f) / (float)VERIFY_QUAD_SIZE; bool tri = fx + fy <= 1.0f; float tx = tri ? fx : (1.0f-fx); float ty = tri ? fy : (1.0f-fy); const Vec4& t0 = tri ? v00 : v11; const Vec4& t1 = tri ? v01 : v10; const Vec4& t2 = tri ? v10 : v01; Vec4 color = t0 + (t1-t0)*tx + (t2-t0)*ty; dstAccess.setPixel(color, x0+x, y0+y); } } } bool VertexArrayVerifier::verify (deUint32 buffer, const deUint8* refPtr, int offset, int numBytes) { const tcu::RenderTarget& renderTarget = m_renderCtx.getRenderTarget(); const int numBytesInVtx = 3; const int numBytesInQuad = numBytesInVtx*4; int maxQuadsX = de::min(128, renderTarget.getWidth() / VERIFY_QUAD_SIZE); int maxQuadsY = de::min(128, renderTarget.getHeight() / VERIFY_QUAD_SIZE); int maxQuadsPerBatch = maxQuadsX*maxQuadsY; int numVerified = 0; deUint32 program = m_program->getProgram(); tcu::RGBA threshold = renderTarget.getPixelFormat().getColorThreshold() + tcu::RGBA(3,3,3,3); bool isOk = true; vector<tcu::Vec2> positions; vector<deUint16> indices; tcu::Surface rendered; tcu::Surface reference; DE_ASSERT(numBytes >= numBytesInQuad); // Can't render full quad with smaller buffers. computePositions(positions, maxQuadsX, maxQuadsY); computeIndices(indices, maxQuadsX, maxQuadsY); // Reset buffer bindings. glBindBuffer (GL_PIXEL_PACK_BUFFER, 0); // Setup rendering state. glViewport (0, 0, maxQuadsX*VERIFY_QUAD_SIZE, maxQuadsY*VERIFY_QUAD_SIZE); glClearColor (0.0f, 0.0f, 0.0f, 1.0f); glUseProgram (program); glBindVertexArray (m_vao); // Upload positions glBindBuffer (GL_ARRAY_BUFFER, m_positionBuf); glBufferData (GL_ARRAY_BUFFER, (glw::GLsizeiptr)(positions.size()*sizeof(positions[0])), &positions[0], GL_STATIC_DRAW); glEnableVertexAttribArray (m_posLoc); glVertexAttribPointer (m_posLoc, 2, GL_FLOAT, GL_FALSE, 0, DE_NULL); // Upload indices glBindBuffer (GL_ELEMENT_ARRAY_BUFFER, m_indexBuf); glBufferData (GL_ELEMENT_ARRAY_BUFFER, (glw::GLsizeiptr)(indices.size()*sizeof(indices[0])), &indices[0], GL_STATIC_DRAW); glEnableVertexAttribArray (m_byteVecLoc); glBindBuffer (GL_ARRAY_BUFFER, buffer); while (numVerified < numBytes) { int numRemaining = numBytes-numVerified; bool isLeftoverBatch = numRemaining < numBytesInQuad; int numBytesToVerify = isLeftoverBatch ? numBytesInQuad : de::min(maxQuadsPerBatch*numBytesInQuad, numRemaining - numRemaining%numBytesInQuad); int curOffset = isLeftoverBatch ? (numBytes-numBytesInQuad) : numVerified; int numQuads = numBytesToVerify/numBytesInQuad; int numCols = de::min(maxQuadsX, numQuads); int numRows = numQuads/maxQuadsX + (numQuads%maxQuadsX != 0 ? 1 : 0); string imageSetDesc = string("Bytes ") + de::toString(offset+curOffset) + " to " + de::toString(offset+curOffset+numBytesToVerify-1); DE_ASSERT(numBytesToVerify > 0 && numBytesToVerify%numBytesInQuad == 0); DE_ASSERT(de::inBounds(curOffset, 0, numBytes)); DE_ASSERT(de::inRange(curOffset+numBytesToVerify, curOffset, numBytes)); // Render batch. glClear (GL_COLOR_BUFFER_BIT); glVertexAttribPointer (m_byteVecLoc, 3, GL_UNSIGNED_BYTE, GL_TRUE, 0, (const glw::GLvoid*)(deUintptr)(offset + curOffset)); glDrawElements (GL_TRIANGLES, numQuads*6, GL_UNSIGNED_SHORT, DE_NULL); renderQuadGridReference(reference, numQuads, numCols, refPtr + offset + curOffset); rendered.setSize(numCols*VERIFY_QUAD_SIZE, numRows*VERIFY_QUAD_SIZE); glu::readPixels(m_renderCtx, 0, 0, rendered.getAccess()); if (!tcu::pixelThresholdCompare(m_log, "RenderResult", imageSetDesc.c_str(), reference, rendered, threshold, tcu::COMPARE_LOG_RESULT)) { isOk = false; break; } numVerified += isLeftoverBatch ? numRemaining : numBytesToVerify; } glBindVertexArray(0); return isOk; } // IndexArrayVerifier IndexArrayVerifier::IndexArrayVerifier (glu::RenderContext& renderCtx, tcu::TestLog& log) : BufferVerifierBase (renderCtx, log) , m_program (DE_NULL) , m_posLoc (0) , m_colorLoc (0) { const glu::ContextType ctxType = renderCtx.getType(); const glu::GLSLVersion glslVersion = glu::isContextTypeES(ctxType) ? glu::GLSL_VERSION_300_ES : glu::GLSL_VERSION_330; DE_ASSERT(glu::isGLSLVersionSupported(ctxType, glslVersion)); m_program = new glu::ShaderProgram(m_renderCtx, glu::makeVtxFragSources( string(glu::getGLSLVersionDeclaration(glslVersion)) + "\n" "in highp vec2 a_position;\n" "in mediump vec3 a_color;\n" "out mediump vec3 v_color;\n" "void main (void)\n" "{\n" " gl_Position = vec4(a_position, 0.0, 1.0);\n" " v_color = a_color;\n" "}\n", string(glu::getGLSLVersionDeclaration(glslVersion)) + "\n" "in mediump vec3 v_color;\n" "layout(location = 0) out mediump vec4 o_color;\n" "void main (void)\n" "{\n" " o_color = vec4(v_color, 1.0);\n" "}\n")); if (!m_program->isOk()) { m_log << *m_program; delete m_program; TCU_FAIL("Compile failed"); } const glw::Functions& gl = m_renderCtx.getFunctions(); m_posLoc = gl.getAttribLocation(m_program->getProgram(), "a_position"); m_colorLoc = gl.getAttribLocation(m_program->getProgram(), "a_color"); gl.genVertexArrays(1, &m_vao); gl.genBuffers(1, &m_positionBuf); gl.genBuffers(1, &m_colorBuf); GLU_EXPECT_NO_ERROR(gl.getError(), "Initialization failed"); } IndexArrayVerifier::~IndexArrayVerifier (void) { const glw::Functions& gl = m_renderCtx.getFunctions(); if (m_vao) gl.deleteVertexArrays(1, &m_vao); if (m_positionBuf) gl.deleteBuffers(1, &m_positionBuf); if (m_colorBuf) gl.deleteBuffers(1, &m_colorBuf); delete m_program; } static void computeIndexVerifierPositions (std::vector<tcu::Vec2>& dst) { const int numPosX = 16; const int numPosY = 16; dst.resize(numPosX*numPosY); for (int y = 0; y < numPosY; y++) { for (int x = 0; x < numPosX; x++) { float xf = float(x) / float(numPosX-1); float yf = float(y) / float(numPosY-1); dst[y*numPosX + x] = tcu::Vec2(2.0f*xf - 1.0f, 2.0f*yf - 1.0f); } } } static void computeIndexVerifierColors (std::vector<tcu::Vec3>& dst) { const int numColors = 256; const float minVal = 0.1f; const float maxVal = 0.5f; de::Random rnd (0xabc231); dst.resize(numColors); for (std::vector<tcu::Vec3>::iterator i = dst.begin(); i != dst.end(); ++i) { i->x() = rnd.getFloat(minVal, maxVal); i->y() = rnd.getFloat(minVal, maxVal); i->z() = rnd.getFloat(minVal, maxVal); } } template<typename T> static void execVertexFetch (T* dst, const T* src, const deUint8* indices, int numIndices) { for (int i = 0; i < numIndices; ++i) dst[i] = src[indices[i]]; } bool IndexArrayVerifier::verify (deUint32 buffer, const deUint8* refPtr, int offset, int numBytes) { const tcu::RenderTarget& renderTarget = m_renderCtx.getRenderTarget(); const int viewportW = de::min<int>(INDEX_ARRAY_DRAW_VIEWPORT_WIDTH, renderTarget.getWidth()); const int viewportH = de::min<int>(INDEX_ARRAY_DRAW_VIEWPORT_HEIGHT, renderTarget.getHeight()); const int minBytesPerBatch = 2; const tcu::RGBA threshold (0,0,0,0); std::vector<tcu::Vec2> positions; std::vector<tcu::Vec3> colors; std::vector<tcu::Vec2> fetchedPos (MAX_LINES_PER_INDEX_ARRAY_DRAW+1); std::vector<tcu::Vec3> fetchedColor (MAX_LINES_PER_INDEX_ARRAY_DRAW+1); tcu::Surface indexBufferImg (viewportW, viewportH); tcu::Surface referenceImg (viewportW, viewportH); int numVerified = 0; bool isOk = true; DE_STATIC_ASSERT(sizeof(tcu::Vec2) == sizeof(float)*2); DE_STATIC_ASSERT(sizeof(tcu::Vec3) == sizeof(float)*3); computeIndexVerifierPositions(positions); computeIndexVerifierColors(colors); // Reset buffer bindings. glBindVertexArray (m_vao); glBindBuffer (GL_PIXEL_PACK_BUFFER, 0); glBindBuffer (GL_ELEMENT_ARRAY_BUFFER, buffer); // Setup rendering state. glViewport (0, 0, viewportW, viewportH); glClearColor (0.0f, 0.0f, 0.0f, 1.0f); glUseProgram (m_program->getProgram()); glEnableVertexAttribArray (m_posLoc); glEnableVertexAttribArray (m_colorLoc); glEnable (GL_BLEND); glBlendFunc (GL_ONE, GL_ONE); glBlendEquation (GL_FUNC_ADD); while (numVerified < numBytes) { int numRemaining = numBytes-numVerified; bool isLeftoverBatch = numRemaining < minBytesPerBatch; int numBytesToVerify = isLeftoverBatch ? minBytesPerBatch : de::min(MAX_LINES_PER_INDEX_ARRAY_DRAW+1, numRemaining); int curOffset = isLeftoverBatch ? (numBytes-minBytesPerBatch) : numVerified; string imageSetDesc = string("Bytes ") + de::toString(offset+curOffset) + " to " + de::toString(offset+curOffset+numBytesToVerify-1); // Step 1: Render using index buffer. glClear (GL_COLOR_BUFFER_BIT); glBindBuffer (GL_ARRAY_BUFFER, m_positionBuf); glBufferData (GL_ARRAY_BUFFER, (glw::GLsizeiptr)(positions.size()*sizeof(positions[0])), &positions[0], GL_STREAM_DRAW); glVertexAttribPointer (m_posLoc, 2, GL_FLOAT, GL_FALSE, 0, DE_NULL); glBindBuffer (GL_ARRAY_BUFFER, m_colorBuf); glBufferData (GL_ARRAY_BUFFER, (glw::GLsizeiptr)(colors.size()*sizeof(colors[0])), &colors[0], GL_STREAM_DRAW); glVertexAttribPointer (m_colorLoc, 3, GL_FLOAT, GL_FALSE, 0, DE_NULL); glDrawElements (GL_LINE_STRIP, numBytesToVerify, GL_UNSIGNED_BYTE, (void*)(deUintptr)(offset+curOffset)); glu::readPixels (m_renderCtx, 0, 0, indexBufferImg.getAccess()); // Step 2: Do manual fetch and render without index buffer. execVertexFetch(&fetchedPos[0], &positions[0], refPtr+offset+curOffset, numBytesToVerify); execVertexFetch(&fetchedColor[0], &colors[0], refPtr+offset+curOffset, numBytesToVerify); glClear (GL_COLOR_BUFFER_BIT); glBindBuffer (GL_ARRAY_BUFFER, m_positionBuf); glBufferData (GL_ARRAY_BUFFER, (glw::GLsizeiptr)(fetchedPos.size()*sizeof(fetchedPos[0])), &fetchedPos[0], GL_STREAM_DRAW); glVertexAttribPointer (m_posLoc, 2, GL_FLOAT, GL_FALSE, 0, DE_NULL); glBindBuffer (GL_ARRAY_BUFFER, m_colorBuf); glBufferData (GL_ARRAY_BUFFER, (glw::GLsizeiptr)(fetchedColor.size()*sizeof(fetchedColor[0])), &fetchedColor[0], GL_STREAM_DRAW); glVertexAttribPointer (m_colorLoc, 3, GL_FLOAT, GL_FALSE, 0, DE_NULL); glDrawArrays (GL_LINE_STRIP, 0, numBytesToVerify); glu::readPixels (m_renderCtx, 0, 0, referenceImg.getAccess()); if (!tcu::pixelThresholdCompare(m_log, "RenderResult", imageSetDesc.c_str(), referenceImg, indexBufferImg, threshold, tcu::COMPARE_LOG_RESULT)) { isOk = false; break; } numVerified += isLeftoverBatch ? numRemaining : numBytesToVerify; } glBindVertexArray(0); return isOk; } const char* getWriteTypeDescription (WriteType write) { static const char* s_desc[] = { "glBufferSubData()", "glMapBufferRange()", "transform feedback", "glReadPixels() into PBO binding" }; return de::getSizedArrayElement<WRITE_LAST>(s_desc, write); } const char* getVerifyTypeDescription (VerifyType verify) { static const char* s_desc[] = { "rendering as vertex data", "rendering as index data", "reading in shader as uniform buffer data", "using as PBO and uploading to texture", "reading back using glMapBufferRange()" }; return de::getSizedArrayElement<VERIFY_LAST>(s_desc, verify); } } // BufferTestUtil } // gls } // deqp