[llvm-dev] Proposal for function vectorization and loop vectorization with function calls

via llvm-dev llvm-dev at lists.llvm.org
Fri Mar 18 03:03:26 PDT 2016


Pinging David and Richard!

Yours,
Andrey

> 3 марта 2016 г., в 11:27, Andrey Bokhanko <andreybokhanko at gmail.com> написал(а):
> 
> Hi David [Majnemer], Richard [Smith],
> 
> Front-end wise, the biggest change in this proposal is introduction of
> new mangling for vector functions.
> 
> May I ask you to look at the mangling part (sections 1 and 2 in the
> "Proposed Implementation" chapter) and review it?
> 
> (Obviously, others who are concerned with how mangling is done in
> Clang are welcome to chime in as well!)
> 
> Yours,
> Andrey
> 
> 
> On Wed, Mar 2, 2016 at 10:49 PM, Tian, Xinmin via llvm-dev
> <llvm-dev at lists.llvm.org> wrote:
>> Proposal for function vectorization and loop vectorization with function calls
>> ==============================================================================
>> Intel Corporation (3/2/2016)
>> 
>> This is a proposal for an initial work towards Clang and LLVM implementation of
>> vectorizing a function annotated with  OpenMP 4.5's "#pragma omp declare simd"
>> (named SIMD-enabled function) and its associated clauses based on the VectorABI
>> [2]. On the caller side, we propose to improve LLVM loopVectorizer such that
>> the code that calls the SIMD-enabled function can be vectorized. On the callee
>> side, we propose to add Clang FE support for "#pragma omp declare simd" syntax
>> and a new pass to transform the SIMD-enabled function body into a SIMD loop.
>> This newly created loop can then be fed to LLVM loopVectorizer (or its future
>> enhancement) for vectorization. This work does leverage LLVM's existing
>> LoopVectorizer.
>> 
>> 
>> Problem Statement
>> =================
>> Currently, if a loop calls a user-defined function or a 3rd party library
>> function, the loop can't be vectorized unless the function is inlined. In the
>> example below the LoopVectorizer fails to vectorize the k loop due to its
>> function call to "dowork" because "dowork" is an external function. Note that
>> inlining the "dowork" function may result in vectorization for some of the
>> cases, but that is not a generally applicable solution. Also, there may be
>> reasons why compiler may not (or can't) inline the "dowork" function call.
>> Therefore, there is value in being able to vectorize the loop with a call to
>> "dowork" function in it.
>> 
>> #include<stdio.h>
>> extern float dowork(float *a, int k);
>> 
>> float a[4096];
>> int main()
>> { int k;
>> #pragma clang loop vectorize(enable)
>>  for (k = 0; k < 4096; k++) {
>>    a[k] = k * 0.5;
>>    a[k] = dowork(a, k);
>>  }
>>  printf("passed %f\n", a[1024]);
>> }
>> 
>> sh-4.1$ clang -c -O2 -Rpass=loop-vectorize -Rpass-missed=loop-vectorize
>>                     -Rpass-analysis=loop-vectorize loopvec.c
>> loopvec.c:15:12: remark: loop not vectorized: call instruction cannot be
>>      vectorized [-Rpass-analysis]
>>    a[k] = dowork(a, k);
>>           ^
>> loopvec.c:13:3: remark: loop not vectorized: use -Rpass-analysis=loop-vectorize
>>      for more info (Force=true) [-Rpass-missed=loop-vectorize]
>>  for (k = 0; k < 4096; k++) {
>>  ^
>> loopvec.c:13:3: warning: loop not vectorized: failed explicitly specified
>>                loop vectorization [-Wpass-failed]
>> 1 warning generated.
>> 
>> 
>> New functionality of Vectorization
>> ==================================
>> New functionalities and enhancements are proposed to address the issues
>> stated above which include: a) Vectorize a function annotated by the
>> programmer using OpenMP* SIMD extensions; b) Enhance LLVM's LoopVectorizer
>> to vectorize a loop containing a call to SIMD-enabled function.
>> 
>> For example, when writing:
>> 
>> #include<stdio.h>
>> 
>> #pragma omp declare simd uniform(a) linear(k)
>> extern float dowork(float *a, int k);
>> 
>> float a[4096];
>> int main()
>> { int k;
>> #pragma clang loop vectorize(enable)
>>  for (k = 0; k < 4096; k++) {
>>    a[k] = k * 0.5;
>>    a[k] = dowork(a, k);
>>  }
>>  printf("passed %f\n", a[1024]);
>> }
>> 
>> the programmer asserts that
>>  a) there will be a vector version of "dowork" available for the compiler to
>>     use (link with, with appropriate signature, explained below) when
>>     vectorizing the k loop; and that
>>  b) no loop-carried backward dependencies are introduced by the "dowork"
>>     call that prevent the vectorization of the k loop.
>> 
>> The expected vector loop (shown as pseudo code, ignoring leftover iterations)
>> resulting from LLVM's LoopVectorizer is
>> 
>>  ... ...
>>  vectorized_for (k = 0; k < 4096; k += VL) {
>>    a[k:VL] = {k, k+1, k+2, k+VL-1} * 0.5;
>>    a[k:VL] = _ZGVb4Nul_dowork(a, k);
>>  }
>>  ... ...
>> 
>> In this example "_ZGVb4Nul_dowork" is a special name mangling where:
>> _ZGV is a prefix based on C/C++ name mangling rule suggested by GCC community,
>> 'b' indicates "xmm" (assume we vectorize here to 128bit xmm vector registers),
>> '4' is VL (assume we vectorize here for length 4),
>> 'N' indicates that the function is vectorized without a mask, M indicates that
>>     the function is vecrized with a mask.
>> 'u' indicates that the first parameter has the "uniform" property,
>> 'l' indicates that the second argement has the "linear" property.
>> 
>> More details (including name mangling scheme) can be found in the following
>> references [2].
>> 
>> References
>> ==========
>> 
>> 1. OpenMP SIMD language extensions: http://www.openmp.org/mp-documents/openmp-4.
>> 5.pdf
>> 
>> 2. VectorABI Documentation:
>> https://www.cilkplus.org/sites/default/files/open_specifications/Intel-ABI-Vecto
>> r-Function-2012-v0.9.5.pdf
>> https://sourceware.org/glibc/wiki/libmvec?action=AttachFile&do=view&target=Vecto
>> rABI.txt
>> 
>> [[Note: VectorABI was reviewed at X86-64 System V Application Binary Interface
>>        mailing list. The discussion was recorded at
>>        https://groups.google.com/forum/#!topic/x86-64-abi/LmppCfN1rZ4 ]]
>> 
>> 3. The first paper on SIMD extensions and implementations:
>> "Compiling C/C++ SIMD Extensions for Function and Loop Vectorizaion on
>> Multicore-SIMD Processors" by Xinmin Tian, Hideki Saito, Milind Girkar,
>> Serguei Preis, Sergey Kozhukhov, et al., IPDPS Workshops 2012, pages 2349--2358
>> [[Note: the first implementation and the paper were done before VectorABI was
>>        finalized with the GCC community and Redhat. The latest VectorABI
>>        version for OpenMP 4.5 is ready to be published]]
>> 
>> 
>> Proposed Implementation
>> =======================
>> 1. Clang FE parses "#pragma omp declare simd [clauses]" and generates mangled
>>   name including these prefixes as vector signatures. These mangled name
>>   prefixes are recorded as function attributes in LLVM function attribute
>>   group. Note that it may be possible to have several mangled names associated
>>   with the same function, which correspond to several desired vectorized
>>   versions. Clang FE generates all function attributes for expected vector
>>   variants to be generated by the back-end. E.g.,
>> 
>>   #pragma omp delcare simd uniform(a) linear(k)
>>   float dowork(float *a, int k)
>>   {
>>      a[k] = sinf(a[k]) + 9.8f;
>>   }
>> 
>>   define __stdcall f32 @_dowork(f32* %a, i32 %k) #0
>>   ... ...
>>   attributes #0 = { nounwind uwtable "_ZGVbM4ul_" "_ZGVbN4ul_" ...}
>> 
>> 2. A new vector function generation pass is introduced to generate vector
>>   variants of the original scalar function based on VectorABI (see [2, 3]).
>>   For example, one vector variant is generated for "_ZGVbN4ul_" attribute
>>   as follows (pseudo code):
>> 
>>   define __stdcall <4 x f32> @_ZGVbN4ul_dowork(f32* %a, i32 %k) #0
>>   {
>>     #pragma clang loop vectorize(enable)
>>     for (int %t = k; %t < %k + 4; %t++) {
>>       %a[t] = sinf(%a[t]) + 9.8f;
>>     }
>>     vec_load xmm0, %a[k:VL]
>>     return xmm0;
>>   }
>> 
>>   The body of the function is wrapped inside a loop having VL iterations,
>>   which correspond to the vector lanes.
>> 
>>   The LLVM LoopVectorizer will vectorize the generated %t loop, expected
>>   to produce the following vectorized code eliminating the loop (pseudo code):
>> 
>>   define __stdcall <4 x f32> @_ZGVbN4ul_dowork(f32* %a, i32 %k) #0
>>   {
>>     vec_load xmm1,  %a[k: VL]
>>     xmm2 = call __svml_sinf(xmm1)
>>     xmm0 = vec_add  xmm2, [9,8f, 9.8f, 9.8f, 9.8f]
>>     store %a[k:VL], xmm0
>>     return xmm0;
>>   }
>> 
>>   [[Note: Vectorizer support for the Short Vector Math Library (SVML)
>>           functions will be a seperate proposal. ]]
>> 
>> 3. The LLVM LoopVectorizer is enhanced to
>>   a) identify loops with calls that have been annotated with
>>      "#pragma omp declare simd" by checking function attribute groups;
>>   b) analyze each call instruction and its parameters in the loop, to
>>      determine if each parameter has the following properties:
>>        * uniform
>>        * linear + stride
>>        * vector
>>        * aligned
>>        * called inside a conditional branch or not
>>          ... ...
>>      Based on these properties, the signature of the vectorized call is
>>      generated; and
>>   c) performs signature matching to obtain the suitable vector variant
>>      among the signatures available for the called function. If no such
>>      signature is found, the call cannot be vectorized.
>> 
>>   Note that a similar enhancement can and should be made also to LLVM's
>>   SLP vectorizer.
>> 
>>   For example:
>> 
>>   #pragma omp declare simd uniform(a) linear(k)
>>   extern float dowork(float *a, int k);
>> 
>>   ... ...
>>   #pragma clang loop vectorize(enable)
>>   for (k = 0; k < 4096; k++) {
>>     a[k] = k * 0.5;
>>     a[k] = dowork(a, k);
>>   }
>>   ... ...
>> 
>>   Step a: "dowork" function is marked as SIMD-enabled function
>>           attributes #0 = { nounwind uwtable "_ZGVbM4ul_" "_ZGVbN4ul_" ...}
>> 
>>   Step b: 1) 'a' is uniform, as it is the base address of array 'a'
>>           2) 'k' is linear, as 'k' is the induction variable with stride=1
>>           3) SIMD "dowork" is called unconditionally in the candidate k loop.
>>           4) it is compiled for SSE4.1 with the Vector Length VL=4.
>>              based on these properties, the signature is "_ZGVbN4ul_"
>> 
>>   [[Notes: For conditional call in the loop, it needs masking support,
>>            the implementation details seen in reference [1][2][3] ]]
>> 
>>   Step c: Check if the signature "_ZGVbN4ul_" exists in function attribute #0;
>>           if yes the suitable vectorized version is found and will be linked
>>           with.
>> 
>>   The below loop is expected to be produced by the LoopVectorizer:
>>   ... ...
>>   vectorized_for (k = 0; k < 4096; k += 4) {
>>     a[k:4] = {k, k+1, k+2, k+3} * 0.5;
>>     a[k:4] = _ZGVb4Nul_dowork(a, k);
>>   }
>>   ... ...
>> 
>> [[Note: Vectorizer support for the Short Vector Math Library (SVML) functions
>>        will be a seperate proposal. ]]
>> 
>> 
>> GCC and ICC Compatibility
>> =========================
>> With this proposal the callee function and the loop containing a call to it
>> can each be compiled and vectorized by a different compiler, including
>> Clang+LLVM with its LoopVectorizer as outlined above, GCC and ICC. The
>> vectorized loop will then be linked with the vectorized callee function.
>> Of-course each of these compilers can also be used to compile both loop and
>> callee function.
>> 
>> 
>> Current Implementation Status and Plan
>> ======================================
>> 1. Clang FE is done by Intel Clang FE team according to #1. Note: Clang FE
>>   syntax process patch is implemented and under community review
>>   (http://reviews.llvm.org/D10599). In general, the review feedback is
>>   very positive from the Clang community.
>> 
>> 2. A new pass for function vectorization is implemented to support #2 and
>>   to be prepared for LLVM community review.
>> 
>> 3. Work is in progress to teach LLVM's LoopVectorizer to vectorize a loop
>>   with user-defined function calls according to #3.
>> 
>> Call for Action
>> ===============
>> 1. Please review this proposal and provide constructive feedback on its
>>   direction and key ideas.
>> 
>> 2. Feel free to ask any technical questions related to this proposal and
>>   to read the associated references.
>> 
>> 3. Help is also highly welcome and appreciated in the development and
>>   upstreaming process.
>> 
>> _______________________________________________
>> LLVM Developers mailing list
>> llvm-dev at lists.llvm.org
>> http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev


More information about the llvm-dev mailing list