/*------------------------------------------------------------------------- * drawElements Quality Program OpenGL ES 2.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 Draw call batching performance tests *//*--------------------------------------------------------------------*/ #include "es2pDrawCallBatchingTests.hpp" #include "gluShaderProgram.hpp" #include "gluRenderContext.hpp" #include "glwDefs.hpp" #include "glwFunctions.hpp" #include "glwEnums.hpp" #include "tcuTestLog.hpp" #include "deRandom.hpp" #include "deStringUtil.hpp" #include "deFile.h" #include "deString.h" #include "deClock.h" #include "deThread.h" #include <cmath> #include <vector> #include <string> #include <sstream> using tcu::TestLog; using namespace glw; using std::vector; using std::string; namespace deqp { namespace gles2 { namespace Performance { namespace { const int CALIBRATION_SAMPLE_COUNT = 34; class DrawCallBatchingTest : public tcu::TestCase { public: struct TestSpec { bool useStaticBuffer; int staticAttributeCount; bool useDynamicBuffer; int dynamicAttributeCount; int triangleCount; int drawCallCount; bool useDrawElements; bool useIndexBuffer; bool dynamicIndices; }; DrawCallBatchingTest (Context& context, const char* name, const char* description, const TestSpec& spec); ~DrawCallBatchingTest (void); void init (void); void deinit (void); IterateResult iterate (void); private: enum State { STATE_LOG_INFO = 0, STATE_WARMUP_BATCHED, STATE_WARMUP_UNBATCHED, STATE_CALC_CALIBRATION, STATE_SAMPLE }; State m_state; glu::RenderContext& m_renderCtx; de::Random m_rnd; int m_sampleIteration; int m_unbatchedSampleCount; int m_batchedSampleCount; TestSpec m_spec; glu::ShaderProgram* m_program; vector<deUint8> m_dynamicIndexData; vector<deUint8> m_staticIndexData; vector<GLuint> m_unbatchedDynamicIndexBuffers; GLuint m_batchedDynamicIndexBuffer; GLuint m_unbatchedStaticIndexBuffer; GLuint m_batchedStaticIndexBuffer; vector<vector<deInt8> > m_staticAttributeDatas; vector<vector<deInt8> > m_dynamicAttributeDatas; vector<GLuint> m_batchedStaticBuffers; vector<GLuint> m_unbatchedStaticBuffers; vector<GLuint> m_batchedDynamicBuffers; vector<vector<GLuint> > m_unbatchedDynamicBuffers; vector<deUint64> m_unbatchedSamplesUs; vector<deUint64> m_batchedSamplesUs; void logTestInfo (void); deUint64 renderUnbatched (void); deUint64 renderBatched (void); void createIndexData (void); void createIndexBuffer (void); void createShader (void); void createAttributeDatas (void); void createArrayBuffers (void); }; DrawCallBatchingTest::DrawCallBatchingTest (Context& context, const char* name, const char* description, const TestSpec& spec) : tcu::TestCase (context.getTestContext(), tcu::NODETYPE_PERFORMANCE, name, description) , m_state (STATE_LOG_INFO) , m_renderCtx (context.getRenderContext()) , m_rnd (deStringHash(name)) , m_sampleIteration (0) , m_unbatchedSampleCount (CALIBRATION_SAMPLE_COUNT) , m_batchedSampleCount (CALIBRATION_SAMPLE_COUNT) , m_spec (spec) , m_program (NULL) , m_batchedDynamicIndexBuffer (0) , m_unbatchedStaticIndexBuffer (0) , m_batchedStaticIndexBuffer (0) { } DrawCallBatchingTest::~DrawCallBatchingTest (void) { deinit(); } void DrawCallBatchingTest::createIndexData (void) { if (m_spec.dynamicIndices) { for (int drawNdx = 0; drawNdx < m_spec.drawCallCount; drawNdx++) { for (int triangleNdx = 0; triangleNdx < m_spec.triangleCount; triangleNdx++) { m_dynamicIndexData.push_back(deUint8(triangleNdx * 3)); m_dynamicIndexData.push_back(deUint8(triangleNdx * 3 + 1)); m_dynamicIndexData.push_back(deUint8(triangleNdx * 3 + 2)); } } } else { for (int drawNdx = 0; drawNdx < m_spec.drawCallCount; drawNdx++) { for (int triangleNdx = 0; triangleNdx < m_spec.triangleCount; triangleNdx++) { m_staticIndexData.push_back(deUint8(triangleNdx * 3)); m_staticIndexData.push_back(deUint8(triangleNdx * 3 + 1)); m_staticIndexData.push_back(deUint8(triangleNdx * 3 + 2)); } } } } void DrawCallBatchingTest::createShader (void) { std::ostringstream vertexShader; std::ostringstream fragmentShader; for (int attributeNdx = 0; attributeNdx < m_spec.staticAttributeCount; attributeNdx++) vertexShader << "attribute mediump vec4 a_static" << attributeNdx << ";\n"; if (m_spec.staticAttributeCount > 0 && m_spec.dynamicAttributeCount > 0) vertexShader << "\n"; for (int attributeNdx = 0; attributeNdx < m_spec.dynamicAttributeCount; attributeNdx++) vertexShader << "attribute mediump vec4 a_dyn" << attributeNdx << ";\n"; vertexShader << "\n" << "varying mediump vec4 v_color;\n" << "\n" << "void main (void)\n" << "{\n"; vertexShader << "\tv_color = "; bool first = true; for (int attributeNdx = 0; attributeNdx < m_spec.staticAttributeCount; attributeNdx++) { if (!first) vertexShader << " + "; first = false; vertexShader << "a_static" << attributeNdx; } for (int attributeNdx = 0; attributeNdx < m_spec.dynamicAttributeCount; attributeNdx++) { if (!first) vertexShader << " + "; first = false; vertexShader << "a_dyn" << attributeNdx; } vertexShader << ";\n"; if (m_spec.dynamicAttributeCount > 0) vertexShader << "\tgl_Position = a_dyn0;\n"; else vertexShader << "\tgl_Position = a_static0;\n"; vertexShader << "}"; fragmentShader << "varying mediump vec4 v_color;\n" << "\n" << "void main(void)\n" << "{\n" << "\tgl_FragColor = v_color;\n" << "}\n"; m_program = new glu::ShaderProgram(m_renderCtx, glu::ProgramSources() << glu::VertexSource(vertexShader.str()) << glu::FragmentSource(fragmentShader.str())); m_testCtx.getLog() << (*m_program); TCU_CHECK(m_program->isOk()); } void DrawCallBatchingTest::createAttributeDatas (void) { // Generate data for static attributes for (int attribute = 0; attribute < m_spec.staticAttributeCount; attribute++) { vector<deInt8> data; if (m_spec.dynamicAttributeCount == 0 && attribute == 0) { data.reserve(4 * 3 * m_spec.triangleCount * m_spec.drawCallCount); for (int i = 0; i < m_spec.triangleCount * m_spec.drawCallCount; i++) { int sign = (m_spec.triangleCount % 2 == 1 || i % 2 == 0 ? 1 : -1); data.push_back(deInt8(-127 * sign)); data.push_back(deInt8(-127 * sign)); data.push_back(0); data.push_back(127); data.push_back(deInt8(127 * sign)); data.push_back(deInt8(-127 * sign)); data.push_back(0); data.push_back(127); data.push_back(deInt8(127 * sign)); data.push_back(deInt8(127 * sign)); data.push_back(0); data.push_back(127); } } else { data.reserve(4 * 3 * m_spec.triangleCount * m_spec.drawCallCount); for (int i = 0; i < 4 * 3 * m_spec.triangleCount * m_spec.drawCallCount; i++) data.push_back((deInt8)m_rnd.getUint32()); } m_staticAttributeDatas.push_back(data); } // Generate data for dynamic attributes for (int attribute = 0; attribute < m_spec.dynamicAttributeCount; attribute++) { vector<deInt8> data; if (attribute == 0) { data.reserve(4 * 3 * m_spec.triangleCount * m_spec.drawCallCount); for (int i = 0; i < m_spec.triangleCount * m_spec.drawCallCount; i++) { int sign = (i % 2 == 0 ? 1 : -1); data.push_back(deInt8(-127 * sign)); data.push_back(deInt8(-127 * sign)); data.push_back(0); data.push_back(127); data.push_back(deInt8(127 * sign)); data.push_back(deInt8(-127 * sign)); data.push_back(0); data.push_back(127); data.push_back(deInt8(127 * sign)); data.push_back(deInt8(127 * sign)); data.push_back(0); data.push_back(127); } } else { data.reserve(4 * 3 * m_spec.triangleCount * m_spec.drawCallCount); for (int i = 0; i < 4 * 3 * m_spec.triangleCount * m_spec.drawCallCount; i++) data.push_back((deInt8)m_rnd.getUint32()); } m_dynamicAttributeDatas.push_back(data); } } void DrawCallBatchingTest::createArrayBuffers (void) { const glw::Functions& gl = m_renderCtx.getFunctions(); if (m_spec.useStaticBuffer) { // Upload static attributes for batched for (int attribute = 0; attribute < m_spec.staticAttributeCount; attribute++) { GLuint buffer; gl.genBuffers(1, &buffer); gl.bindBuffer(GL_ARRAY_BUFFER, buffer); gl.bufferData(GL_ARRAY_BUFFER, 4 * 3 * m_spec.triangleCount * m_spec.drawCallCount, &(m_staticAttributeDatas[attribute][0]), GL_STATIC_DRAW); gl.bindBuffer(GL_ARRAY_BUFFER, 0); GLU_EXPECT_NO_ERROR(gl.getError(), "Creating static buffer failed"); m_batchedStaticBuffers.push_back(buffer); } // Upload static attributes for unbatched for (int attribute = 0; attribute < m_spec.staticAttributeCount; attribute++) { GLuint buffer; gl.genBuffers(1, &buffer); gl.bindBuffer(GL_ARRAY_BUFFER, buffer); gl.bufferData(GL_ARRAY_BUFFER, 4 * 3 * m_spec.triangleCount, &(m_staticAttributeDatas[attribute][0]), GL_STATIC_DRAW); gl.bindBuffer(GL_ARRAY_BUFFER, 0); GLU_EXPECT_NO_ERROR(gl.getError(), "Creating static buffer failed"); m_unbatchedStaticBuffers.push_back(buffer); } } if (m_spec.useDynamicBuffer) { // Upload dynamic attributes for batched for (int attribute = 0; attribute < m_spec.dynamicAttributeCount; attribute++) { GLuint buffer; gl.genBuffers(1, &buffer); gl.bindBuffer(GL_ARRAY_BUFFER, buffer); gl.bufferData(GL_ARRAY_BUFFER, 4 * 3 * m_spec.triangleCount * m_spec.drawCallCount, &(m_dynamicAttributeDatas[attribute][0]), GL_STATIC_DRAW); gl.bindBuffer(GL_ARRAY_BUFFER, 0); GLU_EXPECT_NO_ERROR(gl.getError(), "Creating dynamic buffer failed"); m_batchedDynamicBuffers.push_back(buffer); } // Upload dynamic attributes for unbatched for (int attribute = 0; attribute < m_spec.dynamicAttributeCount; attribute++) { vector<GLuint> buffers; for (int drawNdx = 0; drawNdx < m_spec.drawCallCount; drawNdx++) { GLuint buffer; gl.genBuffers(1, &buffer); gl.bindBuffer(GL_ARRAY_BUFFER, buffer); gl.bufferData(GL_ARRAY_BUFFER, 4 * 3 * m_spec.triangleCount * m_spec.drawCallCount, &(m_dynamicAttributeDatas[attribute][0]), GL_STATIC_DRAW); gl.bindBuffer(GL_ARRAY_BUFFER, 0); GLU_EXPECT_NO_ERROR(gl.getError(), "Creating dynamic buffer failed"); buffers.push_back(buffer); } m_unbatchedDynamicBuffers.push_back(buffers); } } } void DrawCallBatchingTest::createIndexBuffer (void) { const glw::Functions& gl = m_renderCtx.getFunctions(); if (m_spec.dynamicIndices) { for (int drawNdx = 0; drawNdx < m_spec.drawCallCount; drawNdx++) { GLuint buffer; gl.genBuffers(1, &buffer); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, buffer); gl.bufferData(GL_ELEMENT_ARRAY_BUFFER, 3 * m_spec.triangleCount, &(m_dynamicIndexData[drawNdx * m_spec.triangleCount * 3]), GL_STATIC_DRAW); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); GLU_EXPECT_NO_ERROR(gl.getError(), "Creating dynamic index buffer failed"); m_unbatchedDynamicIndexBuffers.push_back(buffer); } { GLuint buffer; gl.genBuffers(1, &buffer); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, buffer); gl.bufferData(GL_ELEMENT_ARRAY_BUFFER, 3 * m_spec.triangleCount * m_spec.drawCallCount, &(m_dynamicIndexData[0]), GL_STATIC_DRAW); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); GLU_EXPECT_NO_ERROR(gl.getError(), "Creating dynamic index buffer failed"); m_batchedDynamicIndexBuffer = buffer; } } else { { GLuint buffer; gl.genBuffers(1, &buffer); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, buffer); gl.bufferData(GL_ELEMENT_ARRAY_BUFFER, 3 * m_spec.triangleCount * m_spec.drawCallCount, &(m_staticIndexData[0]), GL_STATIC_DRAW); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); GLU_EXPECT_NO_ERROR(gl.getError(), "Creating dynamic index buffer failed"); m_batchedStaticIndexBuffer = buffer; } { GLuint buffer; gl.genBuffers(1, &buffer); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, buffer); gl.bufferData(GL_ELEMENT_ARRAY_BUFFER, 3 * m_spec.triangleCount, &(m_staticIndexData[0]), GL_STATIC_DRAW); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); GLU_EXPECT_NO_ERROR(gl.getError(), "Creating dynamic index buffer failed"); m_unbatchedStaticIndexBuffer = buffer; } } } void DrawCallBatchingTest::init (void) { createShader(); createAttributeDatas(); createArrayBuffers(); if (m_spec.useDrawElements) { createIndexData(); if (m_spec.useIndexBuffer) createIndexBuffer(); } } void DrawCallBatchingTest::deinit (void) { const glw::Functions& gl = m_renderCtx.getFunctions(); delete m_program; m_program = NULL; m_dynamicIndexData = vector<deUint8>(); m_staticIndexData = vector<deUint8>(); if (!m_unbatchedDynamicIndexBuffers.empty()) { gl.deleteBuffers((GLsizei)m_unbatchedDynamicIndexBuffers.size(), &(m_unbatchedDynamicIndexBuffers[0])); GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteBuffers()"); m_unbatchedDynamicIndexBuffers = vector<GLuint>(); } if (m_batchedDynamicIndexBuffer) { gl.deleteBuffers((GLsizei)1, &m_batchedDynamicIndexBuffer); GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteBuffers()"); m_batchedDynamicIndexBuffer = 0; } if (m_unbatchedStaticIndexBuffer) { gl.deleteBuffers((GLsizei)1, &m_unbatchedStaticIndexBuffer); GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteBuffers()"); m_unbatchedStaticIndexBuffer = 0; } if (m_batchedStaticIndexBuffer) { gl.deleteBuffers((GLsizei)1, &m_batchedStaticIndexBuffer); GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteBuffers()"); m_batchedStaticIndexBuffer = 0; } m_staticAttributeDatas = vector<vector<deInt8> >(); m_dynamicAttributeDatas = vector<vector<deInt8> >(); if (!m_batchedStaticBuffers.empty()) { gl.deleteBuffers((GLsizei)m_batchedStaticBuffers.size(), &(m_batchedStaticBuffers[0])); GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteBuffers()"); m_batchedStaticBuffers = vector<GLuint>(); } if (!m_unbatchedStaticBuffers.empty()) { gl.deleteBuffers((GLsizei)m_unbatchedStaticBuffers.size(), &(m_unbatchedStaticBuffers[0])); GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteBuffers()"); m_unbatchedStaticBuffers = vector<GLuint>(); } if (!m_batchedDynamicBuffers.empty()) { gl.deleteBuffers((GLsizei)m_batchedDynamicBuffers.size(), &(m_batchedDynamicBuffers[0])); GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteBuffers()"); m_batchedDynamicBuffers = vector<GLuint>(); } for (int i = 0; i < (int)m_unbatchedDynamicBuffers.size(); i++) { gl.deleteBuffers((GLsizei)m_unbatchedDynamicBuffers[i].size(), &(m_unbatchedDynamicBuffers[i][0])); GLU_EXPECT_NO_ERROR(gl.getError(), "glDeleteBuffers()"); } m_unbatchedDynamicBuffers = vector<vector<GLuint> >(); m_unbatchedSamplesUs = vector<deUint64>(); m_batchedSamplesUs = vector<deUint64>(); } deUint64 DrawCallBatchingTest::renderUnbatched (void) { const glw::Functions& gl = m_renderCtx.getFunctions(); deUint64 beginUs = 0; deUint64 endUs = 0; vector<GLint> dynamicAttributeLocations; gl.viewport(0, 0, 32, 32); gl.useProgram(m_program->getProgram()); // Setup static buffers for (int attribNdx = 0; attribNdx < m_spec.staticAttributeCount; attribNdx++) { GLint location = gl.getAttribLocation(m_program->getProgram(), ("a_static" + de::toString(attribNdx)).c_str()); gl.enableVertexAttribArray(location); if (m_spec.useStaticBuffer) { gl.bindBuffer(GL_ARRAY_BUFFER, m_unbatchedStaticBuffers[attribNdx]); gl.vertexAttribPointer(location, 4, GL_BYTE, GL_TRUE, 0, NULL); gl.bindBuffer(GL_ARRAY_BUFFER, 0); } else gl.vertexAttribPointer(location, 4, GL_BYTE, GL_TRUE, 0, &(m_staticAttributeDatas[attribNdx][0])); } // Get locations of dynamic attributes for (int attribNdx = 0; attribNdx < m_spec.dynamicAttributeCount; attribNdx++) { GLint location = gl.getAttribLocation(m_program->getProgram(), ("a_dyn" + de::toString(attribNdx)).c_str()); gl.enableVertexAttribArray(location); dynamicAttributeLocations.push_back(location); } if (m_spec.useDrawElements && m_spec.useIndexBuffer && !m_spec.dynamicIndices) gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_unbatchedStaticIndexBuffer); GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to setup initial state for rendering."); gl.finish(); beginUs = deGetMicroseconds(); for (int drawNdx = 0; drawNdx < m_spec.drawCallCount; drawNdx++) { for (int attribNdx = 0; attribNdx < m_spec.dynamicAttributeCount; attribNdx++) { if (m_spec.useDynamicBuffer) { gl.bindBuffer(GL_ARRAY_BUFFER, m_unbatchedDynamicBuffers[attribNdx][drawNdx]); gl.vertexAttribPointer(dynamicAttributeLocations[attribNdx], 4, GL_BYTE, GL_TRUE, 0, NULL); gl.bindBuffer(GL_ARRAY_BUFFER, 0); } else gl.vertexAttribPointer(dynamicAttributeLocations[attribNdx], 4, GL_BYTE, GL_TRUE, 0, &(m_dynamicAttributeDatas[attribNdx][m_spec.triangleCount * 3 * drawNdx * 4])); } if (m_spec.useDrawElements) { if (m_spec.useIndexBuffer) { if (m_spec.dynamicIndices) { gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_unbatchedDynamicIndexBuffers[drawNdx]); gl.drawElements(GL_TRIANGLES, m_spec.triangleCount * 3, GL_UNSIGNED_BYTE, NULL); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); } else gl.drawElements(GL_TRIANGLES, m_spec.triangleCount * 3, GL_UNSIGNED_BYTE, NULL); } else { if (m_spec.dynamicIndices) gl.drawElements(GL_TRIANGLES, m_spec.triangleCount * 3, GL_UNSIGNED_BYTE, &(m_dynamicIndexData[drawNdx * m_spec.triangleCount * 3])); else gl.drawElements(GL_TRIANGLES, m_spec.triangleCount * 3, GL_UNSIGNED_BYTE, &(m_staticIndexData[0])); } } else gl.drawArrays(GL_TRIANGLES, 0, 3 * m_spec.triangleCount); } gl.finish(); endUs = deGetMicroseconds(); GLU_EXPECT_NO_ERROR(gl.getError(), "Unbatched rendering failed"); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); for (int attribNdx = 0; attribNdx < m_spec.staticAttributeCount; attribNdx++) { GLint location = gl.getAttribLocation(m_program->getProgram(), ("a_static" + de::toString(attribNdx)).c_str()); gl.disableVertexAttribArray(location); } for (int attribNdx = 0; attribNdx < m_spec.dynamicAttributeCount; attribNdx++) gl.disableVertexAttribArray(dynamicAttributeLocations[attribNdx]); GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to reset state after unbatched rendering"); return endUs - beginUs; } deUint64 DrawCallBatchingTest::renderBatched (void) { const glw::Functions& gl = m_renderCtx.getFunctions(); deUint64 beginUs = 0; deUint64 endUs = 0; vector<GLint> dynamicAttributeLocations; gl.viewport(0, 0, 32, 32); gl.useProgram(m_program->getProgram()); // Setup static buffers for (int attribNdx = 0; attribNdx < m_spec.staticAttributeCount; attribNdx++) { GLint location = gl.getAttribLocation(m_program->getProgram(), ("a_static" + de::toString(attribNdx)).c_str()); gl.enableVertexAttribArray(location); if (m_spec.useStaticBuffer) { gl.bindBuffer(GL_ARRAY_BUFFER, m_batchedStaticBuffers[attribNdx]); gl.vertexAttribPointer(location, 4, GL_BYTE, GL_TRUE, 0, NULL); gl.bindBuffer(GL_ARRAY_BUFFER, 0); } else gl.vertexAttribPointer(location, 4, GL_BYTE, GL_TRUE, 0, &(m_staticAttributeDatas[attribNdx][0])); } // Get locations of dynamic attributes for (int attribNdx = 0; attribNdx < m_spec.dynamicAttributeCount; attribNdx++) { GLint location = gl.getAttribLocation(m_program->getProgram(), ("a_dyn" + de::toString(attribNdx)).c_str()); gl.enableVertexAttribArray(location); dynamicAttributeLocations.push_back(location); } if (m_spec.useDrawElements && m_spec.useIndexBuffer && !m_spec.dynamicIndices) gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_batchedStaticIndexBuffer); GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to setup initial state for rendering."); gl.finish(); beginUs = deGetMicroseconds(); for (int attribute = 0; attribute < m_spec.dynamicAttributeCount; attribute++) { if (m_spec.useDynamicBuffer) { gl.bindBuffer(GL_ARRAY_BUFFER, m_batchedDynamicBuffers[attribute]); gl.vertexAttribPointer(dynamicAttributeLocations[attribute], 4, GL_BYTE, GL_TRUE, 0, NULL); gl.bindBuffer(GL_ARRAY_BUFFER, 0); } else gl.vertexAttribPointer(dynamicAttributeLocations[attribute], 4, GL_BYTE, GL_TRUE, 0, &(m_dynamicAttributeDatas[attribute][0])); } if (m_spec.useDrawElements) { if (m_spec.useIndexBuffer) { if (m_spec.dynamicIndices) { gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_batchedDynamicIndexBuffer); gl.drawElements(GL_TRIANGLES, m_spec.triangleCount * 3 * m_spec.drawCallCount, GL_UNSIGNED_BYTE, NULL); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); } else gl.drawElements(GL_TRIANGLES, m_spec.triangleCount * 3 * m_spec.drawCallCount, GL_UNSIGNED_BYTE, NULL); } else { if (m_spec.dynamicIndices) gl.drawElements(GL_TRIANGLES, m_spec.triangleCount * 3 * m_spec.drawCallCount, GL_UNSIGNED_BYTE, &(m_dynamicIndexData[0])); else gl.drawElements(GL_TRIANGLES, m_spec.triangleCount * 3 * m_spec.drawCallCount, GL_UNSIGNED_BYTE, &(m_staticIndexData[0])); } } else gl.drawArrays(GL_TRIANGLES, 0, 3 * m_spec.triangleCount * m_spec.drawCallCount); gl.finish(); endUs = deGetMicroseconds(); GLU_EXPECT_NO_ERROR(gl.getError(), "Batched rendering failed"); gl.bindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); for (int attribNdx = 0; attribNdx < m_spec.staticAttributeCount; attribNdx++) { GLint location = gl.getAttribLocation(m_program->getProgram(), ("a_static" + de::toString(attribNdx)).c_str()); gl.disableVertexAttribArray(location); } for (int attribNdx = 0; attribNdx < m_spec.dynamicAttributeCount; attribNdx++) gl.disableVertexAttribArray(dynamicAttributeLocations[attribNdx]); GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to reset state after batched rendering"); return endUs - beginUs; } struct Statistics { double mean; double standardDeviation; double standardErrorOfMean; }; Statistics calculateStats (const vector<deUint64>& samples) { double mean = 0.0; for (int i = 0; i < (int)samples.size(); i++) mean += (double)samples[i]; mean /= (double)samples.size(); double standardDeviation = 0.0; for (int i = 0; i < (int)samples.size(); i++) { double x = (double)samples[i]; standardDeviation += (x - mean) * (x - mean); } standardDeviation /= (double)samples.size(); standardDeviation = std::sqrt(standardDeviation); double standardErrorOfMean = standardDeviation / std::sqrt((double)samples.size()); Statistics stats; stats.mean = mean; stats.standardDeviation = standardDeviation; stats.standardErrorOfMean = standardErrorOfMean; return stats; } void DrawCallBatchingTest::logTestInfo (void) { TestLog& log = m_testCtx.getLog(); tcu::ScopedLogSection section (log, "Test info", "Test info"); log << TestLog::Message << "Rendering using " << (m_spec.useDrawElements ? "glDrawElements()" : "glDrawArrays()") << "." << TestLog::EndMessage; if (m_spec.useDrawElements) log << TestLog::Message << "Using " << (m_spec.dynamicIndices ? "dynamic " : "") << "indices from " << (m_spec.useIndexBuffer ? "buffer" : "pointer") << "." << TestLog::EndMessage; if (m_spec.staticAttributeCount > 0) log << TestLog::Message << "Using " << m_spec.staticAttributeCount << " static attribute" << (m_spec.staticAttributeCount > 1 ? "s" : "") << " from " << (m_spec.useStaticBuffer ? "buffer" : "pointer") << "." << TestLog::EndMessage; if (m_spec.dynamicAttributeCount > 0) log << TestLog::Message << "Using " << m_spec.dynamicAttributeCount << " dynamic attribute" << (m_spec.dynamicAttributeCount > 1 ? "s" : "") << " from " << (m_spec.useDynamicBuffer ? "buffer" : "pointer") << "." << TestLog::EndMessage; log << TestLog::Message << "Rendering " << m_spec.drawCallCount << " draw calls with " << m_spec.triangleCount << " triangles per call." << TestLog::EndMessage; } tcu::TestCase::IterateResult DrawCallBatchingTest::iterate (void) { if (m_state == STATE_LOG_INFO) { logTestInfo(); m_state = STATE_WARMUP_BATCHED; } else if (m_state == STATE_WARMUP_BATCHED) { renderBatched(); m_state = STATE_WARMUP_UNBATCHED; } else if (m_state == STATE_WARMUP_UNBATCHED) { renderUnbatched(); m_state = STATE_SAMPLE; } else if (m_state == STATE_SAMPLE) { if ((int)m_unbatchedSamplesUs.size() < m_unbatchedSampleCount && ((double)m_unbatchedSamplesUs.size() / ((double)m_unbatchedSampleCount) < (double)m_batchedSamplesUs.size() / ((double)m_batchedSampleCount) || (int)m_batchedSamplesUs.size() >= m_batchedSampleCount)) m_unbatchedSamplesUs.push_back(renderUnbatched()); else if ((int)m_batchedSamplesUs.size() < m_batchedSampleCount) m_batchedSamplesUs.push_back(renderBatched()); else m_state = STATE_CALC_CALIBRATION; } else if (m_state == STATE_CALC_CALIBRATION) { TestLog& log = m_testCtx.getLog(); tcu::ScopedLogSection section(log, ("Sampling iteration " + de::toString(m_sampleIteration)).c_str(), ("Sampling iteration " + de::toString(m_sampleIteration)).c_str()); const double targetSEM = 0.02; const double limitSEM = 0.025; Statistics unbatchedStats = calculateStats(m_unbatchedSamplesUs); Statistics batchedStats = calculateStats(m_batchedSamplesUs); log << TestLog::Message << "Batched samples; Count: " << m_batchedSamplesUs.size() << ", Mean: " << batchedStats.mean << "us, Standard deviation: " << batchedStats.standardDeviation << "us, Standard error of mean: " << batchedStats.standardErrorOfMean << "us(" << (batchedStats.standardErrorOfMean/batchedStats.mean) << ")" << TestLog::EndMessage; log << TestLog::Message << "Unbatched samples; Count: " << m_unbatchedSamplesUs.size() << ", Mean: " << unbatchedStats.mean << "us, Standard deviation: " << unbatchedStats.standardDeviation << "us, Standard error of mean: " << unbatchedStats.standardErrorOfMean << "us(" << (unbatchedStats.standardErrorOfMean/unbatchedStats.mean) << ")" << TestLog::EndMessage; if (m_sampleIteration > 2 || (m_sampleIteration > 0 && (unbatchedStats.standardErrorOfMean/unbatchedStats.mean) + (batchedStats.standardErrorOfMean/batchedStats.mean) <= 2.0 * limitSEM)) { if (m_sampleIteration > 2) log << TestLog::Message << "Maximum iteration count reached." << TestLog::EndMessage; log << TestLog::Message << "Standard errors in target range." << TestLog::EndMessage; log << TestLog::Message << "Batched/Unbatched ratio: " << (batchedStats.mean / unbatchedStats.mean) << TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_PASS, de::floatToString((float)(batchedStats.mean/unbatchedStats.mean), 1).c_str()); return STOP; } else { if ((unbatchedStats.standardErrorOfMean/unbatchedStats.mean) > targetSEM) log << TestLog::Message << "Unbatched standard error of mean outside of range." << TestLog::EndMessage; if ((batchedStats.standardErrorOfMean/batchedStats.mean) > targetSEM) log << TestLog::Message << "Batched standard error of mean outside of range." << TestLog::EndMessage; if (unbatchedStats.standardDeviation > 0.0) { double x = (unbatchedStats.standardDeviation / unbatchedStats.mean) / targetSEM; m_unbatchedSampleCount = std::max((int)m_unbatchedSamplesUs.size(), (int)(x * x)); } else m_unbatchedSampleCount = (int)m_unbatchedSamplesUs.size(); if (batchedStats.standardDeviation > 0.0) { double x = (batchedStats.standardDeviation / batchedStats.mean) / targetSEM; m_batchedSampleCount = std::max((int)m_batchedSamplesUs.size(), (int)(x * x)); } else m_batchedSampleCount = (int)m_batchedSamplesUs.size(); m_batchedSamplesUs.clear(); m_unbatchedSamplesUs.clear(); m_sampleIteration++; m_state = STATE_SAMPLE; } } else DE_ASSERT(false); return CONTINUE; } string specToName (const DrawCallBatchingTest::TestSpec& spec) { std::ostringstream stream; DE_ASSERT(!spec.useStaticBuffer || spec.staticAttributeCount > 0); DE_ASSERT(!spec.useDynamicBuffer|| spec.dynamicAttributeCount > 0); if (spec.staticAttributeCount > 0) stream << spec.staticAttributeCount << "_static_"; if (spec.useStaticBuffer) stream << (spec.staticAttributeCount == 1 ? "buffer_" : "buffers_"); if (spec.dynamicAttributeCount > 0) stream << spec.dynamicAttributeCount << "_dynamic_"; if (spec.useDynamicBuffer) stream << (spec.dynamicAttributeCount == 1 ? "buffer_" : "buffers_"); stream << spec.triangleCount << "_triangles"; return stream.str(); } string specToDescrpition (const DrawCallBatchingTest::TestSpec& spec) { DE_UNREF(spec); return "Test performance of batched rendering against non-batched rendering."; } } // anonymous DrawCallBatchingTests::DrawCallBatchingTests (Context& context) : TestCaseGroup(context, "draw_call_batching", "Draw call batching performance tests.") { } DrawCallBatchingTests::~DrawCallBatchingTests (void) { } void DrawCallBatchingTests::init (void) { int drawCallCounts[] = { 10, 100 }; int triangleCounts[] = { 2, 10 }; int staticAttributeCounts[] = { 1, 0, 4, 8, 0 }; int dynamicAttributeCounts[] = { 0, 1, 4, 0, 8 }; DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(staticAttributeCounts) == DE_LENGTH_OF_ARRAY(dynamicAttributeCounts)); for (int drawType = 0; drawType < 2; drawType++) { bool drawElements = (drawType == 1); for (int indexBufferNdx = 0; indexBufferNdx < 2; indexBufferNdx++) { bool useIndexBuffer = (indexBufferNdx == 1); if (useIndexBuffer && !drawElements) continue; for (int dynamicIndexNdx = 0; dynamicIndexNdx < 2; dynamicIndexNdx++) { bool dynamicIndices = (dynamicIndexNdx == 1); if (dynamicIndices && !drawElements) continue; if (dynamicIndices && !useIndexBuffer) continue; TestCaseGroup* drawTypeGroup = new TestCaseGroup(m_context, (string(dynamicIndices ? "dynamic_" : "") + (useIndexBuffer ? "buffer_" : "" ) + (drawElements ? "draw_elements" : "draw_arrays")).c_str(), (string("Test batched rendering with ") + (drawElements ? "draw_elements" : "draw_arrays")).c_str()); addChild(drawTypeGroup); for (int drawCallCountNdx = 0; drawCallCountNdx < DE_LENGTH_OF_ARRAY(drawCallCounts); drawCallCountNdx++) { int drawCallCount = drawCallCounts[drawCallCountNdx]; TestCaseGroup* callCountGroup = new TestCaseGroup(m_context, (de::toString(drawCallCount) + (drawCallCount == 1 ? "_draw" : "_draws")).c_str(), ("Test batched rendering performance with " + de::toString(drawCallCount) + " draw calls.").c_str()); TestCaseGroup* attributeCount1Group = new TestCaseGroup(m_context, "1_attribute", "Test draw call batching with 1 attribute."); TestCaseGroup* attributeCount8Group = new TestCaseGroup(m_context, "8_attributes", "Test draw call batching with 8 attributes."); callCountGroup->addChild(attributeCount1Group); callCountGroup->addChild(attributeCount8Group); drawTypeGroup->addChild(callCountGroup); for (int attributeCountNdx = 0; attributeCountNdx < DE_LENGTH_OF_ARRAY(dynamicAttributeCounts); attributeCountNdx++) { TestCaseGroup* attributeCountGroup = NULL; int staticAttributeCount = staticAttributeCounts[attributeCountNdx]; int dynamicAttributeCount = dynamicAttributeCounts[attributeCountNdx]; if (staticAttributeCount + dynamicAttributeCount == 1) attributeCountGroup = attributeCount1Group; else if (staticAttributeCount + dynamicAttributeCount == 8) attributeCountGroup = attributeCount8Group; else DE_ASSERT(false); for (int triangleCountNdx = 0; triangleCountNdx < DE_LENGTH_OF_ARRAY(triangleCounts); triangleCountNdx++) { int triangleCount = triangleCounts[triangleCountNdx]; for (int dynamicBufferNdx = 0; dynamicBufferNdx < 2; dynamicBufferNdx++) { bool useDynamicBuffer = (dynamicBufferNdx != 0); for (int staticBufferNdx = 0; staticBufferNdx < 2; staticBufferNdx++) { bool useStaticBuffer = (staticBufferNdx != 0); DrawCallBatchingTest::TestSpec spec; spec.useStaticBuffer = useStaticBuffer; spec.staticAttributeCount = staticAttributeCount; spec.useDynamicBuffer = useDynamicBuffer; spec.dynamicAttributeCount = dynamicAttributeCount; spec.drawCallCount = drawCallCount; spec.triangleCount = triangleCount; spec.useDrawElements = drawElements; spec.useIndexBuffer = useIndexBuffer; spec.dynamicIndices = dynamicIndices; if (spec.useStaticBuffer && spec.staticAttributeCount == 0) continue; if (spec.useDynamicBuffer && spec.dynamicAttributeCount == 0) continue; attributeCountGroup->addChild(new DrawCallBatchingTest(m_context, specToName(spec).c_str(), specToDescrpition(spec).c_str(), spec)); } } } } } } } } } } // Performance } // gles2 } // deqp