blob: 63b138f1d56010bf16cc0fd2d0bb8bbe76cc44fe [file] [log] [blame]
Sanjay Patelb653de12014-09-10 17:58:16 +00001; RUN: opt < %s -loop-vectorize -force-vector-interleave=1 -force-vector-width=4 -dce -instcombine -S | FileCheck %s
Nadav Rotem4f7f7272012-10-19 23:05:40 +00002
3target 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"
4target triple = "x86_64-apple-macosx10.8.0"
5
Stephen Linc1c7a132013-07-14 01:42:54 +00006;CHECK-LABEL: @reduction_sum(
Nadav Rotem4f7f7272012-10-19 23:05:40 +00007;CHECK: phi <4 x i32>
8;CHECK: load <4 x i32>
9;CHECK: add <4 x i32>
Benjamin Kramerf0e5d2f2012-12-18 18:40:20 +000010;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 2, i32 3, i32 undef, i32 undef>
11;CHECK: add <4 x i32>
12;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 1, i32 undef, i32 undef, i32 undef>
13;CHECK: add <4 x i32>
14;CHECK: extractelement <4 x i32> %{{.*}}, i32 0
Nadav Rotem4f7f7272012-10-19 23:05:40 +000015;CHECK: ret i32
16define i32 @reduction_sum(i32 %n, i32* noalias nocapture %A, i32* noalias nocapture %B) nounwind uwtable readonly noinline ssp {
17 %1 = icmp sgt i32 %n, 0
18 br i1 %1, label %.lr.ph, label %._crit_edge
19
20.lr.ph: ; preds = %0, %.lr.ph
21 %indvars.iv = phi i64 [ %indvars.iv.next, %.lr.ph ], [ 0, %0 ]
22 %sum.02 = phi i32 [ %9, %.lr.ph ], [ 0, %0 ]
David Blaikie79e6c742015-02-27 19:29:02 +000023 %2 = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +000024 %3 = load i32, i32* %2, align 4
David Blaikie79e6c742015-02-27 19:29:02 +000025 %4 = getelementptr inbounds i32, i32* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +000026 %5 = load i32, i32* %4, align 4
Nadav Rotem4f7f7272012-10-19 23:05:40 +000027 %6 = trunc i64 %indvars.iv to i32
28 %7 = add i32 %sum.02, %6
29 %8 = add i32 %7, %3
30 %9 = add i32 %8, %5
31 %indvars.iv.next = add i64 %indvars.iv, 1
32 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
33 %exitcond = icmp eq i32 %lftr.wideiv, %n
34 br i1 %exitcond, label %._crit_edge, label %.lr.ph
35
36._crit_edge: ; preds = %.lr.ph, %0
37 %sum.0.lcssa = phi i32 [ 0, %0 ], [ %9, %.lr.ph ]
38 ret i32 %sum.0.lcssa
39}
40
Stephen Linc1c7a132013-07-14 01:42:54 +000041;CHECK-LABEL: @reduction_prod(
Nadav Rotem4f7f7272012-10-19 23:05:40 +000042;CHECK: phi <4 x i32>
43;CHECK: load <4 x i32>
44;CHECK: mul <4 x i32>
Benjamin Kramerf0e5d2f2012-12-18 18:40:20 +000045;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 2, i32 3, i32 undef, i32 undef>
46;CHECK: mul <4 x i32>
47;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 1, i32 undef, i32 undef, i32 undef>
48;CHECK: mul <4 x i32>
49;CHECK: extractelement <4 x i32> %{{.*}}, i32 0
Nadav Rotem4f7f7272012-10-19 23:05:40 +000050;CHECK: ret i32
51define i32 @reduction_prod(i32 %n, i32* noalias nocapture %A, i32* noalias nocapture %B) nounwind uwtable readonly noinline ssp {
52 %1 = icmp sgt i32 %n, 0
53 br i1 %1, label %.lr.ph, label %._crit_edge
54
55.lr.ph: ; preds = %0, %.lr.ph
56 %indvars.iv = phi i64 [ %indvars.iv.next, %.lr.ph ], [ 0, %0 ]
57 %prod.02 = phi i32 [ %9, %.lr.ph ], [ 1, %0 ]
David Blaikie79e6c742015-02-27 19:29:02 +000058 %2 = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +000059 %3 = load i32, i32* %2, align 4
David Blaikie79e6c742015-02-27 19:29:02 +000060 %4 = getelementptr inbounds i32, i32* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +000061 %5 = load i32, i32* %4, align 4
Nadav Rotem4f7f7272012-10-19 23:05:40 +000062 %6 = trunc i64 %indvars.iv to i32
63 %7 = mul i32 %prod.02, %6
64 %8 = mul i32 %7, %3
65 %9 = mul i32 %8, %5
66 %indvars.iv.next = add i64 %indvars.iv, 1
67 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
68 %exitcond = icmp eq i32 %lftr.wideiv, %n
69 br i1 %exitcond, label %._crit_edge, label %.lr.ph
70
71._crit_edge: ; preds = %.lr.ph, %0
72 %prod.0.lcssa = phi i32 [ 1, %0 ], [ %9, %.lr.ph ]
73 ret i32 %prod.0.lcssa
74}
75
Stephen Linc1c7a132013-07-14 01:42:54 +000076;CHECK-LABEL: @reduction_mix(
Nadav Rotem4f7f7272012-10-19 23:05:40 +000077;CHECK: phi <4 x i32>
78;CHECK: load <4 x i32>
Nadav Rotem4cb8cda2012-10-31 21:40:39 +000079;CHECK: mul nsw <4 x i32>
Benjamin Kramerf0e5d2f2012-12-18 18:40:20 +000080;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 2, i32 3, i32 undef, i32 undef>
81;CHECK: add <4 x i32>
82;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 1, i32 undef, i32 undef, i32 undef>
83;CHECK: add <4 x i32>
84;CHECK: extractelement <4 x i32> %{{.*}}, i32 0
Nadav Rotem4f7f7272012-10-19 23:05:40 +000085;CHECK: ret i32
86define i32 @reduction_mix(i32 %n, i32* noalias nocapture %A, i32* noalias nocapture %B) nounwind uwtable readonly noinline ssp {
87 %1 = icmp sgt i32 %n, 0
88 br i1 %1, label %.lr.ph, label %._crit_edge
89
90.lr.ph: ; preds = %0, %.lr.ph
91 %indvars.iv = phi i64 [ %indvars.iv.next, %.lr.ph ], [ 0, %0 ]
92 %sum.02 = phi i32 [ %9, %.lr.ph ], [ 0, %0 ]
David Blaikie79e6c742015-02-27 19:29:02 +000093 %2 = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +000094 %3 = load i32, i32* %2, align 4
David Blaikie79e6c742015-02-27 19:29:02 +000095 %4 = getelementptr inbounds i32, i32* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +000096 %5 = load i32, i32* %4, align 4
Nadav Rotem4f7f7272012-10-19 23:05:40 +000097 %6 = mul nsw i32 %5, %3
98 %7 = trunc i64 %indvars.iv to i32
99 %8 = add i32 %sum.02, %7
100 %9 = add i32 %8, %6
101 %indvars.iv.next = add i64 %indvars.iv, 1
102 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
103 %exitcond = icmp eq i32 %lftr.wideiv, %n
104 br i1 %exitcond, label %._crit_edge, label %.lr.ph
105
106._crit_edge: ; preds = %.lr.ph, %0
107 %sum.0.lcssa = phi i32 [ 0, %0 ], [ %9, %.lr.ph ]
108 ret i32 %sum.0.lcssa
109}
110
Stephen Linc1c7a132013-07-14 01:42:54 +0000111;CHECK-LABEL: @reduction_mul(
Nadav Rotemc1679a92012-10-21 05:52:51 +0000112;CHECK: mul <4 x i32>
Benjamin Kramerf0e5d2f2012-12-18 18:40:20 +0000113;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 2, i32 3, i32 undef, i32 undef>
114;CHECK: mul <4 x i32>
115;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 1, i32 undef, i32 undef, i32 undef>
116;CHECK: mul <4 x i32>
117;CHECK: extractelement <4 x i32> %{{.*}}, i32 0
Nadav Rotem4f7f7272012-10-19 23:05:40 +0000118;CHECK: ret i32
Nadav Rotemc1679a92012-10-21 05:52:51 +0000119define i32 @reduction_mul(i32 %n, i32* noalias nocapture %A, i32* noalias nocapture %B) nounwind uwtable readonly noinline ssp {
Nadav Rotem4f7f7272012-10-19 23:05:40 +0000120 %1 = icmp sgt i32 %n, 0
121 br i1 %1, label %.lr.ph, label %._crit_edge
122
123.lr.ph: ; preds = %0, %.lr.ph
124 %indvars.iv = phi i64 [ %indvars.iv.next, %.lr.ph ], [ 0, %0 ]
Nadav Rotemc1679a92012-10-21 05:52:51 +0000125 %sum.02 = phi i32 [ %9, %.lr.ph ], [ 19, %0 ]
David Blaikie79e6c742015-02-27 19:29:02 +0000126 %2 = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000127 %3 = load i32, i32* %2, align 4
David Blaikie79e6c742015-02-27 19:29:02 +0000128 %4 = getelementptr inbounds i32, i32* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000129 %5 = load i32, i32* %4, align 4
Nadav Rotem4f7f7272012-10-19 23:05:40 +0000130 %6 = trunc i64 %indvars.iv to i32
131 %7 = add i32 %3, %6
132 %8 = add i32 %7, %5
133 %9 = mul i32 %8, %sum.02
134 %indvars.iv.next = add i64 %indvars.iv, 1
135 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
136 %exitcond = icmp eq i32 %lftr.wideiv, %n
137 br i1 %exitcond, label %._crit_edge, label %.lr.ph
138
139._crit_edge: ; preds = %.lr.ph, %0
140 %sum.0.lcssa = phi i32 [ 0, %0 ], [ %9, %.lr.ph ]
141 ret i32 %sum.0.lcssa
142}
Nadav Rotemc1679a92012-10-21 05:52:51 +0000143
Stephen Linc1c7a132013-07-14 01:42:54 +0000144;CHECK-LABEL: @start_at_non_zero(
Nadav Rotemc1679a92012-10-21 05:52:51 +0000145;CHECK: phi <4 x i32>
146;CHECK: <i32 120, i32 0, i32 0, i32 0>
Benjamin Kramerf0e5d2f2012-12-18 18:40:20 +0000147;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 2, i32 3, i32 undef, i32 undef>
148;CHECK: add <4 x i32>
149;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 1, i32 undef, i32 undef, i32 undef>
150;CHECK: add <4 x i32>
151;CHECK: extractelement <4 x i32> %{{.*}}, i32 0
Nadav Rotemc1679a92012-10-21 05:52:51 +0000152;CHECK: ret i32
153define i32 @start_at_non_zero(i32* nocapture %in, i32* nocapture %coeff, i32* nocapture %out, i32 %n) nounwind uwtable readonly ssp {
154entry:
155 %cmp7 = icmp sgt i32 %n, 0
156 br i1 %cmp7, label %for.body, label %for.end
157
158for.body: ; preds = %entry, %for.body
159 %indvars.iv = phi i64 [ %indvars.iv.next, %for.body ], [ 0, %entry ]
160 %sum.09 = phi i32 [ %add, %for.body ], [ 120, %entry ]
David Blaikie79e6c742015-02-27 19:29:02 +0000161 %arrayidx = getelementptr inbounds i32, i32* %in, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000162 %0 = load i32, i32* %arrayidx, align 4
David Blaikie79e6c742015-02-27 19:29:02 +0000163 %arrayidx2 = getelementptr inbounds i32, i32* %coeff, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000164 %1 = load i32, i32* %arrayidx2, align 4
Nadav Rotemc1679a92012-10-21 05:52:51 +0000165 %mul = mul nsw i32 %1, %0
166 %add = add nsw i32 %mul, %sum.09
167 %indvars.iv.next = add i64 %indvars.iv, 1
168 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
169 %exitcond = icmp eq i32 %lftr.wideiv, %n
170 br i1 %exitcond, label %for.end, label %for.body
171
172for.end: ; preds = %for.body, %entry
173 %sum.0.lcssa = phi i32 [ 120, %entry ], [ %add, %for.body ]
174 ret i32 %sum.0.lcssa
175}
176
Stephen Linc1c7a132013-07-14 01:42:54 +0000177;CHECK-LABEL: @reduction_and(
Nadav Rotemb3e8e682012-10-30 18:12:36 +0000178;CHECK: <i32 -1, i32 -1, i32 -1, i32 -1>
James Molloy1e583702015-09-02 10:15:39 +0000179;CHECK: and <4 x i32>
Benjamin Kramerf0e5d2f2012-12-18 18:40:20 +0000180;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 2, i32 3, i32 undef, i32 undef>
181;CHECK: and <4 x i32>
182;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 1, i32 undef, i32 undef, i32 undef>
183;CHECK: and <4 x i32>
184;CHECK: extractelement <4 x i32> %{{.*}}, i32 0
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000185;CHECK: ret i32
186define i32 @reduction_and(i32 %n, i32* nocapture %A, i32* nocapture %B) nounwind uwtable readonly {
187entry:
188 %cmp7 = icmp sgt i32 %n, 0
189 br i1 %cmp7, label %for.body, label %for.end
Nadav Rotemc1679a92012-10-21 05:52:51 +0000190
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000191for.body: ; preds = %entry, %for.body
192 %indvars.iv = phi i64 [ %indvars.iv.next, %for.body ], [ 0, %entry ]
193 %result.08 = phi i32 [ %and, %for.body ], [ -1, %entry ]
David Blaikie79e6c742015-02-27 19:29:02 +0000194 %arrayidx = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000195 %0 = load i32, i32* %arrayidx, align 4
David Blaikie79e6c742015-02-27 19:29:02 +0000196 %arrayidx2 = getelementptr inbounds i32, i32* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000197 %1 = load i32, i32* %arrayidx2, align 4
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000198 %add = add nsw i32 %1, %0
199 %and = and i32 %add, %result.08
200 %indvars.iv.next = add i64 %indvars.iv, 1
201 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
202 %exitcond = icmp eq i32 %lftr.wideiv, %n
203 br i1 %exitcond, label %for.end, label %for.body
204
205for.end: ; preds = %for.body, %entry
206 %result.0.lcssa = phi i32 [ -1, %entry ], [ %and, %for.body ]
207 ret i32 %result.0.lcssa
208}
209
Stephen Linc1c7a132013-07-14 01:42:54 +0000210;CHECK-LABEL: @reduction_or(
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000211;CHECK: or <4 x i32>
Benjamin Kramerf0e5d2f2012-12-18 18:40:20 +0000212;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 2, i32 3, i32 undef, i32 undef>
213;CHECK: or <4 x i32>
214;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 1, i32 undef, i32 undef, i32 undef>
215;CHECK: or <4 x i32>
216;CHECK: extractelement <4 x i32> %{{.*}}, i32 0
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000217;CHECK: ret i32
218define i32 @reduction_or(i32 %n, i32* nocapture %A, i32* nocapture %B) nounwind uwtable readonly {
219entry:
220 %cmp7 = icmp sgt i32 %n, 0
221 br i1 %cmp7, label %for.body, label %for.end
222
223for.body: ; preds = %entry, %for.body
224 %indvars.iv = phi i64 [ %indvars.iv.next, %for.body ], [ 0, %entry ]
225 %result.08 = phi i32 [ %or, %for.body ], [ 0, %entry ]
David Blaikie79e6c742015-02-27 19:29:02 +0000226 %arrayidx = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000227 %0 = load i32, i32* %arrayidx, align 4
David Blaikie79e6c742015-02-27 19:29:02 +0000228 %arrayidx2 = getelementptr inbounds i32, i32* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000229 %1 = load i32, i32* %arrayidx2, align 4
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000230 %add = add nsw i32 %1, %0
231 %or = or i32 %add, %result.08
232 %indvars.iv.next = add i64 %indvars.iv, 1
233 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
234 %exitcond = icmp eq i32 %lftr.wideiv, %n
235 br i1 %exitcond, label %for.end, label %for.body
236
237for.end: ; preds = %for.body, %entry
238 %result.0.lcssa = phi i32 [ 0, %entry ], [ %or, %for.body ]
239 ret i32 %result.0.lcssa
240}
241
Stephen Linc1c7a132013-07-14 01:42:54 +0000242;CHECK-LABEL: @reduction_xor(
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000243;CHECK: xor <4 x i32>
Benjamin Kramerf0e5d2f2012-12-18 18:40:20 +0000244;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 2, i32 3, i32 undef, i32 undef>
245;CHECK: xor <4 x i32>
246;CHECK: shufflevector <4 x i32> %{{.*}}, <4 x i32> undef, <4 x i32> <i32 1, i32 undef, i32 undef, i32 undef>
247;CHECK: xor <4 x i32>
248;CHECK: extractelement <4 x i32> %{{.*}}, i32 0
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000249;CHECK: ret i32
250define i32 @reduction_xor(i32 %n, i32* nocapture %A, i32* nocapture %B) nounwind uwtable readonly {
251entry:
252 %cmp7 = icmp sgt i32 %n, 0
253 br i1 %cmp7, label %for.body, label %for.end
254
255for.body: ; preds = %entry, %for.body
256 %indvars.iv = phi i64 [ %indvars.iv.next, %for.body ], [ 0, %entry ]
257 %result.08 = phi i32 [ %xor, %for.body ], [ 0, %entry ]
David Blaikie79e6c742015-02-27 19:29:02 +0000258 %arrayidx = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000259 %0 = load i32, i32* %arrayidx, align 4
David Blaikie79e6c742015-02-27 19:29:02 +0000260 %arrayidx2 = getelementptr inbounds i32, i32* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000261 %1 = load i32, i32* %arrayidx2, align 4
Nadav Rotem5ffb0492012-10-25 00:08:41 +0000262 %add = add nsw i32 %1, %0
263 %xor = xor i32 %add, %result.08
264 %indvars.iv.next = add i64 %indvars.iv, 1
265 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
266 %exitcond = icmp eq i32 %lftr.wideiv, %n
267 br i1 %exitcond, label %for.end, label %for.body
268
269for.end: ; preds = %for.body, %entry
270 %result.0.lcssa = phi i32 [ 0, %entry ], [ %xor, %for.body ]
271 ret i32 %result.0.lcssa
272}
Paul Redmond874f01e2013-01-04 22:10:16 +0000273
Nadav Roteme9f5bfd2013-01-05 01:15:47 +0000274; In this code the subtracted variable is on the RHS and this is not an induction variable.
Stephen Linc1c7a132013-07-14 01:42:54 +0000275;CHECK-LABEL: @reduction_sub_rhs(
Paul Redmond874f01e2013-01-04 22:10:16 +0000276;CHECK-NOT: phi <4 x i32>
277;CHECK-NOT: sub nsw <4 x i32>
278;CHECK: ret i32
Nadav Roteme9f5bfd2013-01-05 01:15:47 +0000279define i32 @reduction_sub_rhs(i32 %n, i32* noalias nocapture %A) nounwind uwtable readonly {
280entry:
281 %cmp4 = icmp sgt i32 %n, 0
282 br i1 %cmp4, label %for.body, label %for.end
283
284for.body: ; preds = %entry, %for.body
285 %indvars.iv = phi i64 [ %indvars.iv.next, %for.body ], [ 0, %entry ]
286 %x.05 = phi i32 [ %sub, %for.body ], [ 0, %entry ]
David Blaikie79e6c742015-02-27 19:29:02 +0000287 %arrayidx = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000288 %0 = load i32, i32* %arrayidx, align 4
Nadav Roteme9f5bfd2013-01-05 01:15:47 +0000289 %sub = sub nsw i32 %0, %x.05
290 %indvars.iv.next = add i64 %indvars.iv, 1
291 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
292 %exitcond = icmp eq i32 %lftr.wideiv, %n
293 br i1 %exitcond, label %for.end, label %for.body
294
295for.end: ; preds = %for.body, %entry
296 %x.0.lcssa = phi i32 [ 0, %entry ], [ %sub, %for.body ]
297 ret i32 %x.0.lcssa
298}
299
300
301; In this test the reduction variable is on the LHS and we can vectorize it.
Stephen Linc1c7a132013-07-14 01:42:54 +0000302;CHECK-LABEL: @reduction_sub_lhs(
Nadav Roteme9f5bfd2013-01-05 01:15:47 +0000303;CHECK: phi <4 x i32>
304;CHECK: sub nsw <4 x i32>
305;CHECK: ret i32
Paul Redmond874f01e2013-01-04 22:10:16 +0000306define i32 @reduction_sub_lhs(i32 %n, i32* noalias nocapture %A) nounwind uwtable readonly {
307entry:
308 %cmp4 = icmp sgt i32 %n, 0
309 br i1 %cmp4, label %for.body, label %for.end
310
311for.body: ; preds = %entry, %for.body
312 %indvars.iv = phi i64 [ %indvars.iv.next, %for.body ], [ 0, %entry ]
313 %x.05 = phi i32 [ %sub, %for.body ], [ 0, %entry ]
David Blaikie79e6c742015-02-27 19:29:02 +0000314 %arrayidx = getelementptr inbounds i32, i32* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000315 %0 = load i32, i32* %arrayidx, align 4
Nadav Roteme9f5bfd2013-01-05 01:15:47 +0000316 %sub = sub nsw i32 %x.05, %0
Paul Redmond874f01e2013-01-04 22:10:16 +0000317 %indvars.iv.next = add i64 %indvars.iv, 1
318 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
319 %exitcond = icmp eq i32 %lftr.wideiv, %n
320 br i1 %exitcond, label %for.end, label %for.body
321
322for.end: ; preds = %for.body, %entry
323 %x.0.lcssa = phi i32 [ 0, %entry ], [ %sub, %for.body ]
324 ret i32 %x.0.lcssa
325}
Arnold Schwaighofer36101392013-05-07 21:55:37 +0000326
327; We can vectorize conditional reductions with multi-input phis.
328; CHECK: reduction_conditional
329; CHECK: fadd <4 x float>
330
331define float @reduction_conditional(float* %A, float* %B, float* %C, float %S) {
332entry:
333 br label %for.body
334
335for.body:
336 %indvars.iv = phi i64 [ 0, %entry ], [ %indvars.iv.next, %for.inc ]
337 %sum.033 = phi float [ %S, %entry ], [ %sum.1, %for.inc ]
David Blaikie79e6c742015-02-27 19:29:02 +0000338 %arrayidx = getelementptr inbounds float, float* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000339 %0 = load float, float* %arrayidx, align 4
David Blaikie79e6c742015-02-27 19:29:02 +0000340 %arrayidx2 = getelementptr inbounds float, float* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000341 %1 = load float, float* %arrayidx2, align 4
Arnold Schwaighofer36101392013-05-07 21:55:37 +0000342 %cmp3 = fcmp ogt float %0, %1
343 br i1 %cmp3, label %if.then, label %for.inc
344
345if.then:
346 %cmp6 = fcmp ogt float %1, 1.000000e+00
347 br i1 %cmp6, label %if.then8, label %if.else
348
349if.then8:
350 %add = fadd fast float %sum.033, %0
351 br label %for.inc
352
353if.else:
354 %cmp14 = fcmp ogt float %0, 2.000000e+00
355 br i1 %cmp14, label %if.then16, label %for.inc
356
357if.then16:
358 %add19 = fadd fast float %sum.033, %1
359 br label %for.inc
360
361for.inc:
362 %sum.1 = phi float [ %add, %if.then8 ], [ %add19, %if.then16 ], [ %sum.033, %if.else ], [ %sum.033, %for.body ]
363 %indvars.iv.next = add i64 %indvars.iv, 1
364 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
365 %exitcond = icmp ne i32 %lftr.wideiv, 128
366 br i1 %exitcond, label %for.body, label %for.end
367
368for.end:
369 %sum.1.lcssa = phi float [ %sum.1, %for.inc ]
370 ret float %sum.1.lcssa
371}
372
373; We can't vectorize reductions with phi inputs from outside the reduction.
374; CHECK: noreduction_phi
375; CHECK-NOT: fadd <4 x float>
376define float @noreduction_phi(float* %A, float* %B, float* %C, float %S) {
377entry:
378 br label %for.body
379
380for.body:
381 %indvars.iv = phi i64 [ 0, %entry ], [ %indvars.iv.next, %for.inc ]
382 %sum.033 = phi float [ %S, %entry ], [ %sum.1, %for.inc ]
David Blaikie79e6c742015-02-27 19:29:02 +0000383 %arrayidx = getelementptr inbounds float, float* %A, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000384 %0 = load float, float* %arrayidx, align 4
David Blaikie79e6c742015-02-27 19:29:02 +0000385 %arrayidx2 = getelementptr inbounds float, float* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000386 %1 = load float, float* %arrayidx2, align 4
Arnold Schwaighofer36101392013-05-07 21:55:37 +0000387 %cmp3 = fcmp ogt float %0, %1
388 br i1 %cmp3, label %if.then, label %for.inc
389
390if.then:
391 %cmp6 = fcmp ogt float %1, 1.000000e+00
392 br i1 %cmp6, label %if.then8, label %if.else
393
394if.then8:
395 %add = fadd fast float %sum.033, %0
396 br label %for.inc
397
398if.else:
399 %cmp14 = fcmp ogt float %0, 2.000000e+00
400 br i1 %cmp14, label %if.then16, label %for.inc
401
402if.then16:
403 %add19 = fadd fast float %sum.033, %1
404 br label %for.inc
405
406for.inc:
407 %sum.1 = phi float [ %add, %if.then8 ], [ %add19, %if.then16 ], [ 0.000000e+00, %if.else ], [ %sum.033, %for.body ]
408 %indvars.iv.next = add i64 %indvars.iv, 1
409 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
410 %exitcond = icmp ne i32 %lftr.wideiv, 128
411 br i1 %exitcond, label %for.body, label %for.end
412
413for.end:
414 %sum.1.lcssa = phi float [ %sum.1, %for.inc ]
415 ret float %sum.1.lcssa
416}
417
418; We can't vectorize reductions that feed another header PHI.
419; CHECK: noredux_header_phi
420; CHECK-NOT: fadd <4 x float>
421
422define float @noredux_header_phi(float* %A, float* %B, float* %C, float %S) {
423entry:
424 br label %for.body
425
426for.body:
427 %indvars.iv = phi i64 [ 0, %entry ], [ %indvars.iv.next, %for.body ]
428 %sum2.09 = phi float [ 0.000000e+00, %entry ], [ %add1, %for.body ]
429 %sum.08 = phi float [ %S, %entry ], [ %add, %for.body ]
David Blaikie79e6c742015-02-27 19:29:02 +0000430 %arrayidx = getelementptr inbounds float, float* %B, i64 %indvars.iv
David Blaikiea79ac142015-02-27 21:17:42 +0000431 %0 = load float, float* %arrayidx, align 4
Arnold Schwaighofer36101392013-05-07 21:55:37 +0000432 %add = fadd fast float %sum.08, %0
433 %add1 = fadd fast float %sum2.09, %add
434 %indvars.iv.next = add i64 %indvars.iv, 1
435 %lftr.wideiv = trunc i64 %indvars.iv.next to i32
436 %exitcond = icmp ne i32 %lftr.wideiv, 128
437 br i1 %exitcond, label %for.body, label %for.end
438
439for.end:
440 %add1.lcssa = phi float [ %add1, %for.body ]
441 %add.lcssa = phi float [ %add, %for.body ]
442 %add2 = fadd fast float %add.lcssa, %add1.lcssa
443 ret float %add2
444}
Arnold Schwaighofera92eeeb2013-07-13 19:09:29 +0000445
446
447; When vectorizing a reduction whose loop header phi value is used outside the
448; loop special care must be taken. Otherwise, the reduced value feeding into the
449; outside user misses a few iterations (VF-1) of the loop.
450; PR16522
451
Stephen Linc1c7a132013-07-14 01:42:54 +0000452; CHECK-LABEL: @phivalueredux(
Arnold Schwaighofera92eeeb2013-07-13 19:09:29 +0000453; CHECK-NOT: x i32>
454
455define i32 @phivalueredux(i32 %p) {
456entry:
457 br label %for.body
458
459for.body:
460 %t.03 = phi i32 [ 0, %entry ], [ %inc, %for.body ]
461 %p.addr.02 = phi i32 [ %p, %entry ], [ %xor, %for.body ]
462 %xor = xor i32 %p.addr.02, -1
463 %inc = add nsw i32 %t.03, 1
464 %exitcond = icmp eq i32 %inc, 16
465 br i1 %exitcond, label %for.end, label %for.body
466
467for.end:
468 ret i32 %p.addr.02
469}
Arnold Schwaighofer0caddfc2013-10-07 21:05:43 +0000470
471; Don't vectorize a reduction value that is not the last in a reduction cyle. We
472; would loose iterations (VF-1) on the operations after that use.
473; PR17498
474
475; CHECK-LABEL: not_last_operation
476; CHECK-NOT: x i32>
477define i32 @not_last_operation(i32 %p, i32 %val) {
478entry:
479 %tobool = icmp eq i32 %p, 0
480 br label %for.body
481
482for.body:
483 %inc613.1 = phi i32 [ 0, %entry ], [ %inc6.1, %for.body ]
484 %inc511.1 = phi i32 [ %val, %entry ], [ %inc5.1, %for.body ]
485 %0 = zext i1 %tobool to i32
486 %inc4.1 = xor i32 %0, 1
487 %inc511.1.inc4.1 = add nsw i32 %inc511.1, %inc4.1
488 %inc5.1 = add nsw i32 %inc511.1.inc4.1, 1
489 %inc6.1 = add nsw i32 %inc613.1, 1
490 %exitcond.1 = icmp eq i32 %inc6.1, 22
491 br i1 %exitcond.1, label %exit, label %for.body
492
493exit:
494 %inc.2 = add nsw i32 %inc511.1.inc4.1, 2
495 ret i32 %inc.2
496}