diff options
author | Chris Lattner <sabre@nondot.org> | 2012-02-24 19:01:58 +0000 |
---|---|---|
committer | Chris Lattner <sabre@nondot.org> | 2012-02-24 19:01:58 +0000 |
commit | 009e2650d69fe93bda1864340c5f000e56c52fb8 (patch) | |
tree | 0da13e16342e5c497332c81991972cd21517be33 /lib | |
parent | 1a2d061ec08b86ba91d7009b6ffcf08d5bac3f42 (diff) | |
download | external_llvm-009e2650d69fe93bda1864340c5f000e56c52fb8.zip external_llvm-009e2650d69fe93bda1864340c5f000e56c52fb8.tar.gz external_llvm-009e2650d69fe93bda1864340c5f000e56c52fb8.tar.bz2 |
fix PR12075, a regression in a recent transform I added. In unreachable code, gep chains can be infinite. Just like "stripPointerCasts", use a set to keep track of visited instructions so we don't recurse infinitely.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@151383 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib')
-rw-r--r-- | lib/Analysis/InstructionSimplify.cpp | 22 |
1 files changed, 18 insertions, 4 deletions
diff --git a/lib/Analysis/InstructionSimplify.cpp b/lib/Analysis/InstructionSimplify.cpp index 9b0ce11..37253f7 100644 --- a/lib/Analysis/InstructionSimplify.cpp +++ b/lib/Analysis/InstructionSimplify.cpp @@ -1522,13 +1522,27 @@ static Value *ExtractEquivalentCondition(Value *V, CmpInst::Predicate Pred, /// stripPointerAdjustments - This is like Value::stripPointerCasts, but also /// removes inbounds gep operations, regardless of their indices. +static Value *stripPointerAdjustmentsImpl(Value *V, + SmallPtrSet<GEPOperator*, 8> &VisitedGEPs) { + GEPOperator *GEP = dyn_cast<GEPOperator>(V); + if (GEP == 0 || !GEP->isInBounds()) + return V; + + // If we've already seen this GEP, we will end up infinitely looping. This + // can happen in unreachable code. + if (!VisitedGEPs.insert(GEP)) + return V; + + return stripPointerAdjustmentsImpl(GEP->getOperand(0)->stripPointerCasts(), + VisitedGEPs); +} + static Value *stripPointerAdjustments(Value *V) { - if (GEPOperator *GEP = dyn_cast<GEPOperator>(V)) - if (GEP->isInBounds()) - return stripPointerAdjustments(GEP->getOperand(0)->stripPointerCasts()); - return V; + SmallPtrSet<GEPOperator*, 8> VisitedGEPs; + return stripPointerAdjustmentsImpl(V, VisitedGEPs); } + /// SimplifyICmpInst - Given operands for an ICmpInst, see if we can /// fold the result. If not, this returns null. static Value *SimplifyICmpInst(unsigned Predicate, Value *LHS, Value *RHS, |