; RUN: opt < %s -aa-pipeline=tbaa,basic-aa -passes=aa-eval -evaluate-aa-metadata -print-no-aliases -print-may-aliases -disable-output 2>&1 | FileCheck %s
; RUN: opt < %s -aa-pipeline=tbaa,basic-aa -passes=gvn -S | FileCheck %s --check-prefix=OPT
; Generated from clang/test/CodeGen/tbaa.cpp with "-O1 -struct-path-tbaa -disable-llvm-optzns".
%struct.StructA = type { i16, i32, i16, i32 }
%struct.StructB = type { i16, %struct.StructA, i32 }
%struct.StructS = type { i16, i32 }
%struct.StructS2 = type { i16, i32 }
%struct.StructC = type { i16, %struct.StructB, i32 }
%struct.StructD = type { i16, %struct.StructB, i32, i8 }
define i32 @_Z1gPjP7StructAy(ptr %s, ptr %A, i64 %count) #0 {
entry:
; Access to ptr and &(A->f32).
; CHECK: Function
; CHECK: MayAlias: store i32 4, ptr %f32, align 4, !tbaa !8 <-> store i32 1, ptr %0, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i32 4
; OPT: %[[RET:.*]] = load i32, ptr
; OPT: ret i32 %[[RET]]
%s.addr = alloca ptr, align 8
%A.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %s, ptr %s.addr, align 8, !tbaa !0
store ptr %A, ptr %A.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %s.addr, align 8, !tbaa !0
store i32 1, ptr %0, align 4, !tbaa !6
%1 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructA, ptr %1, i32 0, i32 1
store i32 4, ptr %f32, align 4, !tbaa !8
%2 = load ptr, ptr %s.addr, align 8, !tbaa !0
%3 = load i32, ptr %2, align 4, !tbaa !6
ret i32 %3
}
define i32 @_Z2g2PjP7StructAy(ptr %s, ptr %A, i64 %count) #0 {
entry:
; Access to ptr and &(A->f16).
; CHECK: Function
; CHECK: NoAlias: store i16 4, ptr %1, align 2, !tbaa !8 <-> store i32 1, ptr %0, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i16 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%s.addr = alloca ptr, align 8
%A.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %s, ptr %s.addr, align 8, !tbaa !0
store ptr %A, ptr %A.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %s.addr, align 8, !tbaa !0
store i32 1, ptr %0, align 4, !tbaa !6
%1 = load ptr, ptr %A.addr, align 8, !tbaa !0
store i16 4, ptr %1, align 2, !tbaa !11
%2 = load ptr, ptr %s.addr, align 8, !tbaa !0
%3 = load i32, ptr %2, align 4, !tbaa !6
ret i32 %3
}
define i32 @_Z2g3P7StructAP7StructBy(ptr %A, ptr %B, i64 %count) #0 {
entry:
; Access to &(A->f32) and &(B->a.f32).
; CHECK: Function
; CHECK: MayAlias: store i32 4, ptr %f321, align 4, !tbaa !10 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i32 4
; OPT: %[[RET:.*]] = load i32, ptr
; OPT: ret i32 %[[RET]]
%A.addr = alloca ptr, align 8
%B.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %A, ptr %A.addr, align 8, !tbaa !0
store ptr %B, ptr %B.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructA, ptr %0, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !8
%1 = load ptr, ptr %B.addr, align 8, !tbaa !0
%a = getelementptr inbounds %struct.StructB, ptr %1, i32 0, i32 1
%f321 = getelementptr inbounds %struct.StructA, ptr %a, i32 0, i32 1
store i32 4, ptr %f321, align 4, !tbaa !12
%2 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f322 = getelementptr inbounds %struct.StructA, ptr %2, i32 0, i32 1
%3 = load i32, ptr %f322, align 4, !tbaa !8
ret i32 %3
}
define i32 @_Z2g4P7StructAP7StructBy(ptr %A, ptr %B, i64 %count) #0 {
entry:
; Access to &(A->f32) and &(B->a.f16).
; CHECK: Function
; CHECK: NoAlias: store i16 4, ptr %a, align 2, !tbaa !10 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i16 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%A.addr = alloca ptr, align 8
%B.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %A, ptr %A.addr, align 8, !tbaa !0
store ptr %B, ptr %B.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructA, ptr %0, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !8
%1 = load ptr, ptr %B.addr, align 8, !tbaa !0
%a = getelementptr inbounds %struct.StructB, ptr %1, i32 0, i32 1
store i16 4, ptr %a, align 2, !tbaa !14
%2 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f321 = getelementptr inbounds %struct.StructA, ptr %2, i32 0, i32 1
%3 = load i32, ptr %f321, align 4, !tbaa !8
ret i32 %3
}
define i32 @_Z2g5P7StructAP7StructBy(ptr %A, ptr %B, i64 %count) #0 {
entry:
; Access to &(A->f32) and &(B->f32).
; CHECK: Function
; CHECK: NoAlias: store i32 4, ptr %f321, align 4, !tbaa !10 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i32 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%A.addr = alloca ptr, align 8
%B.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %A, ptr %A.addr, align 8, !tbaa !0
store ptr %B, ptr %B.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructA, ptr %0, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !8
%1 = load ptr, ptr %B.addr, align 8, !tbaa !0
%f321 = getelementptr inbounds %struct.StructB, ptr %1, i32 0, i32 2
store i32 4, ptr %f321, align 4, !tbaa !15
%2 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f322 = getelementptr inbounds %struct.StructA, ptr %2, i32 0, i32 1
%3 = load i32, ptr %f322, align 4, !tbaa !8
ret i32 %3
}
define i32 @_Z2g6P7StructAP7StructBy(ptr %A, ptr %B, i64 %count) #0 {
entry:
; Access to &(A->f32) and &(B->a.f32_2).
; CHECK: Function
; CHECK: NoAlias: store i32 4, ptr %f32_2, align 4, !tbaa !10 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i32 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%A.addr = alloca ptr, align 8
%B.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %A, ptr %A.addr, align 8, !tbaa !0
store ptr %B, ptr %B.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructA, ptr %0, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !8
%1 = load ptr, ptr %B.addr, align 8, !tbaa !0
%a = getelementptr inbounds %struct.StructB, ptr %1, i32 0, i32 1
%f32_2 = getelementptr inbounds %struct.StructA, ptr %a, i32 0, i32 3
store i32 4, ptr %f32_2, align 4, !tbaa !16
%2 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f321 = getelementptr inbounds %struct.StructA, ptr %2, i32 0, i32 1
%3 = load i32, ptr %f321, align 4, !tbaa !8
ret i32 %3
}
define i32 @_Z2g7P7StructAP7StructSy(ptr %A, ptr %S, i64 %count) #0 {
entry:
; Access to &(A->f32) and &(S->f32).
; CHECK: Function
; CHECK: NoAlias: store i32 4, ptr %f321, align 4, !tbaa !10 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i32 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%A.addr = alloca ptr, align 8
%S.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %A, ptr %A.addr, align 8, !tbaa !0
store ptr %S, ptr %S.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructA, ptr %0, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !8
%1 = load ptr, ptr %S.addr, align 8, !tbaa !0
%f321 = getelementptr inbounds %struct.StructS, ptr %1, i32 0, i32 1
store i32 4, ptr %f321, align 4, !tbaa !17
%2 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f322 = getelementptr inbounds %struct.StructA, ptr %2, i32 0, i32 1
%3 = load i32, ptr %f322, align 4, !tbaa !8
ret i32 %3
}
define i32 @_Z2g8P7StructAP7StructSy(ptr %A, ptr %S, i64 %count) #0 {
entry:
; Access to &(A->f32) and &(S->f16).
; CHECK: Function
; CHECK: NoAlias: store i16 4, ptr %1, align 2, !tbaa !10 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i16 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%A.addr = alloca ptr, align 8
%S.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %A, ptr %A.addr, align 8, !tbaa !0
store ptr %S, ptr %S.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructA, ptr %0, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !8
%1 = load ptr, ptr %S.addr, align 8, !tbaa !0
store i16 4, ptr %1, align 2, !tbaa !19
%2 = load ptr, ptr %A.addr, align 8, !tbaa !0
%f321 = getelementptr inbounds %struct.StructA, ptr %2, i32 0, i32 1
%3 = load i32, ptr %f321, align 4, !tbaa !8
ret i32 %3
}
define i32 @_Z2g9P7StructSP8StructS2y(ptr %S, ptr %S2, i64 %count) #0 {
entry:
; Access to &(S->f32) and &(S2->f32).
; CHECK: Function
; CHECK: NoAlias: store i32 4, ptr %f321, align 4, !tbaa !10 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i32 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%S.addr = alloca ptr, align 8
%S2.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %S, ptr %S.addr, align 8, !tbaa !0
store ptr %S2, ptr %S2.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %S.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructS, ptr %0, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !17
%1 = load ptr, ptr %S2.addr, align 8, !tbaa !0
%f321 = getelementptr inbounds %struct.StructS2, ptr %1, i32 0, i32 1
store i32 4, ptr %f321, align 4, !tbaa !20
%2 = load ptr, ptr %S.addr, align 8, !tbaa !0
%f322 = getelementptr inbounds %struct.StructS, ptr %2, i32 0, i32 1
%3 = load i32, ptr %f322, align 4, !tbaa !17
ret i32 %3
}
define i32 @_Z3g10P7StructSP8StructS2y(ptr %S, ptr %S2, i64 %count) #0 {
entry:
; Access to &(S->f32) and &(S2->f16).
; CHECK: Function
; CHECK: NoAlias: store i16 4, ptr %1, align 2, !tbaa !10 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i16 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%S.addr = alloca ptr, align 8
%S2.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %S, ptr %S.addr, align 8, !tbaa !0
store ptr %S2, ptr %S2.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %S.addr, align 8, !tbaa !0
%f32 = getelementptr inbounds %struct.StructS, ptr %0, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !17
%1 = load ptr, ptr %S2.addr, align 8, !tbaa !0
store i16 4, ptr %1, align 2, !tbaa !22
%2 = load ptr, ptr %S.addr, align 8, !tbaa !0
%f321 = getelementptr inbounds %struct.StructS, ptr %2, i32 0, i32 1
%3 = load i32, ptr %f321, align 4, !tbaa !17
ret i32 %3
}
define i32 @_Z3g11P7StructCP7StructDy(ptr %C, ptr %D, i64 %count) #0 {
entry:
; Access to &(C->b.a.f32) and &(D->b.a.f32).
; CHECK: Function
; CHECK: NoAlias: store i32 4, ptr %f323, align 4, !tbaa !12 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i32 4
; Remove a load and propagate the value from store.
; OPT: ret i32 1
%C.addr = alloca ptr, align 8
%D.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
store ptr %C, ptr %C.addr, align 8, !tbaa !0
store ptr %D, ptr %D.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %C.addr, align 8, !tbaa !0
%b = getelementptr inbounds %struct.StructC, ptr %0, i32 0, i32 1
%a = getelementptr inbounds %struct.StructB, ptr %b, i32 0, i32 1
%f32 = getelementptr inbounds %struct.StructA, ptr %a, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !23
%1 = load ptr, ptr %D.addr, align 8, !tbaa !0
%b1 = getelementptr inbounds %struct.StructD, ptr %1, i32 0, i32 1
%a2 = getelementptr inbounds %struct.StructB, ptr %b1, i32 0, i32 1
%f323 = getelementptr inbounds %struct.StructA, ptr %a2, i32 0, i32 1
store i32 4, ptr %f323, align 4, !tbaa !25
%2 = load ptr, ptr %C.addr, align 8, !tbaa !0
%b4 = getelementptr inbounds %struct.StructC, ptr %2, i32 0, i32 1
%a5 = getelementptr inbounds %struct.StructB, ptr %b4, i32 0, i32 1
%f326 = getelementptr inbounds %struct.StructA, ptr %a5, i32 0, i32 1
%3 = load i32, ptr %f326, align 4, !tbaa !23
ret i32 %3
}
define i32 @_Z3g12P7StructCP7StructDy(ptr %C, ptr %D, i64 %count) #0 {
entry:
; Access to &(b1->a.f32) and &(b2->a.f32).
; CHECK: Function
; CHECK: MayAlias: store i32 4, ptr %f325, align 4, !tbaa !6 <-> store i32 1, ptr %f32, align 4, !tbaa !6
; OPT: define
; OPT: store i32 1
; OPT: store i32 4
; OPT: %[[RET:.*]] = load i32, ptr
; OPT: ret i32 %[[RET]]
%C.addr = alloca ptr, align 8
%D.addr = alloca ptr, align 8
%count.addr = alloca i64, align 8
%b1 = alloca ptr, align 8
%b2 = alloca ptr, align 8
store ptr %C, ptr %C.addr, align 8, !tbaa !0
store ptr %D, ptr %D.addr, align 8, !tbaa !0
store i64 %count, ptr %count.addr, align 8, !tbaa !4
%0 = load ptr, ptr %C.addr, align 8, !tbaa !0
%b = getelementptr inbounds %struct.StructC, ptr %0, i32 0, i32 1
store ptr %b, ptr %b1, align 8, !tbaa !0
%1 = load ptr, ptr %D.addr, align 8, !tbaa !0
%b3 = getelementptr inbounds %struct.StructD, ptr %1, i32 0, i32 1
store ptr %b3, ptr %b2, align 8, !tbaa !0
%2 = load ptr, ptr %b1, align 8, !tbaa !0
%a = getelementptr inbounds %struct.StructB, ptr %2, i32 0, i32 1
%f32 = getelementptr inbounds %struct.StructA, ptr %a, i32 0, i32 1
store i32 1, ptr %f32, align 4, !tbaa !12
%3 = load ptr, ptr %b2, align 8, !tbaa !0
%a4 = getelementptr inbounds %struct.StructB, ptr %3, i32 0, i32 1
%f325 = getelementptr inbounds %struct.StructA, ptr %a4, i32 0, i32 1
store i32 4, ptr %f325, align 4, !tbaa !12
%4 = load ptr, ptr %b1, align 8, !tbaa !0
%a6 = getelementptr inbounds %struct.StructB, ptr %4, i32 0, i32 1
%f327 = getelementptr inbounds %struct.StructA, ptr %a6, i32 0, i32 1
%5 = load i32, ptr %f327, align 4, !tbaa !12
ret i32 %5
}
attributes #0 = { nounwind "less-precise-fpmad"="false" "frame-pointer"="none" "no-infs-fp-math"="false" "no-nans-fp-math"="false" "unsafe-fp-math"="false" "use-soft-float"="false" }
!0 = !{!1, !1, i64 0}
!1 = !{!"any pointer", !2}
!2 = !{!"omnipotent char", !3}
!3 = !{!"Simple C/C++ TBAA"}
!4 = !{!5, !5, i64 0}
!5 = !{!"long long", !2}
!6 = !{!7, !7, i64 0}
!7 = !{!"int", !2}
!8 = !{!9, !7, i64 4}
!9 = !{!"_ZTS7StructA", !10, i64 0, !7, i64 4, !10, i64 8, !7, i64 12}
!10 = !{!"short", !2}
!11 = !{!9, !10, i64 0}
!12 = !{!13, !7, i64 8}
!13 = !{!"_ZTS7StructB", !10, i64 0, !9, i64 4, !7, i64 20}
!14 = !{!13, !10, i64 4}
!15 = !{!13, !7, i64 20}
!16 = !{!13, !7, i64 16}
!17 = !{!18, !7, i64 4}
!18 = !{!"_ZTS7StructS", !10, i64 0, !7, i64 4}
!19 = !{!18, !10, i64 0}
!20 = !{!21, !7, i64 4}
!21 = !{!"_ZTS8StructS2", !10, i64 0, !7, i64 4}
!22 = !{!21, !10, i64 0}
!23 = !{!24, !7, i64 12}
!24 = !{!"_ZTS7StructC", !10, i64 0, !13, i64 4, !7, i64 28}
!25 = !{!26, !7, i64 12}
!26 = !{!"_ZTS7StructD", !10, i64 0, !13, i64 4, !7, i64 28, !2, i64 32}