[ORC] Add std::tuple support to SimplePackedSerialization.
[llvm-project.git] / llvm / test / Transforms / LoopVectorize / X86 / small-size.ll
blob7558d849ae6bb1d7735cae3217f7e0077965517a
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt < %s -basic-aa -loop-vectorize -force-vector-interleave=1 -force-vector-width=4 -loop-vectorize-with-block-frequency -dce -instcombine -S | FileCheck %s
4 target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64-S128"
5 target triple = "x86_64-apple-macosx10.8.0"
7 @b = common global [2048 x i32] zeroinitializer, align 16
8 @c = common global [2048 x i32] zeroinitializer, align 16
9 @a = common global [2048 x i32] zeroinitializer, align 16
10 @G = common global [32 x [1024 x i32]] zeroinitializer, align 16
11 @ub = common global [1024 x i32] zeroinitializer, align 16
12 @uc = common global [1024 x i32] zeroinitializer, align 16
13 @d = common global [2048 x i32] zeroinitializer, align 16
14 @fa = common global [1024 x float] zeroinitializer, align 16
15 @fb = common global [1024 x float] zeroinitializer, align 16
16 @ic = common global [1024 x i32] zeroinitializer, align 16
17 @da = common global [1024 x float] zeroinitializer, align 16
18 @db = common global [1024 x float] zeroinitializer, align 16
19 @dc = common global [1024 x float] zeroinitializer, align 16
20 @dd = common global [1024 x float] zeroinitializer, align 16
21 @dj = common global [1024 x i32] zeroinitializer, align 16
23 ; We can optimize this test without a tail.
24 define void @example1() optsize {
25 ; CHECK-LABEL: @example1(
26 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
27 ; CHECK:       vector.ph:
28 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
29 ; CHECK:       vector.body:
30 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[VECTOR_BODY]] ]
31 ; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[INDEX]]
32 ; CHECK-NEXT:    [[TMP2:%.*]] = bitcast i32* [[TMP1]] to <4 x i32>*
33 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <4 x i32>, <4 x i32>* [[TMP2]], align 16
34 ; CHECK-NEXT:    [[TMP3:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @c, i64 0, i64 [[INDEX]]
35 ; CHECK-NEXT:    [[TMP4:%.*]] = bitcast i32* [[TMP3]] to <4 x i32>*
36 ; CHECK-NEXT:    [[WIDE_LOAD1:%.*]] = load <4 x i32>, <4 x i32>* [[TMP4]], align 16
37 ; CHECK-NEXT:    [[TMP5:%.*]] = add nsw <4 x i32> [[WIDE_LOAD1]], [[WIDE_LOAD]]
38 ; CHECK-NEXT:    [[TMP6:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @a, i64 0, i64 [[INDEX]]
39 ; CHECK-NEXT:    [[TMP7:%.*]] = bitcast i32* [[TMP6]] to <4 x i32>*
40 ; CHECK-NEXT:    store <4 x i32> [[TMP5]], <4 x i32>* [[TMP7]], align 16
41 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 4
42 ; CHECK-NEXT:    [[TMP8:%.*]] = icmp eq i64 [[INDEX_NEXT]], 256
43 ; CHECK-NEXT:    br i1 [[TMP8]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP0:![0-9]+]]
44 ; CHECK:       middle.block:
45 ; CHECK-NEXT:    br i1 true, label [[TMP10:%.*]], label [[SCALAR_PH]]
46 ; CHECK:       scalar.ph:
47 ; CHECK-NEXT:    br label [[TMP9:%.*]]
48 ; CHECK:       9:
49 ; CHECK-NEXT:    br i1 undef, label [[TMP10]], label [[TMP9]], !llvm.loop [[LOOP2:![0-9]+]]
50 ; CHECK:       10:
51 ; CHECK-NEXT:    ret void
53   br label %1
55 ; <label>:1                                       ; preds = %1, %0
56   %indvars.iv = phi i64 [ 0, %0 ], [ %indvars.iv.next, %1 ]
57   %2 = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 %indvars.iv
58   %3 = load i32, i32* %2, align 4
59   %4 = getelementptr inbounds [2048 x i32], [2048 x i32]* @c, i64 0, i64 %indvars.iv
60   %5 = load i32, i32* %4, align 4
61   %6 = add nsw i32 %5, %3
62   %7 = getelementptr inbounds [2048 x i32], [2048 x i32]* @a, i64 0, i64 %indvars.iv
63   store i32 %6, i32* %7, align 4
64   %indvars.iv.next = add i64 %indvars.iv, 1
65   %lftr.wideiv = trunc i64 %indvars.iv.next to i32
66   %exitcond = icmp eq i32 %lftr.wideiv, 256
67   br i1 %exitcond, label %8, label %1
69 ; <label>:8                                       ; preds = %1
70   ret void
73 ; Can vectorize in 'optsize' mode by masking the needed tail.
74 define void @example2(i32 %n, i32 %x) optsize {
75 ; CHECK-LABEL: @example2(
76 ; CHECK-NEXT:    [[TMP1:%.*]] = icmp sgt i32 [[N:%.*]], 0
77 ; CHECK-NEXT:    br i1 [[TMP1]], label [[DOTLR_PH5_PREHEADER:%.*]], label [[DOTPREHEADER:%.*]]
78 ; CHECK:       .lr.ph5.preheader:
79 ; CHECK-NEXT:    [[TMP2:%.*]] = add i32 [[N]], -1
80 ; CHECK-NEXT:    [[TMP3:%.*]] = zext i32 [[TMP2]] to i64
81 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
82 ; CHECK:       vector.ph:
83 ; CHECK-NEXT:    [[N_RND_UP:%.*]] = add nuw nsw i64 [[TMP3]], 4
84 ; CHECK-NEXT:    [[N_VEC:%.*]] = and i64 [[N_RND_UP]], 8589934588
85 ; CHECK-NEXT:    [[BROADCAST_SPLATINSERT:%.*]] = insertelement <4 x i64> poison, i64 [[TMP3]], i32 0
86 ; CHECK-NEXT:    [[BROADCAST_SPLAT:%.*]] = shufflevector <4 x i64> [[BROADCAST_SPLATINSERT]], <4 x i64> poison, <4 x i32> zeroinitializer
87 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
88 ; CHECK:       vector.body:
89 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[PRED_STORE_CONTINUE6:%.*]] ]
90 ; CHECK-NEXT:    [[VEC_IND:%.*]] = phi <4 x i64> [ <i64 0, i64 1, i64 2, i64 3>, [[VECTOR_PH]] ], [ [[VEC_IND_NEXT:%.*]], [[PRED_STORE_CONTINUE6]] ]
91 ; CHECK-NEXT:    [[TMP4:%.*]] = icmp ule <4 x i64> [[VEC_IND]], [[BROADCAST_SPLAT]]
92 ; CHECK-NEXT:    [[TMP5:%.*]] = extractelement <4 x i1> [[TMP4]], i32 0
93 ; CHECK-NEXT:    br i1 [[TMP5]], label [[PRED_STORE_IF:%.*]], label [[PRED_STORE_CONTINUE:%.*]]
94 ; CHECK:       pred.store.if:
95 ; CHECK-NEXT:    [[TMP6:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[INDEX]]
96 ; CHECK-NEXT:    store i32 [[X:%.*]], i32* [[TMP6]], align 16
97 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE]]
98 ; CHECK:       pred.store.continue:
99 ; CHECK-NEXT:    [[TMP7:%.*]] = extractelement <4 x i1> [[TMP4]], i32 1
100 ; CHECK-NEXT:    br i1 [[TMP7]], label [[PRED_STORE_IF1:%.*]], label [[PRED_STORE_CONTINUE2:%.*]]
101 ; CHECK:       pred.store.if1:
102 ; CHECK-NEXT:    [[TMP8:%.*]] = or i64 [[INDEX]], 1
103 ; CHECK-NEXT:    [[TMP9:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[TMP8]]
104 ; CHECK-NEXT:    store i32 [[X]], i32* [[TMP9]], align 4
105 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE2]]
106 ; CHECK:       pred.store.continue2:
107 ; CHECK-NEXT:    [[TMP10:%.*]] = extractelement <4 x i1> [[TMP4]], i32 2
108 ; CHECK-NEXT:    br i1 [[TMP10]], label [[PRED_STORE_IF3:%.*]], label [[PRED_STORE_CONTINUE4:%.*]]
109 ; CHECK:       pred.store.if3:
110 ; CHECK-NEXT:    [[TMP11:%.*]] = or i64 [[INDEX]], 2
111 ; CHECK-NEXT:    [[TMP12:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[TMP11]]
112 ; CHECK-NEXT:    store i32 [[X]], i32* [[TMP12]], align 8
113 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE4]]
114 ; CHECK:       pred.store.continue4:
115 ; CHECK-NEXT:    [[TMP13:%.*]] = extractelement <4 x i1> [[TMP4]], i32 3
116 ; CHECK-NEXT:    br i1 [[TMP13]], label [[PRED_STORE_IF5:%.*]], label [[PRED_STORE_CONTINUE6]]
117 ; CHECK:       pred.store.if5:
118 ; CHECK-NEXT:    [[TMP14:%.*]] = or i64 [[INDEX]], 3
119 ; CHECK-NEXT:    [[TMP15:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[TMP14]]
120 ; CHECK-NEXT:    store i32 [[X]], i32* [[TMP15]], align 4
121 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE6]]
122 ; CHECK:       pred.store.continue6:
123 ; CHECK-NEXT:    [[INDEX_NEXT]] = add i64 [[INDEX]], 4
124 ; CHECK-NEXT:    [[VEC_IND_NEXT]] = add <4 x i64> [[VEC_IND]], <i64 4, i64 4, i64 4, i64 4>
125 ; CHECK-NEXT:    [[TMP16:%.*]] = icmp eq i64 [[INDEX_NEXT]], [[N_VEC]]
126 ; CHECK-NEXT:    br i1 [[TMP16]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP4:![0-9]+]]
127 ; CHECK:       middle.block:
128 ; CHECK-NEXT:    br i1 true, label [[DOT_PREHEADER_CRIT_EDGE:%.*]], label [[SCALAR_PH]]
129 ; CHECK:       scalar.ph:
130 ; CHECK-NEXT:    br label [[DOTLR_PH5:%.*]]
131 ; CHECK:       ..preheader_crit_edge:
132 ; CHECK-NEXT:    [[PHITMP:%.*]] = sext i32 [[N]] to i64
133 ; CHECK-NEXT:    br label [[DOTPREHEADER]]
134 ; CHECK:       .preheader:
135 ; CHECK-NEXT:    [[I_0_LCSSA:%.*]] = phi i64 [ [[PHITMP]], [[DOT_PREHEADER_CRIT_EDGE]] ], [ 0, [[TMP0:%.*]] ]
136 ; CHECK-NEXT:    [[TMP17:%.*]] = icmp eq i32 [[N]], 0
137 ; CHECK-NEXT:    br i1 [[TMP17]], label [[DOT_CRIT_EDGE:%.*]], label [[DOTLR_PH_PREHEADER:%.*]]
138 ; CHECK:       .lr.ph.preheader:
139 ; CHECK-NEXT:    [[TMP18:%.*]] = add i32 [[N]], -1
140 ; CHECK-NEXT:    [[TMP19:%.*]] = zext i32 [[TMP18]] to i64
141 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH8:%.*]], label [[VECTOR_PH10:%.*]]
142 ; CHECK:       vector.ph10:
143 ; CHECK-NEXT:    [[N_RND_UP11:%.*]] = add nuw nsw i64 [[TMP19]], 4
144 ; CHECK-NEXT:    [[N_VEC13:%.*]] = and i64 [[N_RND_UP11]], 8589934588
145 ; CHECK-NEXT:    [[BROADCAST_SPLATINSERT20:%.*]] = insertelement <4 x i64> poison, i64 [[TMP19]], i32 0
146 ; CHECK-NEXT:    [[BROADCAST_SPLAT21:%.*]] = shufflevector <4 x i64> [[BROADCAST_SPLATINSERT20]], <4 x i64> poison, <4 x i32> zeroinitializer
147 ; CHECK-NEXT:    br label [[VECTOR_BODY9:%.*]]
148 ; CHECK:       vector.body9:
149 ; CHECK-NEXT:    [[INDEX14:%.*]] = phi i64 [ 0, [[VECTOR_PH10]] ], [ [[INDEX_NEXT15:%.*]], [[PRED_STORE_CONTINUE37:%.*]] ]
150 ; CHECK-NEXT:    [[OFFSET_IDX:%.*]] = add i64 [[I_0_LCSSA]], [[INDEX14]]
151 ; CHECK-NEXT:    [[BROADCAST_SPLATINSERT28:%.*]] = insertelement <4 x i64> poison, i64 [[INDEX14]], i32 0
152 ; CHECK-NEXT:    [[BROADCAST_SPLAT29:%.*]] = shufflevector <4 x i64> [[BROADCAST_SPLATINSERT28]], <4 x i64> poison, <4 x i32> zeroinitializer
153 ; CHECK-NEXT:    [[VEC_IV:%.*]] = or <4 x i64> [[BROADCAST_SPLAT29]], <i64 0, i64 1, i64 2, i64 3>
154 ; CHECK-NEXT:    [[TMP20:%.*]] = icmp ule <4 x i64> [[VEC_IV]], [[BROADCAST_SPLAT21]]
155 ; CHECK-NEXT:    [[TMP21:%.*]] = extractelement <4 x i1> [[TMP20]], i32 0
156 ; CHECK-NEXT:    br i1 [[TMP21]], label [[PRED_STORE_IF30:%.*]], label [[PRED_STORE_CONTINUE31:%.*]]
157 ; CHECK:       pred.store.if30:
158 ; CHECK-NEXT:    [[TMP22:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[OFFSET_IDX]]
159 ; CHECK-NEXT:    [[TMP23:%.*]] = load i32, i32* [[TMP22]], align 4
160 ; CHECK-NEXT:    [[TMP24:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @c, i64 0, i64 [[OFFSET_IDX]]
161 ; CHECK-NEXT:    [[TMP25:%.*]] = load i32, i32* [[TMP24]], align 4
162 ; CHECK-NEXT:    [[TMP26:%.*]] = and i32 [[TMP25]], [[TMP23]]
163 ; CHECK-NEXT:    [[TMP27:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @a, i64 0, i64 [[OFFSET_IDX]]
164 ; CHECK-NEXT:    store i32 [[TMP26]], i32* [[TMP27]], align 4
165 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE31]]
166 ; CHECK:       pred.store.continue31:
167 ; CHECK-NEXT:    [[TMP28:%.*]] = extractelement <4 x i1> [[TMP20]], i32 1
168 ; CHECK-NEXT:    br i1 [[TMP28]], label [[PRED_STORE_IF32:%.*]], label [[PRED_STORE_CONTINUE33:%.*]]
169 ; CHECK:       pred.store.if32:
170 ; CHECK-NEXT:    [[TMP29:%.*]] = add i64 [[OFFSET_IDX]], 1
171 ; CHECK-NEXT:    [[TMP30:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[TMP29]]
172 ; CHECK-NEXT:    [[TMP31:%.*]] = load i32, i32* [[TMP30]], align 4
173 ; CHECK-NEXT:    [[TMP32:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @c, i64 0, i64 [[TMP29]]
174 ; CHECK-NEXT:    [[TMP33:%.*]] = load i32, i32* [[TMP32]], align 4
175 ; CHECK-NEXT:    [[TMP34:%.*]] = and i32 [[TMP33]], [[TMP31]]
176 ; CHECK-NEXT:    [[TMP35:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @a, i64 0, i64 [[TMP29]]
177 ; CHECK-NEXT:    store i32 [[TMP34]], i32* [[TMP35]], align 4
178 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE33]]
179 ; CHECK:       pred.store.continue33:
180 ; CHECK-NEXT:    [[TMP36:%.*]] = extractelement <4 x i1> [[TMP20]], i32 2
181 ; CHECK-NEXT:    br i1 [[TMP36]], label [[PRED_STORE_IF34:%.*]], label [[PRED_STORE_CONTINUE35:%.*]]
182 ; CHECK:       pred.store.if34:
183 ; CHECK-NEXT:    [[TMP37:%.*]] = add i64 [[OFFSET_IDX]], 2
184 ; CHECK-NEXT:    [[TMP38:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[TMP37]]
185 ; CHECK-NEXT:    [[TMP39:%.*]] = load i32, i32* [[TMP38]], align 4
186 ; CHECK-NEXT:    [[TMP40:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @c, i64 0, i64 [[TMP37]]
187 ; CHECK-NEXT:    [[TMP41:%.*]] = load i32, i32* [[TMP40]], align 4
188 ; CHECK-NEXT:    [[TMP42:%.*]] = and i32 [[TMP41]], [[TMP39]]
189 ; CHECK-NEXT:    [[TMP43:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @a, i64 0, i64 [[TMP37]]
190 ; CHECK-NEXT:    store i32 [[TMP42]], i32* [[TMP43]], align 4
191 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE35]]
192 ; CHECK:       pred.store.continue35:
193 ; CHECK-NEXT:    [[TMP44:%.*]] = extractelement <4 x i1> [[TMP20]], i32 3
194 ; CHECK-NEXT:    br i1 [[TMP44]], label [[PRED_STORE_IF36:%.*]], label [[PRED_STORE_CONTINUE37]]
195 ; CHECK:       pred.store.if36:
196 ; CHECK-NEXT:    [[TMP45:%.*]] = add i64 [[OFFSET_IDX]], 3
197 ; CHECK-NEXT:    [[TMP46:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 [[TMP45]]
198 ; CHECK-NEXT:    [[TMP47:%.*]] = load i32, i32* [[TMP46]], align 4
199 ; CHECK-NEXT:    [[TMP48:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @c, i64 0, i64 [[TMP45]]
200 ; CHECK-NEXT:    [[TMP49:%.*]] = load i32, i32* [[TMP48]], align 4
201 ; CHECK-NEXT:    [[TMP50:%.*]] = and i32 [[TMP49]], [[TMP47]]
202 ; CHECK-NEXT:    [[TMP51:%.*]] = getelementptr inbounds [2048 x i32], [2048 x i32]* @a, i64 0, i64 [[TMP45]]
203 ; CHECK-NEXT:    store i32 [[TMP50]], i32* [[TMP51]], align 4
204 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE37]]
205 ; CHECK:       pred.store.continue37:
206 ; CHECK-NEXT:    [[INDEX_NEXT15]] = add i64 [[INDEX14]], 4
207 ; CHECK-NEXT:    [[TMP52:%.*]] = icmp eq i64 [[INDEX_NEXT15]], [[N_VEC13]]
208 ; CHECK-NEXT:    br i1 [[TMP52]], label [[MIDDLE_BLOCK7:%.*]], label [[VECTOR_BODY9]], !llvm.loop [[LOOP5:![0-9]+]]
209 ; CHECK:       middle.block7:
210 ; CHECK-NEXT:    br i1 true, label [[DOT_CRIT_EDGE_LOOPEXIT:%.*]], label [[SCALAR_PH8]]
211 ; CHECK:       scalar.ph8:
212 ; CHECK-NEXT:    br label [[DOTLR_PH:%.*]]
213 ; CHECK:       .lr.ph5:
214 ; CHECK-NEXT:    br i1 undef, label [[DOT_PREHEADER_CRIT_EDGE]], label [[DOTLR_PH5]], !llvm.loop [[LOOP6:![0-9]+]]
215 ; CHECK:       .lr.ph:
216 ; CHECK-NEXT:    br i1 undef, label [[DOT_CRIT_EDGE_LOOPEXIT]], label [[DOTLR_PH]], !llvm.loop [[LOOP7:![0-9]+]]
217 ; CHECK:       ._crit_edge.loopexit:
218 ; CHECK-NEXT:    br label [[DOT_CRIT_EDGE]]
219 ; CHECK:       ._crit_edge:
220 ; CHECK-NEXT:    ret void
222   %1 = icmp sgt i32 %n, 0
223   br i1 %1, label %.lr.ph5, label %.preheader
225 ..preheader_crit_edge:                            ; preds = %.lr.ph5
226   %phitmp = sext i32 %n to i64
227   br label %.preheader
229 .preheader:                                       ; preds = %..preheader_crit_edge, %0
230   %i.0.lcssa = phi i64 [ %phitmp, %..preheader_crit_edge ], [ 0, %0 ]
231   %2 = icmp eq i32 %n, 0
232   br i1 %2, label %._crit_edge, label %.lr.ph
234 .lr.ph5:                                          ; preds = %0, %.lr.ph5
235   %indvars.iv6 = phi i64 [ %indvars.iv.next7, %.lr.ph5 ], [ 0, %0 ]
236   %3 = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 %indvars.iv6
237   store i32 %x, i32* %3, align 4
238   %indvars.iv.next7 = add i64 %indvars.iv6, 1
239   %lftr.wideiv = trunc i64 %indvars.iv.next7 to i32
240   %exitcond = icmp eq i32 %lftr.wideiv, %n
241   br i1 %exitcond, label %..preheader_crit_edge, label %.lr.ph5
243 .lr.ph:                                           ; preds = %.preheader, %.lr.ph
244   %indvars.iv = phi i64 [ %indvars.iv.next, %.lr.ph ], [ %i.0.lcssa, %.preheader ]
245   %.02 = phi i32 [ %4, %.lr.ph ], [ %n, %.preheader ]
246   %4 = add nsw i32 %.02, -1
247   %5 = getelementptr inbounds [2048 x i32], [2048 x i32]* @b, i64 0, i64 %indvars.iv
248   %6 = load i32, i32* %5, align 4
249   %7 = getelementptr inbounds [2048 x i32], [2048 x i32]* @c, i64 0, i64 %indvars.iv
250   %8 = load i32, i32* %7, align 4
251   %9 = and i32 %8, %6
252   %10 = getelementptr inbounds [2048 x i32], [2048 x i32]* @a, i64 0, i64 %indvars.iv
253   store i32 %9, i32* %10, align 4
254   %indvars.iv.next = add i64 %indvars.iv, 1
255   %11 = icmp eq i32 %4, 0
256   br i1 %11, label %._crit_edge, label %.lr.ph
258 ._crit_edge:                                      ; preds = %.lr.ph, %.preheader
259   ret void
262 ; Loop has no primary induction as its integer IV has step -1 starting at
263 ; unknown N, but can still be vectorized.
264 define void @example3(i32 %n, i32* noalias nocapture %p, i32* noalias nocapture %q) optsize {
265 ; CHECK-LABEL: @example3(
266 ; CHECK-NEXT:    [[TMP1:%.*]] = icmp eq i32 [[N:%.*]], 0
267 ; CHECK-NEXT:    br i1 [[TMP1]], label [[DOT_CRIT_EDGE:%.*]], label [[DOTLR_PH_PREHEADER:%.*]]
268 ; CHECK:       .lr.ph.preheader:
269 ; CHECK-NEXT:    [[TMP2:%.*]] = add i32 [[N]], -1
270 ; CHECK-NEXT:    [[TMP3:%.*]] = zext i32 [[TMP2]] to i64
271 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
272 ; CHECK:       vector.ph:
273 ; CHECK-NEXT:    [[N_RND_UP:%.*]] = add nuw nsw i64 [[TMP3]], 4
274 ; CHECK-NEXT:    [[N_VEC:%.*]] = and i64 [[N_RND_UP]], 8589934588
275 ; CHECK-NEXT:    [[BROADCAST_SPLATINSERT:%.*]] = insertelement <4 x i64> poison, i64 [[TMP3]], i32 0
276 ; CHECK-NEXT:    [[BROADCAST_SPLAT:%.*]] = shufflevector <4 x i64> [[BROADCAST_SPLATINSERT]], <4 x i64> poison, <4 x i32> zeroinitializer
277 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
278 ; CHECK:       vector.body:
279 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[PRED_STORE_CONTINUE21:%.*]] ]
280 ; CHECK-NEXT:    [[BROADCAST_SPLATINSERT14:%.*]] = insertelement <4 x i64> poison, i64 [[INDEX]], i32 0
281 ; CHECK-NEXT:    [[BROADCAST_SPLAT15:%.*]] = shufflevector <4 x i64> [[BROADCAST_SPLATINSERT14]], <4 x i64> poison, <4 x i32> zeroinitializer
282 ; CHECK-NEXT:    [[VEC_IV:%.*]] = or <4 x i64> [[BROADCAST_SPLAT15]], <i64 0, i64 1, i64 2, i64 3>
283 ; CHECK-NEXT:    [[TMP4:%.*]] = icmp ule <4 x i64> [[VEC_IV]], [[BROADCAST_SPLAT]]
284 ; CHECK-NEXT:    [[TMP5:%.*]] = extractelement <4 x i1> [[TMP4]], i32 0
285 ; CHECK-NEXT:    br i1 [[TMP5]], label [[PRED_STORE_IF:%.*]], label [[PRED_STORE_CONTINUE:%.*]]
286 ; CHECK:       pred.store.if:
287 ; CHECK-NEXT:    [[NEXT_GEP:%.*]] = getelementptr i32, i32* [[P:%.*]], i64 [[INDEX]]
288 ; CHECK-NEXT:    [[NEXT_GEP10:%.*]] = getelementptr i32, i32* [[Q:%.*]], i64 [[INDEX]]
289 ; CHECK-NEXT:    [[TMP6:%.*]] = load i32, i32* [[NEXT_GEP10]], align 16
290 ; CHECK-NEXT:    store i32 [[TMP6]], i32* [[NEXT_GEP]], align 16
291 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE]]
292 ; CHECK:       pred.store.continue:
293 ; CHECK-NEXT:    [[TMP7:%.*]] = extractelement <4 x i1> [[TMP4]], i32 1
294 ; CHECK-NEXT:    br i1 [[TMP7]], label [[PRED_STORE_IF16:%.*]], label [[PRED_STORE_CONTINUE17:%.*]]
295 ; CHECK:       pred.store.if16:
296 ; CHECK-NEXT:    [[TMP8:%.*]] = or i64 [[INDEX]], 1
297 ; CHECK-NEXT:    [[NEXT_GEP7:%.*]] = getelementptr i32, i32* [[P]], i64 [[TMP8]]
298 ; CHECK-NEXT:    [[TMP9:%.*]] = or i64 [[INDEX]], 1
299 ; CHECK-NEXT:    [[NEXT_GEP11:%.*]] = getelementptr i32, i32* [[Q]], i64 [[TMP9]]
300 ; CHECK-NEXT:    [[TMP10:%.*]] = load i32, i32* [[NEXT_GEP11]], align 16
301 ; CHECK-NEXT:    store i32 [[TMP10]], i32* [[NEXT_GEP7]], align 16
302 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE17]]
303 ; CHECK:       pred.store.continue17:
304 ; CHECK-NEXT:    [[TMP11:%.*]] = extractelement <4 x i1> [[TMP4]], i32 2
305 ; CHECK-NEXT:    br i1 [[TMP11]], label [[PRED_STORE_IF18:%.*]], label [[PRED_STORE_CONTINUE19:%.*]]
306 ; CHECK:       pred.store.if18:
307 ; CHECK-NEXT:    [[TMP12:%.*]] = or i64 [[INDEX]], 2
308 ; CHECK-NEXT:    [[NEXT_GEP8:%.*]] = getelementptr i32, i32* [[P]], i64 [[TMP12]]
309 ; CHECK-NEXT:    [[TMP13:%.*]] = or i64 [[INDEX]], 2
310 ; CHECK-NEXT:    [[NEXT_GEP12:%.*]] = getelementptr i32, i32* [[Q]], i64 [[TMP13]]
311 ; CHECK-NEXT:    [[TMP14:%.*]] = load i32, i32* [[NEXT_GEP12]], align 16
312 ; CHECK-NEXT:    store i32 [[TMP14]], i32* [[NEXT_GEP8]], align 16
313 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE19]]
314 ; CHECK:       pred.store.continue19:
315 ; CHECK-NEXT:    [[TMP15:%.*]] = extractelement <4 x i1> [[TMP4]], i32 3
316 ; CHECK-NEXT:    br i1 [[TMP15]], label [[PRED_STORE_IF20:%.*]], label [[PRED_STORE_CONTINUE21]]
317 ; CHECK:       pred.store.if20:
318 ; CHECK-NEXT:    [[TMP16:%.*]] = or i64 [[INDEX]], 3
319 ; CHECK-NEXT:    [[NEXT_GEP9:%.*]] = getelementptr i32, i32* [[P]], i64 [[TMP16]]
320 ; CHECK-NEXT:    [[TMP17:%.*]] = or i64 [[INDEX]], 3
321 ; CHECK-NEXT:    [[NEXT_GEP13:%.*]] = getelementptr i32, i32* [[Q]], i64 [[TMP17]]
322 ; CHECK-NEXT:    [[TMP18:%.*]] = load i32, i32* [[NEXT_GEP13]], align 16
323 ; CHECK-NEXT:    store i32 [[TMP18]], i32* [[NEXT_GEP9]], align 16
324 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE21]]
325 ; CHECK:       pred.store.continue21:
326 ; CHECK-NEXT:    [[INDEX_NEXT]] = add i64 [[INDEX]], 4
327 ; CHECK-NEXT:    [[TMP19:%.*]] = icmp eq i64 [[INDEX_NEXT]], [[N_VEC]]
328 ; CHECK-NEXT:    br i1 [[TMP19]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP8:![0-9]+]]
329 ; CHECK:       middle.block:
330 ; CHECK-NEXT:    br i1 true, label [[DOT_CRIT_EDGE_LOOPEXIT:%.*]], label [[SCALAR_PH]]
331 ; CHECK:       scalar.ph:
332 ; CHECK-NEXT:    br label [[DOTLR_PH:%.*]]
333 ; CHECK:       .lr.ph:
334 ; CHECK-NEXT:    br i1 undef, label [[DOT_CRIT_EDGE_LOOPEXIT]], label [[DOTLR_PH]], !llvm.loop [[LOOP9:![0-9]+]]
335 ; CHECK:       ._crit_edge.loopexit:
336 ; CHECK-NEXT:    br label [[DOT_CRIT_EDGE]]
337 ; CHECK:       ._crit_edge:
338 ; CHECK-NEXT:    ret void
340   %1 = icmp eq i32 %n, 0
341   br i1 %1, label %._crit_edge, label %.lr.ph
343 .lr.ph:                                           ; preds = %0, %.lr.ph
344   %.05 = phi i32 [ %2, %.lr.ph ], [ %n, %0 ]
345   %.014 = phi i32* [ %5, %.lr.ph ], [ %p, %0 ]
346   %.023 = phi i32* [ %3, %.lr.ph ], [ %q, %0 ]
347   %2 = add nsw i32 %.05, -1
348   %3 = getelementptr inbounds i32, i32* %.023, i64 1
349   %4 = load i32, i32* %.023, align 16
350   %5 = getelementptr inbounds i32, i32* %.014, i64 1
351   store i32 %4, i32* %.014, align 16
352   %6 = icmp eq i32 %2, 0
353   br i1 %6, label %._crit_edge, label %.lr.ph
355 ._crit_edge:                                      ; preds = %.lr.ph, %0
356   ret void
359 ; We can't vectorize this one because we need a runtime ptr check.
360 define void @example23(i16* nocapture %src, i32* nocapture %dst) optsize {
361 ; CHECK-LABEL: @example23(
362 ; CHECK-NEXT:    br label [[TMP1:%.*]]
363 ; CHECK:       1:
364 ; CHECK-NEXT:    [[DOT04:%.*]] = phi i16* [ [[SRC:%.*]], [[TMP0:%.*]] ], [ [[TMP2:%.*]], [[TMP1]] ]
365 ; CHECK-NEXT:    [[DOT013:%.*]] = phi i32* [ [[DST:%.*]], [[TMP0]] ], [ [[TMP6:%.*]], [[TMP1]] ]
366 ; CHECK-NEXT:    [[I_02:%.*]] = phi i32 [ 0, [[TMP0]] ], [ [[TMP7:%.*]], [[TMP1]] ]
367 ; CHECK-NEXT:    [[TMP2]] = getelementptr inbounds i16, i16* [[DOT04]], i64 1
368 ; CHECK-NEXT:    [[TMP3:%.*]] = load i16, i16* [[DOT04]], align 2
369 ; CHECK-NEXT:    [[TMP4:%.*]] = zext i16 [[TMP3]] to i32
370 ; CHECK-NEXT:    [[TMP5:%.*]] = shl nuw nsw i32 [[TMP4]], 7
371 ; CHECK-NEXT:    [[TMP6]] = getelementptr inbounds i32, i32* [[DOT013]], i64 1
372 ; CHECK-NEXT:    store i32 [[TMP5]], i32* [[DOT013]], align 4
373 ; CHECK-NEXT:    [[TMP7]] = add nuw nsw i32 [[I_02]], 1
374 ; CHECK-NEXT:    [[EXITCOND:%.*]] = icmp eq i32 [[TMP7]], 256
375 ; CHECK-NEXT:    br i1 [[EXITCOND]], label [[TMP8:%.*]], label [[TMP1]]
376 ; CHECK:       8:
377 ; CHECK-NEXT:    ret void
379   br label %1
381 ; <label>:1                                       ; preds = %1, %0
382   %.04 = phi i16* [ %src, %0 ], [ %2, %1 ]
383   %.013 = phi i32* [ %dst, %0 ], [ %6, %1 ]
384   %i.02 = phi i32 [ 0, %0 ], [ %7, %1 ]
385   %2 = getelementptr inbounds i16, i16* %.04, i64 1
386   %3 = load i16, i16* %.04, align 2
387   %4 = zext i16 %3 to i32
388   %5 = shl nuw nsw i32 %4, 7
389   %6 = getelementptr inbounds i32, i32* %.013, i64 1
390   store i32 %5, i32* %.013, align 4
391   %7 = add nsw i32 %i.02, 1
392   %exitcond = icmp eq i32 %7, 256
393   br i1 %exitcond, label %8, label %1
395 ; <label>:8                                       ; preds = %1
396   ret void
400 ; We CAN vectorize this example because the pointers are marked as noalias.
401 define void @example23b(i16* noalias nocapture %src, i32* noalias nocapture %dst) optsize {
402 ; CHECK-LABEL: @example23b(
403 ; CHECK-NEXT:    br i1 false, label [[SCALAR_PH:%.*]], label [[VECTOR_PH:%.*]]
404 ; CHECK:       vector.ph:
405 ; CHECK-NEXT:    br label [[VECTOR_BODY:%.*]]
406 ; CHECK:       vector.body:
407 ; CHECK-NEXT:    [[INDEX:%.*]] = phi i64 [ 0, [[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], [[VECTOR_BODY]] ]
408 ; CHECK-NEXT:    [[NEXT_GEP:%.*]] = getelementptr i16, i16* [[SRC:%.*]], i64 [[INDEX]]
409 ; CHECK-NEXT:    [[NEXT_GEP4:%.*]] = getelementptr i32, i32* [[DST:%.*]], i64 [[INDEX]]
410 ; CHECK-NEXT:    [[TMP1:%.*]] = bitcast i16* [[NEXT_GEP]] to <4 x i16>*
411 ; CHECK-NEXT:    [[WIDE_LOAD:%.*]] = load <4 x i16>, <4 x i16>* [[TMP1]], align 2
412 ; CHECK-NEXT:    [[TMP2:%.*]] = zext <4 x i16> [[WIDE_LOAD]] to <4 x i32>
413 ; CHECK-NEXT:    [[TMP3:%.*]] = shl nuw nsw <4 x i32> [[TMP2]], <i32 7, i32 7, i32 7, i32 7>
414 ; CHECK-NEXT:    [[TMP4:%.*]] = bitcast i32* [[NEXT_GEP4]] to <4 x i32>*
415 ; CHECK-NEXT:    store <4 x i32> [[TMP3]], <4 x i32>* [[TMP4]], align 4
416 ; CHECK-NEXT:    [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 4
417 ; CHECK-NEXT:    [[TMP5:%.*]] = icmp eq i64 [[INDEX_NEXT]], 256
418 ; CHECK-NEXT:    br i1 [[TMP5]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP10:![0-9]+]]
419 ; CHECK:       middle.block:
420 ; CHECK-NEXT:    br i1 true, label [[TMP7:%.*]], label [[SCALAR_PH]]
421 ; CHECK:       scalar.ph:
422 ; CHECK-NEXT:    br label [[TMP6:%.*]]
423 ; CHECK:       6:
424 ; CHECK-NEXT:    br i1 undef, label [[TMP7]], label [[TMP6]], !llvm.loop [[LOOP11:![0-9]+]]
425 ; CHECK:       7:
426 ; CHECK-NEXT:    ret void
428   br label %1
430 ; <label>:1                                       ; preds = %1, %0
431   %.04 = phi i16* [ %src, %0 ], [ %2, %1 ]
432   %.013 = phi i32* [ %dst, %0 ], [ %6, %1 ]
433   %i.02 = phi i32 [ 0, %0 ], [ %7, %1 ]
434   %2 = getelementptr inbounds i16, i16* %.04, i64 1
435   %3 = load i16, i16* %.04, align 2
436   %4 = zext i16 %3 to i32
437   %5 = shl nuw nsw i32 %4, 7
438   %6 = getelementptr inbounds i32, i32* %.013, i64 1
439   store i32 %5, i32* %.013, align 4
440   %7 = add nsw i32 %i.02, 1
441   %exitcond = icmp eq i32 %7, 256
442   br i1 %exitcond, label %8, label %1
444 ; <label>:8                                       ; preds = %1
445   ret void
448 ; We CAN vectorize this example by folding the tail it entails.
449 define void @example23c(i16* noalias nocapture %src, i32* noalias nocapture %dst) optsize {
450 ; CHECK-LABEL: @example23c(
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_CONTINUE16:%.*]] ]
456 ; CHECK-NEXT:    [[BROADCAST_SPLATINSERT:%.*]] = insertelement <4 x i64> poison, i64 [[INDEX]], i32 0
457 ; CHECK-NEXT:    [[BROADCAST_SPLAT:%.*]] = shufflevector <4 x i64> [[BROADCAST_SPLATINSERT]], <4 x i64> poison, <4 x i32> zeroinitializer
458 ; CHECK-NEXT:    [[INDUCTION:%.*]] = or <4 x i64> [[BROADCAST_SPLAT]], <i64 0, i64 1, i64 2, i64 3>
459 ; CHECK-NEXT:    [[TMP1:%.*]] = icmp ult <4 x i64> [[INDUCTION]], <i64 257, i64 257, i64 257, i64 257>
460 ; CHECK-NEXT:    [[TMP2:%.*]] = extractelement <4 x i1> [[TMP1]], i32 0
461 ; CHECK-NEXT:    br i1 [[TMP2]], label [[PRED_STORE_IF:%.*]], label [[PRED_STORE_CONTINUE:%.*]]
462 ; CHECK:       pred.store.if:
463 ; CHECK-NEXT:    [[NEXT_GEP7:%.*]] = getelementptr i32, i32* [[DST:%.*]], i64 [[INDEX]]
464 ; CHECK-NEXT:    [[NEXT_GEP:%.*]] = getelementptr i16, i16* [[SRC:%.*]], i64 [[INDEX]]
465 ; CHECK-NEXT:    [[TMP3:%.*]] = load i16, i16* [[NEXT_GEP]], align 2
466 ; CHECK-NEXT:    [[TMP4:%.*]] = zext i16 [[TMP3]] to i32
467 ; CHECK-NEXT:    [[TMP5:%.*]] = shl nuw nsw i32 [[TMP4]], 7
468 ; CHECK-NEXT:    store i32 [[TMP5]], i32* [[NEXT_GEP7]], align 4
469 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE]]
470 ; CHECK:       pred.store.continue:
471 ; CHECK-NEXT:    [[TMP6:%.*]] = extractelement <4 x i1> [[TMP1]], i32 1
472 ; CHECK-NEXT:    br i1 [[TMP6]], label [[PRED_STORE_IF11:%.*]], label [[PRED_STORE_CONTINUE12:%.*]]
473 ; CHECK:       pred.store.if11:
474 ; CHECK-NEXT:    [[TMP7:%.*]] = or i64 [[INDEX]], 1
475 ; CHECK-NEXT:    [[NEXT_GEP8:%.*]] = getelementptr i32, i32* [[DST]], i64 [[TMP7]]
476 ; CHECK-NEXT:    [[TMP8:%.*]] = or i64 [[INDEX]], 1
477 ; CHECK-NEXT:    [[NEXT_GEP4:%.*]] = getelementptr i16, i16* [[SRC]], i64 [[TMP8]]
478 ; CHECK-NEXT:    [[TMP9:%.*]] = load i16, i16* [[NEXT_GEP4]], align 2
479 ; CHECK-NEXT:    [[TMP10:%.*]] = zext i16 [[TMP9]] to i32
480 ; CHECK-NEXT:    [[TMP11:%.*]] = shl nuw nsw i32 [[TMP10]], 7
481 ; CHECK-NEXT:    store i32 [[TMP11]], i32* [[NEXT_GEP8]], align 4
482 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE12]]
483 ; CHECK:       pred.store.continue12:
484 ; CHECK-NEXT:    [[TMP12:%.*]] = extractelement <4 x i1> [[TMP1]], i32 2
485 ; CHECK-NEXT:    br i1 [[TMP12]], label [[PRED_STORE_IF13:%.*]], label [[PRED_STORE_CONTINUE14:%.*]]
486 ; CHECK:       pred.store.if13:
487 ; CHECK-NEXT:    [[TMP13:%.*]] = or i64 [[INDEX]], 2
488 ; CHECK-NEXT:    [[NEXT_GEP9:%.*]] = getelementptr i32, i32* [[DST]], i64 [[TMP13]]
489 ; CHECK-NEXT:    [[TMP14:%.*]] = or i64 [[INDEX]], 2
490 ; CHECK-NEXT:    [[NEXT_GEP5:%.*]] = getelementptr i16, i16* [[SRC]], i64 [[TMP14]]
491 ; CHECK-NEXT:    [[TMP15:%.*]] = load i16, i16* [[NEXT_GEP5]], align 2
492 ; CHECK-NEXT:    [[TMP16:%.*]] = zext i16 [[TMP15]] to i32
493 ; CHECK-NEXT:    [[TMP17:%.*]] = shl nuw nsw i32 [[TMP16]], 7
494 ; CHECK-NEXT:    store i32 [[TMP17]], i32* [[NEXT_GEP9]], align 4
495 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE14]]
496 ; CHECK:       pred.store.continue14:
497 ; CHECK-NEXT:    [[TMP18:%.*]] = extractelement <4 x i1> [[TMP1]], i32 3
498 ; CHECK-NEXT:    br i1 [[TMP18]], label [[PRED_STORE_IF15:%.*]], label [[PRED_STORE_CONTINUE16]]
499 ; CHECK:       pred.store.if15:
500 ; CHECK-NEXT:    [[TMP19:%.*]] = or i64 [[INDEX]], 3
501 ; CHECK-NEXT:    [[NEXT_GEP10:%.*]] = getelementptr i32, i32* [[DST]], i64 [[TMP19]]
502 ; CHECK-NEXT:    [[TMP20:%.*]] = or i64 [[INDEX]], 3
503 ; CHECK-NEXT:    [[NEXT_GEP6:%.*]] = getelementptr i16, i16* [[SRC]], i64 [[TMP20]]
504 ; CHECK-NEXT:    [[TMP21:%.*]] = load i16, i16* [[NEXT_GEP6]], align 2
505 ; CHECK-NEXT:    [[TMP22:%.*]] = zext i16 [[TMP21]] to i32
506 ; CHECK-NEXT:    [[TMP23:%.*]] = shl nuw nsw i32 [[TMP22]], 7
507 ; CHECK-NEXT:    store i32 [[TMP23]], i32* [[NEXT_GEP10]], align 4
508 ; CHECK-NEXT:    br label [[PRED_STORE_CONTINUE16]]
509 ; CHECK:       pred.store.continue16:
510 ; CHECK-NEXT:    [[INDEX_NEXT]] = add i64 [[INDEX]], 4
511 ; CHECK-NEXT:    [[TMP24:%.*]] = icmp eq i64 [[INDEX_NEXT]], 260
512 ; CHECK-NEXT:    br i1 [[TMP24]], label [[MIDDLE_BLOCK:%.*]], label [[VECTOR_BODY]], !llvm.loop [[LOOP12:![0-9]+]]
513 ; CHECK:       middle.block:
514 ; CHECK-NEXT:    br i1 true, label [[TMP26:%.*]], label [[SCALAR_PH]]
515 ; CHECK:       scalar.ph:
516 ; CHECK-NEXT:    br label [[TMP25:%.*]]
517 ; CHECK:       25:
518 ; CHECK-NEXT:    br i1 undef, label [[TMP26]], label [[TMP25]], !llvm.loop [[LOOP13:![0-9]+]]
519 ; CHECK:       26:
520 ; CHECK-NEXT:    ret void
522   br label %1
524 ; <label>:1                                       ; preds = %1, %0
525   %.04 = phi i16* [ %src, %0 ], [ %2, %1 ]
526   %.013 = phi i32* [ %dst, %0 ], [ %6, %1 ]
527   %i.02 = phi i64 [ 0, %0 ], [ %7, %1 ]
528   %2 = getelementptr inbounds i16, i16* %.04, i64 1
529   %3 = load i16, i16* %.04, align 2
530   %4 = zext i16 %3 to i32
531   %5 = shl nuw nsw i32 %4, 7
532   %6 = getelementptr inbounds i32, i32* %.013, i64 1
533   store i32 %5, i32* %.013, align 4
534   %7 = add nsw i64 %i.02, 1
535   %exitcond = icmp eq i64 %7, 257
536   br i1 %exitcond, label %8, label %1
538 ; <label>:8                                       ; preds = %1
539   ret void
542 ; We CAN'T vectorize this example because it would entail a tail and an
543 ; induction is used outside the loop.
544 define i64 @example23d(i16* noalias nocapture %src, i32* noalias nocapture %dst) optsize {
545 ; CHECK-LABEL: @example23d(
546 ; CHECK-NEXT:    br label [[TMP1:%.*]]
547 ; CHECK:       1:
548 ; CHECK-NEXT:    [[DOT04:%.*]] = phi i16* [ [[SRC:%.*]], [[TMP0:%.*]] ], [ [[TMP2:%.*]], [[TMP1]] ]
549 ; CHECK-NEXT:    [[DOT013:%.*]] = phi i32* [ [[DST:%.*]], [[TMP0]] ], [ [[TMP6:%.*]], [[TMP1]] ]
550 ; CHECK-NEXT:    [[I_02:%.*]] = phi i64 [ 0, [[TMP0]] ], [ [[TMP7:%.*]], [[TMP1]] ]
551 ; CHECK-NEXT:    [[TMP2]] = getelementptr inbounds i16, i16* [[DOT04]], i64 1
552 ; CHECK-NEXT:    [[TMP3:%.*]] = load i16, i16* [[DOT04]], align 2
553 ; CHECK-NEXT:    [[TMP4:%.*]] = zext i16 [[TMP3]] to i32
554 ; CHECK-NEXT:    [[TMP5:%.*]] = shl nuw nsw i32 [[TMP4]], 7
555 ; CHECK-NEXT:    [[TMP6]] = getelementptr inbounds i32, i32* [[DOT013]], i64 1
556 ; CHECK-NEXT:    store i32 [[TMP5]], i32* [[DOT013]], align 4
557 ; CHECK-NEXT:    [[TMP7]] = add nuw nsw i64 [[I_02]], 1
558 ; CHECK-NEXT:    [[EXITCOND:%.*]] = icmp eq i64 [[TMP7]], 257
559 ; CHECK-NEXT:    br i1 [[EXITCOND]], label [[TMP8:%.*]], label [[TMP1]]
560 ; CHECK:       8:
561 ; CHECK-NEXT:    ret i64 [[TMP7]]
563   br label %1
565 ; <label>:1                                       ; preds = %1, %0
566   %.04 = phi i16* [ %src, %0 ], [ %2, %1 ]
567   %.013 = phi i32* [ %dst, %0 ], [ %6, %1 ]
568   %i.02 = phi i64 [ 0, %0 ], [ %7, %1 ]
569   %2 = getelementptr inbounds i16, i16* %.04, i64 1
570   %3 = load i16, i16* %.04, align 2
571   %4 = zext i16 %3 to i32
572   %5 = shl nuw nsw i32 %4, 7
573   %6 = getelementptr inbounds i32, i32* %.013, i64 1
574   store i32 %5, i32* %.013, align 4
575   %7 = add nsw i64 %i.02, 1
576   %exitcond = icmp eq i64 %7, 257
577   br i1 %exitcond, label %8, label %1
579 ; <label>:8                                       ; preds = %1
580   ret i64 %7