[llvm] 55b480e - [SROA] Allow load-only promotion with read-only captures (#130735)
via llvm-commits
llvm-commits at lists.llvm.org
Thu Mar 13 01:53:06 PDT 2025
Author: Nikita Popov
Date: 2025-03-13T09:53:02+01:00
New Revision: 55b480ec3c870b08b4eb028f9a5eade4d2deeca9
URL: https://github.com/llvm/llvm-project/commit/55b480ec3c870b08b4eb028f9a5eade4d2deeca9
DIFF: https://github.com/llvm/llvm-project/commit/55b480ec3c870b08b4eb028f9a5eade4d2deeca9.diff
LOG: [SROA] Allow load-only promotion with read-only captures (#130735)
It's okay if the address or read-provenance of the pointer is captured.
We only have to make sure that there are no unanalyzable writes to the
pointer.
Added:
Modified:
llvm/lib/Transforms/Scalar/SROA.cpp
llvm/test/Transforms/SROA/readonlynocapture.ll
Removed:
################################################################################
diff --git a/llvm/lib/Transforms/Scalar/SROA.cpp b/llvm/lib/Transforms/Scalar/SROA.cpp
index 69e7ce83f82e4..1b82e5dc58707 100644
--- a/llvm/lib/Transforms/Scalar/SROA.cpp
+++ b/llvm/lib/Transforms/Scalar/SROA.cpp
@@ -1397,10 +1397,10 @@ class AllocaSlices::SliceBuilder : public PtrUseVisitor<SliceBuilder> {
void visitInstruction(Instruction &I) { PI.setAborted(&I); }
void visitCallBase(CallBase &CB) {
- // If the call operand is NoCapture ReadOnly, then we mark it as
- // EscapedReadOnly.
+ // If the call operand is read-only and only does a read-only or address
+ // capture, then we mark it as EscapedReadOnly.
if (CB.isDataOperand(U) &&
- CB.doesNotCapture(U->getOperandNo()) &&
+ !capturesFullProvenance(CB.getCaptureInfo(U->getOperandNo())) &&
CB.onlyReadsMemory(U->getOperandNo())) {
PI.setEscapedReadOnly(&CB);
return;
diff --git a/llvm/test/Transforms/SROA/readonlynocapture.ll b/llvm/test/Transforms/SROA/readonlynocapture.ll
index 611c90ac32b5a..2ad20fcc51dc5 100644
--- a/llvm/test/Transforms/SROA/readonlynocapture.ll
+++ b/llvm/test/Transforms/SROA/readonlynocapture.ll
@@ -406,4 +406,54 @@ define i32 @simple_byval() {
ret i32 %l1
}
+declare void @callee_address_only_capture(ptr readonly captures(address) %p)
+
+define i32 @address_only_capture() {
+; CHECK-LABEL: @address_only_capture(
+; CHECK-NEXT: [[A:%.*]] = alloca i32, align 4
+; CHECK-NEXT: store i32 0, ptr [[A]], align 4
+; CHECK-NEXT: call void @callee_address_only_capture(ptr [[A]])
+; CHECK-NEXT: ret i32 0
+;
+ %a = alloca i32
+ store i32 0, ptr %a
+ call void @callee_address_only_capture(ptr %a)
+ %l1 = load i32, ptr %a
+ ret i32 %l1
+}
+
+declare void @callee_read_only_capture(ptr readonly captures(address, read_provenance) %p)
+
+define i32 @read_only_capture() {
+; CHECK-LABEL: @read_only_capture(
+; CHECK-NEXT: [[A:%.*]] = alloca i32, align 4
+; CHECK-NEXT: store i32 0, ptr [[A]], align 4
+; CHECK-NEXT: call void @callee_read_only_capture(ptr [[A]])
+; CHECK-NEXT: ret i32 0
+;
+ %a = alloca i32
+ store i32 0, ptr %a
+ call void @callee_read_only_capture(ptr %a)
+ %l1 = load i32, ptr %a
+ ret i32 %l1
+}
+
+declare void @callee_provenance_only_capture(ptr readonly captures(provenance) %p)
+
+; Should not be transformed, as write-provenance is captured.
+define i32 @provenance_only_capture() {
+; CHECK-LABEL: @provenance_only_capture(
+; CHECK-NEXT: [[A:%.*]] = alloca i32, align 4
+; CHECK-NEXT: store i32 0, ptr [[A]], align 4
+; CHECK-NEXT: call void @callee_provenance_only_capture(ptr [[A]])
+; CHECK-NEXT: [[L1:%.*]] = load i32, ptr [[A]], align 4
+; CHECK-NEXT: ret i32 [[L1]]
+;
+ %a = alloca i32
+ store i32 0, ptr %a
+ call void @callee_provenance_only_capture(ptr %a)
+ %l1 = load i32, ptr %a
+ ret i32 %l1
+}
+
declare void @llvm.memcpy.p0.p0.i64(ptr, ptr, i64, i1)
More information about the llvm-commits
mailing list