llvm/llvm/test/Transforms/LICM/dropped-tbaa.ll

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

; This test case case is generated from the following C code with -fstrict-aliasing,
; and after passing through -passes=inline,mem2reg,loop-rotate,instcombine
; void add(ptr restrict data, int *restrict addend) {
;    *data += *addend;
; }
;
; void foo(ptr data, int *addend) {
;    for (int i = 0; i < 1000; ++i) {
;        *data += *addend;
;        add(data, addend);
;    }
; }
; We want to make sure the load of addend gets hoisted, independent of the second load
; load having different noalias metadata.

define void @foo(ptr %data, ptr %addend) #0 {
; CHECK:       for.body.lr.ph:
; CHECK-NEXT:    [[TMP1:%.*]] = load i32, ptr [[ADDEND:%.*]], align 4, !tbaa !1
; CHECK-NEXT:    [[CONV:%.*]] = sitofp i32 [[TMP1]] to double
; CHECK-NEXT:    [[TMP2:%.*]] = load i32, ptr [[ADDEND]], align 4, !tbaa !1, !alias.scope !5, !noalias !8
; CHECK-NEXT:    [[CONV_I:%.*]] = sitofp i32 [[TMP2]] to double
entry:
  %i = alloca i32, align 4
  call void @llvm.lifetime.start.p0(i64 4, ptr %i) #2
  store i32 0, ptr %i, align 4, !tbaa !1
  br i1 true, label %for.body.lr.ph, label %for.cond.cleanup

for.body.lr.ph:                                   ; preds = %entry
  br label %for.body

for.cond.for.cond.cleanup_crit_edge:              ; preds = %for.inc
  br label %for.cond.cleanup

for.cond.cleanup:                                 ; preds = %for.cond.for.cond.cleanup_crit_edge, %entry
  call void @llvm.lifetime.end.p0(i64 4, ptr %i) #2
  br label %for.end

for.body:                                         ; preds = %for.body.lr.ph, %for.inc
  %0 = load i32, ptr %addend, align 4, !tbaa !1
  %conv = sitofp i32 %0 to double
  %1 = load i32, ptr %i, align 4, !tbaa !1
  %idxprom = sext i32 %1 to i64
  %arrayidx = getelementptr inbounds double, ptr %data, i64 %idxprom
  %2 = load double, ptr %arrayidx, align 8, !tbaa !5
  %add = fadd double %2, %conv
  store double %add, ptr %arrayidx, align 8, !tbaa !5
  %idxprom1 = sext i32 %1 to i64
  %arrayidx2 = getelementptr inbounds double, ptr %data, i64 %idxprom1
  %3 = load i32, ptr %addend, align 4, !tbaa !1, !alias.scope !7, !noalias !10
  %conv.i = sitofp i32 %3 to double
  %4 = load double, ptr %arrayidx2, align 8, !tbaa !5, !alias.scope !10, !noalias !7
  %add.i = fadd double %4, %conv.i
  store double %add.i, ptr %arrayidx2, align 8, !tbaa !5, !alias.scope !10, !noalias !7
  br label %for.inc

for.inc:                                          ; preds = %for.body
  %5 = load i32, ptr %i, align 4, !tbaa !1
  %inc = add nsw i32 %5, 1
  store i32 %inc, ptr %i, align 4, !tbaa !1
  %cmp = icmp slt i32 %inc, 1000
  br i1 %cmp, label %for.body, label %for.cond.for.cond.cleanup_crit_edge

for.end:                                          ; preds = %for.cond.cleanup
  ret void
}

declare void @llvm.lifetime.start.p0(i64, ptr nocapture) #0
declare void @llvm.lifetime.end.p0(i64, ptr nocapture) #0

attributes #0 = { argmemonly nounwind }

!llvm.ident = !{!0}

!0 = !{!"clang version 5.0.0  (llvm/trunk 299971)"}
!1 = !{!2, !2, i64 0}
!2 = !{!"int", !3, i64 0}
!3 = !{!"omnipotent char", !4, i64 0}
!4 = !{!"Simple C/C++ TBAA"}
!5 = !{!6, !6, i64 0}
!6 = !{!"double", !3, i64 0}
!7 = !{!8}
!8 = distinct !{!8, !9, !"add: %addend"}
!9 = distinct !{!9, !"add"}
!10 = !{!11}
!11 = distinct !{!11, !9, !"add: %data"}