From 2d0f1e5424b728f0b5364c568005c0542ce0cbcd Mon Sep 17 00:00:00 2001 From: John Kessenich <cepheus@frii.com> Date: Fri, 17 May 2013 02:51:45 +0000 Subject: [PATCH] Add precision qualifier propagation for swizzling, texture lookups, built-in funtions mapped to operators, comma op, and more robustly propagate for all binary/unary ops. git-svn-id: https://cvs.khronos.org/svn/repos/ogl/trunk/ecosystem/public/sdk/tools/glslang@21622 e7fa87d3-cd2b-0410-9028-fcbf551c1848 --- Test/precision.frag | 8 ++ glslang/Include/intermediate.h | 2 + glslang/MachineIndependent/Intermediate.cpp | 86 ++++++++++++--------- glslang/MachineIndependent/glslang.y | 9 ++- 4 files changed, 68 insertions(+), 37 deletions(-) diff --git a/Test/precision.frag b/Test/precision.frag index cd601b285..37f3462b7 100644 --- a/Test/precision.frag +++ b/Test/precision.frag @@ -10,6 +10,10 @@ lowp vec2 foo(mediump vec3 mv3) int global_medium; +uniform lowp sampler2D samplerLow; +uniform mediump sampler2D samplerMed; +uniform highp sampler2D samplerHigh; + precision highp int; precision highp ivec2; // ERROR precision mediump int[2]; // ERROR @@ -59,4 +63,8 @@ void main() sum += level1_low3; sum += 4 + ((ivec2(level1_low3) * ivec2(level1_high) + ivec2((/* comma operator */level1_low3, level1_high)))).x; + + texture2D(samplerLow, vec2(0.1, 0.2)); + texture2D(samplerMed, vec2(0.1, 0.2)); + texture2D(samplerHigh, vec2(0.1, 0.2)); } diff --git a/glslang/Include/intermediate.h b/glslang/Include/intermediate.h index e71da208c..299945f32 100644 --- a/glslang/Include/intermediate.h +++ b/glslang/Include/intermediate.h @@ -482,6 +482,7 @@ public: virtual TIntermTyped* getRight() const { return right; } virtual TIntermBinary* getAsBinaryNode() { return this; } virtual bool promote(TInfoSink&); + virtual void updatePrecision(); protected: TIntermTyped* left; TIntermTyped* right; @@ -499,6 +500,7 @@ public: virtual TIntermTyped* getOperand() { return operand; } virtual TIntermUnary* getAsUnaryNode() { return this; } virtual bool promote(TInfoSink&); + virtual void updatePrecision(); protected: TIntermTyped* operand; }; diff --git a/glslang/MachineIndependent/Intermediate.cpp b/glslang/MachineIndependent/Intermediate.cpp index fe4d3a9a7..e69caf99f 100644 --- a/glslang/MachineIndependent/Intermediate.cpp +++ b/glslang/MachineIndependent/Intermediate.cpp @@ -126,6 +126,8 @@ TIntermTyped* TIntermediate::addBinaryMath(TOperator op, TIntermTyped* left, TIn node->setRight(right); if (! node->promote(infoSink)) return 0; + + node->updatePrecision(); // // If they are both constants, they must be folded. @@ -169,6 +171,8 @@ TIntermTyped* TIntermediate::addAssign(TOperator op, TIntermTyped* left, TInterm if (! node->promote(infoSink)) return 0; + node->updatePrecision(); + return node; } @@ -274,6 +278,8 @@ TIntermTyped* TIntermediate::addUnaryMath(TOperator op, TIntermNode* childNode, if (! node->promote(infoSink)) return 0; + node->updatePrecision(); + if (child->getAsConstantUnion()) return child->getAsConstantUnion()->fold(op, node->getType(), infoSink); @@ -309,31 +315,26 @@ TIntermTyped* TIntermediate::addBuiltInFunctionCall(TOperator op, bool unary, TI node->getQualifier().precision = child->getQualifier().precision; // propagate precision down to child - if (node->getQualifier().precision != EpqNone && - child->getQualifier().precision == EpqNone) - child->getQualifier().precision = node->getQualifier().precision; + if (node->getQualifier().precision != EpqNone) + child->propagatePrecision(node->getQualifier().precision); return node; } else { // setAggregateOperater() calls fold() for constant folding TIntermTyped* node = setAggregateOperator(childNode, op, returnType, childNode->getLine()); - - if (returnType.getQualifier().precision == EpqNone && profile == EEsProfile) { - // get maximum precision from arguments, for the built-in's return precision - - TIntermSequence& sequence = node->getAsAggregate()->getSequence(); - TPrecisionQualifier maxPq = EpqNone; - for (unsigned int arg = 0; arg < sequence.size(); ++arg) - maxPq = std::max(maxPq, sequence[arg]->getAsTyped()->getQualifier().precision); - node->getQualifier().precision = maxPq; - } - - if (node->getQualifier().precision != EpqNone) { + + TPrecisionQualifier correctPrecision = returnType.getQualifier().precision; + if (correctPrecision == EpqNone && profile == EEsProfile) { + // find the maximum precision from the arguments, for the built-in's return precision TIntermSequence& sequence = node->getAsAggregate()->getSequence(); for (unsigned int arg = 0; arg < sequence.size(); ++arg) - if (sequence[arg]->getAsTyped()->getQualifier().precision == EpqNone) - sequence[arg]->getAsTyped()->getQualifier().precision = node->getQualifier().precision; + correctPrecision = std::max(correctPrecision, sequence[arg]->getAsTyped()->getQualifier().precision); } + + // Propagate precision through this node and its children. That algorithm stops + // when a precision is found, so start by clearing this subroot precision + node->getQualifier().precision = EpqNone; + node->propagatePrecision(correctPrecision); return node; } @@ -725,12 +726,15 @@ TIntermTyped* TIntermediate::addComma(TIntermTyped* left, TIntermTyped* right, T { if (left->getType().getQualifier().storage == EvqConst && right->getType().getQualifier().storage == EvqConst) { + return right; } else { TIntermTyped *commaAggregate = growAggregate(left, right, line); commaAggregate->getAsAggregate()->setOperator(EOpComma); commaAggregate->setType(right->getType()); commaAggregate->getTypePointer()->getQualifier().storage = EvqTemporary; + commaAggregate->getTypePointer()->getQualifier().precision = right->getTypePointer()->getQualifier().precision; + return commaAggregate; } } @@ -782,6 +786,7 @@ TIntermTyped* TIntermediate::addSelection(TIntermTyped* cond, TIntermTyped* true // TIntermSelection* node = new TIntermSelection(cond, trueBlock, falseBlock, trueBlock->getType()); node->setLine(line); + node->getQualifier().precision = std::max(trueBlock->getQualifier().precision, falseBlock->getQualifier().precision); return node; } @@ -966,6 +971,14 @@ bool TIntermUnary::promote(TInfoSink&) return true; } +void TIntermUnary::updatePrecision() +{ + if (getBasicType() == EbtInt || getBasicType() == EbtUint || getBasicType() == EbtFloat) { + if (operand->getQualifier().precision > getQualifier().precision) + getQualifier().precision = operand->getQualifier().precision; + } +} + // // Establishes the type of the resultant operation, as well as // makes the operator the correct one for the operands. @@ -987,21 +1000,11 @@ bool TIntermBinary::promote(TInfoSink& infoSink) setType(left->getType()); type.getQualifier().storage = EvqTemporary; - // Fix precision qualifiers - if (right->getQualifier().precision > getQualifier().precision) - getQualifier().precision = right->getQualifier().precision; - if (getQualifier().precision != EpqNone) { - left->propagatePrecision(getQualifier().precision); - right->propagatePrecision(getQualifier().precision); - } - // // Array operations. // if (left->isArray()) { - switch (op) { - // // Promote to conditional // @@ -1229,6 +1232,17 @@ bool TIntermBinary::promote(TInfoSink& infoSink) return true; } +void TIntermBinary::updatePrecision() +{ + if (getBasicType() == EbtInt || getBasicType() == EbtUint || getBasicType() == EbtFloat) { + getQualifier().precision = std::max(right->getQualifier().precision, left->getQualifier().precision); + if (getQualifier().precision != EpqNone) { + left->propagatePrecision(getQualifier().precision); + right->propagatePrecision(getQualifier().precision); + } + } +} + void TIntermTyped::propagatePrecision(TPrecisionQualifier newPrecision) { if (getQualifier().precision != EpqNone || (getBasicType() != EbtInt && getBasicType() != EbtUint && getBasicType() != EbtFloat)) @@ -1240,12 +1254,17 @@ void TIntermTyped::propagatePrecision(TPrecisionQualifier newPrecision) if (binaryNode) { binaryNode->getLeft()->propagatePrecision(newPrecision); binaryNode->getRight()->propagatePrecision(newPrecision); + + return; } TIntermUnary* unaryNode = getAsUnaryNode(); - if (unaryNode) + if (unaryNode) { unaryNode->getOperand()->propagatePrecision(newPrecision); + return; + } + TIntermAggregate* aggregateNode = getAsAggregate(); if (aggregateNode) { TIntermSequence operands = aggregateNode->getSequence(); @@ -1255,6 +1274,8 @@ void TIntermTyped::propagatePrecision(TPrecisionQualifier newPrecision) break; typedNode->propagatePrecision(newPrecision); } + + return; } TIntermSelection* selectionNode = getAsSelectionNode(); @@ -1266,14 +1287,9 @@ void TIntermTyped::propagatePrecision(TPrecisionQualifier newPrecision) if (typedNode) typedNode->propagatePrecision(newPrecision); } - } - // TODO: functionality: propagate precision for - // comma operator: just through the last operand - // ":?" and ",": where is this triggered? - // built-in function calls: how much to propagate to arguments? - // length()? - // indexing? + return; + } } TIntermTyped* TIntermediate::promoteConstantUnion(TBasicType promoteTo, TIntermConstantUnion* node) diff --git a/glslang/MachineIndependent/glslang.y b/glslang/MachineIndependent/glslang.y index ed29c50a2..0302e7b03 100644 --- a/glslang/MachineIndependent/glslang.y +++ b/glslang/MachineIndependent/glslang.y @@ -410,12 +410,12 @@ postfix_expression unionArray->setIConst(fields.offsets[0]); TIntermTyped* index = parseContext.intermediate.addConstantUnion(unionArray, TType(EbtInt, EvqConst), $3.line); $$ = parseContext.intermediate.addIndex(EOpIndexDirect, $1, index, $2.line); - $$->setType(TType($1->getBasicType())); + $$->setType(TType($1->getBasicType(), EvqTemporary, $1->getType().getQualifier().precision)); } else { TString vectorString = *$3.string; TIntermTyped* index = parseContext.intermediate.addSwizzle(fields, $3.line); $$ = parseContext.intermediate.addIndex(EOpVectorSwizzle, $1, index, $2.line); - $$->setType(TType($1->getBasicType(), EvqTemporary, (int) vectorString.size())); + $$->setType(TType($1->getBasicType(), EvqTemporary, $1->getType().getQualifier().precision, (int) vectorString.size())); } } } else if ($1->isMatrix()) { @@ -582,6 +582,11 @@ function_call } qualifierList.push_back(qual); } + + // built-in texturing functions get their return value precision from the precision of the sampler + if (builtIn && fnCandidate->getReturnType().getQualifier().precision == EpqNone && + fnCandidate->getParamCount() > 0 && (*fnCandidate)[0].type->getBasicType() == EbtSampler) + $$->getQualifier().precision = $$->getAsAggregate()->getSequence()[0]->getAsTyped()->getQualifier().precision; } } else { // error message was put out by PaFindFunction() -- GitLab