Generating SIMD instructions from Cython code

前端 未结 1 1969
既然无缘
既然无缘 2020-12-01 20:13

I need to get an overview of the performance one can get from using Cython in high performance numerical code. One of the thing I am interested in is to find out if an optim

相关标签:
1条回答
  • 2020-12-01 20:45

    You have two problems in your code (use option -a to make it visible):

    1. The indexing of numpy array isn't efficient
    2. You have forgotten int in cdef sum=0

    Taking this into account we get:

    cpdef int f(np.ndarray[np.int_t] f):  ##HERE
        assert f.dtype == np.int
        cdef int array_length =  f.shape[0]
        cdef int sum = 0                  ##HERE
        cdef int k
        for k in range(array_length):
            sum += f[k]
        return sum
    

    For the loop the following code:

    int __pyx_t_5;
    int __pyx_t_6;
    Py_ssize_t __pyx_t_7;
    ....
    __pyx_t_5 = __pyx_v_array_length;
    for (__pyx_t_6 = 0; __pyx_t_6 < __pyx_t_5; __pyx_t_6+=1) {
       __pyx_v_k = __pyx_t_6;
       __pyx_t_7 = __pyx_v_k;
       __pyx_v_sum = (__pyx_v_sum + (*__Pyx_BufPtrStrided1d(__pyx_t_5numpy_int_t *, __pyx_pybuffernd_f.rcbuffer->pybuffer.buf, __pyx_t_7, __pyx_pybuffernd_f.diminfo[0].strides)));
    

    }

    Which is not that bad, but not as easy for the optimizer as the normal code written by human. As you have already pointed out, __pyx_pybuffernd_f.diminfo[0].strides isn't known at compile time and this prevents vectorization.

    However, you would get better results, when using typed memory views, i.e:

    cpdef int mf(int[::1] f):
        cdef int array_length =  len(f)
    ...
    

    which leads to a less opaque C-code - the one, at least my compiler, can better optimize:

     __pyx_t_2 = __pyx_v_array_length;
      for (__pyx_t_3 = 0; __pyx_t_3 < __pyx_t_2; __pyx_t_3+=1) {
        __pyx_v_k = __pyx_t_3;
        __pyx_t_4 = __pyx_v_k;
        __pyx_v_sum = (__pyx_v_sum + (*((int *) ( /* dim=0 */ ((char *) (((int *) __pyx_v_f.data) + __pyx_t_4)) ))));
      }
    

    The most crucial thing here, is that we make it clear to the cython, that the memory is continuous, i.e. int[::1] compared to int[:] as it is seen for numpy-arrays, for which a possible stride!=1 must be taken into account.

    In this case, the cython-generated C-code results in the same assembler as the code I would have written. As crisb has pointed out, adding -march=native would lead to vectorization, but in this case the assembler of both functions would be slightly different again.

    However, in my experience, compilers have quite often some problems to optimize loops created by cython and/or it is easier to miss a detail which prevents the generation of really good C-code. So my strategy for working-horse-loops is to write them in plain C and use cython for wrapping/accessing them - often it is somewhat faster, because one can also use dedicated compiler flags for this code-snipped without affecting the whole Cython-module.

    0 讨论(0)
提交回复
热议问题