| /*------------------------------------------------------------------------- |
| * 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 Fragment operation test utilities. |
| *//*--------------------------------------------------------------------*/ |
| |
| #include "glsFragmentOpUtil.hpp" |
| #include "gluRenderContext.hpp" |
| #include "gluShaderProgram.hpp" |
| #include "gluDrawUtil.hpp" |
| #include "glwFunctions.hpp" |
| #include "glwEnums.hpp" |
| |
| namespace deqp |
| { |
| namespace gls |
| { |
| namespace FragmentOpUtil |
| { |
| |
| template<typename T> |
| inline T triQuadInterpolate (const T values[4], float xFactor, float yFactor) |
| { |
| if (xFactor + yFactor < 1.0f) |
| return values[0] + (values[2]-values[0])*xFactor + (values[1]-values[0])*yFactor; |
| else |
| return values[3] + (values[1]-values[3])*(1.0f-xFactor) + (values[2]-values[3])*(1.0f-yFactor); |
| } |
| |
| // GLSL ES 1.0 shaders |
| static const char* s_glsl1VertSrc = |
| "attribute highp vec4 a_position;\n" |
| "attribute mediump vec4 a_color;\n" |
| "varying mediump vec4 v_color;\n" |
| "void main()\n" |
| "{\n" |
| " gl_Position = a_position;\n" |
| " v_color = a_color;\n" |
| "}\n"; |
| static const char* s_glsl1FragSrc = |
| "varying mediump vec4 v_color;\n" |
| "void main()\n" |
| "{\n" |
| " gl_FragColor = v_color;\n" |
| "}\n"; |
| |
| // GLSL ES 3.0 shaders |
| static const char* s_glsl3VertSrc = |
| "#version 300 es\n" |
| "in highp vec4 a_position;\n" |
| "in mediump vec4 a_color;\n" |
| "out mediump vec4 v_color;\n" |
| "void main()\n" |
| "{\n" |
| " gl_Position = a_position;\n" |
| " v_color = a_color;\n" |
| "}\n"; |
| static const char* s_glsl3FragSrc = |
| "#version 300 es\n" |
| "in mediump vec4 v_color;\n" |
| "layout(location = 0) out mediump vec4 o_color;\n" |
| "void main()\n" |
| "{\n" |
| " o_color = v_color;\n" |
| "}\n"; |
| |
| // GLSL 3.3 shaders |
| static const char* s_glsl33VertSrc = |
| "#version 330 core\n" |
| "in vec4 a_position;\n" |
| "in vec4 a_color;\n" |
| "in vec4 a_color1;\n" |
| "out vec4 v_color;\n" |
| "out vec4 v_color1;\n" |
| "void main()\n" |
| "{\n" |
| " gl_Position = a_position;\n" |
| " v_color = a_color;\n" |
| " v_color1 = a_color1;\n" |
| "}\n"; |
| static const char* s_glsl33FragSrc = |
| "#version 330 core\n" |
| "in vec4 v_color;\n" |
| "in vec4 v_color1;\n" |
| "layout(location = 0, index = 0) out vec4 o_color;\n" |
| "layout(location = 0, index = 1) out vec4 o_color1;\n" |
| "void main()\n" |
| "{\n" |
| " o_color = v_color;\n" |
| " o_color1 = v_color1;\n" |
| "}\n"; |
| |
| static const char* getVertSrc (glu::GLSLVersion glslVersion) |
| { |
| if (glslVersion == glu::GLSL_VERSION_100_ES) |
| return s_glsl1VertSrc; |
| else if (glslVersion == glu::GLSL_VERSION_300_ES) |
| return s_glsl3VertSrc; |
| else if (glslVersion == glu::GLSL_VERSION_330) |
| return s_glsl33VertSrc; |
| |
| DE_ASSERT(DE_FALSE); |
| return 0; |
| } |
| |
| static const char* getFragSrc (glu::GLSLVersion glslVersion) |
| { |
| if (glslVersion == glu::GLSL_VERSION_100_ES) |
| return s_glsl1FragSrc; |
| else if (glslVersion == glu::GLSL_VERSION_300_ES) |
| return s_glsl3FragSrc; |
| else if (glslVersion == glu::GLSL_VERSION_330) |
| return s_glsl33FragSrc; |
| |
| DE_ASSERT(DE_FALSE); |
| return 0; |
| } |
| |
| QuadRenderer::QuadRenderer (const glu::RenderContext& context, glu::GLSLVersion glslVersion) |
| : m_context (context) |
| , m_program (DE_NULL) |
| , m_positionLoc (0) |
| , m_colorLoc (-1) |
| , m_color1Loc (-1) |
| , m_blendFuncExt (!glu::glslVersionIsES(glslVersion) && (glslVersion >= glu::GLSL_VERSION_330)) |
| { |
| DE_ASSERT(glslVersion == glu::GLSL_VERSION_100_ES || |
| glslVersion == glu::GLSL_VERSION_300_ES || |
| glslVersion == glu::GLSL_VERSION_330); |
| |
| const glw::Functions& gl = context.getFunctions(); |
| const char* vertSrc = getVertSrc(glslVersion); |
| const char* fragSrc = getFragSrc(glslVersion); |
| |
| m_program = new glu::ShaderProgram(m_context, glu::makeVtxFragSources(vertSrc, fragSrc)); |
| if (!m_program->isOk()) |
| { |
| delete m_program; |
| throw tcu::TestError("Failed to compile program", DE_NULL, __FILE__, __LINE__); |
| } |
| |
| m_positionLoc = gl.getAttribLocation(m_program->getProgram(), "a_position"); |
| m_colorLoc = gl.getAttribLocation(m_program->getProgram(), "a_color"); |
| |
| if (m_blendFuncExt) |
| m_color1Loc = gl.getAttribLocation(m_program->getProgram(), "a_color1"); |
| |
| if (m_positionLoc < 0 || m_colorLoc < 0 || (m_blendFuncExt && m_color1Loc < 0)) |
| { |
| delete m_program; |
| throw tcu::TestError("Invalid attribute locations", DE_NULL, __FILE__, __LINE__); |
| } |
| } |
| |
| QuadRenderer::~QuadRenderer (void) |
| { |
| delete m_program; |
| } |
| |
| void QuadRenderer::render (const Quad& quad) const |
| { |
| const float position[] = |
| { |
| quad.posA.x(), quad.posA.y(), quad.depth[0], 1.0f, |
| quad.posA.x(), quad.posB.y(), quad.depth[1], 1.0f, |
| quad.posB.x(), quad.posA.y(), quad.depth[2], 1.0f, |
| quad.posB.x(), quad.posB.y(), quad.depth[3], 1.0f |
| }; |
| const deUint8 indices[] = { 0, 2, 1, 1, 2, 3 }; |
| |
| DE_STATIC_ASSERT(sizeof(tcu::Vec4) == sizeof(float)*4); |
| DE_STATIC_ASSERT(sizeof(quad.color) == sizeof(float)*4*4); |
| DE_STATIC_ASSERT(sizeof(quad.color1) == sizeof(float)*4*4); |
| |
| std::vector<glu::VertexArrayBinding> vertexArrays; |
| |
| vertexArrays.push_back(glu::va::Float(m_positionLoc, 4, 4, 0, &position[0])); |
| vertexArrays.push_back(glu::va::Float(m_colorLoc, 4, 4, 0, (const float*)&quad.color[0])); |
| |
| if (m_blendFuncExt) |
| vertexArrays.push_back(glu::va::Float(m_color1Loc, 4, 4, 0, (const float*)&quad.color1[0])); |
| |
| m_context.getFunctions().useProgram(m_program->getProgram()); |
| glu::draw(m_context, m_program->getProgram(), |
| (int)vertexArrays.size(), &vertexArrays[0], |
| glu::pr::Triangles(DE_LENGTH_OF_ARRAY(indices), &indices[0])); |
| } |
| |
| ReferenceQuadRenderer::ReferenceQuadRenderer (void) |
| : m_fragmentBufferSize(0) |
| { |
| for (int i = 0; i < DE_LENGTH_OF_ARRAY(m_fragmentDepths); i++) |
| m_fragmentDepths[i] = 0.0f; |
| } |
| |
| void ReferenceQuadRenderer::flushFragmentBuffer (const rr::MultisamplePixelBufferAccess& colorBuffer, |
| const rr::MultisamplePixelBufferAccess& depthBuffer, |
| const rr::MultisamplePixelBufferAccess& stencilBuffer, |
| rr::FaceType faceType, |
| const rr::FragmentOperationState& state) |
| { |
| m_fragmentProcessor.render(colorBuffer, depthBuffer, stencilBuffer, &m_fragmentBuffer[0], m_fragmentBufferSize, faceType, state); |
| m_fragmentBufferSize = 0; |
| } |
| |
| void ReferenceQuadRenderer::render (const tcu::PixelBufferAccess& colorBuffer, |
| const tcu::PixelBufferAccess& depthBuffer, |
| const tcu::PixelBufferAccess& stencilBuffer, |
| const IntegerQuad& quad, |
| const rr::FragmentOperationState& state) |
| { |
| bool flipX = quad.posA.x() > quad.posB.x(); |
| bool flipY = quad.posA.y() > quad.posB.y(); |
| rr::FaceType faceType = flipX == flipY ? rr::FACETYPE_FRONT : rr::FACETYPE_BACK; |
| int xFirst = flipX ? quad.posB.x() : quad.posA.x(); |
| int xLast = flipX ? quad.posA.x() : quad.posB.x(); |
| int yFirst = flipY ? quad.posB.y() : quad.posA.y(); |
| int yLast = flipY ? quad.posA.y() : quad.posB.y(); |
| float width = (float)(xLast - xFirst + 1); |
| float height = (float)(yLast - yFirst + 1); |
| |
| for (int y = yFirst; y <= yLast; y++) |
| { |
| // Interpolation factor for y. |
| float yRatio = (0.5f + (float)(y - yFirst)) / height; |
| if (flipY) |
| yRatio = 1.0f - yRatio; |
| |
| for (int x = xFirst; x <= xLast; x++) |
| { |
| // Interpolation factor for x. |
| float xRatio = (0.5f + (float)(x - xFirst)) / width; |
| if (flipX) |
| xRatio = 1.0f - xRatio; |
| |
| tcu::Vec4 color = triQuadInterpolate(quad.color, xRatio, yRatio); |
| tcu::Vec4 color1 = triQuadInterpolate(quad.color1, xRatio, yRatio); |
| float depth = triQuadInterpolate(quad.depth, xRatio, yRatio); |
| |
| // Interpolated color and depth. |
| |
| DE_STATIC_ASSERT(MAX_FRAGMENT_BUFFER_SIZE == DE_LENGTH_OF_ARRAY(m_fragmentBuffer)); |
| |
| if (m_fragmentBufferSize >= MAX_FRAGMENT_BUFFER_SIZE) |
| flushFragmentBuffer(rr::MultisamplePixelBufferAccess::fromMultisampleAccess(colorBuffer), |
| rr::MultisamplePixelBufferAccess::fromMultisampleAccess(depthBuffer), |
| rr::MultisamplePixelBufferAccess::fromMultisampleAccess(stencilBuffer), faceType, state); |
| |
| m_fragmentDepths[m_fragmentBufferSize] = depth; |
| m_fragmentBuffer[m_fragmentBufferSize] = rr::Fragment(tcu::IVec2(x, y), rr::GenericVec4(color), rr::GenericVec4(color1), 1u /* coverage mask */, &m_fragmentDepths[m_fragmentBufferSize]); |
| m_fragmentBufferSize++; |
| } |
| } |
| |
| flushFragmentBuffer(rr::MultisamplePixelBufferAccess::fromMultisampleAccess(colorBuffer), |
| rr::MultisamplePixelBufferAccess::fromMultisampleAccess(depthBuffer), |
| rr::MultisamplePixelBufferAccess::fromMultisampleAccess(stencilBuffer), faceType, state); |
| } |
| |
| tcu::PixelBufferAccess getMultisampleAccess(const tcu::PixelBufferAccess& original) |
| { |
| return tcu::PixelBufferAccess(original.getFormat(), |
| 1, |
| original.getWidth(), |
| original.getHeight(), |
| original.getFormat().getPixelSize(), |
| original.getRowPitch(), |
| original.getDataPtr()); |
| } |
| |
| tcu::ConstPixelBufferAccess getMultisampleAccess(const tcu::ConstPixelBufferAccess& original) |
| { |
| return tcu::ConstPixelBufferAccess(original.getFormat(), |
| 1, |
| original.getWidth(), |
| original.getHeight(), |
| original.getFormat().getPixelSize(), |
| original.getRowPitch(), |
| original.getDataPtr()); |
| } |
| |
| } // FragmentOpUtil |
| } // gls |
| } // deqp |