diff options
-rw-r--r-- | lib/Analysis/ScalarEvolution.cpp | 93 | ||||
-rw-r--r-- | test/Analysis/ScalarEvolution/scev-aa.ll | 7 |
2 files changed, 15 insertions, 85 deletions
diff --git a/lib/Analysis/ScalarEvolution.cpp b/lib/Analysis/ScalarEvolution.cpp index 358d591..d541b55 100644 --- a/lib/Analysis/ScalarEvolution.cpp +++ b/lib/Analysis/ScalarEvolution.cpp @@ -2760,38 +2760,6 @@ const SCEV *ScalarEvolution::createNodeForPHI(PHINode *PN) { return getUnknown(PN); } -/// UseFlag - When creating an operator with operands L and R based on an -/// LLVM IR instruction in basic block BB where the instruction has -/// nsw, nuw, or inbounds, test whether the corresponding flag can be -/// set for the resulting SCEV. -static bool -UseFlag(bool Flag, const SCEV *L, const SCEV *R, const Value *Inst) { - // If the flag is not set, don't use it. This is included here to reduce - // clutter in the callers. - if (!Flag) - return false; - - // Determine the block which contains the instruction with the flag. - const Instruction *I = dyn_cast<Instruction>(Inst); - if (!I) - return false; - const BasicBlock *BB = I->getParent(); - - // Handle an easy case: test if exactly one of the operands is an addrec - // and that the instruction is trivially control-equivalent to the addrec's - // loop's header. - if (const SCEVAddRecExpr *AR = dyn_cast<SCEVAddRecExpr>(L)) { - if (!isa<SCEVAddRecExpr>(R) && - AR->getLoop()->getHeader() == BB) - return true; - } else if (const SCEVAddRecExpr *AR = dyn_cast<SCEVAddRecExpr>(R)) { - if (AR->getLoop()->getHeader() == BB) - return true; - } - - return false; -} - /// createNodeForGEP - Expand GEP instructions into add and multiply /// operations. This allows them to be analyzed by regular SCEV code. /// @@ -2800,9 +2768,7 @@ const SCEV *ScalarEvolution::createNodeForGEP(GEPOperator *GEP) { // Don't blindly transfer the inbounds flag from the GEP instruction to the // Add expression, because the Instruction may be guarded by control flow // and the no-overflow bits may not be valid for the expression in any - // context. However, in the special case where the GEP is in the loop header, - // we know it's trivially control-equivalent to any addrecs for that loop. - bool InBounds = GEP->isInBounds(); + // context. const Type *IntPtrTy = getEffectiveSCEVType(GEP->getType()); Value *Base = GEP->getOperand(0); @@ -2821,13 +2787,8 @@ const SCEV *ScalarEvolution::createNodeForGEP(GEPOperator *GEP) { unsigned FieldNo = cast<ConstantInt>(Index)->getZExtValue(); const SCEV *FieldOffset = getOffsetOfExpr(STy, FieldNo); - // Test if the GEP has the inbounds keyword and is control-equivalent - // to the addrec. - bool HasNUW = UseFlag(InBounds, TotalOffset, FieldOffset, GEP); - // Add the field offset to the running total offset. - TotalOffset = getAddExpr(TotalOffset, FieldOffset, - HasNUW, /*HasNSW=*/false); + TotalOffset = getAddExpr(TotalOffset, FieldOffset); } else { // For an array, add the element offset, explicitly scaled. const SCEV *ElementSize = getSizeOfExpr(*GTI); @@ -2835,33 +2796,19 @@ const SCEV *ScalarEvolution::createNodeForGEP(GEPOperator *GEP) { // Getelementptr indices are signed. IndexS = getTruncateOrSignExtend(IndexS, IntPtrTy); - // Test if the GEP has the inbounds keyword and is control-equivalent - // to the addrec. - bool HasNUW = UseFlag(InBounds, IndexS, ElementSize, GEP); - // Multiply the index by the element size to compute the element offset. - const SCEV *LocalOffset = getMulExpr(IndexS, ElementSize, - HasNUW, /*HasNSW=*/false); - - // Test if the GEP has the inbounds keyword and is control-equivalent - // to the addrec. - HasNUW = UseFlag(InBounds, TotalOffset, LocalOffset, GEP); + const SCEV *LocalOffset = getMulExpr(IndexS, ElementSize); // Add the element offset to the running total offset. - TotalOffset = getAddExpr(TotalOffset, LocalOffset, - HasNUW, /*HasNSW=*/false); + TotalOffset = getAddExpr(TotalOffset, LocalOffset); } } // Get the SCEV for the GEP base. const SCEV *BaseS = getSCEV(Base); - // Test if the GEP has the inbounds keyword and is control-equivalent - // to the addrec. - bool HasNUW = UseFlag(InBounds, BaseS, TotalOffset, GEP); - // Add the total offset from all the GEP indices to the base. - return getAddExpr(BaseS, TotalOffset, HasNUW, /*HasNSW=*/false); + return getAddExpr(BaseS, TotalOffset); } /// GetMinTrailingZeros - Determine the minimum number of zero bits that S is @@ -3253,30 +3200,12 @@ const SCEV *ScalarEvolution::createSCEV(Value *V) { Operator *U = cast<Operator>(V); switch (Opcode) { - case Instruction::Add: { - const SCEV *LHS = getSCEV(U->getOperand(0)); - const SCEV *RHS = getSCEV(U->getOperand(1)); - - // Don't transfer the NSW and NUW bits from the Add instruction to the - // Add expression unless we can prove that it's safe. - AddOperator *Add = cast<AddOperator>(U); - bool HasNUW = UseFlag(Add->hasNoUnsignedWrap(), LHS, RHS, Add); - bool HasNSW = UseFlag(Add->hasNoSignedWrap(), LHS, RHS, Add); - - return getAddExpr(LHS, RHS, HasNUW, HasNSW); - } - case Instruction::Mul: { - const SCEV *LHS = getSCEV(U->getOperand(0)); - const SCEV *RHS = getSCEV(U->getOperand(1)); - - // Don't transfer the NSW and NUW bits from the Mul instruction to the - // Mul expression unless we can prove that it's safe. - MulOperator *Mul = cast<MulOperator>(U); - bool HasNUW = UseFlag(Mul->hasNoUnsignedWrap(), LHS, RHS, Mul); - bool HasNSW = UseFlag(Mul->hasNoSignedWrap(), LHS, RHS, Mul); - - return getMulExpr(LHS, RHS, HasNUW, HasNSW); - } + case Instruction::Add: + return getAddExpr(getSCEV(U->getOperand(0)), + getSCEV(U->getOperand(1))); + case Instruction::Mul: + return getMulExpr(getSCEV(U->getOperand(0)), + getSCEV(U->getOperand(1))); case Instruction::UDiv: return getUDivExpr(getSCEV(U->getOperand(0)), getSCEV(U->getOperand(1))); diff --git a/test/Analysis/ScalarEvolution/scev-aa.ll b/test/Analysis/ScalarEvolution/scev-aa.ll index dd5a66c..866664a 100644 --- a/test/Analysis/ScalarEvolution/scev-aa.ll +++ b/test/Analysis/ScalarEvolution/scev-aa.ll @@ -190,8 +190,9 @@ define void @bar() { ret void } +; TODO: This is theoretically provable to be NoAlias. ; CHECK: Function: nonnegative: 2 pointers, 0 call sites -; CHECK: NoAlias: i64* %arrayidx, i64* %p +; CHECK: MayAlias: i64* %arrayidx, i64* %p define void @nonnegative(i64* %p) nounwind { entry: @@ -210,6 +211,6 @@ for.end: ; preds = %for.body, %entry ret void } -; CHECK: 14 no alias responses -; CHECK: 26 may alias responses +; CHECK: 13 no alias responses +; CHECK: 27 may alias responses ; CHECK: 18 must alias responses |