blob: 8020480f0dc65e2134b8718bfe865ba4df214a18 [file] [edit]
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --check-globals none --version 6
; RUN: opt -S -passes='loop-vectorize,verify' -enable-vplan-native-path -force-vector-width=2 < %s | FileCheck %s
; The noalias.scope.decl intrinsic declares a noalias scope valid for a single
; iteration. During outer loop vectorization, emitting it as a single-scalar
; replicate would incorrectly extend the scope across multiple original
; iterations packed into one vector iteration. Bail out of vectorization for
; outer loops containing noalias.scope.decl.
; FIXME: We could still vectorize by dropping the noalias.scope.decl and
; stripping !alias.scope/!noalias metadata from the loop body.
define void @test_noalias_scope_decl(ptr %a, ptr %b, i64 %n) {
; CHECK-LABEL: define void @test_noalias_scope_decl(
; CHECK-SAME: ptr [[A:%.*]], ptr [[B:%.*]], i64 [[N:%.*]]) {
; CHECK-NEXT: [[SCALAR_PH:.*]]:
; CHECK-NEXT: br label %[[OUTER_HEADER:.*]]
; CHECK: [[OUTER_HEADER]]:
; CHECK-NEXT: [[J:%.*]] = phi i64 [ 0, %[[SCALAR_PH]] ], [ [[J_NEXT:%.*]], %[[OUTER_LATCH:.*]] ]
; CHECK-NEXT: call void @llvm.experimental.noalias.scope.decl(metadata [[META0:![0-9]+]])
; CHECK-NEXT: br label %[[INNER:.*]]
; CHECK: [[INNER]]:
; CHECK-NEXT: [[K:%.*]] = phi i64 [ 0, %[[OUTER_HEADER]] ], [ [[K_NEXT:%.*]], %[[INNER]] ]
; CHECK-NEXT: [[IDX:%.*]] = add i64 [[J]], [[K]]
; CHECK-NEXT: [[PTR_A:%.*]] = getelementptr inbounds i32, ptr [[A]], i64 [[IDX]]
; CHECK-NEXT: [[VAL:%.*]] = load i32, ptr [[PTR_A]], align 4, !alias.scope [[META0]]
; CHECK-NEXT: [[PTR_B:%.*]] = getelementptr inbounds i32, ptr [[B]], i64 [[IDX]]
; CHECK-NEXT: store i32 [[VAL]], ptr [[PTR_B]], align 4, !noalias [[META0]]
; CHECK-NEXT: [[K_NEXT]] = add i64 [[K]], 1
; CHECK-NEXT: [[INNER_COND:%.*]] = icmp eq i64 [[K_NEXT]], [[N]]
; CHECK-NEXT: br i1 [[INNER_COND]], label %[[OUTER_LATCH]], label %[[INNER]]
; CHECK: [[OUTER_LATCH]]:
; CHECK-NEXT: [[J_NEXT]] = add i64 [[J]], 1
; CHECK-NEXT: [[OUTER_COND:%.*]] = icmp eq i64 [[J_NEXT]], [[N]]
; CHECK-NEXT: br i1 [[OUTER_COND]], label %[[EXIT:.*]], label %[[OUTER_HEADER]], !llvm.loop [[LOOP3:![0-9]+]]
; CHECK: [[EXIT]]:
; CHECK-NEXT: ret void
;
entry:
br label %outer.header
outer.header:
%j = phi i64 [ 0, %entry ], [ %j.next, %outer.latch ]
call void @llvm.experimental.noalias.scope.decl(metadata !3)
br label %inner
inner:
%k = phi i64 [ 0, %outer.header ], [ %k.next, %inner ]
%idx = add i64 %j, %k
%ptr.a = getelementptr inbounds i32, ptr %a, i64 %idx
%val = load i32, ptr %ptr.a, align 4, !alias.scope !3
%ptr.b = getelementptr inbounds i32, ptr %b, i64 %idx
store i32 %val, ptr %ptr.b, align 4, !noalias !3
%k.next = add i64 %k, 1
%inner.cond = icmp eq i64 %k.next, %n
br i1 %inner.cond, label %outer.latch, label %inner
outer.latch:
%j.next = add i64 %j, 1
%outer.cond = icmp eq i64 %j.next, %n
br i1 %outer.cond, label %exit, label %outer.header, !llvm.loop !0
exit:
ret void
}
!0 = distinct !{!0, !1, !2}
!1 = !{!"llvm.loop.mustprogress"}
!2 = !{!"llvm.loop.vectorize.enable", i1 true}
!3 = !{!4}
!4 = distinct !{!4, !5}
!5 = distinct !{!5}