[llvm] [BasicAA] Improve escape source analysis for return-only captures (PR #131869)

Nikita Popov via llvm-commits llvm-commits at lists.llvm.org
Tue Mar 18 10:46:04 PDT 2025


https://github.com/nikic updated https://github.com/llvm/llvm-project/pull/131869

>From 761c4d73eb648ba24895da9580c398f9799a3dfc Mon Sep 17 00:00:00 2001
From: Nikita Popov <npopov at redhat.com>
Date: Tue, 18 Mar 2025 18:17:39 +0100
Subject: [PATCH 1/2] Add test

---
 llvm/test/Analysis/BasicAA/captures.ll | 13 +++++++++++++
 1 file changed, 13 insertions(+)

diff --git a/llvm/test/Analysis/BasicAA/captures.ll b/llvm/test/Analysis/BasicAA/captures.ll
index fcc51b54b71b8..07d005dce9d04 100644
--- a/llvm/test/Analysis/BasicAA/captures.ll
+++ b/llvm/test/Analysis/BasicAA/captures.ll
@@ -40,3 +40,16 @@ define void @address_capture_and_full_capture() {
   load i32, ptr %a
   ret void
 }
+
+declare ptr @capture_ret(ptr, ptr)
+
+; CHECK-LABEL: capture_ret_only
+; CHECK: MayAlias:	i8* %a, i8* %ret
+; CHECK: MayAlias:	i8* %b, i8* %ret
+define void @capture_ret_only(ptr noalias %a, ptr noalias %b) {
+  %ret = call ptr @capture_ret(ptr captures(ret: address, provenance) %a, ptr captures(none) %b)
+  load i8, ptr %ret
+  load i8, ptr %a
+  load i8, ptr %b
+  ret void
+}

>From 67708a67040e1d37ecd47bf042ed082d57c0103d Mon Sep 17 00:00:00 2001
From: Nikita Popov <npopov at redhat.com>
Date: Tue, 18 Mar 2025 18:36:53 +0100
Subject: [PATCH 2/2] [BasicAA] Improve escape source analysis for return-only
 captures

This fixes an issue where improving inference from a general capture
to a return-only capture can make alias analysis results worse.

The problem is that if there are return-only captures, the call
can no longer treated as a plain escape source, as the call result
may alias one of the return-only-capture arguments, even if they
don't escape.

Fix this by having isEscapeSource() return the relevant arguments and
explicitly check that these aren't aliased.

Fixes https://github.com/llvm/llvm-project/issues/131168.
---
 llvm/include/llvm/Analysis/AliasAnalysis.h   |  6 +++---
 llvm/include/llvm/IR/InstrTypes.h            |  5 +++--
 llvm/lib/Analysis/AliasAnalysis.cpp          |  9 +++++----
 llvm/lib/Analysis/BasicAliasAnalysis.cpp     | 20 +++++++++++++++-----
 llvm/lib/IR/Instructions.cpp                 |  6 +++---
 llvm/lib/Transforms/Utils/InlineFunction.cpp |  3 ++-
 llvm/test/Analysis/BasicAA/captures.ll       |  2 +-
 7 files changed, 32 insertions(+), 19 deletions(-)

diff --git a/llvm/include/llvm/Analysis/AliasAnalysis.h b/llvm/include/llvm/Analysis/AliasAnalysis.h
index b192a9f5e65e7..b533011a6a4dc 100644
--- a/llvm/include/llvm/Analysis/AliasAnalysis.h
+++ b/llvm/include/llvm/Analysis/AliasAnalysis.h
@@ -888,9 +888,9 @@ bool isIdentifiedFunctionLocal(const Value *V);
 /// can be inbounds w.r.t the actual underlying object.
 bool isBaseOfObject(const Value *V);
 
-/// Returns true if the pointer is one which would have been considered an
-/// escape by isNonEscapingLocalObject.
-bool isEscapeSource(const Value *V);
+/// Returns true if the pointer cannot alias a non-escaping local object,
+/// except via the values returned in the MayAlias argument.
+bool isEscapeSource(const Value *V, SmallVectorImpl<Value *> &MayAlias);
 
 /// Return true if Object memory is not visible after an unwind, in the sense
 /// that program semantics cannot depend on Object containing any particular
diff --git a/llvm/include/llvm/IR/InstrTypes.h b/llvm/include/llvm/IR/InstrTypes.h
index 8e47e3c7b3a7c..8a6de6f7817d9 100644
--- a/llvm/include/llvm/IR/InstrTypes.h
+++ b/llvm/include/llvm/IR/InstrTypes.h
@@ -1692,10 +1692,11 @@ class CallBase : public Instruction {
     return capturesNothing(getCaptureInfo(OpNo));
   }
 
-  /// Returns whether the call has an argument that has an attribute like
+  /// Collects call arguments that have an attribute like
   /// captures(ret: address, provenance), where the return capture components
   /// are not a subset of the other capture components.
-  bool hasArgumentWithAdditionalReturnCaptureComponents() const;
+  void getArgumentsWithAdditionalReturnCaptureComponents(
+      SmallVectorImpl<Value *> &Args) const;
 
   /// Determine whether this argument is passed by value.
   bool isByValArgument(unsigned ArgNo) const {
diff --git a/llvm/lib/Analysis/AliasAnalysis.cpp b/llvm/lib/Analysis/AliasAnalysis.cpp
index 5114137da14ae..76dc41047d16b 100644
--- a/llvm/lib/Analysis/AliasAnalysis.cpp
+++ b/llvm/lib/Analysis/AliasAnalysis.cpp
@@ -840,15 +840,16 @@ bool llvm::isBaseOfObject(const Value *V) {
   return (isa<AllocaInst>(V) || isa<GlobalVariable>(V));
 }
 
-bool llvm::isEscapeSource(const Value *V) {
+bool llvm::isEscapeSource(const Value *V, SmallVectorImpl<Value *> &MayAlias) {
   if (auto *CB = dyn_cast<CallBase>(V)) {
     if (isIntrinsicReturningPointerAliasingArgumentWithoutCapturing(CB, true))
       return false;
 
     // The return value of a function with a captures(ret: address, provenance)
-    // attribute is not necessarily an escape source. The return value may
-    // alias with a non-escaping object.
-    return !CB->hasArgumentWithAdditionalReturnCaptureComponents();
+    // attribute may alias with the corresponding argument. If it doesn't, then
+    // it is an escape source.
+    CB->getArgumentsWithAdditionalReturnCaptureComponents(MayAlias);
+    return true;
   }
 
   // The load case works because isNonEscapingLocalObject considers all
diff --git a/llvm/lib/Analysis/BasicAliasAnalysis.cpp b/llvm/lib/Analysis/BasicAliasAnalysis.cpp
index b41c00ada7ab2..32e538a980add 100644
--- a/llvm/lib/Analysis/BasicAliasAnalysis.cpp
+++ b/llvm/lib/Analysis/BasicAliasAnalysis.cpp
@@ -1606,11 +1606,21 @@ AliasResult BasicAAResult::aliasCheck(const Value *V1, LocationSize V1Size,
     // temporary store the nocapture argument's value in a temporary memory
     // location if that memory location doesn't escape. Or it may pass a
     // nocapture value to other functions as long as they don't capture it.
-    if (isEscapeSource(O1) && AAQI.CA->isNotCapturedBefore(
-                                  O2, dyn_cast<Instruction>(O1), /*OrAt*/ true))
-      return AliasResult::NoAlias;
-    if (isEscapeSource(O2) && AAQI.CA->isNotCapturedBefore(
-                                  O1, dyn_cast<Instruction>(O2), /*OrAt*/ true))
+    auto EscapeSourceDoesNotAliasNonEscapingLocalObject = [&](const Value *O1,
+                                                              const Value *O2) {
+      SmallVector<Value *> MayAlias;
+      if (!isEscapeSource(O1, MayAlias) ||
+          !AAQI.CA->isNotCapturedBefore(O2, dyn_cast<Instruction>(O1),
+                                        /*OrAt=*/true))
+        return false;
+      for (Value *V : MayAlias)
+        if (AAQI.AAR.alias(MemoryLocation::getBeforeOrAfter(O2),
+                           MemoryLocation::getBeforeOrAfter(V), AAQI))
+          return false;
+      return true;
+    };
+    if (EscapeSourceDoesNotAliasNonEscapingLocalObject(O1, O2) ||
+        EscapeSourceDoesNotAliasNonEscapingLocalObject(O2, O1))
       return AliasResult::NoAlias;
   }
 
diff --git a/llvm/lib/IR/Instructions.cpp b/llvm/lib/IR/Instructions.cpp
index d2cf0ae2c1778..6e95c452f31f6 100644
--- a/llvm/lib/IR/Instructions.cpp
+++ b/llvm/lib/IR/Instructions.cpp
@@ -713,7 +713,8 @@ CaptureInfo CallBase::getCaptureInfo(unsigned OpNo) const {
   return OBU.isDeoptOperandBundle() ? CaptureInfo::none() : CaptureInfo::all();
 }
 
-bool CallBase::hasArgumentWithAdditionalReturnCaptureComponents() const {
+void CallBase::getArgumentsWithAdditionalReturnCaptureComponents(
+    SmallVectorImpl<Value *> &Args) const {
   for (unsigned I = 0, E = arg_size(); I < E; ++I) {
     if (!getArgOperand(I)->getType()->isPointerTy())
       continue;
@@ -722,9 +723,8 @@ bool CallBase::hasArgumentWithAdditionalReturnCaptureComponents() const {
     if (auto *Fn = dyn_cast<Function>(getCalledOperand()))
       CI &= Fn->getAttributes().getParamAttrs(I).getCaptureInfo();
     if (capturesAnything(CI.getRetComponents() & ~CI.getOtherComponents()))
-      return true;
+      Args.push_back(getArgOperand(I));
   }
-  return false;
 }
 
 //===----------------------------------------------------------------------===//
diff --git a/llvm/lib/Transforms/Utils/InlineFunction.cpp b/llvm/lib/Transforms/Utils/InlineFunction.cpp
index 66fda5b1ec7c9..5a8f5c4e7daa8 100644
--- a/llvm/lib/Transforms/Utils/InlineFunction.cpp
+++ b/llvm/lib/Transforms/Utils/InlineFunction.cpp
@@ -1273,7 +1273,8 @@ static void AddAliasScopeMetadata(CallBase &CB, ValueToValueMapTy &VMap,
           UsesAliasingPtr = true;
         }
 
-        if (isEscapeSource(V)) {
+        SmallVector<Value *> MayAlias;
+        if (isEscapeSource(V, MayAlias) && MayAlias.empty()) {
           // An escape source can only alias with a noalias argument if it has
           // been captured beforehand.
           RequiresNoCaptureBefore = true;
diff --git a/llvm/test/Analysis/BasicAA/captures.ll b/llvm/test/Analysis/BasicAA/captures.ll
index 07d005dce9d04..1f9197aaa40c8 100644
--- a/llvm/test/Analysis/BasicAA/captures.ll
+++ b/llvm/test/Analysis/BasicAA/captures.ll
@@ -45,7 +45,7 @@ declare ptr @capture_ret(ptr, ptr)
 
 ; CHECK-LABEL: capture_ret_only
 ; CHECK: MayAlias:	i8* %a, i8* %ret
-; CHECK: MayAlias:	i8* %b, i8* %ret
+; CHECK: NoAlias:	i8* %b, i8* %ret
 define void @capture_ret_only(ptr noalias %a, ptr noalias %b) {
   %ret = call ptr @capture_ret(ptr captures(ret: address, provenance) %a, ptr captures(none) %b)
   load i8, ptr %ret



More information about the llvm-commits mailing list