mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-15 04:30:12 +00:00
Optimize conditional branch on i1 phis with non-constant inputs.
This turns: eq: %3 = icmp eq i32 %1, %2 br label %join ne: %4 = icmp ne i32 %1, %2 br label %join join: %5 = phi i1 [%3, %eq], [%4, %ne] br i1 %5, label %yes, label %no => eq: %3 = icmp eq i32 %1, %2 br i1 %3, label %yes, label %no ne: %4 = icmp ne i32 %1, %2 br i1 %4, label %yes, label %no git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@69102 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
5b9c31841f
commit
df2f1189a3
@ -51,6 +51,7 @@ namespace {
|
|||||||
void SimplifyPredecessors(BranchInst *BI);
|
void SimplifyPredecessors(BranchInst *BI);
|
||||||
void SimplifyPredecessors(SwitchInst *SI);
|
void SimplifyPredecessors(SwitchInst *SI);
|
||||||
void RevectorBlockTo(BasicBlock *FromBB, BasicBlock *ToBB);
|
void RevectorBlockTo(BasicBlock *FromBB, BasicBlock *ToBB);
|
||||||
|
bool RevectorBlockTo(BasicBlock *FromBB, Value *Cond, BranchInst *BI);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -160,20 +161,19 @@ void CondProp::SimplifyPredecessors(BranchInst *BI) {
|
|||||||
// Ok, we have this really simple case, walk the PHI operands, looking for
|
// Ok, we have this really simple case, walk the PHI operands, looking for
|
||||||
// constants. Walk from the end to remove operands from the end when
|
// constants. Walk from the end to remove operands from the end when
|
||||||
// possible, and to avoid invalidating "i".
|
// possible, and to avoid invalidating "i".
|
||||||
for (unsigned i = PN->getNumIncomingValues(); i != 0; --i)
|
for (unsigned i = PN->getNumIncomingValues(); i != 0; --i) {
|
||||||
if (ConstantInt *CB = dyn_cast<ConstantInt>(PN->getIncomingValue(i-1))) {
|
Value *InVal = PN->getIncomingValue(i-1);
|
||||||
// If we have a constant, forward the edge from its current to its
|
if (!RevectorBlockTo(PN->getIncomingBlock(i-1), InVal, BI))
|
||||||
// ultimate destination.
|
continue;
|
||||||
RevectorBlockTo(PN->getIncomingBlock(i-1),
|
|
||||||
BI->getSuccessor(CB->isZero()));
|
|
||||||
++NumBrThread;
|
|
||||||
|
|
||||||
// If there were two predecessors before this simplification, or if the
|
++NumBrThread;
|
||||||
// PHI node contained all the same value except for the one we just
|
|
||||||
// substituted, the PHI node may be deleted. Don't iterate through it the
|
// If there were two predecessors before this simplification, or if the
|
||||||
// last time.
|
// PHI node contained all the same value except for the one we just
|
||||||
if (BI->getCondition() != PN) return;
|
// substituted, the PHI node may be deleted. Don't iterate through it the
|
||||||
}
|
// last time.
|
||||||
|
if (BI->getCondition() != PN) return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SimplifyPredecessors(switch) - We know that SI is switch based on a PHI node
|
// SimplifyPredecessors(switch) - We know that SI is switch based on a PHI node
|
||||||
@ -242,3 +242,44 @@ void CondProp::RevectorBlockTo(BasicBlock *FromBB, BasicBlock *ToBB) {
|
|||||||
|
|
||||||
MadeChange = true;
|
MadeChange = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool CondProp::RevectorBlockTo(BasicBlock *FromBB, Value *Cond, BranchInst *BI){
|
||||||
|
BranchInst *FromBr = cast<BranchInst>(FromBB->getTerminator());
|
||||||
|
if (!FromBr->isUnconditional())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Get the old block we are threading through.
|
||||||
|
BasicBlock *OldSucc = FromBr->getSuccessor(0);
|
||||||
|
|
||||||
|
// If the condition is a constant, simply revector the unconditional branch at
|
||||||
|
// the end of FromBB to one of the successors of its current successor.
|
||||||
|
if (ConstantInt *CB = dyn_cast<ConstantInt>(Cond)) {
|
||||||
|
BasicBlock *ToBB = BI->getSuccessor(CB->isZero());
|
||||||
|
|
||||||
|
// OldSucc had multiple successors. If ToBB has multiple predecessors, then
|
||||||
|
// the edge between them would be critical, which we already took care of.
|
||||||
|
// If ToBB has single operand PHI node then take care of it here.
|
||||||
|
FoldSingleEntryPHINodes(ToBB);
|
||||||
|
|
||||||
|
// Update PHI nodes in OldSucc to know that FromBB no longer branches to it.
|
||||||
|
OldSucc->removePredecessor(FromBB);
|
||||||
|
|
||||||
|
// Change FromBr to branch to the new destination.
|
||||||
|
FromBr->setSuccessor(0, ToBB);
|
||||||
|
} else {
|
||||||
|
// Insert the new conditional branch.
|
||||||
|
BranchInst::Create(BI->getSuccessor(0), BI->getSuccessor(1), Cond, FromBr);
|
||||||
|
|
||||||
|
FoldSingleEntryPHINodes(BI->getSuccessor(0));
|
||||||
|
FoldSingleEntryPHINodes(BI->getSuccessor(1));
|
||||||
|
|
||||||
|
// Update PHI nodes in OldSucc to know that FromBB no longer branches to it.
|
||||||
|
OldSucc->removePredecessor(FromBB);
|
||||||
|
|
||||||
|
// Delete the old branch.
|
||||||
|
FromBr->eraseFromParent();
|
||||||
|
}
|
||||||
|
|
||||||
|
MadeChange = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
26
test/Transforms/CondProp/phisimplify3.ll
Normal file
26
test/Transforms/CondProp/phisimplify3.ll
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
; RUN: llvm-as < %s | opt -condprop | llvm-dis | not grep phi
|
||||||
|
|
||||||
|
define i32 @foo(i1, i32, i32) {
|
||||||
|
prologue:
|
||||||
|
br i1 %0, label %eq, label %ne
|
||||||
|
|
||||||
|
eq: ; preds = %prologue
|
||||||
|
store i32 0, i32* inttoptr (i32 10000 to i32*)
|
||||||
|
%3 = icmp eq i32 %1, %2 ; <i1> [#uses=1]
|
||||||
|
br label %join
|
||||||
|
|
||||||
|
ne: ; preds = %prologue
|
||||||
|
%4 = icmp ne i32 %1, %2 ; <i1> [#uses=1]
|
||||||
|
br label %join
|
||||||
|
|
||||||
|
join: ; preds = %ne, %eq
|
||||||
|
%5 = phi i1 [ %3, %eq ], [ %4, %ne ] ; <i1> [#uses=1]
|
||||||
|
br i1 %5, label %yes, label %no
|
||||||
|
|
||||||
|
yes: ; preds = %join
|
||||||
|
store i32 0, i32* inttoptr (i32 20000 to i32*)
|
||||||
|
ret i32 5
|
||||||
|
|
||||||
|
no: ; preds = %join
|
||||||
|
ret i32 20
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user