; 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}