return false;
}
+/// areAllUsesInOneFunction - Return true if all the uses of the given value
+/// are in the same function. Note that this returns false if any of the uses
+/// are from non-instruction values.
+static bool areAllUsesInOneFunction(const Value *V) {
+ const llvm::Function *Fn = 0;
+
+ for (Value::const_use_iterator UI = V->use_begin(), E = V->use_end();
+ UI != E; ++UI) {
+ if (const Instruction *I = dyn_cast<Instruction>(*UI)) {
+ if (!Fn) {
+ Fn = I->getParent()->getParent();
+ continue;
+ }
+
+ if (Fn != I->getParent()->getParent())
+ return false;
+ } else
+ return false;
+ }
+
+ return true;
+}
+
/// isNonEscapingLocalObject - Return true if the pointer is to a function-local
/// object that never escapes from the function.
static bool isNonEscapingLocalObject(const Value *V) {
return true;
return !PointerMayBeCaptured(V, false, /*StoreCaptures=*/true);
}
+
+ // If this is an internal global variable that's only used in this function,
+ // check if it escapes the function.
+ if (const GlobalVariable *GV = dyn_cast<GlobalVariable>(V)) {
+ if (GV->hasInternalLinkage() && areAllUsesInOneFunction(GV)) {
+ return !PointerMayBeCaptured(V, /*ReturnCaptures=*/true,
+ /*StoreCaptures=*/true);
+ }
+ }
+
return false;
}
--- /dev/null
+; RUN: opt -basicaa -aa-eval -print-all-alias-modref-info -disable-output < %s |& FileCheck %s
+
+@global = internal global i32 0
+
+declare void @should_not_be_called()
+declare i32 @f()
+
+; CHECK: Function: g: 2 pointers, 0 call sites
+define void @g(i32* nocapture %p) {
+ store i32 0, i32* @global
+
+ ; @global is internal, is only used in this function, and never has its
+ ; address taken so it can't alias p.
+ ; CHECK: NoAlias: i32* %p, i32* @global
+ store i32 1, i32* %p
+ %1 = load i32* @global
+ ret void
+}
+