Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / llvm / test / Analysis / BasicAA / dereferenceable.ll
blob98bd5e3d5aa6a96f60638daa332c4be9d1976a80
1 ; RUN: opt -aa-pipeline=basic-aa -print-all-alias-modref-info -passes=aa-eval < %s 2>&1 | FileCheck %s
2 ; RUN: opt -aa-pipeline=basic-aa -print-all-alias-modref-info -passes=aa-eval -use-dereferenceable-at-point-semantics=1 < %s 2>&1 | FileCheck %s
4 target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
6 @G = global i32 0, align 4
8 define i64 @global_and_deref_arg_1(ptr dereferenceable(8) %arg) nofree nosync {
9 ; CHECK:     Function: global_and_deref_arg_1: 2 pointers, 0 call sites
10 ; CHECK-NEXT:  NoAlias: i64* %arg, i32* @G
11 bb:
12   store i64 1, ptr %arg, align 8
13   store i32 0, ptr @G, align 4
14   %tmp = load i64, ptr %arg, align 8
15   ret i64 %tmp
18 define i32 @global_and_deref_arg_2(ptr dereferenceable(8) %arg) nofree nosync {
19 ; CHECK:     Function: global_and_deref_arg_2: 2 pointers, 0 call sites
20 ; CHECK-NEXT:  NoAlias: i32* %arg, i32* @G
21 bb:
22   store i32 1, ptr %arg, align 8
23   store i32 0, ptr @G, align 4
24   %tmp = load i32, ptr %arg, align 8
25   ret i32 %tmp
28 define i32 @byval_and_deref_arg_1(ptr byval(i32) %obj, ptr dereferenceable(8) %arg) nofree nosync {
29 ; CHECK:     Function: byval_and_deref_arg_1: 2 pointers, 0 call sites
30 ; CHECK-NEXT:  NoAlias: i64* %arg, i32* %obj
31 bb:
32   store i32 1, ptr %obj, align 4
33   store i64 0, ptr %arg, align 8
34   %tmp = load i32, ptr %obj, align 4
35   ret i32 %tmp
38 define i32 @byval_and_deref_arg_2(ptr byval(i32) %obj, ptr dereferenceable(8) %arg) nofree nosync {
39 ; CHECK:     Function: byval_and_deref_arg_2: 2 pointers, 0 call sites
40 ; CHECK-NEXT:  NoAlias: i32* %arg, i32* %obj
41 bb:
42   store i32 1, ptr %obj, align 4
43   store i32 0, ptr %arg, align 8
44   %tmp = load i32, ptr %obj, align 4
45   ret i32 %tmp
48 declare dereferenceable(8) ptr @get_i32_deref8()
49 declare dereferenceable(8) ptr @get_i64_deref8()
50 declare void @unknown(ptr)
52 define i32 @local_and_deref_ret_1() {
53 ; CHECK:     Function: local_and_deref_ret_1: 2 pointers, 2 call sites
54 ; CHECK-NEXT:  NoAlias: i32* %obj, i64* %ret
55 bb:
56   %obj = alloca i32
57   call void @unknown(ptr %obj)
58   %ret = call dereferenceable(8) ptr @get_i64_deref8()
59   store i32 1, ptr %obj, align 4
60   store i64 0, ptr %ret, align 8
61   %tmp = load i32, ptr %obj, align 4
62   ret i32 %tmp
65 define i32 @local_and_deref_ret_2() {
66 ; CHECK:     Function: local_and_deref_ret_2: 2 pointers, 2 call sites
67 ; CHECK-NEXT:  NoAlias: i32* %obj, i32* %ret
68 bb:
69   %obj = alloca i32
70   call void @unknown(ptr %obj)
71   %ret = call dereferenceable(8) ptr @get_i32_deref8()
72   store i32 1, ptr %obj, align 4
73   store i32 0, ptr %ret, align 8
74   %tmp = load i32, ptr %obj, align 4
75   ret i32 %tmp
79 ; Baseline tests, same as above but with 2 instead of 8 dereferenceable bytes.
81 define i64 @global_and_deref_arg_non_deref_1(ptr dereferenceable(2) %arg) nofree nosync {
82 ; CHECK:     Function: global_and_deref_arg_non_deref_1: 2 pointers, 0 call sites
83 ; CHECK-NEXT:  NoAlias: i64* %arg, i32* @G
84 bb:
85   store i64 1, ptr %arg, align 8
86   store i32 0, ptr @G, align 4
87   %tmp = load i64, ptr %arg, align 8
88   ret i64 %tmp
91 define i32 @global_and_deref_arg_non_deref_2(ptr dereferenceable(2) %arg) nofree nosync {
92 ; CHECK:     Function: global_and_deref_arg_non_deref_2: 2 pointers, 0 call sites
93 ; Different result than above (see @global_and_deref_arg_2).
94 ; CHECK-NEXT:  MayAlias:        i32* %arg, i32* @G
95 bb:
96   store i32 1, ptr %arg, align 8
97   store i32 0, ptr @G, align 4
98   %tmp = load i32, ptr %arg, align 8
99   ret i32 %tmp
102 define i32 @byval_and_deref_arg_non_deref_1(ptr byval(i32) %obj, ptr dereferenceable(2) %arg) nofree nosync {
103 ; CHECK:     Function: byval_and_deref_arg_non_deref_1: 2 pointers, 0 call sites
104 ; CHECK-NEXT:  NoAlias: i64* %arg, i32* %obj
106   store i32 1, ptr %obj, align 4
107   store i64 0, ptr %arg, align 8
108   %tmp = load i32, ptr %obj, align 4
109   ret i32 %tmp
112 define i32 @byval_and_deref_arg_non_deref_2(ptr byval(i32) %obj, ptr dereferenceable(2) %arg) nofree nosync {
113 ; CHECK:     Function: byval_and_deref_arg_non_deref_2: 2 pointers, 0 call sites
114 ; CHECK-NEXT:  NoAlias: i32* %arg, i32* %obj
116   store i32 1, ptr %obj, align 4
117   store i32 0, ptr %arg, align 8
118   %tmp = load i32, ptr %obj, align 4
119   ret i32 %tmp
122 declare dereferenceable(2) ptr @get_i32_deref2()
123 declare dereferenceable(2) ptr @get_i64_deref2()
125 define i32 @local_and_deref_ret_non_deref_1() {
126 ; CHECK:     Function: local_and_deref_ret_non_deref_1: 2 pointers, 2 call sites
127 ; CHECK-NEXT:  NoAlias: i32* %obj, i64* %ret
129   %obj = alloca i32
130   call void @unknown(ptr %obj)
131   %ret = call dereferenceable(2) ptr @get_i64_deref2()
132   store i32 1, ptr %obj, align 4
133   store i64 0, ptr %ret, align 8
134   %tmp = load i32, ptr %obj, align 4
135   ret i32 %tmp
138 define i32 @local_and_deref_ret_non_deref_2() {
139 ; CHECK:     Function: local_and_deref_ret_non_deref_2: 2 pointers, 2 call sites
140 ; Different result than above (see @local_and_deref_ret_2).
141 ; CHECK-NEXT:  MayAlias:        i32* %obj, i32* %ret
143   %obj = alloca i32
144   call void @unknown(ptr %obj)
145   %ret = call dereferenceable(2) ptr @get_i32_deref2()
146   store i32 1, ptr %obj, align 4
147   store i32 0, ptr %ret, align 8
148   %tmp = load i32, ptr %obj, align 4
149   ret i32 %tmp