[llvm] f3c9e58 - Analysis: Fix assertion when load alignment exceeds address space size

Matt Arsenault via llvm-commits llvm-commits at lists.llvm.org
Fri Jun 30 09:31:37 PDT 2023


Author: Matt Arsenault
Date: 2023-06-30T12:31:32-04:00
New Revision: f3c9e5807fe91ef1f96d03e3a11e5971006d1efe

URL: https://github.com/llvm/llvm-project/commit/f3c9e5807fe91ef1f96d03e3a11e5971006d1efe
DIFF: https://github.com/llvm/llvm-project/commit/f3c9e5807fe91ef1f96d03e3a11e5971006d1efe.diff

LOG: Analysis: Fix assertion when load alignment exceeds address space size

Apparently the maximum alignment no longer fits in 32-bits now, which
overflows a 32-bit offset and would fail on the isPowerOf2 assert.

Added: 
    llvm/test/Transforms/ArgumentPromotion/load-alignment-value-overflows-addrspace-size.ll

Modified: 
    llvm/lib/Analysis/Loads.cpp

Removed: 
    


################################################################################
diff  --git a/llvm/lib/Analysis/Loads.cpp b/llvm/lib/Analysis/Loads.cpp
index ab49891c7f85c2..97d21db86abf28 100644
--- a/llvm/lib/Analysis/Loads.cpp
+++ b/llvm/lib/Analysis/Loads.cpp
@@ -29,9 +29,7 @@ using namespace llvm;
 static bool isAligned(const Value *Base, const APInt &Offset, Align Alignment,
                       const DataLayout &DL) {
   Align BA = Base->getPointerAlignment(DL);
-  const APInt APAlign(Offset.getBitWidth(), Alignment.value());
-  assert(APAlign.isPowerOf2() && "must be a power of 2!");
-  return BA >= Alignment && !(Offset & (APAlign - 1));
+  return BA >= Alignment && Offset.isAligned(BA);
 }
 
 /// Test if V is always a pointer to allocated and suitably aligned memory for

diff  --git a/llvm/test/Transforms/ArgumentPromotion/load-alignment-value-overflows-addrspace-size.ll b/llvm/test/Transforms/ArgumentPromotion/load-alignment-value-overflows-addrspace-size.ll
new file mode 100644
index 00000000000000..659d1331700a0f
--- /dev/null
+++ b/llvm/test/Transforms/ArgumentPromotion/load-alignment-value-overflows-addrspace-size.ll
@@ -0,0 +1,137 @@
+; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 2
+; RUN: opt -S -passes=argpromotion < %s | FileCheck %s
+target datalayout = "e-p:64:64-p5:32:32-p6:128:128:128:128"
+
+; Maximum alignment value of the load in a 64-bit address space
+; exceeds the bitwidth of the definition address space.
+define void @entry0() {
+; CHECK-LABEL: define void @entry0() {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    call void @call_load_maxalign_alloca_16()
+; CHECK-NEXT:    ret void
+;
+bb:
+  call void @call_load_maxalign_alloca_16()
+  ret void
+}
+
+define internal void @call_load_maxalign_alloca_16() {
+; CHECK-LABEL: define internal void @call_load_maxalign_alloca_16() {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    [[ALLOCA:%.*]] = alloca [13 x i16], align 16, addrspace(5)
+; CHECK-NEXT:    [[ADDRSPACECAST:%.*]] = addrspacecast ptr addrspace(5) [[ALLOCA]] to ptr
+; CHECK-NEXT:    call void @load_maxalign0(ptr [[ADDRSPACECAST]])
+; CHECK-NEXT:    ret void
+;
+bb:
+  %alloca = alloca [13 x i16], align 16, addrspace(5)
+  %addrspacecast = addrspacecast ptr addrspace(5) %alloca to ptr
+  call void @load_maxalign0(ptr %addrspacecast)
+  ret void
+}
+
+define internal void @load_maxalign0(ptr %arg) {
+; CHECK-LABEL: define internal void @load_maxalign0
+; CHECK-SAME: (ptr [[ARG:%.*]]) {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    br label [[BB1:%.*]]
+; CHECK:       bb1:
+; CHECK-NEXT:    [[LOAD:%.*]] = load i32, ptr [[ARG]], align 4294967296
+; CHECK-NEXT:    ret void
+;
+bb:
+  br label %bb1
+
+bb1:                                              ; preds = %bb
+  %load = load i32, ptr %arg, align 4294967296
+  ret void
+}
+
+; Make sure the early exit alignment check isn't hiding the offset
+; overflow.
+define void @entry1() {
+; CHECK-LABEL: define void @entry1() {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    call void @call_load_maxalign_alloca_maxalign()
+; CHECK-NEXT:    ret void
+;
+bb:
+  call void @call_load_maxalign_alloca_maxalign()
+  ret void
+}
+
+define internal void @call_load_maxalign_alloca_maxalign() {
+; CHECK-LABEL: define internal void @call_load_maxalign_alloca_maxalign() {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    [[ALLOCA:%.*]] = alloca [13 x i16], align 4294967296, addrspace(5)
+; CHECK-NEXT:    [[ADDRSPACECAST:%.*]] = addrspacecast ptr addrspace(5) [[ALLOCA]] to ptr
+; CHECK-NEXT:    [[ADDRSPACECAST_VAL:%.*]] = load i32, ptr [[ADDRSPACECAST]], align 4294967296
+; CHECK-NEXT:    call void @load_maxalign1(i32 [[ADDRSPACECAST_VAL]])
+; CHECK-NEXT:    ret void
+;
+bb:
+  %alloca = alloca [13 x i16], align 4294967296, addrspace(5)
+  %addrspacecast = addrspacecast ptr addrspace(5) %alloca to ptr
+  call void @load_maxalign1(ptr %addrspacecast)
+  ret void
+}
+
+define internal void @load_maxalign1(ptr %arg) {
+; CHECK-LABEL: define internal void @load_maxalign1
+; CHECK-SAME: (i32 [[ARG_0_VAL:%.*]]) {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    br label [[BB1:%.*]]
+; CHECK:       bb1:
+; CHECK-NEXT:    ret void
+;
+bb:
+  br label %bb1
+
+bb1:                                              ; preds = %bb
+  %load = load i32, ptr %arg, align 4294967296
+  ret void
+}
+
+; Alignment value exceeds pointer size, more than 1 past the end
+define void @entry2() {
+; CHECK-LABEL: define void @entry2() {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    call void @call_load_maxalign_alloca_ptr128()
+; CHECK-NEXT:    ret void
+;
+bb:
+  call void @call_load_maxalign_alloca_ptr128()
+  ret void
+}
+
+define internal void @call_load_maxalign_alloca_ptr128() {
+; CHECK-LABEL: define internal void @call_load_maxalign_alloca_ptr128() {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    [[ALLOCA:%.*]] = alloca [13 x i16], align 4294967296, addrspace(6)
+; CHECK-NEXT:    [[ADDRSPACECAST:%.*]] = addrspacecast ptr addrspace(6) [[ALLOCA]] to ptr
+; CHECK-NEXT:    [[ADDRSPACECAST_VAL:%.*]] = load i32, ptr [[ADDRSPACECAST]], align 4294967296
+; CHECK-NEXT:    call void @load_maxalign2(i32 [[ADDRSPACECAST_VAL]])
+; CHECK-NEXT:    ret void
+;
+bb:
+  %alloca = alloca [13 x i16], align 4294967296, addrspace(6)
+  %addrspacecast = addrspacecast ptr addrspace(6) %alloca to ptr
+  call void @load_maxalign2(ptr %addrspacecast)
+  ret void
+}
+
+define internal void @load_maxalign2(ptr %arg) {
+; CHECK-LABEL: define internal void @load_maxalign2
+; CHECK-SAME: (i32 [[ARG_0_VAL:%.*]]) {
+; CHECK-NEXT:  bb:
+; CHECK-NEXT:    br label [[BB1:%.*]]
+; CHECK:       bb1:
+; CHECK-NEXT:    ret void
+;
+bb:
+  br label %bb1
+
+bb1:                                              ; preds = %bb
+  %load = load i32, ptr %arg, align 4294967296
+  ret void
+}


        


More information about the llvm-commits mailing list