blob: 98e010a56701cfca45a5fb7e845ae611061e4811 [file] [log] [blame]
Olli Etuaho822fa842015-04-16 14:26:10 +03001//
2// Copyright (c) 2002-2015 The ANGLE Project Authors. All rights reserved.
3// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6// The SeparateArrayInitialization function splits each array initialization into a declaration and an assignment.
7// Example:
8// type[n] a = initializer;
9// will effectively become
10// type[n] a;
11// a = initializer;
Olli Etuahob1edc4f2015-11-02 17:20:03 +020012//
Olli Etuaho18b9deb2015-11-05 12:14:50 +020013// Note that if the array is declared as const, the initialization may still be split, making the
14// AST technically invalid. Because of that this transformation should only be used when subsequent
15// stages don't care about const qualifiers. However, the initialization will not be split if the
16// initializer can be written as a HLSL literal.
Olli Etuaho822fa842015-04-16 14:26:10 +030017
18#include "compiler/translator/SeparateArrayInitialization.h"
19
20#include "compiler/translator/IntermNode.h"
Olli Etuaho18b9deb2015-11-05 12:14:50 +020021#include "compiler/translator/OutputHLSL.h"
Olli Etuaho822fa842015-04-16 14:26:10 +030022
Jamie Madill45bcc782016-11-07 13:58:48 -050023namespace sh
24{
25
Olli Etuaho822fa842015-04-16 14:26:10 +030026namespace
27{
28
29class SeparateArrayInitTraverser : private TIntermTraverser
30{
31 public:
32 static void apply(TIntermNode *root);
33 private:
34 SeparateArrayInitTraverser();
Olli Etuaho13389b62016-10-16 11:48:18 +010035 bool visitDeclaration(Visit, TIntermDeclaration *node) override;
Olli Etuaho822fa842015-04-16 14:26:10 +030036};
37
38void SeparateArrayInitTraverser::apply(TIntermNode *root)
39{
40 SeparateArrayInitTraverser separateInit;
41 root->traverse(&separateInit);
42 separateInit.updateTree();
43}
44
45SeparateArrayInitTraverser::SeparateArrayInitTraverser()
46 : TIntermTraverser(true, false, false)
47{
48}
49
Olli Etuaho13389b62016-10-16 11:48:18 +010050bool SeparateArrayInitTraverser::visitDeclaration(Visit, TIntermDeclaration *node)
Olli Etuaho822fa842015-04-16 14:26:10 +030051{
Olli Etuaho13389b62016-10-16 11:48:18 +010052 TIntermSequence *sequence = node->getSequence();
53 TIntermBinary *initNode = sequence->back()->getAsBinaryNode();
54 if (initNode != nullptr && initNode->getOp() == EOpInitialize)
Olli Etuaho822fa842015-04-16 14:26:10 +030055 {
Olli Etuaho13389b62016-10-16 11:48:18 +010056 TIntermTyped *initializer = initNode->getRight();
57 if (initializer->isArray() && !sh::OutputHLSL::canWriteAsHLSLLiteral(initializer))
Olli Etuaho822fa842015-04-16 14:26:10 +030058 {
Olli Etuaho13389b62016-10-16 11:48:18 +010059 // We rely on that array declarations have been isolated to single declarations.
60 ASSERT(sequence->size() == 1);
61 TIntermTyped *symbol = initNode->getLeft();
62 TIntermBlock *parentBlock = getParentNode()->getAsBlock();
63 ASSERT(parentBlock != nullptr);
Olli Etuaho822fa842015-04-16 14:26:10 +030064
Olli Etuaho13389b62016-10-16 11:48:18 +010065 TIntermSequence replacements;
Olli Etuaho822fa842015-04-16 14:26:10 +030066
Olli Etuaho13389b62016-10-16 11:48:18 +010067 TIntermDeclaration *replacementDeclaration = new TIntermDeclaration();
68 replacementDeclaration->appendDeclarator(symbol);
69 replacementDeclaration->setLine(symbol->getLine());
70 replacements.push_back(replacementDeclaration);
Olli Etuaho822fa842015-04-16 14:26:10 +030071
Olli Etuaho13389b62016-10-16 11:48:18 +010072 TIntermBinary *replacementAssignment =
73 new TIntermBinary(EOpAssign, symbol, initializer);
74 replacementAssignment->setLine(symbol->getLine());
75 replacements.push_back(replacementAssignment);
Olli Etuaho822fa842015-04-16 14:26:10 +030076
Olli Etuaho13389b62016-10-16 11:48:18 +010077 mMultiReplacements.push_back(
78 NodeReplaceWithMultipleEntry(parentBlock, node, replacements));
Olli Etuaho822fa842015-04-16 14:26:10 +030079 }
Olli Etuaho822fa842015-04-16 14:26:10 +030080 }
Olli Etuaho13389b62016-10-16 11:48:18 +010081 return false;
Olli Etuaho822fa842015-04-16 14:26:10 +030082}
83
84} // namespace
85
86void SeparateArrayInitialization(TIntermNode *root)
87{
88 SeparateArrayInitTraverser::apply(root);
89}
Jamie Madill45bcc782016-11-07 13:58:48 -050090
91} // namespace sh