OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 |
OLD | NEW |