/*------------------------------------------------------------------------- * drawElements Quality Program OpenGL ES 3.1 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 Tests for separate shader objects *//*--------------------------------------------------------------------*/ #include "es31fSeparateShaderTests.hpp" #include "deInt32.h" #include "deString.h" #include "deStringUtil.hpp" #include "deUniquePtr.hpp" #include "deRandom.hpp" #include "deSTLUtil.hpp" #include "tcuCommandLine.hpp" #include "tcuImageCompare.hpp" #include "tcuRenderTarget.hpp" #include "tcuResultCollector.hpp" #include "tcuRGBA.hpp" #include "tcuSurface.hpp" #include "tcuStringTemplate.hpp" #include "gluCallLogWrapper.hpp" #include "gluPixelTransfer.hpp" #include "gluRenderContext.hpp" #include "gluShaderProgram.hpp" #include "gluVarType.hpp" #include "glsShaderLibrary.hpp" #include "glwFunctions.hpp" #include "glwDefs.hpp" #include "glwEnums.hpp" #include <cstdarg> #include <algorithm> #include <map> #include <sstream> #include <string> #include <set> #include <vector> namespace deqp { namespace gles31 { namespace Functional { namespace { using std::map; using std::set; using std::ostringstream; using std::string; using std::vector; using de::MovePtr; using de::Random; using de::UniquePtr; using tcu::MessageBuilder; using tcu::RenderTarget; using tcu::StringTemplate; using tcu::Surface; using tcu::TestLog; using tcu::ResultCollector; using glu::CallLogWrapper; using glu::DataType; using glu::VariableDeclaration; using glu::Precision; using glu::Program; using glu::ProgramPipeline; using glu::ProgramSources; using glu::RenderContext; using glu::ShaderProgram; using glu::ShaderType; using glu::Storage; using glu::VarType; using glu::VertexSource; using glu::FragmentSource; using glu::ProgramSeparable; using namespace glw; #define LOG_CALL(CALL) do \ { \ enableLogging(true); \ CALL; \ enableLogging(false); \ } while (deGetFalse()) enum { VIEWPORT_SIZE = 128 }; enum VaryingInterpolation { VARYINGINTERPOLATION_SMOOTH = 0, VARYINGINTERPOLATION_FLAT, VARYINGINTERPOLATION_CENTROID, VARYINGINTERPOLATION_DEFAULT, VARYINGINTERPOLATION_RANDOM, VARYINGINTERPOLATION_LAST }; DataType randomType (Random& rnd) { using namespace glu; if (rnd.getInt(0, 7) == 0) { const int numCols = rnd.getInt(2, 4), numRows = rnd.getInt(2, 4); return getDataTypeMatrix(numCols, numRows); } else { static const DataType s_types[] = { TYPE_FLOAT, TYPE_INT, TYPE_UINT }; static const float s_weights[] = { 3.0, 1.0, 1.0 }; const int size = rnd.getInt(1, 4); const DataType scalarType = rnd.chooseWeighted<DataType>( DE_ARRAY_BEGIN(s_types), DE_ARRAY_END(s_types), DE_ARRAY_BEGIN(s_weights)); return getDataTypeVector(scalarType, size); } DE_FATAL("Impossible"); return TYPE_INVALID; } VaryingInterpolation randomInterpolation (Random& rnd) { static const VaryingInterpolation s_validInterpolations[] = { VARYINGINTERPOLATION_SMOOTH, VARYINGINTERPOLATION_FLAT, VARYINGINTERPOLATION_CENTROID, VARYINGINTERPOLATION_DEFAULT, }; return s_validInterpolations[rnd.getInt(0, DE_LENGTH_OF_ARRAY(s_validInterpolations)-1)]; } glu::Interpolation getGluInterpolation (VaryingInterpolation interpolation) { switch (interpolation) { case VARYINGINTERPOLATION_SMOOTH: return glu::INTERPOLATION_SMOOTH; case VARYINGINTERPOLATION_FLAT: return glu::INTERPOLATION_FLAT; case VARYINGINTERPOLATION_CENTROID: return glu::INTERPOLATION_CENTROID; case VARYINGINTERPOLATION_DEFAULT: return glu::INTERPOLATION_LAST; //!< Last means no qualifier, i.e. default default: DE_FATAL("Invalid interpolation"); return glu::INTERPOLATION_LAST; } } // used only for debug sanity checks #if defined(DE_DEBUG) VaryingInterpolation getVaryingInterpolation (glu::Interpolation interpolation) { switch (interpolation) { case glu::INTERPOLATION_SMOOTH: return VARYINGINTERPOLATION_SMOOTH; case glu::INTERPOLATION_FLAT: return VARYINGINTERPOLATION_FLAT; case glu::INTERPOLATION_CENTROID: return VARYINGINTERPOLATION_CENTROID; case glu::INTERPOLATION_LAST: return VARYINGINTERPOLATION_DEFAULT; //!< Last means no qualifier, i.e. default default: DE_FATAL("Invalid interpolation"); return VARYINGINTERPOLATION_LAST; } } #endif enum BindingKind { BINDING_NAME, BINDING_LOCATION, BINDING_LAST }; BindingKind randomBinding (Random& rnd) { return rnd.getBool() ? BINDING_LOCATION : BINDING_NAME; } void printInputColor (ostringstream& oss, const VariableDeclaration& input) { using namespace glu; const DataType basicType = input.varType.getBasicType(); string exp = input.name; switch (getDataTypeScalarType(basicType)) { case TYPE_FLOAT: break; case TYPE_INT: case TYPE_UINT: { DataType floatType = getDataTypeFloatScalars(basicType); exp = string() + "(" + getDataTypeName(floatType) + "(" + exp + ") / 255.0" + ")"; break; } default: DE_FATAL("Impossible"); } if (isDataTypeScalarOrVector(basicType)) { switch (getDataTypeScalarSize(basicType)) { case 1: oss << "hsv(vec3(" << exp << ", 1.0, 1.0))"; break; case 2: oss << "hsv(vec3(" << exp << ", 1.0))"; break; case 3: oss << "vec4(" << exp << ", 1.0)"; break; case 4: oss << exp; break; default: DE_FATAL("Impossible"); } } else if (isDataTypeMatrix(basicType)) { int rows = getDataTypeMatrixNumRows(basicType); int columns = getDataTypeMatrixNumColumns(basicType); if (rows == columns) oss << "hsv(vec3(determinant(" << exp << ")))"; else { if (rows != 3 && columns >= 3) { exp = "transpose(" + exp + ")"; std::swap(rows, columns); } exp = exp + "[0]"; if (rows > 3) exp = exp + ".xyz"; oss << "hsv(" << exp << ")"; } } else DE_FATAL("Impossible"); } // Representation for the varyings between vertex and fragment shaders struct VaryingParams { VaryingParams (void) : count (0) , type (glu::TYPE_LAST) , binding (BINDING_LAST) , vtxInterp (VARYINGINTERPOLATION_LAST) , frgInterp (VARYINGINTERPOLATION_LAST) {} int count; DataType type; BindingKind binding; VaryingInterpolation vtxInterp; VaryingInterpolation frgInterp; }; struct VaryingInterface { vector<VariableDeclaration> vtxOutputs; vector<VariableDeclaration> frgInputs; }; // Generate corresponding input and output variable declarations that may vary // in compatible ways. VaryingInterpolation chooseInterpolation (VaryingInterpolation param, DataType type, Random& rnd) { if (glu::getDataTypeScalarType(type) != glu::TYPE_FLOAT) return VARYINGINTERPOLATION_FLAT; if (param == VARYINGINTERPOLATION_RANDOM) return randomInterpolation(rnd); return param; } bool isSSOCompatibleInterpolation (VaryingInterpolation vertexInterpolation, VaryingInterpolation fragmentInterpolation) { // interpolations must be fully specified DE_ASSERT(vertexInterpolation != VARYINGINTERPOLATION_RANDOM); DE_ASSERT(vertexInterpolation < VARYINGINTERPOLATION_LAST); DE_ASSERT(fragmentInterpolation != VARYINGINTERPOLATION_RANDOM); DE_ASSERT(fragmentInterpolation < VARYINGINTERPOLATION_LAST); // interpolation can only be either smooth or flat. Auxiliary storage does not matter. const bool isSmoothVtx = (vertexInterpolation == VARYINGINTERPOLATION_SMOOTH) || //!< trivial (vertexInterpolation == VARYINGINTERPOLATION_DEFAULT) || //!< default to smooth (vertexInterpolation == VARYINGINTERPOLATION_CENTROID); //!< default to smooth, ignore storage const bool isSmoothFrag = (fragmentInterpolation == VARYINGINTERPOLATION_SMOOTH) || //!< trivial (fragmentInterpolation == VARYINGINTERPOLATION_DEFAULT) || //!< default to smooth (fragmentInterpolation == VARYINGINTERPOLATION_CENTROID); //!< default to smooth, ignore storage // Khronos bug #12630: flat / smooth qualifiers must match in SSO return isSmoothVtx == isSmoothFrag; } VaryingInterface genVaryingInterface (const VaryingParams& params, Random& rnd) { using namespace glu; VaryingInterface ret; int offset = 0; for (int varNdx = 0; varNdx < params.count; ++varNdx) { const BindingKind binding = ((params.binding == BINDING_LAST) ? randomBinding(rnd) : params.binding); const DataType type = ((params.type == TYPE_LAST) ? randomType(rnd) : params.type); const VaryingInterpolation vtxInterp = chooseInterpolation(params.vtxInterp, type, rnd); const VaryingInterpolation frgInterp = chooseInterpolation(params.frgInterp, type, rnd); const VaryingInterpolation vtxCompatInterp = (isSSOCompatibleInterpolation(vtxInterp, frgInterp)) ? (vtxInterp) : (frgInterp); const int loc = ((binding == BINDING_LOCATION) ? offset : -1); const string ndxStr = de::toString(varNdx); const string vtxName = ((binding == BINDING_NAME) ? "var" + ndxStr : "vtxVar" + ndxStr); const string frgName = ((binding == BINDING_NAME) ? "var" + ndxStr : "frgVar" + ndxStr); const VarType varType (type, PRECISION_HIGHP); offset += getDataTypeNumLocations(type); // Over 16 locations aren't necessarily supported, so halt here. if (offset > 16) break; ret.vtxOutputs.push_back( VariableDeclaration(varType, vtxName, STORAGE_OUT, getGluInterpolation(vtxCompatInterp), loc)); ret.frgInputs.push_back( VariableDeclaration(varType, frgName, STORAGE_IN, getGluInterpolation(frgInterp), loc)); } return ret; } // Create vertex output variable declarations that are maximally compatible // with the fragment input variables. vector<VariableDeclaration> varyingCompatVtxOutputs (const VaryingInterface& varyings) { vector<VariableDeclaration> outputs = varyings.vtxOutputs; for (size_t i = 0; i < outputs.size(); ++i) { outputs[i].interpolation = varyings.frgInputs[i].interpolation; outputs[i].name = varyings.frgInputs[i].name; } return outputs; } // Shader source generation void printFloat (ostringstream& oss, double d) { oss.setf(oss.fixed | oss.internal); oss.precision(4); oss.width(7); oss << d; } void printFloatDeclaration (ostringstream& oss, const string& varName, bool uniform, GLfloat value = 0.0) { using namespace glu; const VarType varType (TYPE_FLOAT, PRECISION_HIGHP); if (uniform) oss << VariableDeclaration(varType, varName, STORAGE_UNIFORM) << ";\n"; else oss << VariableDeclaration(varType, varName, STORAGE_CONST) << " = " << de::floatToString(value, 6) << ";\n"; } void printRandomInitializer (ostringstream& oss, DataType type, Random& rnd) { using namespace glu; const int size = getDataTypeScalarSize(type); if (size > 0) oss << getDataTypeName(type) << "("; for (int i = 0; i < size; ++i) { oss << (i == 0 ? "" : ", "); switch (getDataTypeScalarType(type)) { case TYPE_FLOAT: printFloat(oss, rnd.getInt(0, 16) / 16.0); break; case TYPE_INT: case TYPE_UINT: oss << rnd.getInt(0, 255); break; case TYPE_BOOL: oss << (rnd.getBool() ? "true" : "false"); break; default: DE_FATAL("Impossible"); } } if (size > 0) oss << ")"; } string genVtxShaderSrc (deUint32 seed, const vector<VariableDeclaration>& outputs, const string& varName, bool uniform, float value = 0.0) { ostringstream oss; Random rnd (seed); enum { NUM_COMPONENTS = 2 }; static const int s_quadrants[][NUM_COMPONENTS] = { {1, 1}, {-1, 1}, {1, -1} }; oss << "#version 310 es\n"; printFloatDeclaration(oss, varName, uniform, value); for (vector<VariableDeclaration>::const_iterator it = outputs.begin(); it != outputs.end(); ++it) oss << *it << ";\n"; oss << "const vec2 triangle[3] = vec2[3](\n"; for (int vertexNdx = 0; vertexNdx < DE_LENGTH_OF_ARRAY(s_quadrants); ++vertexNdx) { oss << "\tvec2("; for (int componentNdx = 0; componentNdx < NUM_COMPONENTS; ++componentNdx) { printFloat(oss, s_quadrants[vertexNdx][componentNdx] * rnd.getInt(4,16) / 16.0); oss << (componentNdx < 1 ? ", " : ""); } oss << ")" << (vertexNdx < 2 ? "," : "") << "\n"; } oss << ");\n"; for (vector<VariableDeclaration>::const_iterator it = outputs.begin(); it != outputs.end(); ++it) { const DataType type = it->varType.getBasicType(); const string typeName = glu::getDataTypeName(type); oss << "const " << typeName << " " << it->name << "Inits[3] = " << typeName << "[3](\n"; for (int i = 0; i < 3; ++i) { oss << (i == 0 ? "\t" : ",\n\t"); printRandomInitializer(oss, type, rnd); } oss << ");\n"; } oss << "void main (void)\n" << "{\n" << "\tgl_Position = vec4(" << varName << " * triangle[gl_VertexID], 0.0, 1.0);\n"; for (vector<VariableDeclaration>::const_iterator it = outputs.begin(); it != outputs.end(); ++it) oss << "\t" << it->name << " = " << it->name << "Inits[gl_VertexID];\n"; oss << "}\n"; return oss.str(); } string genFrgShaderSrc (deUint32 seed, const vector<VariableDeclaration>& inputs, const string& varName, bool uniform, float value = 0.0) { Random rnd (seed); ostringstream oss; oss.precision(4); oss.width(7); oss << "#version 310 es\n"; oss << "precision highp float;\n"; oss << "out vec4 fragColor;\n"; printFloatDeclaration(oss, varName, uniform, value); for (vector<VariableDeclaration>::const_iterator it = inputs.begin(); it != inputs.end(); ++it) oss << *it << ";\n"; // glsl % isn't defined for negative numbers oss << "int imod (int n, int d)" << "\n" << "{" << "\n" << "\t" << "return (n < 0 ? d - 1 - (-1 - n) % d : n % d);" << "\n" << "}" << "\n"; oss << "vec4 hsv (vec3 hsv)" << "{" << "\n" << "\tfloat h = hsv.x * 3.0;\n" << "\tfloat r = max(0.0, 1.0 - h) + max(0.0, h - 2.0);\n" << "\tfloat g = max(0.0, 1.0 - abs(h - 1.0));\n" << "\tfloat b = max(0.0, 1.0 - abs(h - 2.0));\n" << "\tvec3 hs = mix(vec3(1.0), vec3(r, g, b), hsv.y);\n" << "\treturn vec4(hsv.z * hs, 1.0);\n" << "}\n"; oss << "void main (void)\n" << "{\n"; oss << "\t" << "fragColor = vec4(vec3(" << varName << "), 1.0);" << "\n"; if (inputs.size() > 0) { oss << "\t" << "switch (imod(int(0.5 * ("; printFloat(oss, rnd.getFloat(0.5f, 2.0f)); oss << " * gl_FragCoord.x - "; printFloat(oss, rnd.getFloat(0.5f, 2.0f)); oss << " * gl_FragCoord.y)), " << inputs.size() << "))" << "\n" << "\t" << "{" << "\n"; for (size_t i = 0; i < inputs.size(); ++i) { oss << "\t\t" << "case " << i << ":" << "\n" << "\t\t\t" << "fragColor *= "; printInputColor(oss, inputs[i]); oss << ";" << "\n" << "\t\t\t" << "break;" << "\n"; } oss << "\t\t" << "case " << inputs.size() << ":\n" << "\t\t\t" << "fragColor = vec4(1.0, 0.0, 1.0, 1.0);" << "\n"; oss << "\t\t\t" << "break;" << "\n"; oss << "\t\t" << "case -1:\n" << "\t\t\t" << "fragColor = vec4(1.0, 1.0, 0.0, 1.0);" << "\n"; oss << "\t\t\t" << "break;" << "\n"; oss << "\t\t" << "default:" << "\n" << "\t\t\t" << "fragColor = vec4(1.0, 1.0, 0.0, 1.0);" << "\n"; oss << "\t" << "}\n"; } oss << "}\n"; return oss.str(); } // ProgramWrapper class ProgramWrapper { public: virtual ~ProgramWrapper (void) {} virtual GLuint getProgramName (void) = 0; virtual void writeToLog (TestLog& log) = 0; }; class ShaderProgramWrapper : public ProgramWrapper { public: ShaderProgramWrapper (const RenderContext& renderCtx, const ProgramSources& sources) : m_shaderProgram (renderCtx, sources) {} ~ShaderProgramWrapper (void) {} GLuint getProgramName (void) { return m_shaderProgram.getProgram(); } ShaderProgram& getShaderProgram (void) { return m_shaderProgram; } void writeToLog (TestLog& log) { log << m_shaderProgram; } private: ShaderProgram m_shaderProgram; }; class RawProgramWrapper : public ProgramWrapper { public: RawProgramWrapper (const RenderContext& renderCtx, GLuint programName, ShaderType shaderType, const string& source) : m_program (renderCtx, programName) , m_shaderType (shaderType) , m_source (source) {} ~RawProgramWrapper (void) {} GLuint getProgramName (void) { return m_program.getProgram(); } Program& getProgram (void) { return m_program; } void writeToLog (TestLog& log); private: Program m_program; ShaderType m_shaderType; const string m_source; }; void RawProgramWrapper::writeToLog (TestLog& log) { const string info = m_program.getInfoLog(); qpShaderType qpType = glu::getLogShaderType(m_shaderType); log << TestLog::ShaderProgram(true, info) << TestLog::Shader(qpType, m_source, true, "[Shader created by glCreateShaderProgramv()]") << TestLog::EndShaderProgram; } // ProgramParams struct ProgramParams { ProgramParams (deUint32 vtxSeed_, GLfloat vtxScale_, deUint32 frgSeed_, GLfloat frgScale_) : vtxSeed (vtxSeed_) , vtxScale (vtxScale_) , frgSeed (frgSeed_) , frgScale (frgScale_) {} deUint32 vtxSeed; GLfloat vtxScale; deUint32 frgSeed; GLfloat frgScale; }; ProgramParams genProgramParams (Random& rnd) { const deUint32 vtxSeed = rnd.getUint32(); const GLfloat vtxScale = (float)rnd.getInt(8, 16) / 16.0f; const deUint32 frgSeed = rnd.getUint32(); const GLfloat frgScale = (float)rnd.getInt(0, 16) / 16.0f; return ProgramParams(vtxSeed, vtxScale, frgSeed, frgScale); } // TestParams struct TestParams { bool initSingle; bool switchVtx; bool switchFrg; bool useUniform; bool useSameName; bool useCreateHelper; bool useProgramUniform; VaryingParams varyings; }; deUint32 paramsSeed (const TestParams& params) { deUint32 paramCode = (params.initSingle << 0 | params.switchVtx << 1 | params.switchFrg << 2 | params.useUniform << 3 | params.useSameName << 4 | params.useCreateHelper << 5 | params.useProgramUniform << 6); paramCode = deUint32Hash(paramCode) + params.varyings.count; paramCode = deUint32Hash(paramCode) + params.varyings.type; paramCode = deUint32Hash(paramCode) + params.varyings.binding; paramCode = deUint32Hash(paramCode) + params.varyings.vtxInterp; paramCode = deUint32Hash(paramCode) + params.varyings.frgInterp; return deUint32Hash(paramCode); } string paramsCode (const TestParams& params) { using namespace glu; ostringstream oss; oss << (params.initSingle ? "1" : "2") << (params.switchVtx ? "v" : "") << (params.switchFrg ? "f" : "") << (params.useProgramUniform ? "p" : "") << (params.useUniform ? "u" : "") << (params.useSameName ? "s" : "") << (params.useCreateHelper ? "c" : "") << de::toString(params.varyings.count) << (params.varyings.binding == BINDING_NAME ? "n" : params.varyings.binding == BINDING_LOCATION ? "l" : params.varyings.binding == BINDING_LAST ? "r" : "") << (params.varyings.vtxInterp == VARYINGINTERPOLATION_SMOOTH ? "m" : params.varyings.vtxInterp == VARYINGINTERPOLATION_CENTROID ? "e" : params.varyings.vtxInterp == VARYINGINTERPOLATION_FLAT ? "a" : params.varyings.vtxInterp == VARYINGINTERPOLATION_RANDOM ? "r" : "o") << (params.varyings.frgInterp == VARYINGINTERPOLATION_SMOOTH ? "m" : params.varyings.frgInterp == VARYINGINTERPOLATION_CENTROID ? "e" : params.varyings.frgInterp == VARYINGINTERPOLATION_FLAT ? "a" : params.varyings.frgInterp == VARYINGINTERPOLATION_RANDOM ? "r" : "o"); return oss.str(); } bool paramsValid (const TestParams& params) { using namespace glu; // Final pipeline has a single program? if (params.initSingle) { // Cannot have conflicting names for uniforms or constants if (params.useSameName) return false; // CreateShaderProgram would never get called if (!params.switchVtx && !params.switchFrg && params.useCreateHelper) return false; // Must switch either all or nothing if (params.switchVtx != params.switchFrg) return false; } // ProgramUniform would never get called if (params.useProgramUniform && !params.useUniform) return false; // Interpolation is meaningless if we don't use an in/out variable. if (params.varyings.count == 0 && !(params.varyings.vtxInterp == VARYINGINTERPOLATION_LAST && params.varyings.frgInterp == VARYINGINTERPOLATION_LAST)) return false; // Mismatch by flat / smooth is not allowed. See Khronos bug #12630 // \note: iterpolations might be RANDOM, causing generated varyings potentially match / mismatch anyway. // This is checked later on. Here, we just make sure that we don't force the generator to generate // only invalid varying configurations, i.e. there exists a valid varying configuration for this // test param config. if ((params.varyings.vtxInterp != VARYINGINTERPOLATION_RANDOM) && (params.varyings.frgInterp != VARYINGINTERPOLATION_RANDOM) && (params.varyings.vtxInterp == VARYINGINTERPOLATION_FLAT) != (params.varyings.frgInterp == VARYINGINTERPOLATION_FLAT)) return false; return true; } // used only for debug sanity checks #if defined(DE_DEBUG) bool varyingsValid (const VaryingInterface& varyings) { for (int ndx = 0; ndx < (int)varyings.vtxOutputs.size(); ++ndx) { const VaryingInterpolation vertexInterpolation = getVaryingInterpolation(varyings.vtxOutputs[ndx].interpolation); const VaryingInterpolation fragmentInterpolation = getVaryingInterpolation(varyings.frgInputs[ndx].interpolation); if (!isSSOCompatibleInterpolation(vertexInterpolation, fragmentInterpolation)) return false; } return true; } #endif void logParams (TestLog& log, const TestParams& params) { // We don't log operational details here since those are shown // in the log messages during execution. MessageBuilder msg = log.message(); msg << "Pipeline configuration:\n"; msg << "Vertex and fragment shaders have " << (params.useUniform ? "uniform" : "constant") << "s with " << (params.useSameName ? "the same name" : "different names") << ".\n"; if (params.varyings.count == 0) msg << "There are no varyings.\n"; else { if (params.varyings.count == 1) msg << "There is one varying.\n"; else msg << "There are " << params.varyings.count << " varyings.\n"; if (params.varyings.type == glu::TYPE_LAST) msg << "Varyings are of random types.\n"; else msg << "Varyings are of type '" << glu::getDataTypeName(params.varyings.type) << "'.\n"; msg << "Varying outputs and inputs correspond "; switch (params.varyings.binding) { case BINDING_NAME: msg << "by name.\n"; break; case BINDING_LOCATION: msg << "by location.\n"; break; case BINDING_LAST: msg << "randomly either by name or by location.\n"; break; default: DE_FATAL("Impossible"); } msg << "In the vertex shader the varyings are qualified "; if (params.varyings.vtxInterp == VARYINGINTERPOLATION_DEFAULT) msg << "with no interpolation qualifiers.\n"; else if (params.varyings.vtxInterp == VARYINGINTERPOLATION_RANDOM) msg << "with a random interpolation qualifier.\n"; else msg << "'" << glu::getInterpolationName(getGluInterpolation(params.varyings.vtxInterp)) << "'.\n"; msg << "In the fragment shader the varyings are qualified "; if (params.varyings.frgInterp == VARYINGINTERPOLATION_DEFAULT) msg << "with no interpolation qualifiers.\n"; else if (params.varyings.frgInterp == VARYINGINTERPOLATION_RANDOM) msg << "with a random interpolation qualifier.\n"; else msg << "'" << glu::getInterpolationName(getGluInterpolation(params.varyings.frgInterp)) << "'.\n"; } msg << TestLog::EndMessage; log.writeMessage(""); } TestParams genParams (deUint32 seed) { Random rnd (seed); TestParams params; int tryNdx = 0; do { params.initSingle = rnd.getBool(); params.switchVtx = rnd.getBool(); params.switchFrg = rnd.getBool(); params.useUniform = rnd.getBool(); params.useProgramUniform = params.useUniform && rnd.getBool(); params.useCreateHelper = rnd.getBool(); params.useSameName = rnd.getBool(); { int i = rnd.getInt(-1, 3); params.varyings.count = (i == -1 ? 0 : 1 << i); } if (params.varyings.count > 0) { params.varyings.type = glu::TYPE_LAST; params.varyings.binding = BINDING_LAST; params.varyings.vtxInterp = VARYINGINTERPOLATION_RANDOM; params.varyings.frgInterp = VARYINGINTERPOLATION_RANDOM; } else { params.varyings.type = glu::TYPE_INVALID; params.varyings.binding = BINDING_LAST; params.varyings.vtxInterp = VARYINGINTERPOLATION_LAST; params.varyings.frgInterp = VARYINGINTERPOLATION_LAST; } tryNdx += 1; } while (!paramsValid(params) && tryNdx < 16); DE_ASSERT(paramsValid(params)); return params; } // Program pipeline wrapper that retains references to component programs. struct Pipeline { Pipeline (MovePtr<ProgramPipeline> pipeline_, MovePtr<ProgramWrapper> fullProg_, MovePtr<ProgramWrapper> vtxProg_, MovePtr<ProgramWrapper> frgProg_) : pipeline (pipeline_) , fullProg (fullProg_) , vtxProg (vtxProg_) , frgProg (frgProg_) {} ProgramWrapper& getVertexProgram (void) const { return vtxProg ? *vtxProg : *fullProg; } ProgramWrapper& getFragmentProgram (void) const { return frgProg ? *frgProg : *fullProg; } UniquePtr<ProgramPipeline> pipeline; UniquePtr<ProgramWrapper> fullProg; UniquePtr<ProgramWrapper> vtxProg; UniquePtr<ProgramWrapper> frgProg; }; void logPipeline(TestLog& log, const Pipeline& pipeline) { ProgramWrapper& vtxProg = pipeline.getVertexProgram(); ProgramWrapper& frgProg = pipeline.getFragmentProgram(); log.writeMessage("// Failed program pipeline:"); if (&vtxProg == &frgProg) { log.writeMessage("// Common program for both vertex and fragment stages:"); vtxProg.writeToLog(log); } else { log.writeMessage("// Vertex stage program:"); vtxProg.writeToLog(log); log.writeMessage("// Fragment stage program:"); frgProg.writeToLog(log); } } // Rectangle struct Rectangle { Rectangle (int x_, int y_, int width_, int height_) : x (x_) , y (y_) , width (width_) , height (height_) {} int x; int y; int width; int height; }; void setViewport (const RenderContext& renderCtx, const Rectangle& rect) { renderCtx.getFunctions().viewport(rect.x, rect.y, rect.width, rect.height); } void readRectangle (const RenderContext& renderCtx, const Rectangle& rect, Surface& dst) { dst.setSize(rect.width, rect.height); glu::readPixels(renderCtx, rect.x, rect.y, dst.getAccess()); } Rectangle randomViewport (const RenderContext& ctx, Random& rnd, GLint maxWidth, GLint maxHeight) { const RenderTarget& target = ctx.getRenderTarget(); GLint width = de::min(target.getWidth(), maxWidth); GLint xOff = rnd.getInt(0, target.getWidth() - width); GLint height = de::min(target.getHeight(), maxHeight); GLint yOff = rnd.getInt(0, target.getHeight() - height); return Rectangle(xOff, yOff, width, height); } // SeparateShaderTest class SeparateShaderTest : public TestCase, private CallLogWrapper { public: typedef void (SeparateShaderTest::*TestFunc) (MovePtr<Pipeline>& pipeOut); SeparateShaderTest (Context& ctx, const string& name, const string& description, int iterations, const TestParams& params, TestFunc testFunc); IterateResult iterate (void); void testPipelineRendering (MovePtr<Pipeline>& pipeOut); void testCurrentProgPriority (MovePtr<Pipeline>& pipeOut); void testActiveProgramUniform (MovePtr<Pipeline>& pipeOut); void testPipelineQueryActive (MovePtr<Pipeline>& pipeOut); void testPipelineQueryPrograms (MovePtr<Pipeline>& pipeOut); private: TestLog& log (void); const RenderContext& getRenderContext (void); void setUniform (ProgramWrapper& program, const string& uniformName, GLfloat value, bool useProgramUni); void drawSurface (Surface& dst, deUint32 seed = 0); MovePtr<ProgramWrapper> createShaderProgram (const string* vtxSource, const string* frgSource, bool separable); MovePtr<ProgramWrapper> createSingleShaderProgram (ShaderType shaderType, const string& src); MovePtr<Pipeline> createPipeline (const ProgramParams& pp); MovePtr<ProgramWrapper> createReferenceProgram (const ProgramParams& pp); int m_iterations; int m_currentIteration; TestParams m_params; TestFunc m_testFunc; Random m_rnd; ResultCollector m_status; VaryingInterface m_varyings; // Per-iteration state required for logging on exception MovePtr<ProgramWrapper> m_fullProg; MovePtr<ProgramWrapper> m_vtxProg; MovePtr<ProgramWrapper> m_frgProg; }; const RenderContext& SeparateShaderTest::getRenderContext (void) { return m_context.getRenderContext(); } TestLog& SeparateShaderTest::log (void) { return m_testCtx.getLog(); } SeparateShaderTest::SeparateShaderTest (Context& ctx, const string& name, const string& description, int iterations, const TestParams& params, TestFunc testFunc) : TestCase (ctx, name.c_str(), description.c_str()) , CallLogWrapper (ctx.getRenderContext().getFunctions(), log()) , m_iterations (iterations) , m_currentIteration(0) , m_params (params) , m_testFunc (testFunc) , m_rnd (paramsSeed(params)) , m_status (log(), "// ") , m_varyings (genVaryingInterface(params.varyings, m_rnd)) { DE_ASSERT(paramsValid(params)); DE_ASSERT(varyingsValid(m_varyings)); } MovePtr<ProgramWrapper> SeparateShaderTest::createShaderProgram (const string* vtxSource, const string* frgSource, bool separable) { ProgramSources sources; if (vtxSource != DE_NULL) sources << VertexSource(*vtxSource); if (frgSource != DE_NULL) sources << FragmentSource(*frgSource); sources << ProgramSeparable(separable); MovePtr<ShaderProgramWrapper> wrapper (new ShaderProgramWrapper(getRenderContext(), sources)); if (!wrapper->getShaderProgram().isOk()) { log().writeMessage("Couldn't create shader program"); wrapper->writeToLog(log()); TCU_FAIL("Couldn't create shader program"); } return MovePtr<ProgramWrapper>(wrapper.release()); } MovePtr<ProgramWrapper> SeparateShaderTest::createSingleShaderProgram (ShaderType shaderType, const string& src) { const RenderContext& renderCtx = getRenderContext(); if (m_params.useCreateHelper) { const char* const srcStr = src.c_str(); const GLenum glType = glu::getGLShaderType(shaderType); const GLuint programName = glCreateShaderProgramv(glType, 1, &srcStr); if (glGetError() != GL_NO_ERROR || programName == 0) { qpShaderType qpType = glu::getLogShaderType(shaderType); log() << TestLog::Message << "glCreateShaderProgramv() failed" << TestLog::EndMessage << TestLog::ShaderProgram(false, "[glCreateShaderProgramv() failed]") << TestLog::Shader(qpType, src, false, "[glCreateShaderProgramv() failed]") << TestLog::EndShaderProgram; TCU_FAIL("glCreateShaderProgramv() failed"); } RawProgramWrapper* const wrapper = new RawProgramWrapper(renderCtx, programName, shaderType, src); MovePtr<ProgramWrapper> wrapperPtr(wrapper); Program& program = wrapper->getProgram(); if (!program.getLinkStatus()) { log().writeMessage("glCreateShaderProgramv() failed at linking"); wrapper->writeToLog(log()); TCU_FAIL("glCreateShaderProgram() failed at linking"); } return wrapperPtr; } else { switch (shaderType) { case glu::SHADERTYPE_VERTEX: return createShaderProgram(&src, DE_NULL, true); case glu::SHADERTYPE_FRAGMENT: return createShaderProgram(DE_NULL, &src, true); default: DE_FATAL("Impossible case"); } } return MovePtr<ProgramWrapper>(); // Shut up compiler warnings. } void SeparateShaderTest::setUniform (ProgramWrapper& program, const string& uniformName, GLfloat value, bool useProgramUniform) { const GLuint progName = program.getProgramName(); const GLint location = glGetUniformLocation(progName, uniformName.c_str()); MessageBuilder msg = log().message(); msg << "// Set program " << progName << "'s uniform '" << uniformName << "' to " << value; if (useProgramUniform) { msg << " using glProgramUniform1f"; glProgramUniform1f(progName, location, value); } else { msg << " using glUseProgram and glUniform1f"; glUseProgram(progName); glUniform1f(location, value); glUseProgram(0); } msg << TestLog::EndMessage; } MovePtr<Pipeline> SeparateShaderTest::createPipeline (const ProgramParams& pp) { const bool useUniform = m_params.useUniform; const string vtxName = m_params.useSameName ? "scale" : "vtxScale"; const deUint32 initVtxSeed = m_params.switchVtx ? m_rnd.getUint32() : pp.vtxSeed; const string frgName = m_params.useSameName ? "scale" : "frgScale"; const deUint32 initFrgSeed = m_params.switchFrg ? m_rnd.getUint32() : pp.frgSeed; const string frgSource = genFrgShaderSrc(initFrgSeed, m_varyings.frgInputs, frgName, useUniform, pp.frgScale); const RenderContext& renderCtx = getRenderContext(); MovePtr<ProgramPipeline> pipeline (new ProgramPipeline(renderCtx)); MovePtr<ProgramWrapper> fullProg; MovePtr<ProgramWrapper> vtxProg; MovePtr<ProgramWrapper> frgProg; // We cannot allow a situation where we have a single program with a // single uniform, because then the vertex and fragment shader uniforms // would not be distinct in the final pipeline, and we are going to test // that altering one uniform will not affect the other. DE_ASSERT(!(m_params.initSingle && m_params.useSameName && !m_params.switchVtx && !m_params.switchFrg)); if (m_params.initSingle) { string vtxSource = genVtxShaderSrc(initVtxSeed, varyingCompatVtxOutputs(m_varyings), vtxName, useUniform, pp.vtxScale); fullProg = createShaderProgram(&vtxSource, &frgSource, true); pipeline->useProgramStages(GL_VERTEX_SHADER_BIT | GL_FRAGMENT_SHADER_BIT, fullProg->getProgramName()); log() << TestLog::Message << "// Created pipeline " << pipeline->getPipeline() << " with two-shader program " << fullProg->getProgramName() << TestLog::EndMessage; } else { string vtxSource = genVtxShaderSrc(initVtxSeed, m_varyings.vtxOutputs, vtxName, useUniform, pp.vtxScale); vtxProg = createSingleShaderProgram(glu::SHADERTYPE_VERTEX, vtxSource); pipeline->useProgramStages(GL_VERTEX_SHADER_BIT, vtxProg->getProgramName()); frgProg = createSingleShaderProgram(glu::SHADERTYPE_FRAGMENT, frgSource); pipeline->useProgramStages(GL_FRAGMENT_SHADER_BIT, frgProg->getProgramName()); log() << TestLog::Message << "// Created pipeline " << pipeline->getPipeline() << " with vertex program " << vtxProg->getProgramName() << " and fragment program " << frgProg->getProgramName() << TestLog::EndMessage; } m_status.check(pipeline->isValid(), "Pipeline is invalid after initialization"); if (m_params.switchVtx) { string newSource = genVtxShaderSrc(pp.vtxSeed, m_varyings.vtxOutputs, vtxName, useUniform, pp.vtxScale); vtxProg = createSingleShaderProgram(glu::SHADERTYPE_VERTEX, newSource); pipeline->useProgramStages(GL_VERTEX_SHADER_BIT, vtxProg->getProgramName()); log() << TestLog::Message << "// Switched pipeline " << pipeline->getPipeline() << "'s vertex stage to single-shader program " << vtxProg->getProgramName() << TestLog::EndMessage; } if (m_params.switchFrg) { string newSource = genFrgShaderSrc(pp.frgSeed, m_varyings.frgInputs, frgName, useUniform, pp.frgScale); frgProg = createSingleShaderProgram(glu::SHADERTYPE_FRAGMENT, newSource); pipeline->useProgramStages(GL_FRAGMENT_SHADER_BIT, frgProg->getProgramName()); log() << TestLog::Message << "// Switched pipeline " << pipeline->getPipeline() << "'s fragment stage to single-shader program " << frgProg->getProgramName() << TestLog::EndMessage; } if (m_params.switchVtx || m_params.switchFrg) m_status.check(pipeline->isValid(), "Pipeline became invalid after changing a stage's program"); if (m_params.useUniform) { ProgramWrapper& vtxStage = *(vtxProg ? vtxProg : fullProg); ProgramWrapper& frgStage = *(frgProg ? frgProg : fullProg); setUniform(vtxStage, vtxName, pp.vtxScale, m_params.useProgramUniform); setUniform(frgStage, frgName, pp.frgScale, m_params.useProgramUniform); } else log().writeMessage("// Programs use constants instead of uniforms"); return MovePtr<Pipeline>(new Pipeline(pipeline, fullProg, vtxProg, frgProg)); } MovePtr<ProgramWrapper> SeparateShaderTest::createReferenceProgram (const ProgramParams& pp) { bool useUniform = m_params.useUniform; const string vtxSrc = genVtxShaderSrc(pp.vtxSeed, varyingCompatVtxOutputs(m_varyings), "vtxScale", useUniform, pp.vtxScale); const string frgSrc = genFrgShaderSrc(pp.frgSeed, m_varyings.frgInputs, "frgScale", useUniform, pp.frgScale); MovePtr<ProgramWrapper> program = createShaderProgram(&vtxSrc, &frgSrc, false); GLuint progName = program->getProgramName(); log() << TestLog::Message << "// Created monolithic shader program " << progName << TestLog::EndMessage; if (useUniform) { setUniform(*program, "vtxScale", pp.vtxScale, false); setUniform(*program, "frgScale", pp.frgScale, false); } return program; } void SeparateShaderTest::drawSurface (Surface& dst, deUint32 seed) { const RenderContext& renderCtx = getRenderContext(); Random rnd (seed > 0 ? seed : m_rnd.getUint32()); Rectangle viewport = randomViewport(renderCtx, rnd, VIEWPORT_SIZE, VIEWPORT_SIZE); glClearColor(0.125f, 0.25f, 0.5f, 1.f); setViewport(renderCtx, viewport); glClear(GL_COLOR_BUFFER_BIT); GLU_CHECK_CALL(glDrawArrays(GL_TRIANGLES, 0, 3)); readRectangle(renderCtx, viewport, dst); log().writeMessage("// Drew a triangle"); } void SeparateShaderTest::testPipelineRendering (MovePtr<Pipeline>& pipeOut) { ProgramParams pp = genProgramParams(m_rnd); Pipeline& pipeline = *(pipeOut = createPipeline(pp)); GLuint pipeName = pipeline.pipeline->getPipeline(); UniquePtr<ProgramWrapper> refProgram (createReferenceProgram(pp)); GLuint refProgName = refProgram->getProgramName(); Surface refSurface; Surface pipelineSurface; GLuint drawSeed = m_rnd.getUint32(); glUseProgram(refProgName); log() << TestLog::Message << "// Use program " << refProgName << TestLog::EndMessage; drawSurface(refSurface, drawSeed); glUseProgram(0); glBindProgramPipeline(pipeName); log() << TestLog::Message << "// Bind pipeline " << pipeName << TestLog::EndMessage; drawSurface(pipelineSurface, drawSeed); glBindProgramPipeline(0); { const bool result = tcu::fuzzyCompare( m_testCtx.getLog(), "Program pipeline result", "Result of comparing a program pipeline with a monolithic program", refSurface, pipelineSurface, 0.05f, tcu::COMPARE_LOG_RESULT); m_status.check(result, "Pipeline rendering differs from equivalent monolithic program"); } } void SeparateShaderTest::testCurrentProgPriority (MovePtr<Pipeline>& pipeOut) { ProgramParams pipePp = genProgramParams(m_rnd); ProgramParams programPp = genProgramParams(m_rnd); Pipeline& pipeline = *(pipeOut = createPipeline(pipePp)); GLuint pipeName = pipeline.pipeline->getPipeline(); UniquePtr<ProgramWrapper> program (createReferenceProgram(programPp)); Surface pipelineSurface; Surface refSurface; Surface resultSurface; deUint32 drawSeed = m_rnd.getUint32(); LOG_CALL(glBindProgramPipeline(pipeName)); drawSurface(pipelineSurface, drawSeed); LOG_CALL(glBindProgramPipeline(0)); LOG_CALL(glUseProgram(program->getProgramName())); drawSurface(refSurface, drawSeed); LOG_CALL(glUseProgram(0)); LOG_CALL(glUseProgram(program->getProgramName())); LOG_CALL(glBindProgramPipeline(pipeName)); drawSurface(resultSurface, drawSeed); LOG_CALL(glBindProgramPipeline(0)); LOG_CALL(glUseProgram(0)); bool result = tcu::pixelThresholdCompare( m_testCtx.getLog(), "Active program rendering result", "Active program rendering result", refSurface, resultSurface, tcu::RGBA(0, 0, 0, 0), tcu::COMPARE_LOG_RESULT); m_status.check(result, "glBindProgramPipeline() affects glUseProgram()"); if (!result) log() << TestLog::Image("Pipeline image", "Image produced by pipeline", pipelineSurface); } void SeparateShaderTest::testActiveProgramUniform (MovePtr<Pipeline>& pipeOut) { ProgramParams refPp = genProgramParams(m_rnd); Surface refSurface; Surface resultSurface; deUint32 drawSeed = m_rnd.getUint32(); DE_UNREF(pipeOut); { UniquePtr<ProgramWrapper> refProg (createReferenceProgram(refPp)); GLuint refProgName = refProg->getProgramName(); glUseProgram(refProgName); log() << TestLog::Message << "// Use reference program " << refProgName << TestLog::EndMessage; drawSurface(refSurface, drawSeed); glUseProgram(0); } { ProgramParams changePp = genProgramParams(m_rnd); changePp.vtxSeed = refPp.vtxSeed; changePp.frgSeed = refPp.frgSeed; UniquePtr<ProgramWrapper> changeProg (createReferenceProgram(changePp)); GLuint changeName = changeProg->getProgramName(); ProgramPipeline pipeline (getRenderContext()); GLint vtxLoc = glGetUniformLocation(changeName, "vtxScale"); GLint frgLoc = glGetUniformLocation(changeName, "frgScale"); LOG_CALL(glBindProgramPipeline(pipeline.getPipeline())); pipeline.activeShaderProgram(changeName); log() << TestLog::Message << "// Set active shader program to " << changeName << TestLog::EndMessage; glUniform1f(vtxLoc, refPp.vtxScale); log() << TestLog::Message << "// Set uniform 'vtxScale' to " << refPp.vtxScale << " using glUniform1f" << TestLog::EndMessage; glUniform1f(frgLoc, refPp.frgScale); log() << TestLog::Message << "// Set uniform 'frgScale' to " << refPp.frgScale << " using glUniform1f" << TestLog::EndMessage; pipeline.activeShaderProgram(0); LOG_CALL(glBindProgramPipeline(0)); LOG_CALL(glUseProgram(changeName)); drawSurface(resultSurface, drawSeed); LOG_CALL(glUseProgram(0)); } bool result = tcu::fuzzyCompare( m_testCtx.getLog(), "Active program uniform result", "Active program uniform result", refSurface, resultSurface, 0.05f, tcu::COMPARE_LOG_RESULT); m_status.check(result, "glUniform() did not correctly modify " "the active program of the bound pipeline"); } void SeparateShaderTest::testPipelineQueryPrograms (MovePtr<Pipeline>& pipeOut) { ProgramParams pipePp = genProgramParams(m_rnd); Pipeline& pipeline = *(pipeOut = createPipeline(pipePp)); GLuint pipeName = pipeline.pipeline->getPipeline(); GLint queryVtx = 0; GLint queryFrg = 0; LOG_CALL(GLU_CHECK_CALL(glGetProgramPipelineiv(pipeName, GL_VERTEX_SHADER, &queryVtx))); m_status.check(GLuint(queryVtx) == pipeline.getVertexProgram().getProgramName(), "Program pipeline query reported wrong vertex shader program"); LOG_CALL(GLU_CHECK_CALL(glGetProgramPipelineiv(pipeName, GL_FRAGMENT_SHADER, &queryFrg))); m_status.check(GLuint(queryFrg) == pipeline.getFragmentProgram().getProgramName(), "Program pipeline query reported wrong fragment shader program"); } void SeparateShaderTest::testPipelineQueryActive (MovePtr<Pipeline>& pipeOut) { ProgramParams pipePp = genProgramParams(m_rnd); Pipeline& pipeline = *(pipeOut = createPipeline(pipePp)); GLuint pipeName = pipeline.pipeline->getPipeline(); GLuint newActive = pipeline.getVertexProgram().getProgramName(); GLint queryActive = 0; LOG_CALL(GLU_CHECK_CALL(glGetProgramPipelineiv(pipeName, GL_ACTIVE_PROGRAM, &queryActive))); m_status.check(queryActive == 0, "Program pipeline query reported non-zero initial active program"); pipeline.pipeline->activeShaderProgram(newActive); log() << TestLog::Message << "Set pipeline " << pipeName << "'s active shader program to " << newActive << TestLog::EndMessage; LOG_CALL(GLU_CHECK_CALL(glGetProgramPipelineiv(pipeName, GL_ACTIVE_PROGRAM, &queryActive))); m_status.check(GLuint(queryActive) == newActive, "Program pipeline query reported incorrect active program"); pipeline.pipeline->activeShaderProgram(0); } TestCase::IterateResult SeparateShaderTest::iterate (void) { MovePtr<Pipeline> pipeline; DE_ASSERT(m_iterations > 0); if (m_currentIteration == 0) logParams(log(), m_params); ++m_currentIteration; try { (this->*m_testFunc)(pipeline); log().writeMessage(""); } catch (const tcu::Exception&) { if (pipeline) logPipeline(log(), *pipeline); throw; } if (m_status.getResult() != QP_TEST_RESULT_PASS) { if (pipeline) logPipeline(log(), *pipeline); } else if (m_currentIteration < m_iterations) return CONTINUE; m_status.setTestContextResult(m_testCtx); return STOP; } // Group construction utilities enum ParamFlags { PARAMFLAGS_SWITCH_FRAGMENT = 1 << 0, PARAMFLAGS_SWITCH_VERTEX = 1 << 1, PARAMFLAGS_INIT_SINGLE = 1 << 2, PARAMFLAGS_LAST = 1 << 3, PARAMFLAGS_MASK = PARAMFLAGS_LAST - 1 }; bool areCaseParamFlagsValid (ParamFlags flags) { const ParamFlags switchAll = ParamFlags(PARAMFLAGS_SWITCH_VERTEX|PARAMFLAGS_SWITCH_FRAGMENT); if ((flags & PARAMFLAGS_INIT_SINGLE) != 0) return (flags & switchAll) == 0 || (flags & switchAll) == switchAll; else return true; } bool addRenderTest (TestCaseGroup& group, const string& namePrefix, const string& descPrefix, int numIterations, ParamFlags flags, TestParams params) { ostringstream name; ostringstream desc; DE_ASSERT(areCaseParamFlagsValid(flags)); name << namePrefix; desc << descPrefix; params.initSingle = (flags & PARAMFLAGS_INIT_SINGLE) != 0; params.switchVtx = (flags & PARAMFLAGS_SWITCH_VERTEX) != 0; params.switchFrg = (flags & PARAMFLAGS_SWITCH_FRAGMENT) != 0; name << (flags & PARAMFLAGS_INIT_SINGLE ? "single_program" : "separate_programs"); desc << (flags & PARAMFLAGS_INIT_SINGLE ? "Single program with two shaders" : "Separate programs for each shader"); switch (flags & (PARAMFLAGS_SWITCH_FRAGMENT | PARAMFLAGS_SWITCH_VERTEX)) { case 0: break; case PARAMFLAGS_SWITCH_FRAGMENT: name << "_add_fragment"; desc << ", then add a fragment program"; break; case PARAMFLAGS_SWITCH_VERTEX: name << "_add_vertex"; desc << ", then add a vertex program"; break; case PARAMFLAGS_SWITCH_FRAGMENT | PARAMFLAGS_SWITCH_VERTEX: name << "_add_both"; desc << ", then add both vertex and shader programs"; break; } if (!paramsValid(params)) return false; group.addChild(new SeparateShaderTest(group.getContext(), name.str(), desc.str(), numIterations, params, &SeparateShaderTest::testPipelineRendering)); return true; } void describeInterpolation (const string& stage, VaryingInterpolation qual, ostringstream& name, ostringstream& desc) { DE_ASSERT(qual < VARYINGINTERPOLATION_RANDOM); if (qual == VARYINGINTERPOLATION_DEFAULT) { desc << ", unqualified in " << stage << " shader"; return; } else { const string qualName = glu::getInterpolationName(getGluInterpolation(qual)); name << "_" << stage << "_" << qualName; desc << ", qualified '" << qualName << "' in " << stage << " shader"; } } } // anonymous TestCaseGroup* createSeparateShaderTests (Context& ctx) { TestParams defaultParams; int numIterations = 4; TestCaseGroup* group = new TestCaseGroup(ctx, "separate_shader", "Separate shader tests"); defaultParams.useUniform = false; defaultParams.initSingle = false; defaultParams.switchVtx = false; defaultParams.switchFrg = false; defaultParams.useCreateHelper = false; defaultParams.useProgramUniform = false; defaultParams.useSameName = false; defaultParams.varyings.count = 0; defaultParams.varyings.type = glu::TYPE_INVALID; defaultParams.varyings.binding = BINDING_NAME; defaultParams.varyings.vtxInterp = VARYINGINTERPOLATION_LAST; defaultParams.varyings.frgInterp = VARYINGINTERPOLATION_LAST; TestCaseGroup* stagesGroup = new TestCaseGroup(ctx, "pipeline", "Pipeline configuration tests"); group->addChild(stagesGroup); for (deUint32 flags = 0; flags < PARAMFLAGS_LAST << 2; ++flags) { TestParams params = defaultParams; ostringstream name; ostringstream desc; if (!areCaseParamFlagsValid(ParamFlags(flags & PARAMFLAGS_MASK))) continue; if (flags & (PARAMFLAGS_LAST << 1)) { params.useSameName = true; name << "same_"; desc << "Identically named "; } else { name << "different_"; desc << "Differently named "; } if (flags & PARAMFLAGS_LAST) { params.useUniform = true; name << "uniform_"; desc << "uniforms, "; } else { name << "constant_"; desc << "constants, "; } addRenderTest(*stagesGroup, name.str(), desc.str(), numIterations, ParamFlags(flags & PARAMFLAGS_MASK), params); } TestCaseGroup* programUniformGroup = new TestCaseGroup(ctx, "program_uniform", "ProgramUniform tests"); group->addChild(programUniformGroup); for (deUint32 flags = 0; flags < PARAMFLAGS_LAST; ++flags) { TestParams params = defaultParams; if (!areCaseParamFlagsValid(ParamFlags(flags))) continue; params.useUniform = true; params.useProgramUniform = true; addRenderTest(*programUniformGroup, "", "", numIterations, ParamFlags(flags), params); } TestCaseGroup* createShaderProgramGroup = new TestCaseGroup(ctx, "create_shader_program", "CreateShaderProgram tests"); group->addChild(createShaderProgramGroup); for (deUint32 flags = 0; flags < PARAMFLAGS_LAST; ++flags) { TestParams params = defaultParams; if (!areCaseParamFlagsValid(ParamFlags(flags))) continue; params.useCreateHelper = true; addRenderTest(*createShaderProgramGroup, "", "", numIterations, ParamFlags(flags), params); } TestCaseGroup* interfaceGroup = new TestCaseGroup(ctx, "interface", "Shader interface compatibility tests"); group->addChild(interfaceGroup); enum { NUM_INTERPOLATIONS = VARYINGINTERPOLATION_RANDOM, // VARYINGINTERPOLATION_RANDOM is one after last fully specified interpolation INTERFACEFLAGS_LAST = BINDING_LAST * NUM_INTERPOLATIONS * NUM_INTERPOLATIONS }; for (deUint32 flags = 0; flags < INTERFACEFLAGS_LAST; ++flags) { deUint32 tmpFlags = flags; VaryingInterpolation frgInterp = VaryingInterpolation(tmpFlags % NUM_INTERPOLATIONS); VaryingInterpolation vtxInterp = VaryingInterpolation((tmpFlags /= NUM_INTERPOLATIONS) % NUM_INTERPOLATIONS); BindingKind binding = BindingKind((tmpFlags /= NUM_INTERPOLATIONS) % BINDING_LAST); TestParams params = defaultParams; ostringstream name; ostringstream desc; params.varyings.count = 1; params.varyings.type = glu::TYPE_FLOAT; params.varyings.binding = binding; params.varyings.vtxInterp = vtxInterp; params.varyings.frgInterp = frgInterp; switch (binding) { case BINDING_LOCATION: name << "same_location"; desc << "Varyings have same location, "; break; case BINDING_NAME: name << "same_name"; desc << "Varyings have same name, "; break; default: DE_FATAL("Impossible"); } describeInterpolation("vertex", vtxInterp, name, desc); describeInterpolation("fragment", frgInterp, name, desc); if (!paramsValid(params)) continue; interfaceGroup->addChild( new SeparateShaderTest(ctx, name.str(), desc.str(), numIterations, params, &SeparateShaderTest::testPipelineRendering)); } deUint32 baseSeed = ctx.getTestContext().getCommandLine().getBaseSeed(); Random rnd (deStringHash("separate_shader.random") + baseSeed); set<string> seen; TestCaseGroup* randomGroup = new TestCaseGroup( ctx, "random", "Random pipeline configuration tests"); group->addChild(randomGroup); for (deUint32 i = 0; i < 128; ++i) { TestParams params; string code; deUint32 genIterations = 4096; do { params = genParams(rnd.getUint32()); code = paramsCode(params); } while (de::contains(seen, code) && --genIterations > 0); seen.insert(code); string name = de::toString(i); // Would be code but baseSeed can change randomGroup->addChild(new SeparateShaderTest( ctx, name, name, numIterations, params, &SeparateShaderTest::testPipelineRendering)); } TestCaseGroup* apiGroup = new TestCaseGroup(ctx, "api", "Program pipeline API tests"); group->addChild(apiGroup); { // More or less random parameters. These shouldn't have much effect, so just // do a single sample. TestParams params = defaultParams; params.useUniform = true; apiGroup->addChild(new SeparateShaderTest( ctx, "current_program_priority", "Test priority between current program and pipeline binding", 1, params, &SeparateShaderTest::testCurrentProgPriority)); apiGroup->addChild(new SeparateShaderTest( ctx, "active_program_uniform", "Test that glUniform() affects a pipeline's active program", 1, params, &SeparateShaderTest::testActiveProgramUniform)); apiGroup->addChild(new SeparateShaderTest( ctx, "pipeline_programs", "Test queries for programs in program pipeline stages", 1, params, &SeparateShaderTest::testPipelineQueryPrograms)); apiGroup->addChild(new SeparateShaderTest( ctx, "pipeline_active", "Test query for active programs in a program pipeline", 1, params, &SeparateShaderTest::testPipelineQueryActive)); } TestCaseGroup* interfaceMismatchGroup = new TestCaseGroup(ctx, "validation", "Negative program pipeline interface matching"); group->addChild(interfaceMismatchGroup); { gls::ShaderLibrary shaderLibrary (ctx.getTestContext(), ctx.getRenderContext(), ctx.getContextInfo()); const std::vector<tcu::TestNode*> children = shaderLibrary.loadShaderFile("shaders/separate_shader_validation.test"); for (int i = 0; i < (int)children.size(); i++) interfaceMismatchGroup->addChild(children[i]); } return group; } } // Functional } // gles31 } // deqp