/*------------------------------------------------------------------------- * 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 Memory object stress test *//*--------------------------------------------------------------------*/ #include "glsMemoryStressCase.hpp" #include "gluShaderProgram.hpp" #include "tcuTestLog.hpp" #include "tcuCommandLine.hpp" #include "deRandom.hpp" #include "deClock.h" #include "deString.h" #include "glw.h" #include <vector> #include <iostream> using std::vector; using tcu::TestLog; namespace deqp { namespace gls { static const char* glErrorToString (deUint32 error) { switch (error) { case GL_OUT_OF_MEMORY: return "GL_OUT_OF_MEMORY"; break; case GL_INVALID_ENUM: return "GL_INVALID_ENUM"; break; case GL_INVALID_FRAMEBUFFER_OPERATION: return "GL_INVALID_FRAMEBUFFER_OPERATION"; break; case GL_INVALID_OPERATION: return "GL_INVALID_OPERATION"; break; case GL_INVALID_VALUE: return "GL_INVALID_VALUE"; break; case 0: return "<none>"; break; default: // \todo [mika] Handle uknown errors? DE_ASSERT(false); return NULL; break; } } static const float s_quadCoords[] = { -1.0f, -1.0f, 1.0f, -1.0f, 1.0f, 1.0f, -1.0f, 1.0f }; static const GLubyte s_quadIndices[] = { 0, 1, 2, 2, 3, 0 }; class TextureRenderer { public: TextureRenderer (tcu::TestLog& log, glu::RenderContext& renderContext); ~TextureRenderer (void); void render (deUint32 texture); private: glu::ShaderProgram* m_program; glu::RenderContext& m_renderCtx; deUint32 m_coordBuffer; deUint32 m_indexBuffer; deUint32 m_vao; static const char* s_vertexShaderGLES2; static const char* s_fragmentShaderGLES2; static const char* s_vertexShaderGLES3; static const char* s_fragmentShaderGLES3; static const char* s_vertexShaderGL3; static const char* s_fragmentShaderGL3; }; const char* TextureRenderer::s_vertexShaderGLES2 = "attribute mediump vec2 a_coord;\n" "varying mediump vec2 v_texCoord;\n" "void main (void)\n" "{\n" "\tv_texCoord = 0.5 * (a_coord + vec2(1.0));\n" "\tgl_Position = vec4(a_coord, 0.0, 1.0);\n" "}\n"; const char* TextureRenderer::s_fragmentShaderGLES2 = "varying mediump vec2 v_texCoord;\n" "uniform sampler2D u_texture;\n" "void main (void)\n" "{\n" "\tgl_FragColor = texture2D(u_texture, v_texCoord);\n" "}\n"; const char* TextureRenderer::s_vertexShaderGLES3 = "#version 300 es\n" "in mediump vec2 a_coord;\n" "out mediump vec2 v_texCoord;\n" "void main (void)\n" "{\n" "\tv_texCoord = 0.5 * (a_coord + vec2(1.0));\n" "\tgl_Position = vec4(a_coord, 0.0, 1.0);\n" "}\n"; const char* TextureRenderer::s_fragmentShaderGLES3 = "#version 300 es\n" "in mediump vec2 v_texCoord;\n" "uniform sampler2D u_texture;\n" "layout(location = 0) out mediump vec4 dEQP_FragColor;\n" "void main (void)\n" "{\n" "\tdEQP_FragColor = texture(u_texture, v_texCoord);\n" "}\n"; const char* TextureRenderer::s_vertexShaderGL3 = "#version 330\n" "in mediump vec2 a_coord;\n" "out mediump vec2 v_texCoord;\n" "void main (void)\n" "{\n" "\tv_texCoord = 0.5 * (a_coord + vec2(1.0));\n" "\tgl_Position = vec4(a_coord, 0.0, 1.0);\n" "}\n"; const char* TextureRenderer::s_fragmentShaderGL3 = "#version 330\n" "in mediump vec2 v_texCoord;\n" "uniform sampler2D u_texture;\n" "layout(location = 0) out mediump vec4 dEQP_FragColor;\n" "void main (void)\n" "{\n" "\tdEQP_FragColor = texture(u_texture, v_texCoord);\n" "}\n"; TextureRenderer::TextureRenderer (tcu::TestLog& log, glu::RenderContext& renderContext) : m_program (NULL) , m_renderCtx (renderContext) , m_coordBuffer (0) , m_indexBuffer (0) , m_vao (0) { const glu::ContextType ctxType = renderContext.getType(); if (glu::isGLSLVersionSupported(ctxType, glu::GLSL_VERSION_300_ES)) m_program = new glu::ShaderProgram(m_renderCtx, glu::makeVtxFragSources(s_vertexShaderGLES3, s_fragmentShaderGLES3)); else if (glu::isGLSLVersionSupported(ctxType, glu::GLSL_VERSION_100_ES)) m_program = new glu::ShaderProgram(m_renderCtx, glu::makeVtxFragSources(s_vertexShaderGLES2, s_fragmentShaderGLES2)); else if (glu::isGLSLVersionSupported(ctxType, glu::GLSL_VERSION_330)) m_program = new glu::ShaderProgram(m_renderCtx, glu::makeVtxFragSources(s_vertexShaderGL3, s_fragmentShaderGL3)); else DE_ASSERT(false); if (ctxType.getProfile() == glu::PROFILE_CORE) GLU_CHECK_CALL(glGenVertexArrays(1, &m_vao)); GLU_CHECK_CALL(glGenBuffers(1, &m_coordBuffer)); GLU_CHECK_CALL(glBindBuffer(GL_ARRAY_BUFFER, m_coordBuffer)); GLU_CHECK_CALL(glBufferData(GL_ARRAY_BUFFER, sizeof(s_quadCoords), s_quadCoords, GL_STATIC_DRAW)); GLU_CHECK_CALL(glGenBuffers(1, &m_indexBuffer)); GLU_CHECK_CALL(glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_indexBuffer)); GLU_CHECK_CALL(glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(s_quadIndices), s_quadIndices, GL_STATIC_DRAW)); if (!m_program->isOk()) { log << *m_program; TCU_CHECK_MSG(m_program->isOk(), "Shader compilation failed"); } } TextureRenderer::~TextureRenderer (void) { delete m_program; glDeleteBuffers(1, &m_coordBuffer); glDeleteBuffers(1, &m_indexBuffer); } void TextureRenderer::render (deUint32 texture) { deUint32 coordLoc = -1; deUint32 texLoc = -1; GLU_CHECK_CALL(glUseProgram(m_program->getProgram())); coordLoc = glGetAttribLocation(m_program->getProgram(), "a_coord"); GLU_CHECK(); TCU_CHECK(coordLoc != (deUint32)-1); if (m_vao != 0) GLU_CHECK_CALL(glBindVertexArray(m_vao)); GLU_CHECK_CALL(glEnableVertexAttribArray(coordLoc)); GLU_CHECK_CALL(glBindBuffer(GL_ARRAY_BUFFER, m_coordBuffer)); GLU_CHECK_CALL(glVertexAttribPointer(coordLoc, 2, GL_FLOAT, GL_FALSE, 0, NULL)); GLU_CHECK_CALL(glActiveTexture(GL_TEXTURE0)); GLU_CHECK_CALL(glBindTexture(GL_TEXTURE_2D, texture)); texLoc = glGetUniformLocation(m_program->getProgram(), "u_texture"); GLU_CHECK(); TCU_CHECK(texLoc != (deUint32)-1); GLU_CHECK_CALL(glUniform1i(texLoc, 0)); GLU_CHECK_CALL(glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_indexBuffer)); GLU_CHECK_CALL(glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_BYTE, NULL)); GLU_CHECK_CALL(glDisableVertexAttribArray(coordLoc)); if (m_vao != 0) GLU_CHECK_CALL(glBindVertexArray(0)); } class BufferRenderer { public: BufferRenderer (tcu::TestLog& log, glu::RenderContext& renderContext); ~BufferRenderer (void); void render (deUint32 buffer, int size); private: glu::ShaderProgram* m_program; glu::RenderContext& m_renderCtx; deUint32 m_coordBuffer; deUint32 m_indexBuffer; deUint32 m_vao; static const char* s_vertexShaderGLES2; static const char* s_fragmentShaderGLES2; static const char* s_vertexShaderGLES3; static const char* s_fragmentShaderGLES3; static const char* s_vertexShaderGL3; static const char* s_fragmentShaderGL3; }; const char* BufferRenderer::s_vertexShaderGLES2 = "attribute mediump vec2 a_coord;\n" "attribute mediump vec4 a_buffer;\n" "varying mediump vec4 v_buffer;\n" "void main (void)\n" "{\n" "\tv_buffer = a_buffer;\n" "\tgl_Position = vec4(a_coord, 0.0, 1.0);\n" "}\n"; const char* BufferRenderer::s_fragmentShaderGLES2 = "varying mediump vec4 v_buffer;\n" "void main (void)\n" "{\n" "\tgl_FragColor = v_buffer;\n" "}\n"; const char* BufferRenderer::s_vertexShaderGLES3 = "#version 300 es\n" "in mediump vec2 a_coord;\n" "in mediump vec4 a_buffer;\n" "out mediump vec4 v_buffer;\n" "void main (void)\n" "{\n" "\tv_buffer = a_buffer;\n" "\tgl_Position = vec4(a_coord, 0.0, 1.0);\n" "}\n"; const char* BufferRenderer::s_fragmentShaderGLES3 = "#version 300 es\n" "in mediump vec4 v_buffer;\n" "layout(location = 0) out mediump vec4 dEQP_FragColor;\n" "void main (void)\n" "{\n" "\tdEQP_FragColor = v_buffer;\n" "}\n"; const char* BufferRenderer::s_vertexShaderGL3 = "#version 330\n" "in mediump vec2 a_coord;\n" "in mediump vec4 a_buffer;\n" "out mediump vec4 v_buffer;\n" "void main (void)\n" "{\n" "\tv_buffer = a_buffer;\n" "\tgl_Position = vec4(a_coord, 0.0, 1.0);\n" "}\n"; const char* BufferRenderer::s_fragmentShaderGL3 = "#version 330\n" "in mediump vec4 v_buffer;\n" "layout(location = 0) out mediump vec4 dEQP_FragColor;\n" "void main (void)\n" "{\n" "\tdEQP_FragColor = v_buffer;\n" "}\n"; BufferRenderer::BufferRenderer (tcu::TestLog& log, glu::RenderContext& renderContext) : m_program (NULL) , m_renderCtx (renderContext) , m_coordBuffer (0) , m_indexBuffer (0) , m_vao (0) { const glu::ContextType ctxType = renderContext.getType(); if (glu::isGLSLVersionSupported(ctxType, glu::GLSL_VERSION_300_ES)) m_program = new glu::ShaderProgram(m_renderCtx, glu::makeVtxFragSources(s_vertexShaderGLES3, s_fragmentShaderGLES3)); else if (glu::isGLSLVersionSupported(ctxType, glu::GLSL_VERSION_100_ES)) m_program = new glu::ShaderProgram(m_renderCtx, glu::makeVtxFragSources(s_vertexShaderGLES2, s_fragmentShaderGLES2)); else if (glu::isGLSLVersionSupported(ctxType, glu::GLSL_VERSION_330)) m_program = new glu::ShaderProgram(m_renderCtx, glu::makeVtxFragSources(s_vertexShaderGL3, s_fragmentShaderGL3)); else DE_ASSERT(false); if (ctxType.getProfile() == glu::PROFILE_CORE) GLU_CHECK_CALL(glGenVertexArrays(1, &m_vao)); GLU_CHECK_CALL(glGenBuffers(1, &m_coordBuffer)); GLU_CHECK_CALL(glBindBuffer(GL_ARRAY_BUFFER, m_coordBuffer)); GLU_CHECK_CALL(glBufferData(GL_ARRAY_BUFFER, sizeof(s_quadCoords), s_quadCoords, GL_STATIC_DRAW)); GLU_CHECK_CALL(glGenBuffers(1, &m_indexBuffer)); GLU_CHECK_CALL(glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_indexBuffer)); GLU_CHECK_CALL(glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(s_quadIndices), s_quadIndices, GL_STATIC_DRAW)); if (!m_program->isOk()) { log << *m_program; TCU_CHECK_MSG(m_program->isOk(), "Shader compilation failed"); } } BufferRenderer::~BufferRenderer (void) { delete m_program; glDeleteBuffers(1, &m_coordBuffer); glDeleteBuffers(1, &m_indexBuffer); } void BufferRenderer::render (deUint32 buffer, int size) { DE_UNREF(size); DE_ASSERT((size_t)size >= sizeof(GLubyte) * 4 * 6); GLU_CHECK_CALL(glUseProgram(m_program->getProgram())); deUint32 bufferLoc = glGetAttribLocation(m_program->getProgram(), "a_buffer"); TCU_CHECK(bufferLoc != (deUint32)-1); deUint32 coordLoc = glGetAttribLocation(m_program->getProgram(), "a_coord"); TCU_CHECK(coordLoc != (deUint32)-1); if (m_vao != 0) GLU_CHECK_CALL(glBindVertexArray(m_vao)); GLU_CHECK_CALL(glEnableVertexAttribArray(bufferLoc)); GLU_CHECK_CALL(glEnableVertexAttribArray(coordLoc)); GLU_CHECK_CALL(glBindBuffer(GL_ARRAY_BUFFER, m_coordBuffer)); GLU_CHECK_CALL(glVertexAttribPointer(coordLoc, 2, GL_FLOAT, GL_FALSE, 0, NULL)); GLU_CHECK_CALL(glBindBuffer(GL_ARRAY_BUFFER, buffer)); GLU_CHECK_CALL(glVertexAttribPointer(bufferLoc, 4, GL_UNSIGNED_BYTE, GL_TRUE, 0, 0)); GLU_CHECK_CALL(glBindBuffer(GL_ARRAY_BUFFER, 0)); GLU_CHECK_CALL(glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_indexBuffer)); GLU_CHECK_CALL(glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_BYTE, NULL)); GLU_CHECK_CALL(glDisableVertexAttribArray(bufferLoc)); GLU_CHECK_CALL(glDisableVertexAttribArray(coordLoc)); if (m_vao != 0) GLU_CHECK_CALL(glBindVertexArray(0)); } class MemObjectAllocator { public: enum Result { RESULT_GOT_BAD_ALLOC = 0, RESULT_GEN_TEXTURES_FAILED, RESULT_GEN_BUFFERS_FAILED, RESULT_BUFFER_DATA_FAILED, RESULT_BUFFER_SUB_DATA_FAILED, RESULT_TEXTURE_IMAGE_FAILED, RESULT_TEXTURE_SUB_IMAGE_FAILED, RESULT_BIND_TEXTURE_FAILED, RESULT_BIND_BUFFER_FAILED, RESULT_DELETE_TEXTURES_FAILED, RESULT_DELETE_BUFFERS_FAILED, RESULT_RENDER_FAILED, RESULT_LAST }; MemObjectAllocator (tcu::TestLog& log, glu::RenderContext& renderContext, MemObjectType objectTypes, const MemObjectConfig& config, int seed); ~MemObjectAllocator (void); bool allocUntilFailure (void); void clearObjects (void); Result getResult (void) const { return m_result; } deUint32 getGLError (void) const { return m_glError; } int getObjectCount (void) const { return m_objectCount; } deUint32 getBytes (void) const { return m_bytesRequired; } static const char* resultToString (Result result); private: void allocateTexture (de::Random& rnd); void allocateBuffer (de::Random& rnd); vector<deUint32> m_buffers; vector<deUint32> m_textures; int m_seed; int m_objectCount; deUint32 m_bytesRequired; MemObjectType m_objectTypes; Result m_result; MemObjectConfig m_config; deUint32 m_glError; vector<deUint8> m_dummyData; BufferRenderer m_bufferRenderer; TextureRenderer m_textureRenderer; }; MemObjectAllocator::MemObjectAllocator (tcu::TestLog& log, glu::RenderContext& renderContext, MemObjectType objectTypes, const MemObjectConfig& config, int seed) : m_seed (seed) , m_objectCount (0) , m_bytesRequired (0) , m_objectTypes (objectTypes) , m_result (RESULT_LAST) , m_config (config) , m_glError (0) , m_bufferRenderer (log, renderContext) , m_textureRenderer (log, renderContext) { DE_UNREF(renderContext); if (m_config.useDummyData) { int dummySize = deMax32(m_config.maxBufferSize, m_config.maxTextureSize*m_config.maxTextureSize*4); m_dummyData = vector<deUint8>(dummySize); } else if (m_config.write) m_dummyData = vector<deUint8>(128); } MemObjectAllocator::~MemObjectAllocator (void) { } bool MemObjectAllocator::allocUntilFailure (void) { de::Random rnd(m_seed); GLU_CHECK_MSG("Error in init"); try { const deUint64 timeoutUs = 10000000; // 10s deUint64 beginTimeUs = deGetMicroseconds(); deUint64 currentTimeUs; do { GLU_CHECK_MSG("Unkown Error"); switch (m_objectTypes) { case MEMOBJECTTYPE_TEXTURE: allocateTexture(rnd); break; case MEMOBJECTTYPE_BUFFER: allocateBuffer(rnd); break; default: { if (rnd.getBool()) allocateBuffer(rnd); else allocateTexture(rnd); break; } } if (m_result != RESULT_LAST) { glFinish(); return true; } currentTimeUs = deGetMicroseconds(); } while (currentTimeUs - beginTimeUs < timeoutUs); // Timeout if (currentTimeUs - beginTimeUs >= timeoutUs) return false; else return true; } catch (const std::bad_alloc&) { m_result = RESULT_GOT_BAD_ALLOC; return true; } } void MemObjectAllocator::clearObjects (void) { deUint32 error = 0; if (!m_textures.empty()) { glDeleteTextures((GLsizei)m_textures.size(), &(m_textures[0])); error = glGetError(); if (error != 0) { m_result = RESULT_DELETE_TEXTURES_FAILED; m_glError = error; } m_textures.clear(); } if (!m_buffers.empty()) { glDeleteBuffers((GLsizei)m_buffers.size(), &(m_buffers[0])); error = glGetError(); if (error != 0) { m_result = RESULT_DELETE_BUFFERS_FAILED; m_glError = error; } m_buffers.clear(); } } void MemObjectAllocator::allocateTexture (de::Random& rnd) { const int vectorBlockSize = 128; deUint32 tex = 0; deUint32 error = 0; int width = rnd.getInt(m_config.minTextureSize, m_config.maxTextureSize); int height = rnd.getInt(m_config.minTextureSize, m_config.maxTextureSize); glGenTextures(1, &tex); error = glGetError(); if (error != 0) { m_result = RESULT_GEN_TEXTURES_FAILED; m_glError = error; return; } if (m_textures.size() % vectorBlockSize == 0) m_textures.reserve(m_textures.size() + vectorBlockSize); m_textures.push_back(tex); glBindTexture(GL_TEXTURE_2D, tex); error = glGetError(); if (error != 0) { m_result = RESULT_BIND_TEXTURE_FAILED; m_glError = error; return; } if (m_config.useDummyData) { DE_ASSERT((int)m_dummyData.size() >= width*height*4); glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, &(m_dummyData[0])); } else glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, NULL); error = glGetError(); if (error != 0) { m_result = RESULT_TEXTURE_IMAGE_FAILED; m_glError = error; return; } if (m_config.write) glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, 1, 1, GL_RGBA, GL_UNSIGNED_BYTE, &(m_dummyData[0])); error = glGetError(); if (error != 0) { m_result = RESULT_TEXTURE_SUB_IMAGE_FAILED; m_glError = error; return; } if (m_config.use) { try { m_textureRenderer.render(tex); } catch (const glu::Error& err) { m_result = RESULT_RENDER_FAILED; m_glError = err.getError(); return; } catch (const glu::OutOfMemoryError&) { m_result = RESULT_RENDER_FAILED; m_glError = GL_OUT_OF_MEMORY; return; } } glBindTexture(GL_TEXTURE_2D, 0); error = glGetError(); if (error != 0) { m_result = RESULT_BIND_TEXTURE_FAILED; m_glError = error; return; } m_objectCount++; m_bytesRequired += width*height*4; } void MemObjectAllocator::allocateBuffer (de::Random& rnd) { const int vectorBlockSize = 128; deUint32 buffer = 0; deUint32 error = 0; int size = rnd.getInt(m_config.minBufferSize, m_config.maxBufferSize); glGenBuffers(1, &buffer); error = glGetError(); if (error != 0) { m_result = RESULT_GEN_BUFFERS_FAILED; m_glError = error; return; } glBindBuffer(GL_ARRAY_BUFFER, buffer); error = glGetError(); if (error != 0) { m_result = RESULT_BIND_BUFFER_FAILED; m_glError = error; return; } if (m_buffers.size() % vectorBlockSize == 0) m_buffers.reserve(m_buffers.size() + vectorBlockSize); m_buffers.push_back(buffer); if (m_config.useDummyData) { DE_ASSERT((int)m_dummyData.size() >= size); glBufferData(GL_ARRAY_BUFFER, size, &(m_dummyData[0]), GL_DYNAMIC_DRAW); } else glBufferData(GL_ARRAY_BUFFER, size, NULL, GL_DYNAMIC_DRAW); error = glGetError(); if (error != 0) { m_result = RESULT_BUFFER_DATA_FAILED; m_glError = error; return; } if (m_config.write) glBufferSubData(GL_ARRAY_BUFFER, 0, 1, &(m_dummyData[0])); error = glGetError(); if (error != 0) { m_result = RESULT_BUFFER_SUB_DATA_FAILED; m_glError = error; return; } if (m_config.use) { try { m_bufferRenderer.render(buffer, size); } catch (const glu::Error& err) { m_result = RESULT_RENDER_FAILED; m_glError = err.getError(); return; } catch (const glu::OutOfMemoryError&) { m_result = RESULT_RENDER_FAILED; m_glError = GL_OUT_OF_MEMORY; return; } } glBindBuffer(GL_ARRAY_BUFFER, 0); error = glGetError(); if (error != 0) { m_result = RESULT_BIND_BUFFER_FAILED; m_glError = error; return; } m_objectCount++; m_bytesRequired += size; } const char* MemObjectAllocator::resultToString (Result result) { switch (result) { case RESULT_GOT_BAD_ALLOC: return "Caught std::bad_alloc"; break; case RESULT_GEN_TEXTURES_FAILED: return "glGenTextures failed"; break; case RESULT_GEN_BUFFERS_FAILED: return "glGenBuffers failed"; break; case RESULT_BUFFER_DATA_FAILED: return "glBufferData failed"; break; case RESULT_BUFFER_SUB_DATA_FAILED: return "glBufferSubData failed"; break; case RESULT_TEXTURE_IMAGE_FAILED: return "glTexImage2D failed"; break; case RESULT_TEXTURE_SUB_IMAGE_FAILED: return "glTexSubImage2D failed"; break; case RESULT_BIND_TEXTURE_FAILED: return "glBindTexture failed"; break; case RESULT_BIND_BUFFER_FAILED: return "glBindBuffer failed"; break; case RESULT_DELETE_TEXTURES_FAILED: return "glDeleteTextures failed"; break; case RESULT_DELETE_BUFFERS_FAILED: return "glDeleteBuffers failed"; break; case RESULT_RENDER_FAILED: return "Rendering result failed"; break; default: DE_ASSERT(false); return NULL; } } MemoryStressCase::MemoryStressCase (tcu::TestContext& ctx, glu::RenderContext& renderContext, deUint32 objectTypes, int minTextureSize, int maxTextureSize, int minBufferSize, int maxBufferSize, bool write, bool use, bool useDummyData, bool clearAfterOOM, const char* name, const char* desc) : tcu::TestCase (ctx, name, desc) , m_iteration (0) , m_iterationCount (5) , m_objectTypes ((MemObjectType)objectTypes) , m_zeroAlloc (false) , m_clearAfterOOM (clearAfterOOM) , m_renderCtx (renderContext) { m_allocated.reserve(m_iterationCount); m_config.maxTextureSize = maxTextureSize; m_config.minTextureSize = minTextureSize; m_config.maxBufferSize = maxBufferSize; m_config.minBufferSize = minBufferSize; m_config.useDummyData = useDummyData; m_config.write = write; m_config.use = use; } MemoryStressCase::~MemoryStressCase (void) { } void MemoryStressCase::init (void) { if (!m_testCtx.getCommandLine().isOutOfMemoryTestEnabled()) { m_testCtx.getLog() << TestLog::Message << "Tests that exhaust memory are disabled, use --deqp-test-oom=enable command line option to enable." << TestLog::EndMessage; throw tcu::NotSupportedError("OOM tests disabled"); } } void MemoryStressCase::deinit (void) { TCU_CHECK(!m_zeroAlloc); } tcu::TestCase::IterateResult MemoryStressCase::iterate (void) { bool end = false; tcu::TestLog& log = m_testCtx.getLog(); MemObjectAllocator allocator(log, m_renderCtx, m_objectTypes, m_config, deStringHash(getName())); if (!allocator.allocUntilFailure()) { // Allocation timed out allocator.clearObjects(); log << TestLog::Message << "Timeout. Couldn't exhaust memory in timelimit. Allocated " << allocator.getObjectCount() << " objects." << TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); return STOP; } // Try to cancel rendering operations if (m_clearAfterOOM) GLU_CHECK_CALL(glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT)); allocator.clearObjects(); m_allocated.push_back(allocator.getObjectCount()); if (m_iteration != 0 && allocator.getObjectCount() == 0) m_zeroAlloc = true; log << TestLog::Message << "Got error when allocation object count: " << allocator.getObjectCount() << " bytes: " << allocator.getBytes() << TestLog::EndMessage; if ((allocator.getGLError() == 0) && (allocator.getResult() == MemObjectAllocator::RESULT_GOT_BAD_ALLOC)) { log << TestLog::Message << "std::bad_alloc" << TestLog::EndMessage; end = true; m_testCtx.setTestResult(QP_TEST_RESULT_RESOURCE_ERROR, "Memory allocation failed"); } else if (allocator.getGLError() != GL_OUT_OF_MEMORY) { log << TestLog::Message << "Invalid Error " << MemObjectAllocator::resultToString(allocator.getResult()) << " GLError: " << glErrorToString(allocator.getGLError()) << TestLog::EndMessage; end = true; m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail"); } if ((m_iteration+1) == m_iterationCount) { int min = m_allocated[0]; int max = m_allocated[0]; float threshold = 50.0f; for (int allocNdx = 0; allocNdx < (int)m_allocated.size(); allocNdx++) { min = deMin32(m_allocated[allocNdx], min); max = deMax32(m_allocated[allocNdx], max); } if (min == 0 && max != 0) { log << TestLog::Message << "Allocation count zero" << TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail"); } else { const float change = (float)(min - max) / (float)(max); if (change > threshold) { log << TestLog::Message << "Allocated objects max: " << max << ", min: " << min << ", difference: " << change << "% threshold: " << threshold << "%" << TestLog::EndMessage; m_testCtx.setTestResult(QP_TEST_RESULT_QUALITY_WARNING, "Allocation count variation"); } else m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass"); } end = true; } GLU_CHECK_CALL(glFinish()); m_iteration++; if (end) return STOP; else return CONTINUE; } } // gls } // deqp