Fix a grammar mistake, and add a line about the two phases that the BB/SLP vectorizer...
[oota-llvm.git] / docs / Vectorizers.rst
1 ==========================
2 Auto-Vectorization in LLVM
3 ==========================
4
5 .. contents::
6    :local:
7
8 LLVM has two vectorizers: The :ref:`Loop Vectorizer <loop-vectorizer>`,
9 which operates on Loops, and the :ref:`SLP Vectorizer
10 <slp-vectorizer>`, which optimizes straight-line code. These vectorizers
11 focus on different optimization opportunities and use different techniques.
12 The SLP vectorizer merges multiple scalars that are found in the code into
13 vectors while the Loop Vectorizer widens instructions in the original loop
14 to operate on multiple consecutive loop iterations.
15
16 .. _loop-vectorizer:
17
18 The Loop Vectorizer
19 ===================
20
21 Usage
22 -----
23
24 LLVM's Loop Vectorizer is now enabled by default for -O3.
25 We plan to enable parts of the Loop Vectorizer on -O2 and -Os in future releases.
26 The vectorizer can be disabled using the command line:
27
28 .. code-block:: console
29
30    $ clang ... -fno-vectorize  file.c
31
32 Command line flags
33 ^^^^^^^^^^^^^^^^^^
34
35 The loop vectorizer uses a cost model to decide on the optimal vectorization factor
36 and unroll factor. However, users of the vectorizer can force the vectorizer to use
37 specific values. Both 'clang' and 'opt' support the flags below.
38
39 Users can control the vectorization SIMD width using the command line flag "-force-vector-width".
40
41 .. code-block:: console
42
43   $ clang  -mllvm -force-vector-width=8 ...
44   $ opt -loop-vectorize -force-vector-width=8 ...
45
46 Users can control the unroll factor using the command line flag "-force-vector-unroll"
47
48 .. code-block:: console
49
50   $ clang  -mllvm -force-vector-unroll=2 ...
51   $ opt -loop-vectorize -force-vector-unroll=2 ...
52
53 Features
54 --------
55
56 The LLVM Loop Vectorizer has a number of features that allow it to vectorize
57 complex loops.
58
59 Loops with unknown trip count
60 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
61
62 The Loop Vectorizer supports loops with an unknown trip count.
63 In the loop below, the iteration ``start`` and ``finish`` points are unknown,
64 and the Loop Vectorizer has a mechanism to vectorize loops that do not start
65 at zero. In this example, 'n' may not be a multiple of the vector width, and
66 the vectorizer has to execute the last few iterations as scalar code. Keeping
67 a scalar copy of the loop increases the code size.
68
69 .. code-block:: c++
70
71   void bar(float *A, float* B, float K, int start, int end) {
72     for (int i = start; i < end; ++i)
73       A[i] *= B[i] + K;
74   }
75
76 Runtime Checks of Pointers
77 ^^^^^^^^^^^^^^^^^^^^^^^^^^
78
79 In the example below, if the pointers A and B point to consecutive addresses,
80 then it is illegal to vectorize the code because some elements of A will be
81 written before they are read from array B.
82
83 Some programmers use the 'restrict' keyword to notify the compiler that the
84 pointers are disjointed, but in our example, the Loop Vectorizer has no way of
85 knowing that the pointers A and B are unique. The Loop Vectorizer handles this
86 loop by placing code that checks, at runtime, if the arrays A and B point to
87 disjointed memory locations. If arrays A and B overlap, then the scalar version
88 of the loop is executed.
89
90 .. code-block:: c++
91
92   void bar(float *A, float* B, float K, int n) {
93     for (int i = 0; i < n; ++i)
94       A[i] *= B[i] + K;
95   }
96
97
98 Reductions
99 ^^^^^^^^^^
100
101 In this example the ``sum`` variable is used by consecutive iterations of
102 the loop. Normally, this would prevent vectorization, but the vectorizer can
103 detect that 'sum' is a reduction variable. The variable 'sum' becomes a vector
104 of integers, and at the end of the loop the elements of the array are added
105 together to create the correct result. We support a number of different
106 reduction operations, such as addition, multiplication, XOR, AND and OR.
107
108 .. code-block:: c++
109
110   int foo(int *A, int *B, int n) {
111     unsigned sum = 0;
112     for (int i = 0; i < n; ++i)
113       sum += A[i] + 5;
114     return sum;
115   }
116
117 We support floating point reduction operations when `-ffast-math` is used.
118
119 Inductions
120 ^^^^^^^^^^
121
122 In this example the value of the induction variable ``i`` is saved into an
123 array. The Loop Vectorizer knows to vectorize induction variables.
124
125 .. code-block:: c++
126
127   void bar(float *A, float* B, float K, int n) {
128     for (int i = 0; i < n; ++i)
129       A[i] = i;
130   }
131
132 If Conversion
133 ^^^^^^^^^^^^^
134
135 The Loop Vectorizer is able to "flatten" the IF statement in the code and
136 generate a single stream of instructions. The Loop Vectorizer supports any
137 control flow in the innermost loop. The innermost loop may contain complex
138 nesting of IFs, ELSEs and even GOTOs.
139
140 .. code-block:: c++
141
142   int foo(int *A, int *B, int n) {
143     unsigned sum = 0;
144     for (int i = 0; i < n; ++i)
145       if (A[i] > B[i])
146         sum += A[i] + 5;
147     return sum;
148   }
149
150 Pointer Induction Variables
151 ^^^^^^^^^^^^^^^^^^^^^^^^^^^
152
153 This example uses the "accumulate" function of the standard c++ library. This
154 loop uses C++ iterators, which are pointers, and not integer indices.
155 The Loop Vectorizer detects pointer induction variables and can vectorize
156 this loop. This feature is important because many C++ programs use iterators.
157
158 .. code-block:: c++
159
160   int baz(int *A, int n) {
161     return std::accumulate(A, A + n, 0);
162   }
163
164 Reverse Iterators
165 ^^^^^^^^^^^^^^^^^
166
167 The Loop Vectorizer can vectorize loops that count backwards.
168
169 .. code-block:: c++
170
171   int foo(int *A, int *B, int n) {
172     for (int i = n; i > 0; --i)
173       A[i] +=1;
174   }
175
176 Scatter / Gather
177 ^^^^^^^^^^^^^^^^
178
179 The Loop Vectorizer can vectorize code that becomes a sequence of scalar instructions 
180 that scatter/gathers memory.
181
182 .. code-block:: c++
183
184   int foo(int *A, int *B, int n, int k) {
185     for (int i = 0; i < n; ++i)
186       A[i*7] += B[i*k];
187   }
188
189 Vectorization of Mixed Types
190 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
191
192 The Loop Vectorizer can vectorize programs with mixed types. The Vectorizer
193 cost model can estimate the cost of the type conversion and decide if
194 vectorization is profitable.
195
196 .. code-block:: c++
197
198   int foo(int *A, char *B, int n, int k) {
199     for (int i = 0; i < n; ++i)
200       A[i] += 4 * B[i];
201   }
202
203 Global Structures Alias Analysis
204 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
205
206 Access to global structures can also be vectorized, with alias analysis being
207 used to make sure accesses don't alias. Run-time checks can also be added on
208 pointer access to structure members.
209
210 Many variations are supported, but some that rely on undefined behaviour being
211 ignored (as other compilers do) are still being left un-vectorized.
212
213 .. code-block:: c++
214
215   struct { int A[100], K, B[100]; } Foo;
216
217   int foo() {
218     for (int i = 0; i < 100; ++i)
219       Foo.A[i] = Foo.B[i] + 100;
220   }
221
222 Vectorization of function calls
223 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
224
225 The Loop Vectorize can vectorize intrinsic math functions.
226 See the table below for a list of these functions.
227
228 +-----+-----+---------+
229 | pow | exp |  exp2   |
230 +-----+-----+---------+
231 | sin | cos |  sqrt   |
232 +-----+-----+---------+
233 | log |log2 |  log10  |
234 +-----+-----+---------+
235 |fabs |floor|  ceil   |
236 +-----+-----+---------+
237 |fma  |trunc|nearbyint|
238 +-----+-----+---------+
239 |     |     | fmuladd |
240 +-----+-----+---------+
241
242 The loop vectorizer knows about special instructions on the target and will
243 vectorize a loop containing a function call that maps to the instructions. For
244 example, the loop below will be vectorized on Intel x86 if the SSE4.1 roundps
245 instruction is available.
246
247 .. code-block:: c++
248
249   void foo(float *f) {
250     for (int i = 0; i != 1024; ++i)
251       f[i] = floorf(f[i]);
252   }
253
254 Partial unrolling during vectorization
255 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
256
257 Modern processors feature multiple execution units, and only programs that contain a
258 high degree of parallelism can fully utilize the entire width of the machine. 
259 The Loop Vectorizer increases the instruction level parallelism (ILP) by 
260 performing partial-unrolling of loops.
261
262 In the example below the entire array is accumulated into the variable 'sum'.
263 This is inefficient because only a single execution port can be used by the processor.
264 By unrolling the code the Loop Vectorizer allows two or more execution ports
265 to be used simultaneously.
266
267 .. code-block:: c++
268
269   int foo(int *A, int *B, int n) {
270     unsigned sum = 0;
271     for (int i = 0; i < n; ++i)
272         sum += A[i];
273     return sum;
274   }
275
276 The Loop Vectorizer uses a cost model to decide when it is profitable to unroll loops.
277 The decision to unroll the loop depends on the register pressure and the generated code size. 
278
279 Performance
280 -----------
281
282 This section shows the the execution time of Clang on a simple benchmark:
283 `gcc-loops <http://llvm.org/viewvc/llvm-project/test-suite/trunk/SingleSource/UnitTests/Vectorizer/>`_.
284 This benchmarks is a collection of loops from the GCC autovectorization
285 `page <http://gcc.gnu.org/projects/tree-ssa/vectorization.html>`_ by Dorit Nuzman.
286
287 The chart below compares GCC-4.7, ICC-13, and Clang-SVN with and without loop vectorization at -O3, tuned for "corei7-avx", running on a Sandybridge iMac.
288 The Y-axis shows the time in msec. Lower is better. The last column shows the geomean of all the kernels.
289
290 .. image:: gcc-loops.png
291
292 And Linpack-pc with the same configuration. Result is Mflops, higher is better.
293
294 .. image:: linpack-pc.png
295
296 .. _slp-vectorizer:
297
298 The SLP Vectorizer
299 ==================
300
301 Details
302 -------
303
304 The goal of SLP vectorization (a.k.a. superword-level parallelism) is
305 to combine similar independent instructions within simple control-flow regions
306 into vector instructions. Memory accesses, arithemetic operations, comparison
307 operations and some math functions can all be vectorized using this technique
308 (subject to the capabilities of the target architecture).
309
310 For example, the following function performs very similar operations on its
311 inputs (a1, b1) and (a2, b2). The basic-block vectorizer may combine these
312 into vector operations.
313
314 .. code-block:: c++
315
316   void foo(int a1, int a2, int b1, int b2, int *A) {
317     A[0] = a1*(a1 + b1)/b1 + 50*b1/a1;
318     A[1] = a2*(a2 + b2)/b2 + 50*b2/a2;
319   }
320
321 The SLP-vectorizer has two phases, bottom-up, and top-down. The top-down vectorization
322 phase is more aggressive, but takes more time to run.
323
324 Usage
325 ------
326
327 The SLP Vectorizer is not enabled by default, but it can be enabled
328 through clang using the command line flag:
329
330 .. code-block:: console
331
332    $ clang -fslp-vectorize file.c
333
334 LLVM has a second basic block vectorization phase
335 which is more compile-time intensive (The BB vectorizer). This optimization
336 can be enabled through clang using the command line flag:
337
338 .. code-block:: console
339
340    $ clang -fslp-vectorize-aggressive file.c
341
342
343 The SLP vectorizer is in early development stages but can already vectorize
344 and accelerate many programs in the LLVM test suite.
345
346 =======================   ============
347 Benchmark Name              Gain
348 =======================   ============
349 Misc/flops-7               -32.70%
350 Misc/matmul_f64_4x4        -23.23%
351 Olden/power                -21.45%
352 Misc/flops-4               -14.90%
353 ASC_Sequoia/AMGmk          -13.85%
354 TSVC/LoopRerolling-flt     -11.76%
355 Misc/flops-6               -9.70%
356 Misc/flops-5               -8.54%
357 Misc/flops                 -8.12%
358 TSVC/NodeSplitting-dbl     -6.96%
359 Misc-C++/sphereflake       -6.74%
360 Ptrdist/yacr2              -6.31%
361 =======================   ============
362