Edit

kc3-lang/angle/src/compiler/translator/IntermNodePatternMatcher.cpp

Branch :

  • Show log

    Commit

  • Author : Corentin Wallez
    Date : 2016-11-23 13:44:05
    Hash : 1212bcac
    Message : translator: separate declarations after rewriting loops Otherwise when trying to add the declarations back, things might fail because the loop initialization is a sequence and not a block. BUG=668028 Change-Id: I8d84a25c25765e9655c16ce56604ae08f0f8176c Reviewed-on: https://chromium-review.googlesource.com/414305 Commit-Queue: Corentin Wallez <cwallez@chromium.org> Reviewed-by: Olli Etuaho <oetuaho@nvidia.com>

  • src/compiler/translator/IntermNodePatternMatcher.cpp
  • //
    // Copyright (c) 2016 The ANGLE Project Authors. All rights reserved.
    // Use of this source code is governed by a BSD-style license that can be
    // found in the LICENSE file.
    //
    // IntermNodePatternMatcher is a helper class for matching node trees to given patterns.
    // It can be used whenever the same checks for certain node structures are common to multiple AST
    // traversers.
    //
    
    #include "compiler/translator/IntermNodePatternMatcher.h"
    
    #include "compiler/translator/IntermNode.h"
    
    namespace sh
    {
    
    IntermNodePatternMatcher::IntermNodePatternMatcher(const unsigned int mask) : mMask(mask)
    {
    }
    
    // static
    bool IntermNodePatternMatcher::IsDynamicIndexingOfVectorOrMatrix(TIntermBinary *node)
    {
        return node->getOp() == EOpIndexIndirect && !node->getLeft()->isArray() &&
               node->getLeft()->getBasicType() != EbtStruct;
    }
    
    bool IntermNodePatternMatcher::matchInternal(TIntermBinary *node, TIntermNode *parentNode)
    {
        if ((mMask & kExpressionReturningArray) != 0)
        {
            if (node->isArray() && node->getOp() == EOpAssign && parentNode != nullptr &&
                !parentNode->getAsBlock())
            {
                return true;
            }
        }
    
        if ((mMask & kUnfoldedShortCircuitExpression) != 0)
        {
            if (node->getRight()->hasSideEffects() &&
                (node->getOp() == EOpLogicalOr || node->getOp() == EOpLogicalAnd))
            {
                return true;
            }
        }
        return false;
    }
    
    bool IntermNodePatternMatcher::match(TIntermBinary *node, TIntermNode *parentNode)
    {
        // L-value tracking information is needed to check for dynamic indexing in L-value.
        // Traversers that don't track l-values can still use this class and match binary nodes with
        // this variation of this method if they don't need to check for dynamic indexing in l-values.
        ASSERT((mMask & kDynamicIndexingOfVectorOrMatrixInLValue) == 0);
        return matchInternal(node, parentNode);
    }
    
    bool IntermNodePatternMatcher::match(TIntermBinary *node,
                                         TIntermNode *parentNode,
                                         bool isLValueRequiredHere)
    {
        if (matchInternal(node, parentNode))
        {
            return true;
        }
        if ((mMask & kDynamicIndexingOfVectorOrMatrixInLValue) != 0)
        {
            if (isLValueRequiredHere && IsDynamicIndexingOfVectorOrMatrix(node))
            {
                return true;
            }
        }
        return false;
    }
    
    bool IntermNodePatternMatcher::match(TIntermAggregate *node, TIntermNode *parentNode)
    {
        if ((mMask & kExpressionReturningArray) != 0)
        {
            if (parentNode != nullptr)
            {
                TIntermBinary *parentBinary = parentNode->getAsBinaryNode();
                bool parentIsAssignment =
                    (parentBinary != nullptr &&
                     (parentBinary->getOp() == EOpAssign || parentBinary->getOp() == EOpInitialize));
    
                if (node->getType().isArray() && !parentIsAssignment &&
                    (node->isConstructor() || node->getOp() == EOpFunctionCall) &&
                    !parentNode->getAsBlock())
                {
                    return true;
                }
            }
        }
        return false;
    }
    
    bool IntermNodePatternMatcher::match(TIntermTernary *node)
    {
        if ((mMask & kUnfoldedShortCircuitExpression) != 0)
        {
            return true;
        }
        return false;
    }
    
    bool IntermNodePatternMatcher::match(TIntermDeclaration *node)
    {
        if ((mMask & kMultiDeclaration) != 0)
        {
            return node->getSequence()->size() > 1;
        }
        return false;
    }
    
    }  // namespace sh