Add a parameter to ConstantFoldTerminator() that callers can use to ask it to also clean up the condition of any conditional terminator it folds to be unconditional, if that turns the condition into dead code. This just means it calls RecursivelyDeleteTriviallyDeadInstructions() in strategic spots. It defaults to the old behavior.

I also changed -simplifycfg, -jump-threading and -codegenprepare to use this to produce slightly better code without any extra cleanup passes (AFAICT this was the only place in -simplifycfg where now-dead conditions of replaced terminators weren't being cleaned up). The only other user of this function is -sccp, but I didn't read that thoroughly enough to figure out whether it might be holding pointers to instructions that could be deleted by this.



git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@131855 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Frits van Bommel 2011-05-22 16:24:18 +00:00
parent 8480aba5ec
commit 5649ba70fb
7 changed files with 78 additions and 12 deletions

View File

@ -43,8 +43,10 @@ template<typename T> class SmallVectorImpl;
/// constant value, convert it into an unconditional branch to the constant /// constant value, convert it into an unconditional branch to the constant
/// destination. This is a nontrivial operation because the successors of this /// destination. This is a nontrivial operation because the successors of this
/// basic block must have their PHI nodes updated. /// basic block must have their PHI nodes updated.
/// /// Also calls RecursivelyDeleteTriviallyDeadInstructions() on any branch/switch
bool ConstantFoldTerminator(BasicBlock *BB); /// conditions and indirectbr addresses this might make dead if
/// DeleteDeadConditions is true.
bool ConstantFoldTerminator(BasicBlock *BB, bool DeleteDeadConditions = false);
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
// Local dead code elimination. // Local dead code elimination.

View File

@ -147,7 +147,7 @@ bool CodeGenPrepare::runOnFunction(Function &F) {
if (!DisableBranchOpts) { if (!DisableBranchOpts) {
MadeChange = false; MadeChange = false;
for (Function::iterator BB = F.begin(), E = F.end(); BB != E; ++BB) for (Function::iterator BB = F.begin(), E = F.end(); BB != E; ++BB)
MadeChange |= ConstantFoldTerminator(BB); MadeChange |= ConstantFoldTerminator(BB, true);
if (MadeChange) if (MadeChange)
ModifiedDT = true; ModifiedDT = true;

View File

@ -706,7 +706,7 @@ bool JumpThreading::ProcessBlock(BasicBlock *BB) {
DEBUG(dbgs() << " In block '" << BB->getName() DEBUG(dbgs() << " In block '" << BB->getName()
<< "' folding terminator: " << *BB->getTerminator() << '\n'); << "' folding terminator: " << *BB->getTerminator() << '\n');
++NumFolds; ++NumFolds;
ConstantFoldTerminator(BB); ConstantFoldTerminator(BB, true);
return true; return true;
} }

View File

@ -163,7 +163,7 @@ static bool MarkAliveBlocks(BasicBlock *BB,
Changed = true; Changed = true;
} }
Changed |= ConstantFoldTerminator(BB); Changed |= ConstantFoldTerminator(BB, true);
for (succ_iterator SI = succ_begin(BB), SE = succ_end(BB); SI != SE; ++SI) for (succ_iterator SI = succ_begin(BB), SE = succ_end(BB); SI != SE; ++SI)
Worklist.push_back(*SI); Worklist.push_back(*SI);
} while (!Worklist.empty()); } while (!Worklist.empty());

View File

@ -44,11 +44,14 @@ using namespace llvm;
// Local constant propagation. // Local constant propagation.
// //
// ConstantFoldTerminator - If a terminator instruction is predicated on a /// ConstantFoldTerminator - If a terminator instruction is predicated on a
// constant value, convert it into an unconditional branch to the constant /// constant value, convert it into an unconditional branch to the constant
// destination. /// destination. This is a nontrivial operation because the successors of this
// /// basic block must have their PHI nodes updated.
bool llvm::ConstantFoldTerminator(BasicBlock *BB) { /// Also calls RecursivelyDeleteTriviallyDeadInstructions() on any branch/switch
/// conditions and indirectbr addresses this might make dead if
/// DeleteDeadConditions is true.
bool llvm::ConstantFoldTerminator(BasicBlock *BB, bool DeleteDeadConditions) {
TerminatorInst *T = BB->getTerminator(); TerminatorInst *T = BB->getTerminator();
IRBuilder<> Builder(T); IRBuilder<> Builder(T);
@ -89,7 +92,10 @@ bool llvm::ConstantFoldTerminator(BasicBlock *BB) {
// Replace the conditional branch with an unconditional one. // Replace the conditional branch with an unconditional one.
Builder.CreateBr(Dest1); Builder.CreateBr(Dest1);
Value *Cond = BI->getCondition();
BI->eraseFromParent(); BI->eraseFromParent();
if (DeleteDeadConditions)
RecursivelyDeleteTriviallyDeadInstructions(Cond);
return true; return true;
} }
return false; return false;
@ -152,7 +158,10 @@ bool llvm::ConstantFoldTerminator(BasicBlock *BB) {
} }
// Delete the old switch. // Delete the old switch.
BB->getInstList().erase(SI); Value *Cond = SI->getCondition();
SI->eraseFromParent();
if (DeleteDeadConditions)
RecursivelyDeleteTriviallyDeadInstructions(Cond);
return true; return true;
} }
@ -186,7 +195,10 @@ bool llvm::ConstantFoldTerminator(BasicBlock *BB) {
else else
IBI->getDestination(i)->removePredecessor(IBI->getParent()); IBI->getDestination(i)->removePredecessor(IBI->getParent());
} }
Value *Address = IBI->getAddress();
IBI->eraseFromParent(); IBI->eraseFromParent();
if (DeleteDeadConditions)
RecursivelyDeleteTriviallyDeadInstructions(Address);
// If we didn't find our destination in the IBI successor list, then we // If we didn't find our destination in the IBI successor list, then we
// have undefined behavior. Replace the unconditional branch with an // have undefined behavior. Replace the unconditional branch with an

View File

@ -2651,7 +2651,7 @@ bool SimplifyCFGOpt::run(BasicBlock *BB) {
// Check to see if we can constant propagate this terminator instruction // Check to see if we can constant propagate this terminator instruction
// away... // away...
Changed |= ConstantFoldTerminator(BB); Changed |= ConstantFoldTerminator(BB, true);
// Check for and eliminate duplicate PHI nodes in this block. // Check for and eliminate duplicate PHI nodes in this block.
Changed |= EliminateDuplicatePHINodes(BB); Changed |= EliminateDuplicatePHINodes(BB);

View File

@ -0,0 +1,52 @@
; RUN: opt -S <%s -simplifycfg | FileCheck %s
define void @test_br(i32 %x) {
entry:
; CHECK: @test_br
; CHECK-NEXT: entry:
; CHECK-NEXT: ret void
%cmp = icmp eq i32 %x, 10
br i1 %cmp, label %if.then, label %if.then
if.then: ; preds = %entry
br label %if.end
if.end: ; preds = %if.else, %if.then
ret void
}
define void @test_switch(i32 %x) nounwind {
entry:
; CHECK: @test_switch
; CHECK-NEXT: entry:
; CHECK-NEXT: ret void
%rem = srem i32 %x, 3
switch i32 %rem, label %sw.bb [
i32 1, label %sw.bb
i32 10, label %sw.bb
]
sw.bb: ; preds = %sw.default, %entry, %entry
br label %sw.epilog
sw.epilog: ; preds = %sw.bb
ret void
}
define void @test_indirectbr(i32 %x) {
entry:
; CHECK: @test_indirectbr
; CHECK-NEXT: entry:
; Ideally this should now check:
; CHK-NEXT: ret void
; But that doesn't happen yet. Instead:
; CHECK-NEXT: br label %L1
%label = bitcast i8* blockaddress(@test_indirectbr, %L1) to i8*
indirectbr i8* %label, [label %L1, label %L2]
L1: ; preds = %entry
ret void
L2: ; preds = %entry
ret void
}