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