diff --git a/Test/baseResults/reflection.vert.out b/Test/baseResults/reflection.vert.out
index f5591ce5656ceeb8eca6ce1f620a33e60fc20b7a..583afa35af2fa05d597eaba4e6bdbc19ae1d53af 100644
--- a/Test/baseResults/reflection.vert.out
+++ b/Test/baseResults/reflection.vert.out
@@ -19,7 +19,7 @@ scalarAfterm23: offset 48, type 1404, size 1, index 0
 c_m23: offset 16, type 8b67, size 1, index 2
 c_scalarAfterm23: offset 64, type 1404, size 1, index 2
 scalarBeforeArray: offset 96, type 1404, size 1, index 0
-floatArray: offset 112, type 1406, size 5, index 0
+floatArray: offset 112, type 1406, size 3, index 0
 scalarAfterArray: offset 192, type 1404, size 1, index 0
 ablock.memvec2: offset 48, type 8b50, size 1, index 1
 ablock.memf1: offset 56, type 1406, size 1, index 1
@@ -27,8 +27,8 @@ ablock.memf2: offset 60, type 8b56, size 1, index 1
 ablock.memf3: offset 64, type 1404, size 1, index 1
 ablock.memvec2a: offset 72, type 8b50, size 1, index 1
 ablock.m22: offset 80, type 8b5a, size 7, index 1
-dm22: offset -1, type 8b5a, size 10, index -1
-m22: offset 208, type 8b5a, size 9, index 0
+dm22: offset -1, type 8b5a, size 4, index -1
+m22: offset 208, type 8b5a, size 3, index 0
 nest.foo.n1.a: offset 0, type 1406, size 1, index 3
 nest.foo.n2.b: offset 16, type 1406, size 1, index 3
 nest.foo.n2.c: offset 20, type 1406, size 1, index 3
diff --git a/glslang/MachineIndependent/reflection.cpp b/glslang/MachineIndependent/reflection.cpp
index 27cd53d5a48ee46776f15f429caab5cef6732921..8c0977a3b5184e0f54fadf8828eab154887f0c95 100644
--- a/glslang/MachineIndependent/reflection.cpp
+++ b/glslang/MachineIndependent/reflection.cpp
@@ -235,21 +235,41 @@ public:
         return size;
     }
 
-    // Add a complex uniform reference where blocks/struct/arrays are involved in the access.
-    // Handles the situation where the left node is at too coarse a granularity to be a single 
-    // uniform, while the result of the operation at 'node' is at the right granuarity.
+    // Add a uniform dereference where blocks/struct/arrays are involved in the access.
+    // Handles the situation where the left node is at the correct or too coarse a
+    // granularity for reflection.  (That is, further dereferences up the tree will be 
+    // skipped.) Earlier dereferences, down the tree, will be handled
+    // at the same time, and logged to prevent reprocessing as the tree is traversed.
     //
-    // Note: Simpler things like the following are already handled elsewhere:
-    //  - a simple non-array, non-struct variable
-    //  - a variable that's an array of non-struct
+    // Note: Other things like the following must be caught elsewhere:
+    //  - a simple non-array, non-struct variable (no dereference even conceivable)
+    //  - an aggregrate consumed en masse, without a dereference
     //
     // So, this code is for cases like 
-    //   - a struct/block holding a member (member is array or not)
+    //   - a struct/block dereferencing a member (whether the member is array or not)
     //   - an array of struct
     //   - structs/arrays containing the above
     //
-    void addDereferencedUniform(TIntermSymbol* base, TIntermBinary* topNode)
+    void addDereferencedUniform(TIntermBinary* topNode)
     {
+        // See if too fine-grained to process (wait to get further down the tree)
+        const TType& leftType = topNode->getLeft()->getType();
+        if ((leftType.isVector() || leftType.isMatrix()) && ! leftType.isArray())
+            return;
+
+        // We have an array or structure or block dereference, see if it's a uniform 
+        // based dereference (if not, skip it).
+        TIntermSymbol* base = findBase(topNode);
+        if (! base || base->getQualifier().storage != EvqUniform)
+            return;
+            
+        // See if we've already processed this (e.g., in the middle of something 
+        // we did earlier), and if so skip it
+        if (processedDerefs.find(topNode) != processedDerefs.end())
+            return;
+
+        // Process this uniform dereference
+
         int offset = -1;
         int blockIndex = -1;
         bool anonymous = false;
@@ -269,21 +289,49 @@ public:
                 blockIndex = it->second;
         }
 
-        // Process the dereference chain, backward, accumulating the pieces on a stack
+        // If the derefenced entity to record is an array, note the maximum array size.
+        int maxArraySize;
+        const TType* reflectionType;
+        if (isReflectionGranularity(topNode->getLeft()->getType()) && topNode->getLeft()->isArray()) {
+            reflectionType = &topNode->getLeft()->getType();
+            switch (topNode->getOp()) {
+            case EOpIndexIndirect:
+                maxArraySize = topNode->getLeft()->getType().getArraySize();
+                break;
+            case EOpIndexDirect:
+                maxArraySize = topNode->getRight()->getAsConstantUnion()->getConstArray()[0].getIConst() + 1;
+                break;
+            default:
+                assert(0);
+                maxArraySize = 1;
+                break;
+            }
+        } else {
+            reflectionType = &topNode->getType();
+            maxArraySize = 1;
+        }
+
+        // TODO: fully expand a partially dereferenced aggregate
+
+        // Process the dereference chain, backward, accumulating the pieces on a stack.
+        // If the topNode is a simple array dereference, don't include that.
         if (block)
             offset = 0;
         std::list<TString> derefs;
         for (TIntermBinary* visitNode = topNode; visitNode; visitNode = visitNode->getLeft()->getAsBinaryNode()) {
+            processedDerefs.insert(visitNode);
             int index;
             switch (visitNode->getOp()) {
             case EOpIndexIndirect:
                 // TODO handle indirect references in mid-chain: enumerate all possibilities?
-                derefs.push_back(TString("[") + String(0) + "]");
+                if (! isReflectionGranularity(visitNode->getLeft()->getType()))
+                    derefs.push_back(TString("[") + String(0) + "]");
                 break;
             case EOpIndexDirect:
-                // TODO: reflection: track the highest used index for an array, to reduce the array's size
-                index = visitNode->getRight()->getAsConstantUnion()->getConstArray()[0].getIConst();
-                derefs.push_back(TString("[") + String(index) + "]");
+                if (! isReflectionGranularity(visitNode->getLeft()->getType())) {
+                    index = visitNode->getRight()->getAsConstantUnion()->getConstArray()[0].getIConst();
+                    derefs.push_back(TString("[") + String(index) + "]");
+                }
                 break;
             case EOpIndexDirectStruct:
                 index = visitNode->getRight()->getAsConstantUnion()->getConstArray()[0].getIConst();
@@ -311,7 +359,7 @@ public:
         if (name.size() > 0) {
             if (reflection.nameToIndex.find(name) == reflection.nameToIndex.end()) {
                 reflection.nameToIndex[name] = reflection.indexToUniform.size();                        
-                reflection.indexToUniform.push_back(TObjectReflection(name, offset, mapToGlType(topNode->getType()), mapToGlArraySize(topNode->getType()), blockIndex));
+                reflection.indexToUniform.push_back(TObjectReflection(name, offset, mapToGlType(*reflectionType), maxArraySize, blockIndex));
             }
         }
     }
@@ -595,6 +643,7 @@ public:
     TFunctionStack functions;
     const TIntermediate& intermediate;
     TReflection& reflection;
+    std::set<TIntermNode*> processedDerefs;
 };
 
 const int TLiveTraverser::baseAlignmentVec4Std140 = 16;
@@ -617,6 +666,7 @@ bool LiveAggregate(bool /* preVisit */, TIntermAggregate* node, TIntermTraverser
 }
 
 // To catch dereferenced aggregates that must be reflected.
+// This catches them at the highest level possible in the tree.
 bool LiveBinary(bool /* preVisit */, TIntermBinary* node, TIntermTraverser* it)
 {
     TLiveTraverser* oit = static_cast<TLiveTraverser*>(it);
@@ -625,24 +675,18 @@ bool LiveBinary(bool /* preVisit */, TIntermBinary* node, TIntermTraverser* it)
     case EOpIndexDirect:
     case EOpIndexIndirect:
     case EOpIndexDirectStruct:
-        // If the left side is already small enough granularity to report, ignore
-        // this operation, and pick it up when the left side is visited.
-        if (! oit->isReflectionGranularity(node->getLeft()->getType()) &&
-            oit->isReflectionGranularity(node->getType())) {            
-            // right granularity; see if this really is a uniform-based dereference,
-            // and if so, process it
-            TIntermSymbol* base = oit->findBase(node);
-            if (base && base->getQualifier().storage == EvqUniform)
-                oit->addDereferencedUniform(base, node);
-        }
+        oit->addDereferencedUniform(node);
+        break;
     default:
         break;
     }
 
-    return true;  // still need to visit everything below
+    // still need to visit everything below, which could contain sub-expressions 
+    // containing different uniforms
+    return true;
 }
 
-// To catch non-dereferenced objects that must be reflected.
+// To reflect non-dereferenced objects.
 void LiveSymbol(TIntermSymbol* symbol, TIntermTraverser* it)
 {
     TLiveTraverser* oit = static_cast<TLiveTraverser*>(it);