Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(29)

Unified Diff: src/IceCfgNode.cpp

Issue 2069923004: Short Circuit Evaluation (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: Created 4 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« src/IceCfg.cpp ('K') | « src/IceCfgNode.h ('k') | src/IceClFlags.def » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: src/IceCfgNode.cpp
diff --git a/src/IceCfgNode.cpp b/src/IceCfgNode.cpp
index 78e3d18f414a94f1baa6f68923eb977604f30f42..7c59827f801945e4796e137ba39ec1358ea63a5b 100644
--- a/src/IceCfgNode.cpp
+++ b/src/IceCfgNode.cpp
@@ -51,6 +51,22 @@ void CfgNode::appendInst(Inst *Instr) {
}
}
+void CfgNode::replaceInEdge(CfgNode *Old, CfgNode *New) {
+ for (SizeT i = 0; i < InEdges.size(); ++i) {
+ if (InEdges[i] == Old) {
+ InEdges[i] = New;
+ }
+ }
+ for (auto &Inst : getPhis()) {
+ InstPhi &Phi = llvm::cast<InstPhi>(Inst);
+ for (SizeT i = 0; i < Phi.getSrcSize(); ++i) {
+ if (Phi.getLabel(i) == Old) {
+ Phi.setLabel(i, New);
+ }
+ }
+ }
+}
+
namespace {
template <typename List> void removeDeletedAndRenumber(List *L, Cfg *Func) {
const bool DoDelete =
@@ -1471,5 +1487,156 @@ void CfgNode::profileExecutionCount(VariableDeclaration *Var) {
Instr->addArg(OrderAcquireRelease);
Insts.push_front(Instr);
}
+void CfgNode::removeInEdge(CfgNode *In) {
John 2016/06/16 13:41:10 add empty line before function definition?
manasijm 2016/06/20 20:45:36 Done.
+ InEdges.erase(std::find(InEdges.begin(), InEdges.end(), In));
+}
+CfgNode *CfgNode::shortCircuit() {
John 2016/06/16 13:41:11 add empty line before function definition?
manasijm 2016/06/20 20:45:36 Done.
+ auto *Func = getCfg();
+ auto *Last = &getInsts().back();
+ Variable *Goal = nullptr;
John 2016/06/16 13:41:10 How about renaming this variable (Goal) to Conditi
manasijm 2016/06/20 20:45:36 Done.
+ InstBr *Br = nullptr;
+ if ((Br = llvm::dyn_cast<InstBr>(Last))) {
+ if (!Br->isUnconditional()) {
+ Goal = llvm::dyn_cast<Variable>(Br->getCondition());
+ }
+ }
+ if (Goal == nullptr)
+ return nullptr;
+
+ auto *JumpOnTrue = Br->getTargetTrue();
+ auto *JumpOnFalse = Br->getTargetFalse();
+
+ bool FoundOr = false;
+ bool FoundAnd = false;
+
+ CfgUnorderedSet<Variable *> Operands;
John 2016/06/16 13:41:10 Instead of creating the unordered set, save a poin
manasijm 2016/06/20 20:45:36 Done.
+ InstArithmetic *TopLevelBoolOp = nullptr;
+
+ for (auto &Inst : reverse_range(getInsts())) {
+ if (Inst.isDeleted())
+ continue;
+ if (Inst.getDest() == Goal) {
+ if (auto *Arith = llvm::dyn_cast<InstArithmetic>(&Inst)) {
+
+ FoundOr = (Arith->getOp() == InstArithmetic::OpKind::Or);
+ FoundAnd = (Arith->getOp() == InstArithmetic::OpKind::And);
+
+ if (FoundOr || FoundAnd) {
+ for (SizeT i = 0; i < Arith->getSrcSize(); ++i) {
+ if (auto *Var = llvm::dyn_cast<Variable>(Arith->getSrc(i))) {
+ Operands.insert(Var);
+ }
+ }
+ TopLevelBoolOp = Arith;
+ break;
+ }
+ }
+ }
+ }
+
+ Inst *FirstOperandDef = nullptr;
+ if ((FoundAnd || FoundOr) && Operands.size() > 1) {
John 2016/06/16 13:41:10 This seems to be optimizing the a = or i32 x, x
+ for (auto &Inst : getInsts()) {
+ if (Operands.find(Inst.getDest()) != Operands.end()) {
+ FirstOperandDef = &Inst;
+ break;
+ }
+ }
+ } else {
+ return nullptr;
+ }
+ if (FirstOperandDef == nullptr) {
John 2016/06/16 13:41:11 this should never happens, by contruction, right?
manasijm 2016/06/20 20:45:36 It could happen for valid code, if the condition o
+ return nullptr;
+ }
+
+ // Check for side effects, might be too conservative. Checking for last use
+ // instead of isSingleBlock might introduce more opportunities.
John 2016/06/16 13:41:11 actually, there are good reasons for this. 1) you
manasijm 2016/06/20 20:45:36 Acknowledged.
+ auto It = Ice::instToIterator(FirstOperandDef);
+ bool hasSideEffects = false;
+ while (It != getInsts().end()) {
+ if (It->isDeleted()) {
+ It++;
+ continue;
+ }
+ if (llvm::isa<InstBr>(It) || llvm::isa<InstRet>(It)) {
+ break;
+ }
+ auto Dest = It->getDest();
John 2016/06/16 13:41:10 auto*
+ if (It->getDest() == nullptr || It->hasSideEffects() ||
+ !Func->getVMetadata()->isSingleBlock(Dest)) {
+ // Relying on short cicuit eval here.
+ // getVMetadata()->isSingleBlock(Dest)
+ // will segfault if It->getDest() == nullptr
+ hasSideEffects = true;
John 2016/06/16 13:41:11 just return nullptr here?
manasijm 2016/06/20 20:45:36 Done.
+ }
+ if (hasSideEffects) {
+ break;
+ }
+ It++;
+ }
+
+ if (hasSideEffects) {
+ return nullptr;
+ }
+
+ auto *NewNode = Func->makeNode();
+ NewNode->setLoopNestDepth(getLoopNestDepth());
+ It = Ice::instToIterator(FirstOperandDef);
+ It++; // Have to split after the def
+
+ NewNode->getInsts().splice(NewNode->getInsts().begin(), getInsts(), It,
+ getInsts().end());
+
+ NewNode->setName(getName().append("_ext_"));
+
+ // Point edges properly
+ NewNode->addInEdge(this);
+ for (auto *Out : getOutEdges()) {
+ NewNode->addOutEdge(Out);
+ Out->addInEdge(NewNode);
+ }
+ removeAllOutEdges();
+ addOutEdge(NewNode);
+
+ //Manage Phi instructions of successors
+ for (auto *Succ : NewNode->getOutEdges()) {
+ for (auto &Inst : Succ->getPhis()) {
+ auto *Phi = llvm::dyn_cast<InstPhi>(&Inst);
+ for (SizeT i = 0; i < Phi->getSrcSize(); ++i) {
+ if (Phi->getLabel(i) == this) {
+ Phi->addArgument(Phi->getSrc(i), NewNode);
+ }
+ }
+ }
+ }
+
+ //Create new Br instruction
+ InstBr *NewInst = nullptr;
+ if (FoundOr) {
+ addOutEdge(JumpOnTrue);
+ JumpOnFalse->removeInEdge(this);
+ NewInst =
+ InstBr::create(Func, FirstOperandDef->getDest(), JumpOnTrue, NewNode);
+ } else if (FoundAnd) {
+ addOutEdge(JumpOnFalse);
+ JumpOnTrue->removeInEdge(this);
+ NewInst =
+ InstBr::create(Func, FirstOperandDef->getDest(), NewNode, JumpOnFalse);
+ }
+ appendInst(NewInst);
+
+ Operand *UnusedOperand = nullptr;
+ assert(TopLevelBoolOp->getSrcSize() == 2);
+ if (TopLevelBoolOp->getSrc(0) == FirstOperandDef->getDest())
+ UnusedOperand = TopLevelBoolOp->getSrc(1);
+ else if (TopLevelBoolOp->getSrc(1) == FirstOperandDef->getDest())
+ UnusedOperand = TopLevelBoolOp->getSrc(0);
+ assert(UnusedOperand);
+
+ Br->replaceSource(0, UnusedOperand); // Index 0 has the condition of the Br
+
+ TopLevelBoolOp->setDeleted();
+ return NewNode;
+}
} // end of namespace Ice
« src/IceCfg.cpp ('K') | « src/IceCfgNode.h ('k') | src/IceClFlags.def » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698