| /*------------------------------------------------------------------------- |
| * drawElements Quality Program Random Shader Generator |
| * ---------------------------------------------------- |
| * |
| * 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 Utilities. |
| *//*--------------------------------------------------------------------*/ |
| |
| #include "rsgUtils.hpp" |
| |
| #include <set> |
| #include <string> |
| |
| using std::set; |
| using std::string; |
| using std::vector; |
| |
| namespace rsg |
| { |
| |
| void addNewUniforms (vector<const ShaderInput*>& uniforms, set<string>& addedUniforms, const Shader& shader) |
| { |
| const vector<ShaderInput*>& shaderUniforms = shader.getUniforms(); |
| for (vector<ShaderInput*>::const_iterator i = shaderUniforms.begin(); i != shaderUniforms.end(); i++) |
| { |
| const ShaderInput* uniform = *i; |
| if (addedUniforms.find(uniform->getVariable()->getName()) == addedUniforms.end()) |
| { |
| addedUniforms.insert(uniform->getVariable()->getName()); |
| uniforms.push_back(uniform); |
| } |
| } |
| } |
| |
| void computeUnifiedUniforms (const Shader& vertexShader, const Shader& fragmentShader, std::vector<const ShaderInput*>& uniforms) |
| { |
| set<string> addedUniforms; |
| addNewUniforms(uniforms, addedUniforms, vertexShader); |
| addNewUniforms(uniforms, addedUniforms, fragmentShader); |
| } |
| |
| void computeRandomValue (de::Random& rnd, ValueAccess dst, ConstValueRangeAccess valueRange) |
| { |
| const VariableType& type = dst.getType(); |
| |
| switch (type.getBaseType()) |
| { |
| case VariableType::TYPE_FLOAT: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| const float quantizeStep = 1.0f/8.0f; |
| float minVal = valueRange.component(ndx).getMin().asFloat(); |
| float maxVal = valueRange.component(ndx).getMax().asFloat(); |
| dst.component(ndx).asFloat() = getQuantizedFloat(rnd, minVal, maxVal, quantizeStep); |
| } |
| break; |
| |
| case VariableType::TYPE_BOOL: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| int minVal = valueRange.component(ndx).getMin().asBool() ? 1 : 0; |
| int maxVal = valueRange.component(ndx).getMin().asBool() ? 1 : 0; |
| dst.component(ndx).asBool() = rnd.getInt(minVal, maxVal) == 1; |
| } |
| break; |
| |
| case VariableType::TYPE_INT: |
| case VariableType::TYPE_SAMPLER_2D: |
| case VariableType::TYPE_SAMPLER_CUBE: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| int minVal = valueRange.component(ndx).getMin().asInt(); |
| int maxVal = valueRange.component(ndx).getMax().asInt(); |
| dst.component(ndx).asInt() = rnd.getInt(minVal, maxVal); |
| } |
| break; |
| |
| case VariableType::TYPE_ARRAY: |
| { |
| int numElements = type.getNumElements(); |
| for (int ndx = 0; ndx < numElements; ndx++) |
| computeRandomValue(rnd, dst.arrayElement(ndx), valueRange.arrayElement(ndx)); |
| break; |
| } |
| |
| case VariableType::TYPE_STRUCT: |
| { |
| int numMembers = (int)type.getMembers().size(); |
| for (int ndx = 0; ndx < numMembers; ndx++) |
| computeRandomValue(rnd, dst.member(ndx), valueRange.member(ndx)); |
| break; |
| } |
| |
| default: |
| TCU_FAIL("Invalid type"); |
| } |
| } |
| |
| void computeUniformValues (de::Random& rnd, std::vector<VariableValue>& values, const std::vector<const ShaderInput*>& uniforms) |
| { |
| DE_ASSERT(values.empty()); |
| for (vector<const ShaderInput*>::const_iterator i = uniforms.begin(); i != uniforms.end(); i++) |
| { |
| const ShaderInput* uniform = *i; |
| values.push_back(VariableValue(uniform->getVariable())); |
| computeRandomValue(rnd, values[values.size()-1].getValue(), uniform->getValueRange()); |
| } |
| } |
| |
| bool isUndefinedValueRange (ConstValueRangeAccess valueRange) |
| { |
| switch (valueRange.getType().getBaseType()) |
| { |
| case VariableType::TYPE_FLOAT: |
| case VariableType::TYPE_INT: |
| { |
| bool isFloat = valueRange.getType().getBaseType() == VariableType::TYPE_FLOAT; |
| Scalar infMin = isFloat ? Scalar::min<float>() : Scalar::min<int>(); |
| Scalar infMax = isFloat ? Scalar::max<float>() : Scalar::max<int>(); |
| |
| for (int ndx = 0; ndx < valueRange.getType().getNumElements(); ndx++) |
| { |
| if (valueRange.getMin().component(ndx).asScalar() != infMin || |
| valueRange.getMax().component(ndx).asScalar() != infMax) |
| return false; |
| } |
| return true; |
| } |
| |
| case VariableType::TYPE_BOOL: |
| return false; |
| |
| default: |
| TCU_FAIL("Unsupported type"); |
| } |
| } |
| |
| VariableType computeRandomType (GeneratorState& state, int maxScalars) |
| { |
| DE_ASSERT(maxScalars >= 1); |
| |
| static const VariableType::Type baseTypes[] = |
| { |
| VariableType::TYPE_BOOL, |
| VariableType::TYPE_INT, |
| VariableType::TYPE_FLOAT |
| // \todo [pyry] Other types |
| }; |
| |
| VariableType::Type baseType = VariableType::TYPE_LAST; |
| state.getRandom().choose(baseTypes, baseTypes + DE_LENGTH_OF_ARRAY(baseTypes), &baseType, 1); |
| |
| switch (baseType) |
| { |
| case VariableType::TYPE_BOOL: |
| case VariableType::TYPE_INT: |
| case VariableType::TYPE_FLOAT: |
| { |
| const int minVecLength = 1; |
| const int maxVecLength = 4; |
| return VariableType(baseType, state.getRandom().getInt(minVecLength, de::min(maxScalars, maxVecLength))); |
| } |
| |
| default: |
| DE_ASSERT(DE_FALSE); |
| throw Exception("computeRandomType(): Unsupported type"); |
| } |
| } |
| |
| void computeRandomValueRange (GeneratorState& state, ValueRangeAccess valueRange) |
| { |
| const VariableType& type = valueRange.getType(); |
| de::Random& rnd = state.getRandom(); |
| |
| switch (type.getBaseType()) |
| { |
| case VariableType::TYPE_BOOL: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| bool minVal = rnd.getBool(); |
| bool maxVal = minVal ? true : rnd.getBool(); |
| valueRange.getMin().component(ndx).asBool() = minVal; |
| valueRange.getMax().component(ndx).asBool() = maxVal; |
| } |
| break; |
| |
| case VariableType::TYPE_INT: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| const int minIntVal = -16; |
| const int maxIntVal = 16; |
| const int maxRangeLen = maxIntVal - minIntVal; |
| |
| int rangeLen = rnd.getInt(0, maxRangeLen); |
| int minVal = minIntVal + rnd.getInt(0, maxRangeLen-rangeLen); |
| int maxVal = minVal + rangeLen; |
| |
| valueRange.getMin().component(ndx).asInt() = minVal; |
| valueRange.getMax().component(ndx).asInt() = maxVal; |
| } |
| break; |
| |
| case VariableType::TYPE_FLOAT: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| const float step = 0.1f; |
| const int maxSteps = 320; |
| const float minFloatVal = -16.0f; |
| |
| int rangeLen = rnd.getInt(0, maxSteps); |
| int minStep = rnd.getInt(0, maxSteps-rangeLen); |
| |
| float minVal = minFloatVal + step*(float)minStep; |
| float maxVal = minVal + step*(float)rangeLen; |
| |
| valueRange.getMin().component(ndx).asFloat() = minVal; |
| valueRange.getMax().component(ndx).asFloat() = maxVal; |
| } |
| break; |
| |
| default: |
| DE_ASSERT(DE_FALSE); |
| throw Exception("computeRandomValueRange(): Unsupported type"); |
| } |
| } |
| |
| int getTypeConstructorDepth (const VariableType& type) |
| { |
| switch (type.getBaseType()) |
| { |
| case VariableType::TYPE_STRUCT: |
| { |
| const vector<VariableType::Member>& members = type.getMembers(); |
| int maxDepth = 0; |
| for (vector<VariableType::Member>::const_iterator i = members.begin(); i != members.end(); i++) |
| { |
| const VariableType& memberType = i->getType(); |
| int depth = 0; |
| switch (memberType.getBaseType()) |
| { |
| case VariableType::TYPE_STRUCT: |
| depth = getTypeConstructorDepth(memberType); |
| break; |
| |
| case VariableType::TYPE_BOOL: |
| case VariableType::TYPE_FLOAT: |
| case VariableType::TYPE_INT: |
| depth = memberType.getNumElements() == 1 ? 1 : 2; |
| break; |
| |
| default: |
| DE_ASSERT(DE_FALSE); |
| break; |
| } |
| |
| maxDepth = de::max(maxDepth, depth); |
| } |
| return maxDepth + 1; |
| } |
| |
| case VariableType::TYPE_BOOL: |
| case VariableType::TYPE_FLOAT: |
| case VariableType::TYPE_INT: |
| return 2; // One node for ctor, another for value |
| |
| default: |
| DE_ASSERT(DE_FALSE); |
| return 0; |
| } |
| } |
| |
| int getConservativeValueExprDepth (const GeneratorState& state, ConstValueRangeAccess valueRange) |
| { |
| // \todo [2011-03-22 pyry] Do a look-up into variable manager? |
| DE_UNREF(state); |
| return getTypeConstructorDepth(valueRange.getType()); |
| } |
| |
| static float computeRangeLengthSum (ConstValueRangeAccess valueRange) |
| { |
| const VariableType& type = valueRange.getType(); |
| float rangeLength = 0.0f; |
| |
| switch (type.getBaseType()) |
| { |
| case VariableType::TYPE_FLOAT: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| float minVal = valueRange.component(ndx).getMin().asFloat(); |
| float maxVal = valueRange.component(ndx).getMax().asFloat(); |
| rangeLength += maxVal - minVal; |
| } |
| break; |
| |
| case VariableType::TYPE_BOOL: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| int minVal = valueRange.component(ndx).getMin().asBool() ? 1 : 0; |
| int maxVal = valueRange.component(ndx).getMin().asBool() ? 1 : 0; |
| rangeLength += (float)(maxVal - minVal); |
| } |
| break; |
| |
| case VariableType::TYPE_INT: |
| case VariableType::TYPE_SAMPLER_2D: |
| case VariableType::TYPE_SAMPLER_CUBE: |
| for (int ndx = 0; ndx < type.getNumElements(); ndx++) |
| { |
| int minVal = valueRange.component(ndx).getMin().asInt(); |
| int maxVal = valueRange.component(ndx).getMax().asInt(); |
| rangeLength += (float)(maxVal - minVal); |
| } |
| break; |
| |
| case VariableType::TYPE_ARRAY: |
| { |
| int numElements = type.getNumElements(); |
| for (int ndx = 0; ndx < numElements; ndx++) |
| rangeLength += computeRangeLengthSum(valueRange.arrayElement(ndx)); |
| break; |
| } |
| |
| case VariableType::TYPE_STRUCT: |
| { |
| int numMembers = (int)type.getMembers().size(); |
| for (int ndx = 0; ndx < numMembers; ndx++) |
| rangeLength += computeRangeLengthSum(valueRange.member(ndx)); |
| break; |
| } |
| |
| default: |
| TCU_FAIL("Invalid type"); |
| } |
| |
| return rangeLength; |
| } |
| |
| float computeDynamicRangeWeight (ConstValueRangeAccess valueRange) |
| { |
| const VariableType& type = valueRange.getType(); |
| float rangeLenSum = computeRangeLengthSum(valueRange); |
| int numScalars = type.getScalarSize(); |
| |
| return rangeLenSum / (float)numScalars; |
| } |
| |
| } // rsg |