[TTI] getTypeBasedIntrinsicInstrCost - add basic handling for strided load/store...
[llvm-project.git] / llvm / test / Transforms / DeadStoreElimination / batchaa-caching-new-pointers.ll
blobee9bd6912e2ae4e717da25e3c26a6cba9b1639b2
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 4
2 ; RUN: opt -S -passes=dse < %s | FileCheck %s
4 ; DSE kills `store i32 44, ptr %struct.byte.4, align 4` but should not kill
5 ; `call void @llvm.memset.p0.i64(...)`  because it has a clobber read:
6 ; `%ret = load ptr, ptr %struct.byte.8`
9 %struct.type = type { ptr, ptr }
11 define ptr @foo(ptr noundef %ptr) {
12 ; CHECK-LABEL: define ptr @foo(
13 ; CHECK-SAME: ptr noundef [[PTR:%.*]]) {
14 ; CHECK-NEXT:    [[STRUCT_ALLOCA:%.*]] = alloca [[STRUCT_TYPE:%.*]], align 8
15 ; CHECK-NEXT:    call void @llvm.lifetime.start.p0(i64 56, ptr nonnull [[STRUCT_ALLOCA]]) #[[ATTR6:[0-9]+]]
16 ; CHECK-NEXT:    [[STRUCT_BYTE_8:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_ALLOCA]], i64 8
17 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_BYTE_8]], i64 4
18 ; CHECK-NEXT:    call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 [[TMP1]], i8 42, i64 4, i1 false)
19 ; CHECK-NEXT:    store i32 43, ptr [[STRUCT_BYTE_8]], align 4
20 ; CHECK-NEXT:    [[RET:%.*]] = load ptr, ptr [[STRUCT_BYTE_8]], align 8
21 ; CHECK-NEXT:    call void @llvm.lifetime.end.p0(i64 56, ptr nonnull [[STRUCT_ALLOCA]]) #[[ATTR6]]
22 ; CHECK-NEXT:    ret ptr [[RET]]
24   %struct.alloca = alloca %struct.type, align 8
25   call void @llvm.lifetime.start.p0(i64 56, ptr nonnull %struct.alloca) nounwind
26   %struct.byte.8 = getelementptr inbounds i8, ptr %struct.alloca, i64 8
27   ; Set %struct.alloca[8, 16) to 42.
28   call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 %struct.byte.8, i8 42, i64 8, i1 false)
29   ; Set %struct.alloca[8, 12) to 43.
30   store i32 43, ptr %struct.byte.8, align 4
31   ; Set %struct.alloca[4, 8) to 44.
32   %struct.byte.4 = getelementptr inbounds i8, ptr %struct.alloca, i64 4
33   store i32 44, ptr %struct.byte.4, align 4
34   ; Return %struct.alloca[8, 16).
35   %ret = load ptr, ptr %struct.byte.8
36   call void @llvm.lifetime.end.p0(i64 56, ptr nonnull %struct.alloca) nounwind
37   ret ptr %ret
40 ; Set of tests based on @foo, but where the memset's operands cannot be erased
41 ; due to other uses. Instead, they contain a number of removable MemoryDefs;
42 ; with non-void types result types.
44 define ptr @foo_with_removable_malloc() {
45 ; CHECK-LABEL: define ptr @foo_with_removable_malloc() {
46 ; CHECK-NEXT:    [[STRUCT_ALLOCA:%.*]] = alloca [[STRUCT_TYPE:%.*]], align 8
47 ; CHECK-NEXT:    call void @llvm.lifetime.start.p0(i64 56, ptr nonnull [[STRUCT_ALLOCA]]) #[[ATTR6]]
48 ; CHECK-NEXT:    [[STRUCT_BYTE_4:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_ALLOCA]], i64 4
49 ; CHECK-NEXT:    [[STRUCT_BYTE_8:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_ALLOCA]], i64 8
50 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_BYTE_8]], i64 4
51 ; CHECK-NEXT:    call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 [[TMP1]], i8 42, i64 4, i1 false)
52 ; CHECK-NEXT:    store i32 43, ptr [[STRUCT_BYTE_8]], align 4
53 ; CHECK-NEXT:    [[RET:%.*]] = load ptr, ptr [[STRUCT_BYTE_8]], align 8
54 ; CHECK-NEXT:    call void @readnone(ptr [[STRUCT_BYTE_4]])
55 ; CHECK-NEXT:    call void @readnone(ptr [[STRUCT_BYTE_8]])
56 ; CHECK-NEXT:    call void @llvm.lifetime.end.p0(i64 56, ptr nonnull [[STRUCT_ALLOCA]]) #[[ATTR6]]
57 ; CHECK-NEXT:    ret ptr [[RET]]
59   %struct.alloca = alloca %struct.type, align 8
60   call void @llvm.lifetime.start.p0(i64 56, ptr nonnull %struct.alloca) nounwind
61   %struct.byte.4 = getelementptr inbounds i8, ptr %struct.alloca, i64 4
62   %struct.byte.8 = getelementptr inbounds i8, ptr %struct.alloca, i64 8
64   ; Set of removable memory deffs
65   %m2 = tail call ptr @malloc(i64 4)
66   %m1 = tail call ptr @malloc(i64 4)
67   store i32 0, ptr %struct.byte.8
68   store i32 0, ptr %struct.byte.8
69   store i32 123, ptr %m1
70   store i32 123, ptr %m2
72   ; Set %struct.alloca[8, 16) to 42.
73   call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 %struct.byte.8, i8 42, i64 8, i1 false)
74   ; Set %struct.alloca[8, 12) to 43.
75   store i32 43, ptr %struct.byte.8, align 4
76   ; Set %struct.alloca[4, 8) to 44.
77   store i32 44, ptr %struct.byte.4, align 4
78   ; Return %struct.alloca[8, 16).
79   %ret = load ptr, ptr %struct.byte.8
80   call void @readnone(ptr %struct.byte.4);
81   call void @readnone(ptr %struct.byte.8);
82   call void @llvm.lifetime.end.p0(i64 56, ptr nonnull %struct.alloca) nounwind
83   ret ptr %ret
86 define ptr @foo_with_removable_malloc_free() {
87 ; CHECK-LABEL: define ptr @foo_with_removable_malloc_free() {
88 ; CHECK-NEXT:    [[STRUCT_ALLOCA:%.*]] = alloca [[STRUCT_TYPE:%.*]], align 8
89 ; CHECK-NEXT:    [[M1:%.*]] = tail call ptr @malloc(i64 4)
90 ; CHECK-NEXT:    call void @llvm.lifetime.start.p0(i64 56, ptr nonnull [[STRUCT_ALLOCA]]) #[[ATTR6]]
91 ; CHECK-NEXT:    [[STRUCT_BYTE_4:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_ALLOCA]], i64 4
92 ; CHECK-NEXT:    [[STRUCT_BYTE_8:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_ALLOCA]], i64 8
93 ; CHECK-NEXT:    [[M2:%.*]] = tail call ptr @malloc(i64 4)
94 ; CHECK-NEXT:    call void @free(ptr [[M1]])
95 ; CHECK-NEXT:    call void @free(ptr [[M2]])
96 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_BYTE_8]], i64 4
97 ; CHECK-NEXT:    call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 [[TMP1]], i8 42, i64 4, i1 false)
98 ; CHECK-NEXT:    store i32 43, ptr [[STRUCT_BYTE_8]], align 4
99 ; CHECK-NEXT:    [[RET:%.*]] = load ptr, ptr [[STRUCT_BYTE_8]], align 8
100 ; CHECK-NEXT:    call void @readnone(ptr [[STRUCT_BYTE_4]])
101 ; CHECK-NEXT:    call void @readnone(ptr [[STRUCT_BYTE_8]])
102 ; CHECK-NEXT:    call void @llvm.lifetime.end.p0(i64 56, ptr nonnull [[STRUCT_ALLOCA]]) #[[ATTR6]]
103 ; CHECK-NEXT:    ret ptr [[RET]]
105   %struct.alloca = alloca %struct.type, align 8
106   %m1 = tail call ptr @malloc(i64 4)
107   call void @llvm.lifetime.start.p0(i64 56, ptr nonnull %struct.alloca) nounwind
108   %struct.byte.4 = getelementptr inbounds i8, ptr %struct.alloca, i64 4
109   %struct.byte.8 = getelementptr inbounds i8, ptr %struct.alloca, i64 8
111   store i32 0, ptr %struct.byte.4
112   store i32 0, ptr %struct.byte.8
113   %m2 = tail call ptr @malloc(i64 4)
114   store i32 123, ptr %m1
115   call void @free(ptr %m1);
116   store i32 123, ptr %m2
117   call void @free(ptr %m2);
119   ; Set %struct.alloca[8, 16) to 42.
120   call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 %struct.byte.8, i8 42, i64 8, i1 false)
121   ; Set %struct.alloca[8, 12) to 43.
122   store i32 43, ptr %struct.byte.8, align 4
123   ; Set %struct.alloca[4, 8) to 44.
124   store i32 44, ptr %struct.byte.4, align 4
125   ; Return %struct.alloca[8, 16).
126   %ret = load ptr, ptr %struct.byte.8
127   call void @readnone(ptr %struct.byte.4);
128   call void @readnone(ptr %struct.byte.8);
129   call void @llvm.lifetime.end.p0(i64 56, ptr nonnull %struct.alloca) nounwind
130   ret ptr %ret
133 define ptr @foo_with_malloc_to_calloc() {
134 ; CHECK-LABEL: define ptr @foo_with_malloc_to_calloc() {
135 ; CHECK-NEXT:    [[STRUCT_ALLOCA:%.*]] = alloca [[STRUCT_TYPE:%.*]], align 8
136 ; CHECK-NEXT:    call void @llvm.lifetime.start.p0(i64 56, ptr nonnull [[STRUCT_ALLOCA]]) #[[ATTR6]]
137 ; CHECK-NEXT:    [[STRUCT_BYTE_8:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_ALLOCA]], i64 8
138 ; CHECK-NEXT:    [[STRUCT_BYTE_4:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_ALLOCA]], i64 4
139 ; CHECK-NEXT:    [[CALLOC1:%.*]] = call ptr @calloc(i64 1, i64 4)
140 ; CHECK-NEXT:    [[CALLOC:%.*]] = call ptr @calloc(i64 1, i64 4)
141 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[STRUCT_BYTE_8]], i64 4
142 ; CHECK-NEXT:    call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 [[TMP1]], i8 42, i64 4, i1 false)
143 ; CHECK-NEXT:    store i32 43, ptr [[STRUCT_BYTE_8]], align 4
144 ; CHECK-NEXT:    [[RET:%.*]] = load ptr, ptr [[STRUCT_BYTE_8]], align 8
145 ; CHECK-NEXT:    call void @readnone(ptr [[STRUCT_BYTE_4]])
146 ; CHECK-NEXT:    call void @readnone(ptr [[STRUCT_BYTE_8]])
147 ; CHECK-NEXT:    call void @llvm.lifetime.end.p0(i64 56, ptr nonnull [[STRUCT_ALLOCA]]) #[[ATTR6]]
148 ; CHECK-NEXT:    call void @use(ptr [[CALLOC1]])
149 ; CHECK-NEXT:    call void @use(ptr [[CALLOC]])
150 ; CHECK-NEXT:    ret ptr [[RET]]
152   %struct.alloca = alloca %struct.type, align 8
153   call void @llvm.lifetime.start.p0(i64 56, ptr nonnull %struct.alloca) nounwind
154   %struct.byte.8 = getelementptr inbounds i8, ptr %struct.alloca, i64 8
155   %struct.byte.4 = getelementptr inbounds i8, ptr %struct.alloca, i64 4
157   ; Set of removable memory deffs
158   %m1 = tail call ptr @malloc(i64 4)
159   %m2 = tail call ptr @malloc(i64 4)
160   store i32 0, ptr %struct.byte.4
161   store i32 0, ptr %struct.byte.8
162   call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 %m2, i8 0, i64 4, i1 false)
163   call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 %m1, i8 0, i64 4, i1 false)
165   ; Set %struct.alloca[8, 16) to 42.
166   call void @llvm.memset.p0.i64(ptr noundef nonnull align 4 %struct.byte.8, i8 42, i64 8, i1 false)
167   ; Set %struct.alloca[8, 12) to 43.
168   store i32 43, ptr %struct.byte.8, align 4
169   ; Set %struct.alloca[4, 8) to 44.
170   store i32 44, ptr %struct.byte.4, align 4
171   ; Return %struct.alloca[8, 16).
172   %ret = load ptr, ptr %struct.byte.8
173   call void @readnone(ptr %struct.byte.4);
174   call void @readnone(ptr %struct.byte.8);
175   call void @llvm.lifetime.end.p0(i64 56, ptr nonnull %struct.alloca) nounwind
176   call void @use(ptr %m1)
177   call void @use(ptr %m2)
178   ret ptr %ret
181 declare void @llvm.memset.p0.i64(ptr nocapture writeonly, i8, i64, i1 immarg)
182 declare void @llvm.lifetime.end.p0(i64 immarg, ptr nocapture)
183 declare void @llvm.lifetime.start.p0(i64 immarg, ptr nocapture)
185 declare noalias ptr @malloc(i64) willreturn allockind("alloc,uninitialized") "alloc-family"="malloc"
186 declare void @readnone(ptr) readnone nounwind
187 declare void @free(ptr nocapture) allockind("free") "alloc-family"="malloc"
189 declare void @use(ptr)