diff options
author | Dan Gohman <gohman@apple.com> | 2010-08-06 18:24:38 +0000 |
---|---|---|
committer | Dan Gohman <gohman@apple.com> | 2010-08-06 18:24:38 +0000 |
commit | e68958fcdfb02296df153598c1610cefd915d51e (patch) | |
tree | 472946511f16ade3365f87abc2fed4fba9b43dea | |
parent | 0cb2c7a25ed5496febb619a90ad68c7b1857e733 (diff) | |
download | bcm5719-llvm-e68958fcdfb02296df153598c1610cefd915d51e.tar.gz bcm5719-llvm-e68958fcdfb02296df153598c1610cefd915d51e.zip |
Implement a proper getModRefInfo for va_arg.
llvm-svn: 110458
-rw-r--r-- | llvm/include/llvm/Analysis/AliasAnalysis.h | 5 | ||||
-rw-r--r-- | llvm/lib/Analysis/AliasAnalysis.cpp | 17 | ||||
-rw-r--r-- | llvm/test/Analysis/BasicAA/modref.ll | 11 |
3 files changed, 29 insertions, 4 deletions
diff --git a/llvm/include/llvm/Analysis/AliasAnalysis.h b/llvm/include/llvm/Analysis/AliasAnalysis.h index bf8f5185cbb..7b003d8a691 100644 --- a/llvm/include/llvm/Analysis/AliasAnalysis.h +++ b/llvm/include/llvm/Analysis/AliasAnalysis.h @@ -243,6 +243,7 @@ public: /// Convenience functions... ModRefResult getModRefInfo(const LoadInst *L, const Value *P, unsigned Size); ModRefResult getModRefInfo(const StoreInst *S, const Value *P, unsigned Size); + ModRefResult getModRefInfo(const VAArgInst* I, const Value* P, unsigned Size); ModRefResult getModRefInfo(const CallInst *C, const Value *P, unsigned Size) { return getModRefInfo(ImmutableCallSite(C), P, Size); } @@ -250,10 +251,6 @@ public: const Value *P, unsigned Size) { return getModRefInfo(ImmutableCallSite(I), P, Size); } - ModRefResult getModRefInfo(const VAArgInst* I, - const Value* P, unsigned Size) { - return AliasAnalysis::ModRef; - } ModRefResult getModRefInfo(const Instruction *I, const Value *P, unsigned Size) { switch (I->getOpcode()) { diff --git a/llvm/lib/Analysis/AliasAnalysis.cpp b/llvm/lib/Analysis/AliasAnalysis.cpp index 0b15334e145..1f2528fa560 100644 --- a/llvm/lib/Analysis/AliasAnalysis.cpp +++ b/llvm/lib/Analysis/AliasAnalysis.cpp @@ -229,6 +229,23 @@ AliasAnalysis::getModRefInfo(const StoreInst *S, const Value *P, unsigned Size) return Mod; } +AliasAnalysis::ModRefResult +AliasAnalysis::getModRefInfo(const VAArgInst *V, const Value *P, unsigned Size) { + // If the va_arg address cannot alias the pointer in question, then the + // specified memory cannot be accessed by the va_arg. + if (!alias(V->getOperand(0), UnknownSize, P, Size)) + return NoModRef; + + // If the pointer is a pointer to constant memory, then it could not have been + // modified by this va_arg. + if (pointsToConstantMemory(P)) + return NoModRef; + + // Otherwise, a va_arg reads and writes. + return ModRef; +} + + AliasAnalysis::ModRefBehavior AliasAnalysis::getIntrinsicModRefBehavior(unsigned iid) { #define GET_INTRINSIC_MODREF_BEHAVIOR diff --git a/llvm/test/Analysis/BasicAA/modref.ll b/llvm/test/Analysis/BasicAA/modref.ll index a2aabf135f6..b9a3c5e58f6 100644 --- a/llvm/test/Analysis/BasicAA/modref.ll +++ b/llvm/test/Analysis/BasicAA/modref.ll @@ -123,3 +123,14 @@ define i32 @test5(i8* %P, i32 %Len) { ; CHECK: sub i32 %tmp, %tmp } +define i8 @test6(i8* %p, i8* noalias %a) { + %x = load i8* %a + %t = va_arg i8* %p, float + %y = load i8* %a + %z = add i8 %x, %y + ret i8 %z +; CHECK: @test6 +; CHECK: load i8* %a +; CHECK-NOT: load +; CHECK: ret +} |