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

Side by Side 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 unified diff | Download patch
OLDNEW
1 //===- subzero/src/IceCfgNode.cpp - Basic block (node) implementation -----===// 1 //===- subzero/src/IceCfgNode.cpp - Basic block (node) implementation -----===//
2 // 2 //
3 // The Subzero Code Generator 3 // The Subzero Code Generator
4 // 4 //
5 // This file is distributed under the University of Illinois Open Source 5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details. 6 // License. See LICENSE.TXT for details.
7 // 7 //
8 //===----------------------------------------------------------------------===// 8 //===----------------------------------------------------------------------===//
9 /// 9 ///
10 /// \file 10 /// \file
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after
44 if (!Insts.empty()) { 44 if (!Insts.empty()) {
45 Func->setError("Phi instruction added to the middle of a block"); 45 Func->setError("Phi instruction added to the middle of a block");
46 return; 46 return;
47 } 47 }
48 Phis.push_back(Phi); 48 Phis.push_back(Phi);
49 } else { 49 } else {
50 Insts.push_back(Instr); 50 Insts.push_back(Instr);
51 } 51 }
52 } 52 }
53 53
54 void CfgNode::replaceInEdge(CfgNode *Old, CfgNode *New) {
55 for (SizeT i = 0; i < InEdges.size(); ++i) {
56 if (InEdges[i] == Old) {
57 InEdges[i] = New;
58 }
59 }
60 for (auto &Inst : getPhis()) {
61 InstPhi &Phi = llvm::cast<InstPhi>(Inst);
62 for (SizeT i = 0; i < Phi.getSrcSize(); ++i) {
63 if (Phi.getLabel(i) == Old) {
64 Phi.setLabel(i, New);
65 }
66 }
67 }
68 }
69
54 namespace { 70 namespace {
55 template <typename List> void removeDeletedAndRenumber(List *L, Cfg *Func) { 71 template <typename List> void removeDeletedAndRenumber(List *L, Cfg *Func) {
56 const bool DoDelete = 72 const bool DoDelete =
57 BuildDefs::minimal() || !getFlags().getKeepDeletedInsts(); 73 BuildDefs::minimal() || !getFlags().getKeepDeletedInsts();
58 auto I = L->begin(), E = L->end(), Next = I; 74 auto I = L->begin(), E = L->end(), Next = I;
59 for (++Next; I != E; I = Next++) { 75 for (++Next; I != E; I = Next++) {
60 if (DoDelete && I->isDeleted()) { 76 if (DoDelete && I->isDeleted()) {
61 L->erase(I); 77 L->erase(I);
62 } else { 78 } else {
63 I->renumber(Func); 79 I->renumber(Func);
(...skipping 1400 matching lines...) Expand 10 before | Expand all | Expand 10 after
1464 Ctx->getConstantInt32(Intrinsics::MemoryOrderAcquireRelease); 1480 Ctx->getConstantInt32(Intrinsics::MemoryOrderAcquireRelease);
1465 1481
1466 auto *Instr = InstIntrinsicCall::create( 1482 auto *Instr = InstIntrinsicCall::create(
1467 Func, 5, Func->makeVariable(IceType_i64), RMWI64Name, Info->Info); 1483 Func, 5, Func->makeVariable(IceType_i64), RMWI64Name, Info->Info);
1468 Instr->addArg(AtomicRMWOp); 1484 Instr->addArg(AtomicRMWOp);
1469 Instr->addArg(Counter); 1485 Instr->addArg(Counter);
1470 Instr->addArg(One); 1486 Instr->addArg(One);
1471 Instr->addArg(OrderAcquireRelease); 1487 Instr->addArg(OrderAcquireRelease);
1472 Insts.push_front(Instr); 1488 Insts.push_front(Instr);
1473 } 1489 }
1490 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.
1491 InEdges.erase(std::find(InEdges.begin(), InEdges.end(), In));
1492 }
1493 CfgNode *CfgNode::shortCircuit() {
John 2016/06/16 13:41:11 add empty line before function definition?
manasijm 2016/06/20 20:45:36 Done.
1494 auto *Func = getCfg();
1495 auto *Last = &getInsts().back();
1496 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.
1497 InstBr *Br = nullptr;
1498 if ((Br = llvm::dyn_cast<InstBr>(Last))) {
1499 if (!Br->isUnconditional()) {
1500 Goal = llvm::dyn_cast<Variable>(Br->getCondition());
1501 }
1502 }
1503 if (Goal == nullptr)
1504 return nullptr;
1505
1506 auto *JumpOnTrue = Br->getTargetTrue();
1507 auto *JumpOnFalse = Br->getTargetFalse();
1508
1509 bool FoundOr = false;
1510 bool FoundAnd = false;
1511
1512 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.
1513 InstArithmetic *TopLevelBoolOp = nullptr;
1514
1515 for (auto &Inst : reverse_range(getInsts())) {
1516 if (Inst.isDeleted())
1517 continue;
1518 if (Inst.getDest() == Goal) {
1519 if (auto *Arith = llvm::dyn_cast<InstArithmetic>(&Inst)) {
1520
1521 FoundOr = (Arith->getOp() == InstArithmetic::OpKind::Or);
1522 FoundAnd = (Arith->getOp() == InstArithmetic::OpKind::And);
1523
1524 if (FoundOr || FoundAnd) {
1525 for (SizeT i = 0; i < Arith->getSrcSize(); ++i) {
1526 if (auto *Var = llvm::dyn_cast<Variable>(Arith->getSrc(i))) {
1527 Operands.insert(Var);
1528 }
1529 }
1530 TopLevelBoolOp = Arith;
1531 break;
1532 }
1533 }
1534 }
1535 }
1536
1537 Inst *FirstOperandDef = nullptr;
1538 if ((FoundAnd || FoundOr) && Operands.size() > 1) {
John 2016/06/16 13:41:10 This seems to be optimizing the a = or i32 x, x
1539 for (auto &Inst : getInsts()) {
1540 if (Operands.find(Inst.getDest()) != Operands.end()) {
1541 FirstOperandDef = &Inst;
1542 break;
1543 }
1544 }
1545 } else {
1546 return nullptr;
1547 }
1548 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
1549 return nullptr;
1550 }
1551
1552 // Check for side effects, might be too conservative. Checking for last use
1553 // 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.
1554 auto It = Ice::instToIterator(FirstOperandDef);
1555 bool hasSideEffects = false;
1556 while (It != getInsts().end()) {
1557 if (It->isDeleted()) {
1558 It++;
1559 continue;
1560 }
1561 if (llvm::isa<InstBr>(It) || llvm::isa<InstRet>(It)) {
1562 break;
1563 }
1564 auto Dest = It->getDest();
John 2016/06/16 13:41:10 auto*
1565 if (It->getDest() == nullptr || It->hasSideEffects() ||
1566 !Func->getVMetadata()->isSingleBlock(Dest)) {
1567 // Relying on short cicuit eval here.
1568 // getVMetadata()->isSingleBlock(Dest)
1569 // will segfault if It->getDest() == nullptr
1570 hasSideEffects = true;
John 2016/06/16 13:41:11 just return nullptr here?
manasijm 2016/06/20 20:45:36 Done.
1571 }
1572 if (hasSideEffects) {
1573 break;
1574 }
1575 It++;
1576 }
1577
1578 if (hasSideEffects) {
1579 return nullptr;
1580 }
1581
1582 auto *NewNode = Func->makeNode();
1583 NewNode->setLoopNestDepth(getLoopNestDepth());
1584 It = Ice::instToIterator(FirstOperandDef);
1585 It++; // Have to split after the def
1586
1587 NewNode->getInsts().splice(NewNode->getInsts().begin(), getInsts(), It,
1588 getInsts().end());
1589
1590 NewNode->setName(getName().append("_ext_"));
1591
1592 // Point edges properly
1593 NewNode->addInEdge(this);
1594 for (auto *Out : getOutEdges()) {
1595 NewNode->addOutEdge(Out);
1596 Out->addInEdge(NewNode);
1597 }
1598 removeAllOutEdges();
1599 addOutEdge(NewNode);
1600
1601 //Manage Phi instructions of successors
1602 for (auto *Succ : NewNode->getOutEdges()) {
1603 for (auto &Inst : Succ->getPhis()) {
1604 auto *Phi = llvm::dyn_cast<InstPhi>(&Inst);
1605 for (SizeT i = 0; i < Phi->getSrcSize(); ++i) {
1606 if (Phi->getLabel(i) == this) {
1607 Phi->addArgument(Phi->getSrc(i), NewNode);
1608 }
1609 }
1610 }
1611 }
1612
1613 //Create new Br instruction
1614 InstBr *NewInst = nullptr;
1615 if (FoundOr) {
1616 addOutEdge(JumpOnTrue);
1617 JumpOnFalse->removeInEdge(this);
1618 NewInst =
1619 InstBr::create(Func, FirstOperandDef->getDest(), JumpOnTrue, NewNode);
1620 } else if (FoundAnd) {
1621 addOutEdge(JumpOnFalse);
1622 JumpOnTrue->removeInEdge(this);
1623 NewInst =
1624 InstBr::create(Func, FirstOperandDef->getDest(), NewNode, JumpOnFalse);
1625 }
1626 appendInst(NewInst);
1627
1628 Operand *UnusedOperand = nullptr;
1629 assert(TopLevelBoolOp->getSrcSize() == 2);
1630 if (TopLevelBoolOp->getSrc(0) == FirstOperandDef->getDest())
1631 UnusedOperand = TopLevelBoolOp->getSrc(1);
1632 else if (TopLevelBoolOp->getSrc(1) == FirstOperandDef->getDest())
1633 UnusedOperand = TopLevelBoolOp->getSrc(0);
1634 assert(UnusedOperand);
1635
1636 Br->replaceSource(0, UnusedOperand); // Index 0 has the condition of the Br
1637
1638 TopLevelBoolOp->setDeleted();
1639 return NewNode;
1640 }
1474 1641
1475 } // end of namespace Ice 1642 } // end of namespace Ice
OLDNEW
« 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