Revert "Warn when unique objects might be duplicated in shared libraries (#117622)"
[llvm-project.git] / llvm / test / Transforms / LoopVectorize / load-deref-pred-align.ll
blobcbc483fabc1844ece2607699c6d6fe2b87208dca
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt -passes=loop-vectorize -force-vector-width=2 -force-vector-interleave=1 -S %s | FileCheck %s
4 target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128-ni:1"
6 declare void @init(ptr nocapture nofree)
8 ; Test case where the predicated load in the loop has an access size of 2 but
9 ; has an alignment of 4.
10 define i16 @test_access_size_not_multiple_of_align(i64 %len, ptr %test_base) {
11 ; CHECK-LABEL: @test_access_size_not_multiple_of_align(
12 ; CHECK-NEXT:  entry:
13 ; CHECK-NEXT:    [[ALLOCA:%.*]] = alloca [163840 x i16], align 4
14 ; CHECK-NEXT:    call void @init(ptr [[ALLOCA]])
15 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
16 ; CHECK:       vector.ph:
17 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
18 ; CHECK:       vector.body:
19 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[PRED_LOAD_CONTINUE2:%.*]] ]
20 ; CHECK-NEXT:    [[VEC_PHI:%.*]] = phi <2 x i16> [ zeroinitializer, [[VECTOR_PH]] ], [ [[TMP15:%.*]], [[PRED_LOAD_CONTINUE2]] ]
21 ; CHECK-NEXT:    [[TMP0:%.*]] = add i64 [[INDEX]], 0
22 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[TEST_BASE:%.*]], i64 [[TMP0]]
23 ; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i8, ptr [[TMP1]], i32 0
24 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <2 x i8>, ptr [[TMP2]], align 1
25 ; CHECK-NEXT:    [[TMP3:%.*]] = icmp sge <2 x i8> [[WIDE_LOAD]], zeroinitializer
26 ; CHECK-NEXT:    [[TMP4:%.*]] = extractelement <2 x i1> [[TMP3]], i32 0
27 ; CHECK-NEXT:    br i1 [[TMP4]], label [[PRED_LOAD_IF:%.*]], label [[PRED_LOAD_CONTINUE:%.*]]
28 ; CHECK:       pred.load.if:
29 ; CHECK-NEXT:    [[TMP5:%.*]] = getelementptr inbounds i16, ptr [[ALLOCA]], i64 [[TMP0]]
30 ; CHECK-NEXT:    [[TMP6:%.*]] = load i16, ptr [[TMP5]], align 4
31 ; CHECK-NEXT:    [[TMP7:%.*]] = insertelement <2 x i16> poison, i16 [[TMP6]], i32 0
32 ; CHECK-NEXT:    br label [[PRED_LOAD_CONTINUE]]
33 ; CHECK:       pred.load.continue:
34 ; CHECK-NEXT:    [[TMP8:%.*]] = phi <2 x i16> [ poison, [[VECTOR_BODY]] ], [ [[TMP7]], [[PRED_LOAD_IF]] ]
35 ; CHECK-NEXT:    [[TMP9:%.*]] = extractelement <2 x i1> [[TMP3]], i32 1
36 ; CHECK-NEXT:    br i1 [[TMP9]], label [[PRED_LOAD_IF1:%.*]], label [[PRED_LOAD_CONTINUE2]]
37 ; CHECK:       pred.load.if1:
38 ; CHECK-NEXT:    [[TMP10:%.*]] = add i64 [[INDEX]], 1
39 ; CHECK-NEXT:    [[TMP11:%.*]] = getelementptr inbounds i16, ptr [[ALLOCA]], i64 [[TMP10]]
40 ; CHECK-NEXT:    [[TMP12:%.*]] = load i16, ptr [[TMP11]], align 4
41 ; CHECK-NEXT:    [[TMP13:%.*]] = insertelement <2 x i16> [[TMP8]], i16 [[TMP12]], i32 1
42 ; CHECK-NEXT:    br label [[PRED_LOAD_CONTINUE2]]
43 ; CHECK:       pred.load.continue2:
44 ; CHECK-NEXT:    [[TMP14:%.*]] = phi <2 x i16> [ [[TMP8]], [[PRED_LOAD_CONTINUE]] ], [ [[TMP13]], [[PRED_LOAD_IF1]] ]
45 ; CHECK-NEXT:    [[PREDPHI:%.*]] = select <2 x i1> [[TMP3]], <2 x i16> [[TMP14]], <2 x i16> zeroinitializer
46 ; CHECK-NEXT:    [[TMP15]] = add <2 x i16> [[VEC_PHI]], [[PREDPHI]]
47 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 2
48 ; CHECK-NEXT:    [[TMP16:%.*]] = icmp eq i64 [[INDEX_NEXT]], 4096
49 ; CHECK-NEXT:    br i1 [[TMP16]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP0:![0-9]+]]
50 ; CHECK:       middle.block:
51 ; CHECK-NEXT:    [[TMP17:%.*]] = call i16 @llvm.vector.reduce.add.v2i16(<2 x i16> [[TMP15]])
52 ; CHECK-NEXT:    br i1 true, label [[LOOP_EXIT:%.*]], label [[SCALAR_PH]]
53 ; CHECK:       scalar.ph:
54 ; CHECK-NEXT:    [[BC_RESUME_VAL:%.*]] = phi i64 [ 4096, [[MIDDLE_BLOCK]] ], [ 0, [[ENTRY:%.*]] ]
55 ; CHECK-NEXT:    [[BC_MERGE_RDX:%.*]] = phi i16 [ [[TMP17]], [[MIDDLE_BLOCK]] ], [ 0, [[ENTRY]] ]
56 ; CHECK-NEXT:    br label [[LOOP:%.*]]
57 ; CHECK:       loop:
58 ; CHECK-NEXT:    [[IV:%.*]] = phi i64 [ [[BC_RESUME_VAL]], [[SCALAR_PH]] ], [ [[IV_NEXT:%.*]], [[LATCH:%.*]] ]
59 ; CHECK-NEXT:    [[ACCUM:%.*]] = phi i16 [ [[BC_MERGE_RDX]], [[SCALAR_PH]] ], [ [[ACCUM_NEXT:%.*]], [[LATCH]] ]
60 ; CHECK-NEXT:    [[IV_NEXT]] = add i64 [[IV]], 1
61 ; CHECK-NEXT:    [[TEST_ADDR:%.*]] = getelementptr inbounds i8, ptr [[TEST_BASE]], i64 [[IV]]
62 ; CHECK-NEXT:    [[L_T:%.*]] = load i8, ptr [[TEST_ADDR]], align 1
63 ; CHECK-NEXT:    [[CMP:%.*]] = icmp sge i8 [[L_T]], 0
64 ; CHECK-NEXT:    br i1 [[CMP]], label [[PRED:%.*]], label [[LATCH]]
65 ; CHECK:       pred:
66 ; CHECK-NEXT:    [[ADDR:%.*]] = getelementptr inbounds i16, ptr [[ALLOCA]], i64 [[IV]]
67 ; CHECK-NEXT:    [[VAL:%.*]] = load i16, ptr [[ADDR]], align 4
68 ; CHECK-NEXT:    br label [[LATCH]]
69 ; CHECK:       latch:
70 ; CHECK-NEXT:    [[VAL_PHI:%.*]] = phi i16 [ 0, [[LOOP]] ], [ [[VAL]], [[PRED]] ]
71 ; CHECK-NEXT:    [[ACCUM_NEXT]] = add i16 [[ACCUM]], [[VAL_PHI]]
72 ; CHECK-NEXT:    [[EXIT:%.*]] = icmp eq i64 [[IV]], 4095
73 ; CHECK-NEXT:    br i1 [[EXIT]], label [[LOOP_EXIT]], label [[LOOP]], !llvm.loop [[LOOP3:![0-9]+]]
74 ; CHECK:       loop_exit:
75 ; CHECK-NEXT:    [[ACCUM_NEXT_LCSSA:%.*]] = phi i16 [ [[ACCUM_NEXT]], [[LATCH]] ], [ [[TMP17]], [[MIDDLE_BLOCK]] ]
76 ; CHECK-NEXT:    ret i16 [[ACCUM_NEXT_LCSSA]]
78 entry:
79   %alloca = alloca [163840 x i16], align 4
80   call void @init(ptr %alloca)
81   br label %loop
82 loop:
83   %iv = phi i64 [ 0, %entry ], [ %iv.next, %latch ]
84   %accum = phi i16 [ 0, %entry ], [ %accum.next, %latch ]
85   %iv.next = add i64 %iv, 1
86   %test_addr = getelementptr inbounds i8, ptr %test_base, i64 %iv
87   %l.t = load i8, ptr %test_addr
88   %cmp = icmp sge i8 %l.t, 0
89   br i1 %cmp, label %pred, label %latch
90 pred:
91   %addr = getelementptr inbounds i16, ptr %alloca, i64 %iv
92   %val = load i16, ptr %addr, align 4
93   br label %latch
94 latch:
95   %val.phi = phi i16 [0, %loop], [%val, %pred]
96   %accum.next = add i16 %accum, %val.phi
97   %exit = icmp eq i64 %iv, 4095
98   br i1 %exit, label %loop_exit, label %loop
100 loop_exit:
101   ret i16 %accum.next
104 ; Test case where the predicated load in the loop has an access size of 4 and
105 ; an alignment of 4, but the start pointer is offset by 1.
106 define i32 @test_access_size_multiple_of_align_but_offset_by_1(i64 %len, ptr %test_base) {
107 ; CHECK-LABEL: @test_access_size_multiple_of_align_but_offset_by_1(
108 ; CHECK-NEXT:  entry:
109 ; CHECK-NEXT:    [[ALLOCA:%.*]] = alloca [163840 x i32], align 4
110 ; CHECK-NEXT:    call void @init(ptr [[ALLOCA]])
111 ; CHECK-NEXT:    [[START:%.*]] = getelementptr i8, ptr [[ALLOCA]], i64 2
112 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
113 ; CHECK:       vector.ph:
114 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
115 ; CHECK:       vector.body:
116 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[PRED_LOAD_CONTINUE2:%.*]] ]
117 ; CHECK-NEXT:    [[VEC_PHI:%.*]] = phi <2 x i32> [ zeroinitializer, [[VECTOR_PH]] ], [ [[TMP15:%.*]], [[PRED_LOAD_CONTINUE2]] ]
118 ; CHECK-NEXT:    [[TMP0:%.*]] = add i64 [[INDEX]], 0
119 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[TEST_BASE:%.*]], i64 [[TMP0]]
120 ; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i8, ptr [[TMP1]], i32 0
121 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <2 x i8>, ptr [[TMP2]], align 1
122 ; CHECK-NEXT:    [[TMP3:%.*]] = icmp sge <2 x i8> [[WIDE_LOAD]], zeroinitializer
123 ; CHECK-NEXT:    [[TMP4:%.*]] = extractelement <2 x i1> [[TMP3]], i32 0
124 ; CHECK-NEXT:    br i1 [[TMP4]], label [[PRED_LOAD_IF:%.*]], label [[PRED_LOAD_CONTINUE:%.*]]
125 ; CHECK:       pred.load.if:
126 ; CHECK-NEXT:    [[TMP5:%.*]] = getelementptr inbounds i32, ptr [[START]], i64 [[TMP0]]
127 ; CHECK-NEXT:    [[TMP6:%.*]] = load i32, ptr [[TMP5]], align 4
128 ; CHECK-NEXT:    [[TMP7:%.*]] = insertelement <2 x i32> poison, i32 [[TMP6]], i32 0
129 ; CHECK-NEXT:    br label [[PRED_LOAD_CONTINUE]]
130 ; CHECK:       pred.load.continue:
131 ; CHECK-NEXT:    [[TMP8:%.*]] = phi <2 x i32> [ poison, [[VECTOR_BODY]] ], [ [[TMP7]], [[PRED_LOAD_IF]] ]
132 ; CHECK-NEXT:    [[TMP9:%.*]] = extractelement <2 x i1> [[TMP3]], i32 1
133 ; CHECK-NEXT:    br i1 [[TMP9]], label [[PRED_LOAD_IF1:%.*]], label [[PRED_LOAD_CONTINUE2]]
134 ; CHECK:       pred.load.if1:
135 ; CHECK-NEXT:    [[TMP10:%.*]] = add i64 [[INDEX]], 1
136 ; CHECK-NEXT:    [[TMP11:%.*]] = getelementptr inbounds i32, ptr [[START]], i64 [[TMP10]]
137 ; CHECK-NEXT:    [[TMP12:%.*]] = load i32, ptr [[TMP11]], align 4
138 ; CHECK-NEXT:    [[TMP13:%.*]] = insertelement <2 x i32> [[TMP8]], i32 [[TMP12]], i32 1
139 ; CHECK-NEXT:    br label [[PRED_LOAD_CONTINUE2]]
140 ; CHECK:       pred.load.continue2:
141 ; CHECK-NEXT:    [[TMP14:%.*]] = phi <2 x i32> [ [[TMP8]], [[PRED_LOAD_CONTINUE]] ], [ [[TMP13]], [[PRED_LOAD_IF1]] ]
142 ; CHECK-NEXT:    [[PREDPHI:%.*]] = select <2 x i1> [[TMP3]], <2 x i32> [[TMP14]], <2 x i32> zeroinitializer
143 ; CHECK-NEXT:    [[TMP15]] = add <2 x i32> [[VEC_PHI]], [[PREDPHI]]
144 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 2
145 ; CHECK-NEXT:    [[TMP16:%.*]] = icmp eq i64 [[INDEX_NEXT]], 4096
146 ; CHECK-NEXT:    br i1 [[TMP16]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP4:![0-9]+]]
147 ; CHECK:       middle.block:
148 ; CHECK-NEXT:    [[TMP17:%.*]] = call i32 @llvm.vector.reduce.add.v2i32(<2 x i32> [[TMP15]])
149 ; CHECK-NEXT:    br i1 true, label [[LOOP_EXIT:%.*]], label [[SCALAR_PH]]
150 ; CHECK:       scalar.ph:
151 ; CHECK-NEXT:    [[BC_RESUME_VAL:%.*]] = phi i64 [ 4096, [[MIDDLE_BLOCK]] ], [ 0, [[ENTRY:%.*]] ]
152 ; CHECK-NEXT:    [[BC_MERGE_RDX:%.*]] = phi i32 [ [[TMP17]], [[MIDDLE_BLOCK]] ], [ 0, [[ENTRY]] ]
153 ; CHECK-NEXT:    br label [[LOOP:%.*]]
154 ; CHECK:       loop:
155 ; CHECK-NEXT:    [[IV:%.*]] = phi i64 [ [[BC_RESUME_VAL]], [[SCALAR_PH]] ], [ [[IV_NEXT:%.*]], [[LATCH:%.*]] ]
156 ; CHECK-NEXT:    [[ACCUM:%.*]] = phi i32 [ [[BC_MERGE_RDX]], [[SCALAR_PH]] ], [ [[ACCUM_NEXT:%.*]], [[LATCH]] ]
157 ; CHECK-NEXT:    [[IV_NEXT]] = add i64 [[IV]], 1
158 ; CHECK-NEXT:    [[TEST_ADDR:%.*]] = getelementptr inbounds i8, ptr [[TEST_BASE]], i64 [[IV]]
159 ; CHECK-NEXT:    [[L_T:%.*]] = load i8, ptr [[TEST_ADDR]], align 1
160 ; CHECK-NEXT:    [[CMP:%.*]] = icmp sge i8 [[L_T]], 0
161 ; CHECK-NEXT:    br i1 [[CMP]], label [[PRED:%.*]], label [[LATCH]]
162 ; CHECK:       pred:
163 ; CHECK-NEXT:    [[ADDR:%.*]] = getelementptr inbounds i32, ptr [[START]], i64 [[IV]]
164 ; CHECK-NEXT:    [[VAL:%.*]] = load i32, ptr [[ADDR]], align 4
165 ; CHECK-NEXT:    br label [[LATCH]]
166 ; CHECK:       latch:
167 ; CHECK-NEXT:    [[VAL_PHI:%.*]] = phi i32 [ 0, [[LOOP]] ], [ [[VAL]], [[PRED]] ]
168 ; CHECK-NEXT:    [[ACCUM_NEXT]] = add i32 [[ACCUM]], [[VAL_PHI]]
169 ; CHECK-NEXT:    [[EXIT:%.*]] = icmp eq i64 [[IV]], 4095
170 ; CHECK-NEXT:    br i1 [[EXIT]], label [[LOOP_EXIT]], label [[LOOP]], !llvm.loop [[LOOP5:![0-9]+]]
171 ; CHECK:       loop_exit:
172 ; CHECK-NEXT:    [[ACCUM_NEXT_LCSSA:%.*]] = phi i32 [ [[ACCUM_NEXT]], [[LATCH]] ], [ [[TMP17]], [[MIDDLE_BLOCK]] ]
173 ; CHECK-NEXT:    ret i32 [[ACCUM_NEXT_LCSSA]]
175 entry:
176   %alloca = alloca [163840 x i32], align 4
177   call void @init(ptr %alloca)
178   %start = getelementptr i8, ptr %alloca, i64 2
179   br label %loop
180 loop:
181   %iv = phi i64 [ 0, %entry ], [ %iv.next, %latch ]
182   %accum = phi i32 [ 0, %entry ], [ %accum.next, %latch ]
183   %iv.next = add i64 %iv, 1
184   %test_addr = getelementptr inbounds i8, ptr %test_base, i64 %iv
185   %l.t = load i8, ptr %test_addr
186   %cmp = icmp sge i8 %l.t, 0
187   br i1 %cmp, label %pred, label %latch
188 pred:
189   %addr = getelementptr inbounds i32, ptr %start, i64 %iv
190   %val = load i32, ptr %addr, align 4
191   br label %latch
192 latch:
193   %val.phi = phi i32 [0, %loop], [%val, %pred]
194   %accum.next = add i32 %accum, %val.phi
195   %exit = icmp eq i64 %iv, 4095
196   br i1 %exit, label %loop_exit, label %loop
198 loop_exit:
199   ret i32 %accum.next
203 define i32 @loop_requires_scev_predicate(ptr %dest, i32 %end) {
204 ; CHECK-LABEL: @loop_requires_scev_predicate(
205 ; CHECK-NEXT:  entry:
206 ; CHECK-NEXT:    [[P1:%.*]] = alloca [1024 x i32], align 4
207 ; CHECK-NEXT:    [[P2:%.*]] = alloca [1024 x i32], align 4
208 ; CHECK-NEXT:    call void @init(ptr [[P1]])
209 ; CHECK-NEXT:    call void @init(ptr [[P2]])
210 ; CHECK-NEXT:    [[END_CLAMPED:%.*]] = and i32 [[END:%.*]], 1023
211 ; CHECK-NEXT:    [[TMP0:%.*]] = trunc i32 [[END]] to i10
212 ; CHECK-NEXT:    [[TMP1:%.*]] = zext i10 [[TMP0]] to i64
213 ; CHECK-NEXT:    [[UMAX1:%.*]] = call i64 @llvm.umax.i64(i64 [[TMP1]], i64 1)
214 ; CHECK-NEXT:    [[MIN_ITERS_CHECK:%.*]] = icmp ult i64 [[UMAX1]], 2
215 ; CHECK-NEXT:    br i1 [[MIN_ITERS_CHECK]], label [[SCALAR_PH:%.*]], label [[VECTOR_SCEVCHECK:%.*]]
216 ; CHECK:       vector.scevcheck:
217 ; CHECK-NEXT:    [[UMAX:%.*]] = call i32 @llvm.umax.i32(i32 [[END_CLAMPED]], i32 1)
218 ; CHECK-NEXT:    [[TMP2:%.*]] = add nsw i32 [[UMAX]], -1
219 ; CHECK-NEXT:    [[TMP3:%.*]] = trunc i32 [[TMP2]] to i8
220 ; CHECK-NEXT:    [[TMP4:%.*]] = add i8 1, [[TMP3]]
221 ; CHECK-NEXT:    [[TMP5:%.*]] = icmp ult i8 [[TMP4]], 1
222 ; CHECK-NEXT:    [[TMP6:%.*]] = icmp ugt i32 [[TMP2]], 255
223 ; CHECK-NEXT:    [[TMP7:%.*]] = or i1 [[TMP5]], [[TMP6]]
224 ; CHECK-NEXT:    br i1 [[TMP7]], label [[SCALAR_PH]], label [[VECTOR_PH:%.*]]
225 ; CHECK:       vector.ph:
226 ; CHECK-NEXT:    [[N_MOD_VF:%.*]] = urem i64 [[UMAX1]], 2
227 ; CHECK-NEXT:    [[N_VEC:%.*]] = sub i64 [[UMAX1]], [[N_MOD_VF]]
228 ; CHECK-NEXT:    [[IND_END:%.*]] = trunc i64 [[N_VEC]] to i8
229 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
230 ; CHECK:       vector.body:
231 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[PRED_STORE_CONTINUE4:%.*]] ]
232 ; CHECK-NEXT:    [[TMP8:%.*]] = add i64 [[INDEX]], 0
233 ; CHECK-NEXT:    [[TMP9:%.*]] = getelementptr inbounds i32, ptr [[P1]], i64 [[TMP8]]
234 ; CHECK-NEXT:    [[TMP10:%.*]] = getelementptr inbounds i32, ptr [[TMP9]], i32 0
235 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <2 x i32>, ptr [[TMP10]], align 4
236 ; CHECK-NEXT:    [[TMP11:%.*]] = icmp ne <2 x i32> [[WIDE_LOAD]], zeroinitializer
237 ; CHECK-NEXT:    [[TMP12:%.*]] = getelementptr i32, ptr [[P2]], i64 [[TMP8]]
238 ; CHECK-NEXT:    [[TMP13:%.*]] = getelementptr i32, ptr [[TMP12]], i32 0
239 ; CHECK-NEXT:    [[WIDE_LOAD3:%.*]] = load <2 x i32>, ptr [[TMP13]], align 4
240 ; CHECK-NEXT:    [[TMP14:%.*]] = extractelement <2 x i1> [[TMP11]], i32 0
241 ; CHECK-NEXT:    br i1 [[TMP14]], label [[PRED_STORE_IF:%.*]], label [[PRED_STORE_CONTINUE:%.*]]
242 ; CHECK:       pred.store.if:
243 ; CHECK-NEXT:    [[TMP15:%.*]] = getelementptr inbounds i32, ptr [[DEST:%.*]], i64 [[TMP8]]
244 ; CHECK-NEXT:    [[TMP16:%.*]] = extractelement <2 x i32> [[WIDE_LOAD]], i32 0
245 ; CHECK-NEXT:    [[TMP17:%.*]] = extractelement <2 x i32> [[WIDE_LOAD3]], i32 0
246 ; CHECK-NEXT:    [[TMP18:%.*]] = add i32 [[TMP16]], [[TMP17]]
247 ; CHECK-NEXT:    store i32 [[TMP18]], ptr [[TMP15]], align 4
248 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE]]
249 ; CHECK:       pred.store.continue:
250 ; CHECK-NEXT:    [[TMP19:%.*]] = extractelement <2 x i1> [[TMP11]], i32 1
251 ; CHECK-NEXT:    br i1 [[TMP19]], label [[PRED_STORE_IF3:%.*]], label [[PRED_STORE_CONTINUE4]]
252 ; CHECK:       pred.store.if3:
253 ; CHECK-NEXT:    [[TMP20:%.*]] = add i64 [[INDEX]], 1
254 ; CHECK-NEXT:    [[TMP21:%.*]] = getelementptr inbounds i32, ptr [[DEST]], i64 [[TMP20]]
255 ; CHECK-NEXT:    [[TMP22:%.*]] = extractelement <2 x i32> [[WIDE_LOAD]], i32 1
256 ; CHECK-NEXT:    [[TMP23:%.*]] = extractelement <2 x i32> [[WIDE_LOAD3]], i32 1
257 ; CHECK-NEXT:    [[TMP24:%.*]] = add i32 [[TMP22]], [[TMP23]]
258 ; CHECK-NEXT:    store i32 [[TMP24]], ptr [[TMP21]], align 4
259 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE4]]
260 ; CHECK:       pred.store.continue4:
261 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 2
262 ; CHECK-NEXT:    [[TMP25:%.*]] = icmp eq i64 [[INDEX_NEXT]], [[N_VEC]]
263 ; CHECK-NEXT:    br i1 [[TMP25]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP8:![0-9]+]]
264 ; CHECK:       middle.block:
265 ; CHECK-NEXT:    [[CMP_N:%.*]] = icmp eq i64 [[UMAX1]], [[N_VEC]]
266 ; CHECK-NEXT:    br i1 [[CMP_N]], label [[EXIT:%.*]], label [[SCALAR_PH]]
267 ; CHECK:       scalar.ph:
268 ; CHECK-NEXT:    [[BC_RESUME_VAL:%.*]] = phi i8 [ [[IND_END]], [[MIDDLE_BLOCK]] ], [ 0, [[VECTOR_SCEVCHECK]] ], [ 0, [[ENTRY:%.*]] ]
269 ; CHECK-NEXT:    [[BC_RESUME_VAL2:%.*]] = phi i64 [ [[N_VEC]], [[MIDDLE_BLOCK]] ], [ 0, [[VECTOR_SCEVCHECK]] ], [ 0, [[ENTRY]] ]
270 ; CHECK-NEXT:    br label [[FOR_BODY:%.*]]
271 ; CHECK:       for.body:
272 ; CHECK-NEXT:    [[IND:%.*]] = phi i8 [ [[IND_NEXT:%.*]], [[FOR_INC:%.*]] ], [ [[BC_RESUME_VAL]], [[SCALAR_PH]] ]
273 ; CHECK-NEXT:    [[GEP_IND:%.*]] = phi i64 [ [[GEP_IND_NEXT:%.*]], [[FOR_INC]] ], [ [[BC_RESUME_VAL2]], [[SCALAR_PH]] ]
274 ; CHECK-NEXT:    [[ARRAYIDX:%.*]] = getelementptr inbounds i32, ptr [[P1]], i64 [[GEP_IND]]
275 ; CHECK-NEXT:    [[TMP26:%.*]] = load i32, ptr [[ARRAYIDX]], align 4
276 ; CHECK-NEXT:    [[DOWORK:%.*]] = icmp ne i32 [[TMP26]], 0
277 ; CHECK-NEXT:    br i1 [[DOWORK]], label [[FOR_DOWORK:%.*]], label [[FOR_INC]]
278 ; CHECK:       for.dowork:
279 ; CHECK-NEXT:    [[ARRAYIDX3:%.*]] = getelementptr inbounds i32, ptr [[P2]], i64 [[GEP_IND]]
280 ; CHECK-NEXT:    [[TMP27:%.*]] = load i32, ptr [[ARRAYIDX3]], align 4
281 ; CHECK-NEXT:    [[ADD:%.*]] = add i32 [[TMP26]], [[TMP27]]
282 ; CHECK-NEXT:    [[ARRAYIDX5:%.*]] = getelementptr inbounds i32, ptr [[DEST]], i64 [[GEP_IND]]
283 ; CHECK-NEXT:    store i32 [[ADD]], ptr [[ARRAYIDX5]], align 4
284 ; CHECK-NEXT:    br label [[FOR_INC]]
285 ; CHECK:       for.inc:
286 ; CHECK-NEXT:    [[IND_NEXT]] = add i8 [[IND]], 1
287 ; CHECK-NEXT:    [[CONV:%.*]] = zext i8 [[IND_NEXT]] to i32
288 ; CHECK-NEXT:    [[GEP_IND_NEXT]] = add i64 [[GEP_IND]], 1
289 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[CONV]], [[END_CLAMPED]]
290 ; CHECK-NEXT:    br i1 [[CMP]], label [[FOR_BODY]], label [[EXIT]], !llvm.loop [[LOOP9:![0-9]+]]
291 ; CHECK:       exit:
292 ; CHECK-NEXT:    ret i32 0
294 entry:
295   %p1 = alloca [1024 x i32]
296   %p2 = alloca [1024 x i32]
297   call void @init(ptr %p1)
298   call void @init(ptr %p2)
299   %end.clamped = and i32 %end, 1023
300   br label %for.body
302 for.body:
303   %ind = phi i8 [ %ind.next, %for.inc ], [ 0, %entry ]
304   %gep.ind = phi i64 [ %gep.ind.next, %for.inc ], [ 0, %entry ]
305   %arrayidx = getelementptr inbounds i32, ptr %p1, i64 %gep.ind
306   %0 = load i32, ptr %arrayidx, align 4
307   %dowork = icmp ne i32 %0, 0
308   br i1 %dowork, label %for.dowork, label %for.inc
310 for.dowork:
311   %arrayidx3 = getelementptr inbounds i32, ptr %p2, i64 %gep.ind
312   %1 = load i32, ptr %arrayidx3, align 4
313   %add = add i32 %0, %1
314   %arrayidx5 = getelementptr inbounds i32, ptr %dest, i64 %gep.ind
315   store i32 %add, ptr %arrayidx5, align 4
316   br label %for.inc
318 for.inc:
319   %ind.next = add i8 %ind, 1
320   %conv = zext i8 %ind.next to i32
321   %gep.ind.next = add i64 %gep.ind, 1
322   %cmp = icmp ult i32 %conv, %end.clamped
323   br i1 %cmp, label %for.body, label %exit
325 exit:
326   ret i32 0
330 ; Test reverse loops where we should be able to prove loads in predicated blocks
331 ; are safe to load unconditionally.
332 define void @test_rev_loops_deref_loads(ptr nocapture noundef writeonly %dest) {
333 ; CHECK-LABEL: @test_rev_loops_deref_loads(
334 ; CHECK-NEXT:  entry:
335 ; CHECK-NEXT:    [[LOCAL_DEST:%.*]] = alloca [1024 x i32], align 4
336 ; CHECK-NEXT:    [[LOCAL_SRC:%.*]] = alloca [1024 x i32], align 4
337 ; CHECK-NEXT:    [[LOCAL_CMP:%.*]] = alloca [1024 x i32], align 4
338 ; CHECK-NEXT:    call void @init(ptr [[LOCAL_SRC]])
339 ; CHECK-NEXT:    call void @init(ptr [[LOCAL_CMP]])
340 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
341 ; CHECK:       vector.ph:
342 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
343 ; CHECK:       vector.body:
344 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[PRED_STORE_CONTINUE4:%.*]] ]
345 ; CHECK-NEXT:    [[OFFSET_IDX:%.*]] = sub i64 1023, [[INDEX]]
346 ; CHECK-NEXT:    [[TMP0:%.*]] = add i64 [[OFFSET_IDX]], 0
347 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_CMP]], i64 0, i64 [[TMP0]]
348 ; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i32, ptr [[TMP1]], i32 0
349 ; CHECK-NEXT:    [[TMP3:%.*]] = getelementptr inbounds i32, ptr [[TMP2]], i32 -1
350 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <2 x i32>, ptr [[TMP3]], align 4
351 ; CHECK-NEXT:    [[REVERSE:%.*]] = shufflevector <2 x i32> [[WIDE_LOAD]], <2 x i32> poison, <2 x i32> <i32 1, i32 0>
352 ; CHECK-NEXT:    [[TMP4:%.*]] = icmp eq <2 x i32> [[REVERSE]], splat (i32 3)
353 ; CHECK-NEXT:    [[TMP5:%.*]] = xor <2 x i1> [[TMP4]], splat (i1 true)
354 ; CHECK-NEXT:    [[TMP6:%.*]] = getelementptr [1024 x i32], ptr [[LOCAL_SRC]], i64 0, i64 [[TMP0]]
355 ; CHECK-NEXT:    [[TMP7:%.*]] = getelementptr i32, ptr [[TMP6]], i32 0
356 ; CHECK-NEXT:    [[TMP8:%.*]] = getelementptr i32, ptr [[TMP7]], i32 -1
357 ; CHECK-NEXT:    [[WIDE_LOAD1:%.*]] = load <2 x i32>, ptr [[TMP8]], align 4
358 ; CHECK-NEXT:    [[REVERSE2:%.*]] = shufflevector <2 x i32> [[WIDE_LOAD1]], <2 x i32> poison, <2 x i32> <i32 1, i32 0>
359 ; CHECK-NEXT:    [[TMP9:%.*]] = extractelement <2 x i1> [[TMP5]], i32 0
360 ; CHECK-NEXT:    br i1 [[TMP9]], label [[PRED_STORE_IF:%.*]], label [[PRED_STORE_CONTINUE:%.*]]
361 ; CHECK:       pred.store.if:
362 ; CHECK-NEXT:    [[TMP10:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[TMP0]]
363 ; CHECK-NEXT:    [[TMP11:%.*]] = extractelement <2 x i32> [[REVERSE2]], i32 0
364 ; CHECK-NEXT:    [[TMP12:%.*]] = shl nsw i32 [[TMP11]], 2
365 ; CHECK-NEXT:    store i32 [[TMP12]], ptr [[TMP10]], align 4
366 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE]]
367 ; CHECK:       pred.store.continue:
368 ; CHECK-NEXT:    [[TMP13:%.*]] = extractelement <2 x i1> [[TMP5]], i32 1
369 ; CHECK-NEXT:    br i1 [[TMP13]], label [[PRED_STORE_IF3:%.*]], label [[PRED_STORE_CONTINUE4]]
370 ; CHECK:       pred.store.if3:
371 ; CHECK-NEXT:    [[TMP14:%.*]] = add i64 [[OFFSET_IDX]], -1
372 ; CHECK-NEXT:    [[TMP15:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[TMP14]]
373 ; CHECK-NEXT:    [[TMP16:%.*]] = extractelement <2 x i32> [[REVERSE2]], i32 1
374 ; CHECK-NEXT:    [[TMP17:%.*]] = shl nsw i32 [[TMP16]], 2
375 ; CHECK-NEXT:    store i32 [[TMP17]], ptr [[TMP15]], align 4
376 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE4]]
377 ; CHECK:       pred.store.continue4:
378 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 2
379 ; CHECK-NEXT:    [[TMP18:%.*]] = icmp eq i64 [[INDEX_NEXT]], 1024
380 ; CHECK-NEXT:    br i1 [[TMP18]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP10:![0-9]+]]
381 ; CHECK:       middle.block:
382 ; CHECK-NEXT:    br i1 true, label [[EXIT:%.*]], label [[SCALAR_PH]]
383 ; CHECK:       scalar.ph:
384 ; CHECK-NEXT:    [[BC_RESUME_VAL:%.*]] = phi i64 [ -1, [[MIDDLE_BLOCK]] ], [ 1023, [[ENTRY:%.*]] ]
385 ; CHECK-NEXT:    br label [[FOR_BODY:%.*]]
386 ; CHECK:       for.body:
387 ; CHECK-NEXT:    [[IV:%.*]] = phi i64 [ [[BC_RESUME_VAL]], [[SCALAR_PH]] ], [ [[IV_NEXT:%.*]], [[FOR_INC:%.*]] ]
388 ; CHECK-NEXT:    [[ARRAYIDX:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_CMP]], i64 0, i64 [[IV]]
389 ; CHECK-NEXT:    [[TMP19:%.*]] = load i32, ptr [[ARRAYIDX]], align 4
390 ; CHECK-NEXT:    [[CMP3_NOT:%.*]] = icmp eq i32 [[TMP19]], 3
391 ; CHECK-NEXT:    br i1 [[CMP3_NOT]], label [[FOR_INC]], label [[IF_THEN:%.*]]
392 ; CHECK:       if.then:
393 ; CHECK-NEXT:    [[ARRAYIDX5:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_SRC]], i64 0, i64 [[IV]]
394 ; CHECK-NEXT:    [[TMP20:%.*]] = load i32, ptr [[ARRAYIDX5]], align 4
395 ; CHECK-NEXT:    [[MUL:%.*]] = shl nsw i32 [[TMP20]], 2
396 ; CHECK-NEXT:    [[ARRAYIDX7:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[IV]]
397 ; CHECK-NEXT:    store i32 [[MUL]], ptr [[ARRAYIDX7]], align 4
398 ; CHECK-NEXT:    br label [[FOR_INC]]
399 ; CHECK:       for.inc:
400 ; CHECK-NEXT:    [[IV_NEXT]] = add nsw i64 [[IV]], -1
401 ; CHECK-NEXT:    [[CMP2_NOT:%.*]] = icmp eq i64 [[IV]], 0
402 ; CHECK-NEXT:    br i1 [[CMP2_NOT]], label [[EXIT]], label [[FOR_BODY]], !llvm.loop [[LOOP11:![0-9]+]]
403 ; CHECK:       exit:
404 ; CHECK-NEXT:    call void @llvm.memcpy.p0.p0.i64(ptr [[DEST:%.*]], ptr [[LOCAL_DEST]], i64 1024, i1 false)
405 ; CHECK-NEXT:    ret void
407 entry:
408   %local_dest = alloca [1024 x i32], align 4
409   %local_src = alloca [1024 x i32], align 4
410   %local_cmp = alloca [1024 x i32], align 4
411   call void @init(ptr %local_src)
412   call void @init(ptr %local_cmp)
413   br label %for.body
415 for.body:
416   %iv = phi i64 [ 1023, %entry ], [ %iv.next, %for.inc ]
417   %arrayidx = getelementptr inbounds [1024 x i32], ptr %local_cmp, i64 0, i64 %iv
418   %0 = load i32, ptr %arrayidx, align 4
419   %cmp3.not = icmp eq i32 %0, 3
420   br i1 %cmp3.not, label %for.inc, label %if.then
422 if.then:
423   %arrayidx5 = getelementptr inbounds [1024 x i32], ptr %local_src, i64 0, i64 %iv
424   %1 = load i32, ptr %arrayidx5, align 4
425   %mul = shl nsw i32 %1, 2
426   %arrayidx7 = getelementptr inbounds [1024 x i32], ptr %local_dest, i64 0, i64 %iv
427   store i32 %mul, ptr %arrayidx7, align 4
428   br label %for.inc
430 for.inc:
431   %iv.next = add nsw i64 %iv, -1
432   %cmp2.not = icmp eq i64 %iv, 0
433   br i1 %cmp2.not, label %exit, label %for.body
435 exit:
436   call void @llvm.memcpy.p0.p0.i64(ptr %dest, ptr %local_dest, i64 1024, i1 false)
437   ret void
441 ; Test reverse loops where we *cannot* prove loads in predicated blocks are safe
442 ; to load unconditionally.
443 define void @test_rev_loops_non_deref_loads(ptr nocapture noundef writeonly %dest) {
444 ; CHECK-LABEL: @test_rev_loops_non_deref_loads(
445 ; CHECK-NEXT:  entry:
446 ; CHECK-NEXT:    [[LOCAL_DEST:%.*]] = alloca [1024 x i32], align 4
447 ; CHECK-NEXT:    [[LOCAL_SRC:%.*]] = alloca [1024 x i32], align 4
448 ; CHECK-NEXT:    [[LOCAL_CMP:%.*]] = alloca [1024 x i32], align 4
449 ; CHECK-NEXT:    call void @init(ptr [[LOCAL_SRC]])
450 ; CHECK-NEXT:    call void @init(ptr [[LOCAL_CMP]])
451 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
452 ; CHECK:       vector.ph:
453 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
454 ; CHECK:       vector.body:
455 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[PRED_STORE_CONTINUE2:%.*]] ]
456 ; CHECK-NEXT:    [[VEC_IND:%.*]] = phi <2 x i64> [ <i64 1023, i64 1022>, [[VECTOR_PH]] ], [ [[VEC_IND_NEXT:%.*]], [[PRED_STORE_CONTINUE2]] ]
457 ; CHECK-NEXT:    [[TMP0:%.*]] = add <2 x i64> [[VEC_IND]], splat (i64 -1)
458 ; CHECK-NEXT:    [[TMP1:%.*]] = extractelement <2 x i64> [[TMP0]], i32 0
459 ; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_CMP]], i64 0, i64 [[TMP1]]
460 ; CHECK-NEXT:    [[TMP3:%.*]] = getelementptr inbounds i32, ptr [[TMP2]], i32 0
461 ; CHECK-NEXT:    [[TMP4:%.*]] = getelementptr inbounds i32, ptr [[TMP3]], i32 -1
462 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <2 x i32>, ptr [[TMP4]], align 4
463 ; CHECK-NEXT:    [[REVERSE:%.*]] = shufflevector <2 x i32> [[WIDE_LOAD]], <2 x i32> poison, <2 x i32> <i32 1, i32 0>
464 ; CHECK-NEXT:    [[TMP5:%.*]] = icmp eq <2 x i32> [[REVERSE]], splat (i32 3)
465 ; CHECK-NEXT:    [[TMP6:%.*]] = xor <2 x i1> [[TMP5]], splat (i1 true)
466 ; CHECK-NEXT:    [[TMP7:%.*]] = extractelement <2 x i1> [[TMP6]], i32 0
467 ; CHECK-NEXT:    br i1 [[TMP7]], label [[PRED_STORE_IF:%.*]], label [[PRED_STORE_CONTINUE:%.*]]
468 ; CHECK:       pred.store.if:
469 ; CHECK-NEXT:    [[TMP8:%.*]] = extractelement <2 x i64> [[TMP0]], i32 0
470 ; CHECK-NEXT:    [[TMP9:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_SRC]], i64 0, i64 [[TMP8]]
471 ; CHECK-NEXT:    [[TMP10:%.*]] = load i32, ptr [[TMP9]], align 4
472 ; CHECK-NEXT:    [[TMP11:%.*]] = extractelement <2 x i64> [[TMP0]], i32 0
473 ; CHECK-NEXT:    [[TMP12:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[TMP11]]
474 ; CHECK-NEXT:    [[TMP13:%.*]] = shl nsw i32 [[TMP10]], 2
475 ; CHECK-NEXT:    store i32 [[TMP13]], ptr [[TMP12]], align 4
476 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE]]
477 ; CHECK:       pred.store.continue:
478 ; CHECK-NEXT:    [[TMP14:%.*]] = extractelement <2 x i1> [[TMP6]], i32 1
479 ; CHECK-NEXT:    br i1 [[TMP14]], label [[PRED_STORE_IF1:%.*]], label [[PRED_STORE_CONTINUE2]]
480 ; CHECK:       pred.store.if1:
481 ; CHECK-NEXT:    [[TMP15:%.*]] = extractelement <2 x i64> [[TMP0]], i32 1
482 ; CHECK-NEXT:    [[TMP16:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_SRC]], i64 0, i64 [[TMP15]]
483 ; CHECK-NEXT:    [[TMP17:%.*]] = load i32, ptr [[TMP16]], align 4
484 ; CHECK-NEXT:    [[TMP18:%.*]] = extractelement <2 x i64> [[TMP0]], i32 1
485 ; CHECK-NEXT:    [[TMP19:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[TMP18]]
486 ; CHECK-NEXT:    [[TMP20:%.*]] = shl nsw i32 [[TMP17]], 2
487 ; CHECK-NEXT:    store i32 [[TMP20]], ptr [[TMP19]], align 4
488 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE2]]
489 ; CHECK:       pred.store.continue2:
490 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 2
491 ; CHECK-NEXT:    [[VEC_IND_NEXT]] = add <2 x i64> [[VEC_IND]], splat (i64 -2)
492 ; CHECK-NEXT:    [[TMP21:%.*]] = icmp eq i64 [[INDEX_NEXT]], 1024
493 ; CHECK-NEXT:    br i1 [[TMP21]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP12:![0-9]+]]
494 ; CHECK:       middle.block:
495 ; CHECK-NEXT:    br i1 true, label [[EXIT:%.*]], label [[SCALAR_PH]]
496 ; CHECK:       scalar.ph:
497 ; CHECK-NEXT:    [[BC_RESUME_VAL:%.*]] = phi i64 [ -1, [[MIDDLE_BLOCK]] ], [ 1023, [[ENTRY:%.*]] ]
498 ; CHECK-NEXT:    br label [[FOR_BODY:%.*]]
499 ; CHECK:       for.body:
500 ; CHECK-NEXT:    [[IV:%.*]] = phi i64 [ [[BC_RESUME_VAL]], [[SCALAR_PH]] ], [ [[IV_NEXT:%.*]], [[FOR_INC:%.*]] ]
501 ; CHECK-NEXT:    [[OFF:%.*]] = add i64 [[IV]], -1
502 ; CHECK-NEXT:    [[ARRAYIDX:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_CMP]], i64 0, i64 [[OFF]]
503 ; CHECK-NEXT:    [[TMP22:%.*]] = load i32, ptr [[ARRAYIDX]], align 4
504 ; CHECK-NEXT:    [[CMP3_NOT:%.*]] = icmp eq i32 [[TMP22]], 3
505 ; CHECK-NEXT:    br i1 [[CMP3_NOT]], label [[FOR_INC]], label [[IF_THEN:%.*]]
506 ; CHECK:       if.then:
507 ; CHECK-NEXT:    [[ARRAYIDX5:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_SRC]], i64 0, i64 [[OFF]]
508 ; CHECK-NEXT:    [[TMP23:%.*]] = load i32, ptr [[ARRAYIDX5]], align 4
509 ; CHECK-NEXT:    [[MUL:%.*]] = shl nsw i32 [[TMP23]], 2
510 ; CHECK-NEXT:    [[ARRAYIDX7:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[OFF]]
511 ; CHECK-NEXT:    store i32 [[MUL]], ptr [[ARRAYIDX7]], align 4
512 ; CHECK-NEXT:    br label [[FOR_INC]]
513 ; CHECK:       for.inc:
514 ; CHECK-NEXT:    [[IV_NEXT]] = add nsw i64 [[IV]], -1
515 ; CHECK-NEXT:    [[CMP2_NOT:%.*]] = icmp eq i64 [[IV]], 0
516 ; CHECK-NEXT:    br i1 [[CMP2_NOT]], label [[EXIT]], label [[FOR_BODY]], !llvm.loop [[LOOP13:![0-9]+]]
517 ; CHECK:       exit:
518 ; CHECK-NEXT:    call void @llvm.memcpy.p0.p0.i64(ptr [[DEST:%.*]], ptr [[LOCAL_DEST]], i64 1024, i1 false)
519 ; CHECK-NEXT:    ret void
521 entry:
522   %local_dest = alloca [1024 x i32], align 4
523   %local_src = alloca [1024 x i32], align 4
524   %local_cmp = alloca [1024 x i32], align 4
525   call void @init(ptr %local_src)
526   call void @init(ptr %local_cmp)
527   br label %for.body
529 for.body:
530   %iv = phi i64 [ 1023, %entry ], [ %iv.next, %for.inc ]
531   %off = add i64 %iv, -1
532   %arrayidx = getelementptr inbounds [1024 x i32], ptr %local_cmp, i64 0, i64 %off
533   %0 = load i32, ptr %arrayidx, align 4
534   %cmp3.not = icmp eq i32 %0, 3
535   br i1 %cmp3.not, label %for.inc, label %if.then
537 if.then:
538   %arrayidx5 = getelementptr inbounds [1024 x i32], ptr %local_src, i64 0, i64 %off
539   %1 = load i32, ptr %arrayidx5, align 4
540   %mul = shl nsw i32 %1, 2
541   %arrayidx7 = getelementptr inbounds [1024 x i32], ptr %local_dest, i64 0, i64 %off
542   store i32 %mul, ptr %arrayidx7, align 4
543   br label %for.inc
545 for.inc:
546   %iv.next = add nsw i64 %iv, -1
547   %cmp2.not = icmp eq i64 %iv, 0
548   br i1 %cmp2.not, label %exit, label %for.body
550 exit:
551   call void @llvm.memcpy.p0.p0.i64(ptr %dest, ptr %local_dest, i64 1024, i1 false)
552   ret void
556 ; Test a loop with a positive step recurrence that has a strided access
557 define i16 @test_strided_access(i64 %len, ptr %test_base) {
558 ; CHECK-LABEL: @test_strided_access(
559 ; CHECK-NEXT:  entry:
560 ; CHECK-NEXT:    [[ALLOCA:%.*]] = alloca [163840 x i16], align 4
561 ; CHECK-NEXT:    call void @init(ptr [[ALLOCA]])
562 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
563 ; CHECK:       vector.ph:
564 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
565 ; CHECK:       vector.body:
566 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[VECTOR_BODY]] ]
567 ; CHECK-NEXT:    [[VEC_IND:%.*]] = phi <2 x i64> [ <i64 0, i64 1>, [[VECTOR_PH]] ], [ [[VEC_IND_NEXT:%.*]], [[VECTOR_BODY]] ]
568 ; CHECK-NEXT:    [[VEC_PHI:%.*]] = phi <2 x i16> [ zeroinitializer, [[VECTOR_PH]] ], [ [[TMP13:%.*]], [[VECTOR_BODY]] ]
569 ; CHECK-NEXT:    [[TMP0:%.*]] = add i64 [[INDEX]], 0
570 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[TEST_BASE:%.*]], i64 [[TMP0]]
571 ; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i8, ptr [[TMP1]], i32 0
572 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <2 x i8>, ptr [[TMP2]], align 1
573 ; CHECK-NEXT:    [[TMP3:%.*]] = icmp sge <2 x i8> [[WIDE_LOAD]], zeroinitializer
574 ; CHECK-NEXT:    [[TMP4:%.*]] = mul <2 x i64> [[VEC_IND]], splat (i64 2)
575 ; CHECK-NEXT:    [[TMP5:%.*]] = extractelement <2 x i64> [[TMP4]], i32 0
576 ; CHECK-NEXT:    [[TMP6:%.*]] = getelementptr inbounds i16, ptr [[ALLOCA]], i64 [[TMP5]]
577 ; CHECK-NEXT:    [[TMP7:%.*]] = extractelement <2 x i64> [[TMP4]], i32 1
578 ; CHECK-NEXT:    [[TMP8:%.*]] = getelementptr inbounds i16, ptr [[ALLOCA]], i64 [[TMP7]]
579 ; CHECK-NEXT:    [[TMP9:%.*]] = load i16, ptr [[TMP6]], align 2
580 ; CHECK-NEXT:    [[TMP10:%.*]] = load i16, ptr [[TMP8]], align 2
581 ; CHECK-NEXT:    [[TMP11:%.*]] = insertelement <2 x i16> poison, i16 [[TMP9]], i32 0
582 ; CHECK-NEXT:    [[TMP12:%.*]] = insertelement <2 x i16> [[TMP11]], i16 [[TMP10]], i32 1
583 ; CHECK-NEXT:    [[PREDPHI:%.*]] = select <2 x i1> [[TMP3]], <2 x i16> [[TMP12]], <2 x i16> zeroinitializer
584 ; CHECK-NEXT:    [[TMP13]] = add <2 x i16> [[VEC_PHI]], [[PREDPHI]]
585 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 2
586 ; CHECK-NEXT:    [[VEC_IND_NEXT]] = add <2 x i64> [[VEC_IND]], splat (i64 2)
587 ; CHECK-NEXT:    [[TMP14:%.*]] = icmp eq i64 [[INDEX_NEXT]], 4096
588 ; CHECK-NEXT:    br i1 [[TMP14]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP14:![0-9]+]]
589 ; CHECK:       middle.block:
590 ; CHECK-NEXT:    [[TMP15:%.*]] = call i16 @llvm.vector.reduce.add.v2i16(<2 x i16> [[TMP13]])
591 ; CHECK-NEXT:    br i1 true, label [[LOOP_EXIT:%.*]], label [[SCALAR_PH]]
592 ; CHECK:       scalar.ph:
593 ; CHECK-NEXT:    [[BC_RESUME_VAL:%.*]] = phi i64 [ 4096, [[MIDDLE_BLOCK]] ], [ 0, [[ENTRY:%.*]] ]
594 ; CHECK-NEXT:    [[BC_MERGE_RDX:%.*]] = phi i16 [ [[TMP15]], [[MIDDLE_BLOCK]] ], [ 0, [[ENTRY]] ]
595 ; CHECK-NEXT:    br label [[LOOP:%.*]]
596 ; CHECK:       loop:
597 ; CHECK-NEXT:    [[IV:%.*]] = phi i64 [ [[BC_RESUME_VAL]], [[SCALAR_PH]] ], [ [[IV_NEXT:%.*]], [[LATCH:%.*]] ]
598 ; CHECK-NEXT:    [[ACCUM:%.*]] = phi i16 [ [[BC_MERGE_RDX]], [[SCALAR_PH]] ], [ [[ACCUM_NEXT:%.*]], [[LATCH]] ]
599 ; CHECK-NEXT:    [[IV_NEXT]] = add i64 [[IV]], 1
600 ; CHECK-NEXT:    [[TEST_ADDR:%.*]] = getelementptr inbounds i8, ptr [[TEST_BASE]], i64 [[IV]]
601 ; CHECK-NEXT:    [[L_T:%.*]] = load i8, ptr [[TEST_ADDR]], align 1
602 ; CHECK-NEXT:    [[CMP:%.*]] = icmp sge i8 [[L_T]], 0
603 ; CHECK-NEXT:    br i1 [[CMP]], label [[PRED:%.*]], label [[LATCH]]
604 ; CHECK:       pred:
605 ; CHECK-NEXT:    [[IV_STRIDE:%.*]] = mul i64 [[IV]], 2
606 ; CHECK-NEXT:    [[ADDR:%.*]] = getelementptr inbounds i16, ptr [[ALLOCA]], i64 [[IV_STRIDE]]
607 ; CHECK-NEXT:    [[VAL:%.*]] = load i16, ptr [[ADDR]], align 2
608 ; CHECK-NEXT:    br label [[LATCH]]
609 ; CHECK:       latch:
610 ; CHECK-NEXT:    [[VAL_PHI:%.*]] = phi i16 [ 0, [[LOOP]] ], [ [[VAL]], [[PRED]] ]
611 ; CHECK-NEXT:    [[ACCUM_NEXT]] = add i16 [[ACCUM]], [[VAL_PHI]]
612 ; CHECK-NEXT:    [[EXIT:%.*]] = icmp eq i64 [[IV]], 4095
613 ; CHECK-NEXT:    br i1 [[EXIT]], label [[LOOP_EXIT]], label [[LOOP]], !llvm.loop [[LOOP15:![0-9]+]]
614 ; CHECK:       loop_exit:
615 ; CHECK-NEXT:    [[ACCUM_NEXT_LCSSA:%.*]] = phi i16 [ [[ACCUM_NEXT]], [[LATCH]] ], [ [[TMP15]], [[MIDDLE_BLOCK]] ]
616 ; CHECK-NEXT:    ret i16 [[ACCUM_NEXT_LCSSA]]
618 entry:
619   %alloca = alloca [163840 x i16], align 4
620   call void @init(ptr %alloca)
621   br label %loop
622 loop:
623   %iv = phi i64 [ 0, %entry ], [ %iv.next, %latch ]
624   %accum = phi i16 [ 0, %entry ], [ %accum.next, %latch ]
625   %iv.next = add i64 %iv, 1
626   %test_addr = getelementptr inbounds i8, ptr %test_base, i64 %iv
627   %l.t = load i8, ptr %test_addr
628   %cmp = icmp sge i8 %l.t, 0
629   br i1 %cmp, label %pred, label %latch
630 pred:
631   %iv.stride = mul i64 %iv, 2
632   %addr = getelementptr inbounds i16, ptr %alloca, i64 %iv.stride
633   %val = load i16, ptr %addr, align 2
634   br label %latch
635 latch:
636   %val.phi = phi i16 [0, %loop], [%val, %pred]
637   %accum.next = add i16 %accum, %val.phi
638   %exit = icmp eq i64 %iv, 4095
639   br i1 %exit, label %loop_exit, label %loop
641 loop_exit:
642   ret i16 %accum.next
646 ; Test a loop with a negative step recurrence that has a strided access
647 define void @test_rev_loops_strided_deref_loads(ptr nocapture noundef writeonly %dest) {
648 ; CHECK-LABEL: @test_rev_loops_strided_deref_loads(
649 ; CHECK-NEXT:  entry:
650 ; CHECK-NEXT:    [[LOCAL_DEST:%.*]] = alloca [1024 x i32], align 4
651 ; CHECK-NEXT:    [[LOCAL_SRC:%.*]] = alloca [1024 x i32], align 4
652 ; CHECK-NEXT:    [[LOCAL_CMP:%.*]] = alloca [1024 x i32], align 4
653 ; CHECK-NEXT:    call void @init(ptr [[LOCAL_SRC]])
654 ; CHECK-NEXT:    call void @init(ptr [[LOCAL_CMP]])
655 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
656 ; CHECK:       vector.ph:
657 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
658 ; CHECK:       vector.body:
659 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[PRED_STORE_CONTINUE2:%.*]] ]
660 ; CHECK-NEXT:    [[VEC_IND:%.*]] = phi <2 x i64> [ <i64 511, i64 510>, [[VECTOR_PH]] ], [ [[VEC_IND_NEXT:%.*]], [[PRED_STORE_CONTINUE2]] ]
661 ; CHECK-NEXT:    [[OFFSET_IDX:%.*]] = sub i64 511, [[INDEX]]
662 ; CHECK-NEXT:    [[TMP0:%.*]] = add i64 [[OFFSET_IDX]], 0
663 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_CMP]], i64 0, i64 [[TMP0]]
664 ; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i32, ptr [[TMP1]], i32 0
665 ; CHECK-NEXT:    [[TMP3:%.*]] = getelementptr inbounds i32, ptr [[TMP2]], i32 -1
666 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <2 x i32>, ptr [[TMP3]], align 4
667 ; CHECK-NEXT:    [[REVERSE:%.*]] = shufflevector <2 x i32> [[WIDE_LOAD]], <2 x i32> poison, <2 x i32> <i32 1, i32 0>
668 ; CHECK-NEXT:    [[TMP4:%.*]] = icmp eq <2 x i32> [[REVERSE]], splat (i32 3)
669 ; CHECK-NEXT:    [[TMP5:%.*]] = xor <2 x i1> [[TMP4]], splat (i1 true)
670 ; CHECK-NEXT:    [[TMP6:%.*]] = mul <2 x i64> [[VEC_IND]], splat (i64 2)
671 ; CHECK-NEXT:    [[TMP7:%.*]] = extractelement <2 x i64> [[TMP6]], i32 0
672 ; CHECK-NEXT:    [[TMP8:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_SRC]], i64 0, i64 [[TMP7]]
673 ; CHECK-NEXT:    [[TMP9:%.*]] = extractelement <2 x i64> [[TMP6]], i32 1
674 ; CHECK-NEXT:    [[TMP10:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_SRC]], i64 0, i64 [[TMP9]]
675 ; CHECK-NEXT:    [[TMP11:%.*]] = load i32, ptr [[TMP8]], align 4
676 ; CHECK-NEXT:    [[TMP12:%.*]] = load i32, ptr [[TMP10]], align 4
677 ; CHECK-NEXT:    [[TMP13:%.*]] = extractelement <2 x i1> [[TMP5]], i32 0
678 ; CHECK-NEXT:    br i1 [[TMP13]], label [[PRED_STORE_IF:%.*]], label [[PRED_STORE_CONTINUE:%.*]]
679 ; CHECK:       pred.store.if:
680 ; CHECK-NEXT:    [[TMP14:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[TMP0]]
681 ; CHECK-NEXT:    [[TMP15:%.*]] = shl nsw i32 [[TMP11]], 2
682 ; CHECK-NEXT:    store i32 [[TMP15]], ptr [[TMP14]], align 4
683 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE]]
684 ; CHECK:       pred.store.continue:
685 ; CHECK-NEXT:    [[TMP16:%.*]] = extractelement <2 x i1> [[TMP5]], i32 1
686 ; CHECK-NEXT:    br i1 [[TMP16]], label [[PRED_STORE_IF1:%.*]], label [[PRED_STORE_CONTINUE2]]
687 ; CHECK:       pred.store.if1:
688 ; CHECK-NEXT:    [[TMP17:%.*]] = add i64 [[OFFSET_IDX]], -1
689 ; CHECK-NEXT:    [[TMP18:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[TMP17]]
690 ; CHECK-NEXT:    [[TMP19:%.*]] = shl nsw i32 [[TMP12]], 2
691 ; CHECK-NEXT:    store i32 [[TMP19]], ptr [[TMP18]], align 4
692 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE2]]
693 ; CHECK:       pred.store.continue2:
694 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 2
695 ; CHECK-NEXT:    [[VEC_IND_NEXT]] = add <2 x i64> [[VEC_IND]], splat (i64 -2)
696 ; CHECK-NEXT:    [[TMP20:%.*]] = icmp eq i64 [[INDEX_NEXT]], 512
697 ; CHECK-NEXT:    br i1 [[TMP20]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP16:![0-9]+]]
698 ; CHECK:       middle.block:
699 ; CHECK-NEXT:    br i1 true, label [[EXIT:%.*]], label [[SCALAR_PH]]
700 ; CHECK:       scalar.ph:
701 ; CHECK-NEXT:    [[BC_RESUME_VAL:%.*]] = phi i64 [ -1, [[MIDDLE_BLOCK]] ], [ 511, [[ENTRY:%.*]] ]
702 ; CHECK-NEXT:    br label [[FOR_BODY:%.*]]
703 ; CHECK:       for.body:
704 ; CHECK-NEXT:    [[IV:%.*]] = phi i64 [ [[BC_RESUME_VAL]], [[SCALAR_PH]] ], [ [[IV_NEXT:%.*]], [[FOR_INC:%.*]] ]
705 ; CHECK-NEXT:    [[ARRAYIDX:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_CMP]], i64 0, i64 [[IV]]
706 ; CHECK-NEXT:    [[TMP21:%.*]] = load i32, ptr [[ARRAYIDX]], align 4
707 ; CHECK-NEXT:    [[CMP3_NOT:%.*]] = icmp eq i32 [[TMP21]], 3
708 ; CHECK-NEXT:    br i1 [[CMP3_NOT]], label [[FOR_INC]], label [[IF_THEN:%.*]]
709 ; CHECK:       if.then:
710 ; CHECK-NEXT:    [[IV_STRIDED:%.*]] = mul i64 [[IV]], 2
711 ; CHECK-NEXT:    [[ARRAYIDX5:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_SRC]], i64 0, i64 [[IV_STRIDED]]
712 ; CHECK-NEXT:    [[TMP22:%.*]] = load i32, ptr [[ARRAYIDX5]], align 4
713 ; CHECK-NEXT:    [[MUL:%.*]] = shl nsw i32 [[TMP22]], 2
714 ; CHECK-NEXT:    [[ARRAYIDX7:%.*]] = getelementptr inbounds [1024 x i32], ptr [[LOCAL_DEST]], i64 0, i64 [[IV]]
715 ; CHECK-NEXT:    store i32 [[MUL]], ptr [[ARRAYIDX7]], align 4
716 ; CHECK-NEXT:    br label [[FOR_INC]]
717 ; CHECK:       for.inc:
718 ; CHECK-NEXT:    [[IV_NEXT]] = add nsw i64 [[IV]], -1
719 ; CHECK-NEXT:    [[CMP2_NOT:%.*]] = icmp eq i64 [[IV]], 0
720 ; CHECK-NEXT:    br i1 [[CMP2_NOT]], label [[EXIT]], label [[FOR_BODY]], !llvm.loop [[LOOP17:![0-9]+]]
721 ; CHECK:       exit:
722 ; CHECK-NEXT:    call void @llvm.memcpy.p0.p0.i64(ptr [[DEST:%.*]], ptr [[LOCAL_DEST]], i64 1024, i1 false)
723 ; CHECK-NEXT:    ret void
725 entry:
726   %local_dest = alloca [1024 x i32], align 4
727   %local_src = alloca [1024 x i32], align 4
728   %local_cmp = alloca [1024 x i32], align 4
729   call void @init(ptr %local_src)
730   call void @init(ptr %local_cmp)
731   br label %for.body
733 for.body:
734   %iv = phi i64 [ 511, %entry ], [ %iv.next, %for.inc ]
735   %arrayidx = getelementptr inbounds [1024 x i32], ptr %local_cmp, i64 0, i64 %iv
736   %0 = load i32, ptr %arrayidx, align 4
737   %cmp3.not = icmp eq i32 %0, 3
738   br i1 %cmp3.not, label %for.inc, label %if.then
740 if.then:
741   %iv.strided = mul i64 %iv, 2
742   %arrayidx5 = getelementptr inbounds [1024 x i32], ptr %local_src, i64 0, i64 %iv.strided
743   %1 = load i32, ptr %arrayidx5, align 4
744   %mul = shl nsw i32 %1, 2
745   %arrayidx7 = getelementptr inbounds [1024 x i32], ptr %local_dest, i64 0, i64 %iv
746   store i32 %mul, ptr %arrayidx7, align 4
747   br label %for.inc
749 for.inc:
750   %iv.next = add nsw i64 %iv, -1
751   %cmp2.not = icmp eq i64 %iv, 0
752   br i1 %cmp2.not, label %exit, label %for.body
754 exit:
755   call void @llvm.memcpy.p0.p0.i64(ptr %dest, ptr %local_dest, i64 1024, i1 false)
756   ret void