Hash :
d7aa0130
Author :
Date :
2021-04-26T16:56:15
Upstream Apple's direct-to-Metal backend: compile translator. This change is meant to merge the translator changes from Apple's direct-to-Metal backend. Taken from Kyle Piddington's CL: https://chromium-review.googlesource.com/c/angle/angle/+/2857366/ The goal of this CL is to merge the translator code in a state that compiles, but not to switch the Metal backend over to use this translator backend yet. Bug: angleproject:5505 Change-Id: I68a6354604498cd5fd1eb96c13fc56f3b38f2bd0 Reviewed-on: https://chromium-review.googlesource.com/c/angle/angle/+/2897536 Reviewed-by: Jonah Ryan-Davis <jonahr@google.com> Commit-Queue: Jonah Ryan-Davis <jonahr@google.com>
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
//
// Copyright 2020 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.
//
#include "compiler/translator/TranslatorMetalDirect/WrapMain.h"
#include "compiler/translator/Compiler.h"
#include "compiler/translator/TranslatorMetalDirect/AstHelpers.h"
using namespace sh;
////////////////////////////////////////////////////////////////////////////////
namespace
{
class Wrapper : public TIntermTraverser
{
private:
IdGen &mIdGen;
public:
Wrapper(TSymbolTable &symbolTable, IdGen &idGen)
: TIntermTraverser(false, false, true, &symbolTable), mIdGen(idGen)
{}
bool visitBlock(Visit, TIntermBlock *blockNode) override
{
if (blockNode != getRootNode())
{
return true;
}
for (TIntermNode *node : *blockNode->getSequence())
{
if (TIntermFunctionDefinition *funcDefNode = node->getAsFunctionDefinition())
{
const TFunction &func = *funcDefNode->getFunction();
if (func.isMain())
{
visitMain(*blockNode, funcDefNode);
break;
}
}
}
return true;
}
private:
void visitMain(TIntermBlock &root, TIntermFunctionDefinition *funcDefNode)
{
const TFunction &func = *funcDefNode->getFunction();
ASSERT(func.isMain());
ASSERT(func.getReturnType().getBasicType() == TBasicType::EbtVoid);
ASSERT(func.getParamCount() == 0);
const TFunction &externalMainFunc = *funcDefNode->getFunction();
const TFunction &internalMainFunc = CloneFunction(*mSymbolTable, mIdGen, externalMainFunc);
TIntermFunctionPrototype *externalMainProto = funcDefNode->getFunctionPrototype();
TIntermFunctionPrototype *internalMainProto =
new TIntermFunctionPrototype(&internalMainFunc);
TIntermBlock *externalMainBody = new TIntermBlock();
externalMainBody->appendStatement(
TIntermAggregate::CreateFunctionCall(internalMainFunc, new TIntermSequence()));
TIntermBlock *internalMainBody = funcDefNode->getBody();
TIntermFunctionDefinition *externalMainDef =
new TIntermFunctionDefinition(externalMainProto, externalMainBody);
TIntermFunctionDefinition *internalMainDef =
new TIntermFunctionDefinition(internalMainProto, internalMainBody);
mMultiReplacements.push_back(NodeReplaceWithMultipleEntry(
&root, funcDefNode, TIntermSequence{internalMainDef, externalMainDef}));
}
};
} // namespace
bool sh::WrapMain(TCompiler &compiler, IdGen &idGen, TIntermBlock &root)
{
TSymbolTable &symbolTable = compiler.getSymbolTable();
Wrapper wrapper(symbolTable, idGen);
root.traverse(&wrapper);
if (!wrapper.updateTree(&compiler, &root))
{
return false;
}
return true;
}