llvm/llvm/test/Transforms/MemCpyOpt/pr54682.ll

; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt -S -passes=memcpyopt < %s | FileCheck %s

; The memcpy here is *not* dead, because it reads memory written in a previous
; loop iteration.

define void @test(i1 %c, ptr nocapture noundef readonly %path, ptr noundef writeonly %name) {
; CHECK-LABEL: @test(
; CHECK-NEXT:  entry:
; CHECK-NEXT:    [[TMP:%.*]] = alloca [260 x i8], align 16
; CHECK-NEXT:    br label [[WHILE_BODY:%.*]]
; CHECK:       while.body:
; CHECK-NEXT:    [[IV:%.*]] = phi i64 [ [[IV_NEXT:%.*]], [[WHILE_BODY]] ], [ 259, [[ENTRY:%.*]] ]
; CHECK-NEXT:    [[IV_NEXT]] = add nsw i64 [[IV]], -1
; CHECK-NEXT:    [[TMP_IV:%.*]] = getelementptr inbounds [260 x i8], ptr [[TMP]], i64 0, i64 [[IV]]
; CHECK-NEXT:    store i8 42, ptr [[TMP_IV]], align 1
; CHECK-NEXT:    br i1 [[C:%.*]], label [[WHILE_BODY]], label [[EXIT:%.*]]
; CHECK:       exit:
; CHECK-NEXT:    [[TMP_IV_1:%.*]] = getelementptr inbounds i8, ptr [[TMP_IV]], i64 1
; CHECK-NEXT:    [[LEN:%.*]] = sub nsw i64 259, [[IV]]
; CHECK-NEXT:    call void @llvm.memcpy.p0.p0.i64(ptr align 1 [[NAME:%.*]], ptr nonnull align 1 [[TMP_IV_1]], i64 [[LEN]], i1 false)
; CHECK-NEXT:    ret void
;
entry:
  %tmp = alloca [260 x i8], align 16
  br label %while.body

while.body:
  %iv = phi i64 [ %iv.next, %while.body ], [ 259, %entry ]
  %iv.next = add nsw i64 %iv, -1
  %tmp.iv = getelementptr inbounds [260 x i8], ptr %tmp, i64 0, i64 %iv
  store i8 42, ptr %tmp.iv, align 1
  br i1 %c, label %while.body, label %exit

exit:
  %tmp.iv.1 = getelementptr inbounds i8, ptr %tmp.iv, i64 1
  %len = sub nsw i64 259, %iv
  call void @llvm.memcpy.p0.p0.i64(ptr align 1 %name, ptr nonnull align 1 %tmp.iv.1, i64 %len, i1 false)
  ret void
}

declare void @llvm.memcpy.p0.p0.i64(ptr noalias nocapture writeonly %0, ptr noalias nocapture readonly %1, i64 %2, i1 immarg %3)