diff options
| author | Chris Lattner <sabre@nondot.org> | 2009-01-19 23:03:13 +0000 |
|---|---|---|
| committer | Chris Lattner <sabre@nondot.org> | 2009-01-19 23:03:13 +0000 |
| commit | ea9f1d3c47a5fa42bee7207e8c588f61bb31b183 (patch) | |
| tree | 07ff839efb5643162e94b2d98757c07b11070e79 /llvm/lib | |
| parent | d21d7226b1db00fa1a20e1d8a1127dce2aa538ec (diff) | |
| download | bcm5719-llvm-ea9f1d3c47a5fa42bee7207e8c588f61bb31b183.tar.gz bcm5719-llvm-ea9f1d3c47a5fa42bee7207e8c588f61bb31b183.zip | |
Fix a problem exposed by PR3354: simplifycfg was making a potentially
trapping instruction be executed unconditionally.
llvm-svn: 62541
Diffstat (limited to 'llvm/lib')
| -rw-r--r-- | llvm/lib/Transforms/Utils/SimplifyCFG.cpp | 15 |
1 files changed, 14 insertions, 1 deletions
diff --git a/llvm/lib/Transforms/Utils/SimplifyCFG.cpp b/llvm/lib/Transforms/Utils/SimplifyCFG.cpp index b0feadffadd..4abc2eb0422 100644 --- a/llvm/lib/Transforms/Utils/SimplifyCFG.cpp +++ b/llvm/lib/Transforms/Utils/SimplifyCFG.cpp @@ -1423,11 +1423,21 @@ static bool FoldBranchToCommonDest(BranchInst *BI) { if ((!isa<CmpInst>(Cond) && !isa<BinaryOperator>(Cond)) || Cond->getParent() != BB || &BB->front() != Cond || !Cond->hasOneUse()) return false; - + // Make sure the instruction after the condition is the cond branch. BasicBlock::iterator CondIt = Cond; ++CondIt; if (&*CondIt != BI) return false; + + // Cond is known to be a compare or binary operator. Check to make sure that + // neither operand is a potentially-trapping constant expression. + if (ConstantExpr *CE = dyn_cast<ConstantExpr>(Cond->getOperand(0))) + if (CE->canTrap()) + return false; + if (ConstantExpr *CE = dyn_cast<ConstantExpr>(Cond->getOperand(1))) + if (CE->canTrap()) + return false; + // Finally, don't infinitely unroll conditional loops. BasicBlock *TrueDest = BI->getSuccessor(0); @@ -1438,6 +1448,7 @@ static bool FoldBranchToCommonDest(BranchInst *BI) { for (pred_iterator PI = pred_begin(BB), E = pred_end(BB); PI != E; ++PI) { BasicBlock *PredBlock = *PI; BranchInst *PBI = dyn_cast<BranchInst>(PredBlock->getTerminator()); + // Check that we have two conditional branches. If there is a PHI node in // the common successor, verify that the same value flows in from both // blocks. @@ -1459,6 +1470,8 @@ static bool FoldBranchToCommonDest(BranchInst *BI) { else continue; + DOUT << "FOLDING BRANCH TO COMMON DEST:\n" << *PBI << *BB; + // If we need to invert the condition in the pred block to match, do so now. if (InvertPredCond) { Value *NewCond = |

