diff options
| author | Chris Lattner <sabre@nondot.org> | 2008-06-16 06:19:11 +0000 |
|---|---|---|
| committer | Chris Lattner <sabre@nondot.org> | 2008-06-16 06:19:11 +0000 |
| commit | b35d9b5e0788538214a59129e1e0bdc8b186143a (patch) | |
| tree | 2296b4ad2f6bda48a3df85af42d5988e5f839ae5 /llvm/lib/Analysis | |
| parent | d813091cdecf3b50eba9c2fe09b6a9a47df73c25 (diff) | |
| download | bcm5719-llvm-b35d9b5e0788538214a59129e1e0bdc8b186143a.tar.gz bcm5719-llvm-b35d9b5e0788538214a59129e1e0bdc8b186143a.zip | |
If we are checking to see if the result of a call aliases a
pointer derived from a local allocation, if the local allocation
never escapes, the pointers can't alias. This implements PR2436
llvm-svn: 52301
Diffstat (limited to 'llvm/lib/Analysis')
| -rw-r--r-- | llvm/lib/Analysis/BasicAliasAnalysis.cpp | 22 |
1 files changed, 21 insertions, 1 deletions
diff --git a/llvm/lib/Analysis/BasicAliasAnalysis.cpp b/llvm/lib/Analysis/BasicAliasAnalysis.cpp index 7e096cd0bc2..6ab7d941ee3 100644 --- a/llvm/lib/Analysis/BasicAliasAnalysis.cpp +++ b/llvm/lib/Analysis/BasicAliasAnalysis.cpp @@ -317,6 +317,18 @@ static bool isKnownNonNull(const Value *V) { return false; } +/// isNonEscapingLocalObject - Return true if the pointer is to a function-local +/// object that never escapes from the function. +static bool isNonEscapingLocalObject(const Value *V) { + // If this is a local allocation or byval argument, check to see if it + // escapes. + if (isa<AllocationInst>(V) || + (isa<Argument>(V) && cast<Argument>(V)->hasByValAttr())) + return !AddressMightEscape(V); + return false; +} + + /// isObjectSmallerThan - Return true if we can prove that the object specified /// by V is smaller than Size. static bool isObjectSmallerThan(const Value *V, unsigned Size, @@ -393,7 +405,15 @@ BasicAliasAnalysis::alias(const Value *V1, unsigned V1Size, (V2Size != ~0U && isObjectSmallerThan(O1, V2Size, TD))) return NoAlias; - + // If one pointer is the result of a call/invoke and the other is a + // non-escaping local object, then we know the object couldn't escape to a + // point where the call could return it. + if ((isa<CallInst>(O1) || isa<InvokeInst>(O1)) && + isNonEscapingLocalObject(O2)) + return NoAlias; + if ((isa<CallInst>(O2) || isa<InvokeInst>(O2)) && + isNonEscapingLocalObject(O1)) + return NoAlias; // If we have two gep instructions with must-alias'ing base pointers, figure // out if the indexes to the GEP tell us anything about the derived pointer. |

