Remove leftover cpp file
Change-Id: I30e4b37a43d1301d9e8011c9952874f3ba8d4f8b Reviewed-on: https://skia-review.googlesource.com/c/skia/+/383702 Reviewed-by: Kevin Lubick <kjlubick@google.com> Reviewed-by: Brian Osman <brianosman@google.com> Commit-Queue: Kevin Lubick <kjlubick@google.com>
This commit is contained in:
parent
6679eb9786
commit
65e50128f2
@ -1,674 +0,0 @@
|
||||
/*
|
||||
* Copyright 2016 Google Inc.
|
||||
*
|
||||
* Use of this source code is governed by a BSD-style license that can be
|
||||
* found in the LICENSE file.
|
||||
*/
|
||||
|
||||
#include "src/sksl/SkSLCFGGenerator.h"
|
||||
|
||||
#include "src/sksl/SkSLOperators.h"
|
||||
#include "src/sksl/ir/SkSLBinaryExpression.h"
|
||||
#include "src/sksl/ir/SkSLConstructor.h"
|
||||
#include "src/sksl/ir/SkSLDoStatement.h"
|
||||
#include "src/sksl/ir/SkSLExpressionStatement.h"
|
||||
#include "src/sksl/ir/SkSLExternalFunctionCall.h"
|
||||
#include "src/sksl/ir/SkSLFieldAccess.h"
|
||||
#include "src/sksl/ir/SkSLForStatement.h"
|
||||
#include "src/sksl/ir/SkSLFunctionCall.h"
|
||||
#include "src/sksl/ir/SkSLIfStatement.h"
|
||||
#include "src/sksl/ir/SkSLIndexExpression.h"
|
||||
#include "src/sksl/ir/SkSLNop.h"
|
||||
#include "src/sksl/ir/SkSLPostfixExpression.h"
|
||||
#include "src/sksl/ir/SkSLPrefixExpression.h"
|
||||
#include "src/sksl/ir/SkSLReturnStatement.h"
|
||||
#include "src/sksl/ir/SkSLSwitchStatement.h"
|
||||
#include "src/sksl/ir/SkSLSwizzle.h"
|
||||
#include "src/sksl/ir/SkSLTernaryExpression.h"
|
||||
|
||||
#include <tuple>
|
||||
|
||||
namespace SkSL {
|
||||
|
||||
void BasicBlockNode::setExpression(std::unique_ptr<Expression> expr, ProgramUsage* usage) {
|
||||
SkASSERT(!this->isStatement());
|
||||
usage->remove(fExpression->get());
|
||||
*fExpression = std::move(expr);
|
||||
}
|
||||
|
||||
std::unique_ptr<Statement> BasicBlockNode::setStatement(std::unique_ptr<Statement> stmt,
|
||||
ProgramUsage* usage) {
|
||||
SkASSERT(!this->isExpression());
|
||||
// See comment in header - we assume that stmt was already counted in usage (it was a subset
|
||||
// of fStatement). There is no way to verify that, unfortunately.
|
||||
usage->remove(fStatement->get());
|
||||
std::unique_ptr<Statement> result;
|
||||
if (stmt->is<Nop>()) {
|
||||
result = std::move(*fStatement);
|
||||
}
|
||||
*fStatement = std::move(stmt);
|
||||
return result;
|
||||
}
|
||||
|
||||
BlockId CFG::newBlock() {
|
||||
BlockId result = fBlocks.size();
|
||||
fBlocks.emplace_back();
|
||||
if (fBlocks.size() > 1) {
|
||||
this->addExit(fCurrent, result);
|
||||
}
|
||||
fCurrent = result;
|
||||
return result;
|
||||
}
|
||||
|
||||
BlockId CFG::newIsolatedBlock() {
|
||||
BlockId result = fBlocks.size();
|
||||
fBlocks.emplace_back();
|
||||
return result;
|
||||
}
|
||||
|
||||
void CFG::addExit(BlockId from, BlockId to) {
|
||||
BasicBlock::ExitArray& exits = fBlocks[from].fExits;
|
||||
if (std::find(exits.begin(), exits.end(), to) == exits.end()) {
|
||||
exits.push_back(to);
|
||||
}
|
||||
if (from == 0 || fBlocks[from].fIsReachable) {
|
||||
fBlocks[to].fIsReachable = true;
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef SK_DEBUG
|
||||
void CFG::dump() const {
|
||||
for (size_t i = 0; i < fBlocks.size(); i++) {
|
||||
printf("Block %zu\n-------\n", i);
|
||||
fBlocks[i].dump();
|
||||
}
|
||||
}
|
||||
|
||||
void BasicBlock::dump() const {
|
||||
printf("Before: [");
|
||||
const char* separator = "";
|
||||
for (const auto& [var, expr] : fBefore) {
|
||||
printf("%s%s = %s",
|
||||
separator,
|
||||
var->description().c_str(),
|
||||
expr ? *expr ? (*expr)->description().c_str() : "NULL" : "<undefined>");
|
||||
separator = ", ";
|
||||
}
|
||||
printf("]\nIs Reachable: [%s]\n", fIsReachable ? "yes" : "no");
|
||||
for (size_t j = 0; j < fNodes.size(); j++) {
|
||||
const BasicBlock::Node& n = fNodes[j];
|
||||
printf("Node %zu (%p): %s\n", j, &n, n.description().c_str());
|
||||
}
|
||||
printf("Exits: [");
|
||||
separator = "";
|
||||
for (BlockId b : fExits) {
|
||||
printf("%s%zu", separator, b);
|
||||
separator = ", ";
|
||||
}
|
||||
printf("]\n\n");
|
||||
}
|
||||
#endif
|
||||
|
||||
bool BasicBlock::tryRemoveExpressionBefore(std::vector<BasicBlock::Node>::iterator* iter,
|
||||
Expression* e) {
|
||||
if (e->is<TernaryExpression>()) {
|
||||
return false;
|
||||
}
|
||||
bool result;
|
||||
if ((*iter)->isExpression()) {
|
||||
SkASSERT((*iter)->expression()->get() != e);
|
||||
// Remember the expression that we started on.
|
||||
Expression* old = (*iter)->expression()->get();
|
||||
// Back up `iter` until we find the expression that we want to remove. (If we don't find
|
||||
// that expression at all, fail and rescan.)
|
||||
do {
|
||||
if ((*iter) == fNodes.begin()) {
|
||||
return false;
|
||||
}
|
||||
--(*iter);
|
||||
} while (!(*iter)->isExpression() || (*iter)->expression()->get() != e);
|
||||
|
||||
// `iter` now points to our expression that needs removal. Erase it.
|
||||
result = this->tryRemoveExpression(iter);
|
||||
|
||||
// Move `iter` forward again until we find the expression we started on.
|
||||
while (!(*iter)->isExpression() || (*iter)->expression()->get() != old) {
|
||||
SkASSERT(*iter != fNodes.end());
|
||||
++(*iter);
|
||||
}
|
||||
} else {
|
||||
// Remember the statement that we started on.
|
||||
Statement* old = (*iter)->statement()->get();
|
||||
// Back up `iter` until we find the expression that we want to remove. (If we don't find
|
||||
// that expression at all, fail and rescan.)
|
||||
do {
|
||||
if ((*iter) == fNodes.begin()) {
|
||||
return false;
|
||||
}
|
||||
--(*iter);
|
||||
} while (!(*iter)->isExpression() || (*iter)->expression()->get() != e);
|
||||
|
||||
// `iter` now points to our expression that needs removal. Erase it.
|
||||
result = this->tryRemoveExpression(iter);
|
||||
|
||||
// Move `iter` forward again until we find the statement we started on.
|
||||
while (!(*iter)->isStatement() || (*iter)->statement()->get() != old) {
|
||||
SkASSERT(*iter != fNodes.end());
|
||||
++(*iter);
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
bool BasicBlock::tryRemoveLValueBefore(std::vector<BasicBlock::Node>::iterator* iter,
|
||||
Expression* lvalue) {
|
||||
switch (lvalue->kind()) {
|
||||
case Expression::Kind::kVariableReference:
|
||||
return true;
|
||||
case Expression::Kind::kSwizzle:
|
||||
return this->tryRemoveLValueBefore(iter, lvalue->as<Swizzle>().base().get());
|
||||
case Expression::Kind::kFieldAccess:
|
||||
return this->tryRemoveLValueBefore(iter, lvalue->as<FieldAccess>().base().get());
|
||||
case Expression::Kind::kIndex: {
|
||||
IndexExpression& indexExpr = lvalue->as<IndexExpression>();
|
||||
if (!this->tryRemoveLValueBefore(iter, indexExpr.base().get())) {
|
||||
return false;
|
||||
}
|
||||
return this->tryRemoveExpressionBefore(iter, indexExpr.index().get());
|
||||
}
|
||||
case Expression::Kind::kTernary: {
|
||||
TernaryExpression& ternary = lvalue->as<TernaryExpression>();
|
||||
if (!this->tryRemoveExpressionBefore(iter, ternary.test().get())) {
|
||||
return false;
|
||||
}
|
||||
if (!this->tryRemoveLValueBefore(iter, ternary.ifTrue().get())) {
|
||||
return false;
|
||||
}
|
||||
return this->tryRemoveLValueBefore(iter, ternary.ifFalse().get());
|
||||
}
|
||||
default:
|
||||
SkDEBUGFAILF("invalid lvalue: %s\n", lvalue->description().c_str());
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
bool BasicBlock::tryRemoveExpression(std::vector<BasicBlock::Node>::iterator* iter) {
|
||||
Expression* expr = (*iter)->expression()->get();
|
||||
switch (expr->kind()) {
|
||||
case Expression::Kind::kBinary: {
|
||||
BinaryExpression& b = expr->as<BinaryExpression>();
|
||||
if (b.getOperator().kind() == Token::Kind::TK_EQ) {
|
||||
if (!this->tryRemoveLValueBefore(iter, b.left().get())) {
|
||||
return false;
|
||||
}
|
||||
} else if (!this->tryRemoveExpressionBefore(iter, b.left().get())) {
|
||||
return false;
|
||||
}
|
||||
if (!this->tryRemoveExpressionBefore(iter, b.right().get())) {
|
||||
return false;
|
||||
}
|
||||
SkASSERT((*iter)->expression()->get() == expr);
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kTernary: {
|
||||
// ternaries cross basic block boundaries, must regenerate the CFG to remove it
|
||||
return false;
|
||||
}
|
||||
case Expression::Kind::kFieldAccess: {
|
||||
FieldAccess& f = expr->as<FieldAccess>();
|
||||
if (!this->tryRemoveExpressionBefore(iter, f.base().get())) {
|
||||
return false;
|
||||
}
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kSwizzle: {
|
||||
Swizzle& s = expr->as<Swizzle>();
|
||||
if (s.base() && !this->tryRemoveExpressionBefore(iter, s.base().get())) {
|
||||
return false;
|
||||
}
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kIndex: {
|
||||
IndexExpression& idx = expr->as<IndexExpression>();
|
||||
if (!this->tryRemoveExpressionBefore(iter, idx.base().get())) {
|
||||
return false;
|
||||
}
|
||||
if (!this->tryRemoveExpressionBefore(iter, idx.index().get())) {
|
||||
return false;
|
||||
}
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kConstructor: {
|
||||
Constructor& c = expr->as<Constructor>();
|
||||
for (auto& arg : c.arguments()) {
|
||||
if (!this->tryRemoveExpressionBefore(iter, arg.get())) {
|
||||
return false;
|
||||
}
|
||||
SkASSERT((*iter)->expression()->get() == expr);
|
||||
}
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kFunctionCall: {
|
||||
FunctionCall& f = expr->as<FunctionCall>();
|
||||
for (auto& arg : f.arguments()) {
|
||||
if (!this->tryRemoveExpressionBefore(iter, arg.get())) {
|
||||
return false;
|
||||
}
|
||||
SkASSERT((*iter)->expression()->get() == expr);
|
||||
}
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kPrefix:
|
||||
if (!this->tryRemoveExpressionBefore(iter,
|
||||
expr->as<PrefixExpression>().operand().get())) {
|
||||
return false;
|
||||
}
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
case Expression::Kind::kPostfix:
|
||||
if (!this->tryRemoveExpressionBefore(iter,
|
||||
expr->as<PostfixExpression>().operand().get())) {
|
||||
return false;
|
||||
}
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
case Expression::Kind::kBoolLiteral: // fall through
|
||||
case Expression::Kind::kFloatLiteral: // fall through
|
||||
case Expression::Kind::kIntLiteral: // fall through
|
||||
case Expression::Kind::kSetting: // fall through
|
||||
case Expression::Kind::kVariableReference:
|
||||
*iter = fNodes.erase(*iter);
|
||||
return true;
|
||||
default:
|
||||
SkDEBUGFAILF("unhandled expression: %s\n", expr->description().c_str());
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
bool BasicBlock::tryInsertExpression(std::vector<BasicBlock::Node>::iterator* iter,
|
||||
std::unique_ptr<Expression>* expr) {
|
||||
switch ((*expr)->kind()) {
|
||||
case Expression::Kind::kBinary: {
|
||||
BinaryExpression& b = expr->get()->as<BinaryExpression>();
|
||||
if (!this->tryInsertExpression(iter, &b.right())) {
|
||||
return false;
|
||||
}
|
||||
|
||||
++(*iter);
|
||||
if (!this->tryInsertExpression(iter, &b.left())) {
|
||||
return false;
|
||||
}
|
||||
++(*iter);
|
||||
*iter = fNodes.insert(*iter,
|
||||
BasicBlock::MakeExpression(expr, /*constantPropagation=*/true));
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kBoolLiteral: // fall through
|
||||
case Expression::Kind::kFloatLiteral: // fall through
|
||||
case Expression::Kind::kIntLiteral: // fall through
|
||||
case Expression::Kind::kVariableReference: {
|
||||
*iter = fNodes.insert(*iter,
|
||||
BasicBlock::MakeExpression(expr, /*constantPropagation=*/true));
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kConstructor: {
|
||||
Constructor& c = expr->get()->as<Constructor>();
|
||||
for (auto& arg : c.arguments()) {
|
||||
if (!this->tryInsertExpression(iter, &arg)) {
|
||||
return false;
|
||||
}
|
||||
++(*iter);
|
||||
}
|
||||
*iter = fNodes.insert(*iter,
|
||||
BasicBlock::MakeExpression(expr, /*constantPropagation=*/true));
|
||||
return true;
|
||||
}
|
||||
case Expression::Kind::kSwizzle: {
|
||||
Swizzle& s = expr->get()->as<Swizzle>();
|
||||
if (!this->tryInsertExpression(iter, &s.base())) {
|
||||
return false;
|
||||
}
|
||||
++(*iter);
|
||||
*iter = fNodes.insert(*iter,
|
||||
BasicBlock::MakeExpression(expr, /*constantPropagation=*/true));
|
||||
return true;
|
||||
}
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
void CFGGenerator::addExpression(CFG& cfg, std::unique_ptr<Expression>* e, bool constantPropagate) {
|
||||
SkASSERT(e);
|
||||
switch ((*e)->kind()) {
|
||||
case Expression::Kind::kBinary: {
|
||||
BinaryExpression& b = e->get()->as<BinaryExpression>();
|
||||
Operator op = b.getOperator();
|
||||
switch (op.kind()) {
|
||||
case Token::Kind::TK_LOGICALAND: // fall through
|
||||
case Token::Kind::TK_LOGICALOR: {
|
||||
// this isn't as precise as it could be -- we don't bother to track that if we
|
||||
// early exit from a logical and/or, we know which branch of an 'if' we're going
|
||||
// to hit -- but it won't make much difference in practice.
|
||||
this->addExpression(cfg, &b.left(), constantPropagate);
|
||||
BlockId start = cfg.fCurrent;
|
||||
cfg.newBlock();
|
||||
this->addExpression(cfg, &b.right(), constantPropagate);
|
||||
cfg.newBlock();
|
||||
cfg.addExit(start, cfg.fCurrent);
|
||||
cfg.currentBlock().fNodes.push_back(
|
||||
BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
}
|
||||
case Token::Kind::TK_EQ: {
|
||||
this->addExpression(cfg, &b.right(), constantPropagate);
|
||||
this->addLValue(cfg, &b.left());
|
||||
cfg.currentBlock().fNodes.push_back(
|
||||
BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
}
|
||||
default:
|
||||
this->addExpression(cfg, &b.left(), !b.getOperator().isAssignment());
|
||||
this->addExpression(cfg, &b.right(), constantPropagate);
|
||||
cfg.currentBlock().fNodes.push_back(
|
||||
BasicBlock::MakeExpression(e, constantPropagate));
|
||||
}
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kCodeString:
|
||||
SkDEBUGFAIL("shouldn't be able to receive kCodeString here");
|
||||
break;
|
||||
case Expression::Kind::kConstructor: {
|
||||
Constructor& c = e->get()->as<Constructor>();
|
||||
for (auto& arg : c.arguments()) {
|
||||
this->addExpression(cfg, &arg, constantPropagate);
|
||||
}
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kExternalFunctionCall: {
|
||||
ExternalFunctionCall& c = e->get()->as<ExternalFunctionCall>();
|
||||
for (auto& arg : c.arguments()) {
|
||||
this->addExpression(cfg, &arg, constantPropagate);
|
||||
}
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kFunctionCall: {
|
||||
FunctionCall& c = e->get()->as<FunctionCall>();
|
||||
for (auto& arg : c.arguments()) {
|
||||
this->addExpression(cfg, &arg, constantPropagate);
|
||||
}
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kFieldAccess: {
|
||||
this->addExpression(cfg, &e->get()->as<FieldAccess>().base(), constantPropagate);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kIndex: {
|
||||
IndexExpression& indexExpr = e->get()->as<IndexExpression>();
|
||||
|
||||
this->addExpression(cfg, &indexExpr.base(), constantPropagate);
|
||||
this->addExpression(cfg, &indexExpr.index(), constantPropagate);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kPrefix: {
|
||||
PrefixExpression& p = e->get()->as<PrefixExpression>();
|
||||
this->addExpression(cfg, &p.operand(),
|
||||
constantPropagate &&
|
||||
p.getOperator().kind() != Token::Kind::TK_PLUSPLUS &&
|
||||
p.getOperator().kind() != Token::Kind::TK_MINUSMINUS);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kPostfix:
|
||||
this->addExpression(cfg, &e->get()->as<PostfixExpression>().operand(), false);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
case Expression::Kind::kSwizzle:
|
||||
this->addExpression(cfg, &e->get()->as<Swizzle>().base(), constantPropagate);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
case Expression::Kind::kBoolLiteral: // fall through
|
||||
case Expression::Kind::kFloatLiteral: // fall through
|
||||
case Expression::Kind::kIntLiteral: // fall through
|
||||
case Expression::Kind::kSetting: // fall through
|
||||
case Expression::Kind::kVariableReference:
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
break;
|
||||
case Expression::Kind::kTernary: {
|
||||
TernaryExpression& t = e->get()->as<TernaryExpression>();
|
||||
this->addExpression(cfg, &t.test(), constantPropagate);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeExpression(e, constantPropagate));
|
||||
BlockId start = cfg.fCurrent;
|
||||
cfg.newBlock();
|
||||
this->addExpression(cfg, &t.ifTrue(), constantPropagate);
|
||||
BlockId next = cfg.newBlock();
|
||||
cfg.fCurrent = start;
|
||||
cfg.newBlock();
|
||||
this->addExpression(cfg, &t.ifFalse(), constantPropagate);
|
||||
cfg.addExit(cfg.fCurrent, next);
|
||||
cfg.fCurrent = next;
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kExternalFunctionReference: // fall through
|
||||
case Expression::Kind::kFunctionReference: // fall through
|
||||
case Expression::Kind::kTypeReference: // fall through
|
||||
case Expression::Kind::kDefined:
|
||||
SkASSERT(false);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// adds expressions that are evaluated as part of resolving an lvalue
|
||||
void CFGGenerator::addLValue(CFG& cfg, std::unique_ptr<Expression>* e) {
|
||||
switch ((*e)->kind()) {
|
||||
case Expression::Kind::kFieldAccess:
|
||||
this->addLValue(cfg, &e->get()->as<FieldAccess>().base());
|
||||
break;
|
||||
case Expression::Kind::kIndex: {
|
||||
IndexExpression& indexExpr = e->get()->as<IndexExpression>();
|
||||
this->addLValue(cfg, &indexExpr.base());
|
||||
this->addExpression(cfg, &indexExpr.index(), /*constantPropagate=*/true);
|
||||
break;
|
||||
}
|
||||
case Expression::Kind::kSwizzle:
|
||||
this->addLValue(cfg, &e->get()->as<Swizzle>().base());
|
||||
break;
|
||||
case Expression::Kind::kVariableReference:
|
||||
break;
|
||||
case Expression::Kind::kTernary: {
|
||||
TernaryExpression& ternary = e->get()->as<TernaryExpression>();
|
||||
this->addExpression(cfg, &ternary.test(), /*constantPropagate=*/true);
|
||||
// Technically we will of course only evaluate one or the other, but if the test turns
|
||||
// out to be constant, the ternary will get collapsed down to just one branch anyway. So
|
||||
// it should be ok to pretend that we always evaluate both branches here.
|
||||
this->addLValue(cfg, &ternary.ifTrue());
|
||||
this->addLValue(cfg, &ternary.ifFalse());
|
||||
break;
|
||||
}
|
||||
default:
|
||||
// not an lvalue, can't happen
|
||||
SkASSERT(false);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void CFGGenerator::addStatement(CFG& cfg, std::unique_ptr<Statement>* s) {
|
||||
switch ((*s)->kind()) {
|
||||
case Statement::Kind::kBlock: {
|
||||
Block& block = (*s)->as<Block>();
|
||||
for (std::unique_ptr<Statement>& child : block.children()) {
|
||||
addStatement(cfg, &child);
|
||||
}
|
||||
break;
|
||||
}
|
||||
case Statement::Kind::kIf: {
|
||||
IfStatement& ifs = (*s)->as<IfStatement>();
|
||||
this->addExpression(cfg, &ifs.test(), /*constantPropagate=*/true);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeStatement(s));
|
||||
BlockId start = cfg.fCurrent;
|
||||
cfg.newBlock();
|
||||
this->addStatement(cfg, &ifs.ifTrue());
|
||||
BlockId next = cfg.newBlock();
|
||||
if (ifs.ifFalse()) {
|
||||
cfg.fCurrent = start;
|
||||
cfg.newBlock();
|
||||
this->addStatement(cfg, &ifs.ifFalse());
|
||||
cfg.addExit(cfg.fCurrent, next);
|
||||
cfg.fCurrent = next;
|
||||
} else {
|
||||
cfg.addExit(start, next);
|
||||
}
|
||||
break;
|
||||
}
|
||||
case Statement::Kind::kExpression: {
|
||||
this->addExpression(cfg, &(*s)->as<ExpressionStatement>().expression(),
|
||||
/*constantPropagate=*/true);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeStatement(s));
|
||||
break;
|
||||
}
|
||||
case Statement::Kind::kVarDeclaration: {
|
||||
VarDeclaration& vd = (*s)->as<VarDeclaration>();
|
||||
if (vd.value()) {
|
||||
this->addExpression(cfg, &vd.value(), /*constantPropagate=*/true);
|
||||
}
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeStatement(s));
|
||||
break;
|
||||
}
|
||||
case Statement::Kind::kDiscard:
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeStatement(s));
|
||||
cfg.fCurrent = cfg.newIsolatedBlock();
|
||||
break;
|
||||
case Statement::Kind::kReturn: {
|
||||
ReturnStatement& r = (*s)->as<ReturnStatement>();
|
||||
if (r.expression()) {
|
||||
this->addExpression(cfg, &r.expression(), /*constantPropagate=*/true);
|
||||
}
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeStatement(s));
|
||||
cfg.fCurrent = cfg.newIsolatedBlock();
|
||||
break;
|
||||
}
|
||||
case Statement::Kind::kBreak:
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeStatement(s));
|
||||
cfg.addExit(cfg.fCurrent, fLoopExits.top());
|
||||
cfg.fCurrent = cfg.newIsolatedBlock();
|
||||
break;
|
||||
case Statement::Kind::kContinue:
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeStatement(s));
|
||||
cfg.addExit(cfg.fCurrent, fLoopContinues.top());
|
||||
cfg.fCurrent = cfg.newIsolatedBlock();
|
||||
break;
|
||||
case Statement::Kind::kDo: {
|
||||
DoStatement& d = (*s)->as<DoStatement>();
|
||||
BlockId loopStart = cfg.newBlock();
|
||||
fLoopContinues.push(loopStart);
|
||||
BlockId loopExit = cfg.newIsolatedBlock();
|
||||
fLoopExits.push(loopExit);
|
||||
this->addStatement(cfg, &d.statement());
|
||||
this->addExpression(cfg, &d.test(), /*constantPropagate=*/true);
|
||||
cfg.addExit(cfg.fCurrent, loopExit);
|
||||
cfg.addExit(cfg.fCurrent, loopStart);
|
||||
fLoopContinues.pop();
|
||||
fLoopExits.pop();
|
||||
cfg.fCurrent = loopExit;
|
||||
break;
|
||||
}
|
||||
case Statement::Kind::kFor: {
|
||||
ForStatement& f = (*s)->as<ForStatement>();
|
||||
if (f.initializer()) {
|
||||
this->addStatement(cfg, &f.initializer());
|
||||
}
|
||||
BlockId loopStart = cfg.newBlock();
|
||||
BlockId next = cfg.newIsolatedBlock();
|
||||
fLoopContinues.push(next);
|
||||
BlockId loopExit = cfg.newIsolatedBlock();
|
||||
fLoopExits.push(loopExit);
|
||||
if (f.test()) {
|
||||
this->addExpression(cfg, &f.test(), /*constantPropagate=*/true);
|
||||
// this isn't quite right; we should have an exit from here to the loop exit, and
|
||||
// remove the exit from the loop body to the loop exit. Structuring it like this
|
||||
// forces the optimizer to believe that the loop body is always executed at least
|
||||
// once. While not strictly correct, this avoids incorrect "variable not assigned"
|
||||
// errors on variables which are assigned within the loop. The correct solution to
|
||||
// this is to analyze the loop to see whether or not at least one iteration is
|
||||
// guaranteed to happen, but for the time being we take the easy way out.
|
||||
}
|
||||
cfg.newBlock();
|
||||
this->addStatement(cfg, &f.statement());
|
||||
cfg.addExit(cfg.fCurrent, next);
|
||||
cfg.fCurrent = next;
|
||||
if (f.next()) {
|
||||
this->addExpression(cfg, &f.next(), /*constantPropagate=*/true);
|
||||
}
|
||||
// The increment expression of a for loop is allowed to be unreachable, because GLSL
|
||||
// ES2 requires us to provide an increment expression for our for-loops whether or not
|
||||
// it can be reached. Reporting it as "unreachable" isn't helpful if the alternative
|
||||
// is an invalid program.
|
||||
cfg.currentBlock().fAllowUnreachable = true;
|
||||
cfg.addExit(cfg.fCurrent, loopStart);
|
||||
cfg.addExit(cfg.fCurrent, loopExit);
|
||||
fLoopContinues.pop();
|
||||
fLoopExits.pop();
|
||||
cfg.fCurrent = loopExit;
|
||||
break;
|
||||
}
|
||||
case Statement::Kind::kSwitch: {
|
||||
SwitchStatement& ss = (*s)->as<SwitchStatement>();
|
||||
this->addExpression(cfg, &ss.value(), /*constantPropagate=*/true);
|
||||
cfg.currentBlock().fNodes.push_back(BasicBlock::MakeStatement(s));
|
||||
BlockId start = cfg.fCurrent;
|
||||
BlockId switchExit = cfg.newIsolatedBlock();
|
||||
fLoopExits.push(switchExit);
|
||||
for (std::unique_ptr<Statement>& stmt : ss.cases()) {
|
||||
SwitchCase& c = stmt->as<SwitchCase>();
|
||||
cfg.newBlock();
|
||||
cfg.addExit(start, cfg.fCurrent);
|
||||
if (c.value()) {
|
||||
// technically this should go in the start block, but it doesn't actually matter
|
||||
// because it must be constant. Not worth running two loops for.
|
||||
this->addExpression(cfg, &c.value(), /*constantPropagate=*/true);
|
||||
}
|
||||
this->addStatement(cfg, &c.statement());
|
||||
}
|
||||
cfg.addExit(cfg.fCurrent, switchExit);
|
||||
// note that unlike GLSL, our grammar requires the default case to be last
|
||||
if (ss.cases().empty() || ss.cases().back()->as<SwitchCase>().value()) {
|
||||
// switch does not have a default clause, mark that it can skip straight to the end
|
||||
cfg.addExit(start, switchExit);
|
||||
}
|
||||
fLoopExits.pop();
|
||||
cfg.fCurrent = switchExit;
|
||||
break;
|
||||
}
|
||||
case Statement::Kind::kInlineMarker:
|
||||
case Statement::Kind::kNop:
|
||||
break;
|
||||
default:
|
||||
SkDEBUGFAILF("unsupported statement: %s\n", (*s)->description().c_str());
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
CFG CFGGenerator::getCFG(FunctionDefinition& f) {
|
||||
CFG result;
|
||||
result.fStart = result.newBlock();
|
||||
result.fCurrent = result.fStart;
|
||||
// The starting block is "reached" implicitly, even if nothing points to it.
|
||||
result.currentBlock().fAllowUnreachable = true;
|
||||
this->addStatement(result, &f.body());
|
||||
result.newBlock();
|
||||
result.fExit = result.fCurrent;
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace SkSL
|
Loading…
Reference in New Issue
Block a user