2016-02-25 7 views
7

Unten ist die C++ - Implementierung, die die Zeit vergleicht, die von Eigen und For Loop benötigt wird, um Matrix-Matrix-Produkte auszuführen. Die For-Schleife wurde optimiert, um Cache-Misses zu minimieren. Die for-Schleife ist anfangs schneller als Eigen, wird dann aber langsamer (bis zu einem Faktor von 2 für 500 mal 500 Matrizen). Was sollte ich noch tun, um mit Eigen zu konkurrieren? Blockiert der Grund für die bessere Eigenleistung? Wenn ja, wie sollte ich die for-Schleife blockieren?Wie schreibe ich ein Matrix-Matrix-Produkt, das mit Eigen konkurrieren kann?

#include<iostream> 
#include<Eigen/Dense> 
#include<ctime> 

int main(int argc, char* argv[]) { 
    srand(time(NULL)); 
    // Input the size of the matrix from the user 
    int N = atoi(argv[1]); 

    int M = N*N; 

    // The matrices stored as row-wise vectors 
    double a[M]; 
    double b[M]; 
    double c[M]; 

    // Initializing Eigen Matrices 
    Eigen::MatrixXd a_E = Eigen::MatrixXd::Random(N,N); 
    Eigen::MatrixXd b_E = Eigen::MatrixXd::Random(N,N); 
    Eigen::MatrixXd c_E(N,N); 

    double CPS = CLOCKS_PER_SEC; 

    clock_t start, end; 


    // Matrix vector product by Eigen 
    start = clock(); 
    c_E  = a_E*b_E; 
    end  = clock(); 

    std::cout << "\nTime taken by Eigen is: " << (end-start)/CPS << "\n"; 

    // Initializing For loop vectors 
    int count = 0; 
    for (int j=0; j<N; ++j) { 
     for (int k=0; k<N; ++k) { 
      a[count] = a_E(j,k); 
      b[count] = b_E(j,k); 
      ++count; 
     } 
    } 

    // Matrix vector product by For loop 
    start = clock(); 
    count = 0; 
    int count1, count2; 
    for (int j=0; j<N; ++j) { 
     count1 = j*N; 
     for (int k=0; k<N; ++k) { 
      c[count] = a[count1]*b[k]; 
      ++count; 
     } 
    } 

    for (int j=0; j<N; ++j) { 
     count2 = N; 
     for (int l=1; l<N; ++l) { 
      count = j*N; 
      count1 = count+l; 
      for (int k=0; k<N; ++k) { 
       c[count]+=a[count1]*b[count2]; 
       ++count; 
       ++count2; 
      } 
     } 
    } 
    end  = clock(); 

    std::cout << "\nTime taken by for-loop is: " << (end-start)/CPS << "\n"; 
} 
+3

Hier ist ein schönes Tutorial, das zeigt, wie die Matrix-Matrix-Produkt beschleunigen: [GEMM: Vom reinen C bis SSE Optimierte Mikro-Kernel] (http://apfel.mathematik.uni-ulm.de/~lehn/shpc/gemm/) – davidhigh

+2

Basierend auf dem gleichen Konzept, zeigte ich auch, wie die Matrix-Matrix in boost :: ublas erhalten werden könnte hier aufgepimpt: [Matrix-Matrix-Produktversuche mit uBLAS] (http://www.mathematik.uni-ulm.de/~lehn/test_ublas/index.html) und [Matrix-Matrix-Produktversuche mit BLAZE] (http: //www.mathematik.uni-ulm.de/~lehn/test_blaze/index.html). Es beginnt mit einer "reinen C++" Implementierung. Neben der Verwendung von Assembly-Code für die Vektorisierung enthält es auch ein Beispiel mit den GCC-Vektor-Erweiterungen. –

Antwort

5

Es ist nicht verwunderlich, wie eine leistungsstarke Implementierung des Matrix-Matrix-Produkts erreicht werden kann. In der Tat brauchen wir mehr Leute, die davon wissen, um zukünftigen Herausforderungen im Hochleistungsrechnen zu begegnen. Um in dieses Thema zu gelangen, ist das Lesen BLIS: A Framework for Rapidly Instantiating BLAS Functionality ein guter Ausgangspunkt.

Also um die Frage zu entmystifizieren und zu beantworten (Wie schreibe ich ein Matrix-Matrix-Produkt, das mit Eigen konkurrieren kann) habe ich den von ggael geposteten Code auf insgesamt 400 Zeilen erweitert. Ich habe es gerade auf einem AVX-Rechner (Intel® Core ™ i5-3470 CPU @ 3.20GHz) getestet. Hier einige Ergebnisse:

g++-5.3 -O3 -DNDEBUG -std=c++11 -mavx -m64 -I ../eigen.3.2.8/ gemm.cc -lrt 

[email protected]:~/work/test_eigen$ ./a.out 500 
Time taken by Eigen is: 0.0190425 
Time taken by for-loop is: 0.0121688 

[email protected]:~/work/test_eigen$ ./a.out 1000 
Time taken by Eigen is: 0.147991 
Time taken by for-loop is: 0.0959097 

[email protected]:~/work/test_eigen$ ./a.out 1500 
Time taken by Eigen is: 0.492858 
Time taken by for-loop is: 0.322442 

[email protected]:~/work/test_eigen$ ./a.out 5000 
Time taken by Eigen is: 18.3666 
Time taken by for-loop is: 12.1023 

Wenn Sie FMA haben, können Sie mit

g++-5.3 -O3 -DNDEBUG -std=c++11 -mfma -m64 -I ../eigen.3.2.8/ -DHAVE_FMA gemm.cc -lrt 

kompilieren Wenn Sie auch mit OpenMP Multithreading wollen auch mit -fopenmp

die komplette Code basiert auf den Ideen kompilieren von das BLIS-Papier. Es ist in sich geschlossen, außer dass sie die vollständige Eigen-Quelldateien benötigt als ggael bereits erwähnt:

#include<iostream> 
#include<Eigen/Dense> 
#include<bench/BenchTimer.h> 
#if defined(_OPENMP) 
#include <omp.h> 
#endif 
//-- malloc with alignment -------------------------------------------------------- 
void * 
malloc_(std::size_t alignment, std::size_t size) 
{ 
    alignment = std::max(alignment, alignof(void *)); 
    size  += alignment; 

    void *ptr = std::malloc(size); 
    void *ptr2 = (void *)(((uintptr_t)ptr + alignment) & ~(alignment-1)); 
    void **vp = (void**) ptr2 - 1; 
    *vp  = ptr; 
    return ptr2; 
} 

void 
free_(void *ptr) 
{ 
    std::free(*((void**)ptr-1)); 
} 

//-- Config -------------------------------------------------------------------- 

// SIMD-Register width in bits 
// SSE:   128 
// AVX/FMA:  256 
// AVX-512:  512 
#ifndef SIMD_REGISTER_WIDTH 
#define SIMD_REGISTER_WIDTH 256 
#endif 

#ifdef HAVE_FMA 

# ifndef BS_D_MR 
# define BS_D_MR 4 
# endif 

# ifndef BS_D_NR 
# define BS_D_NR 12 
# endif 

# ifndef BS_D_MC 
# define BS_D_MC 256 
# endif 

# ifndef BS_D_KC 
# define BS_D_KC 512 
# endif 

# ifndef BS_D_NC 
# define BS_D_NC 4092 
# endif 

#endif 



#ifndef BS_D_MR 
#define BS_D_MR 4 
#endif 

#ifndef BS_D_NR 
#define BS_D_NR 8 
#endif 

#ifndef BS_D_MC 
#define BS_D_MC 256 
#endif 

#ifndef BS_D_KC 
#define BS_D_KC 256 
#endif 

#ifndef BS_D_NC 
#define BS_D_NC 4096 
#endif 

template <typename T> 
struct BlockSize 
{ 
    static constexpr int MC = 64; 
    static constexpr int KC = 64; 
    static constexpr int NC = 256; 
    static constexpr int MR = 8; 
    static constexpr int NR = 8; 

    static constexpr int rwidth = 0; 
    static constexpr int align = alignof(T); 
    static constexpr int vlen = 0; 

    static_assert(MC>0 && KC>0 && NC>0 && MR>0 && NR>0, "Invalid block size."); 
    static_assert(MC % MR == 0, "MC must be a multiple of MR."); 
    static_assert(NC % NR == 0, "NC must be a multiple of NR."); 
}; 


template <> 
struct BlockSize<double> 
{ 
    static constexpr int MC  = BS_D_MC; 
    static constexpr int KC  = BS_D_KC; 
    static constexpr int NC  = BS_D_NC; 
    static constexpr int MR  = BS_D_MR; 
    static constexpr int NR  = BS_D_NR; 

    static constexpr int rwidth = SIMD_REGISTER_WIDTH; 
    static constexpr int align = rwidth/8; 
    static constexpr int vlen = rwidth/(8*sizeof(double)); 

    static_assert(MC>0 && KC>0 && NC>0 && MR>0 && NR>0, "Invalid block size."); 
    static_assert(MC % MR == 0, "MC must be a multiple of MR."); 
    static_assert(NC % NR == 0, "NC must be a multiple of NR."); 
    static_assert(rwidth % sizeof(double) == 0, "SIMD register width not sane."); 
}; 

//-- aux routines -------------------------------------------------------------- 
template <typename Index, typename Alpha, typename TX, typename TY> 
void 
geaxpy(Index m, Index n, 
     const Alpha &alpha, 
     const TX *X, Index incRowX, Index incColX, 
     TY  *Y, Index incRowY, Index incColY) 
{ 
    for (Index j=0; j<n; ++j) { 
     for (Index i=0; i<m; ++i) { 
      Y[i*incRowY+j*incColY] += alpha*X[i*incRowX+j*incColX]; 
     } 
    } 
} 

template <typename Index, typename Alpha, typename TX> 
void 
gescal(Index m, Index n, 
     const Alpha &alpha, 
     TX *X, Index incRowX, Index incColX) 
{ 
    if (alpha!=Alpha(0)) { 
     for (Index j=0; j<n; ++j) { 
      for (Index i=0; i<m; ++i) { 
       X[i*incRowX+j*incColX] *= alpha; 
      } 
     } 
    } else { 
     for (Index j=0; j<n; ++j) { 
      for (Index i=0; i<m; ++i) { 
       X[i*incRowX+j*incColX] = Alpha(0); 
      } 
     } 
    } 
} 


//-- Micro Kernel -------------------------------------------------------------- 
template <typename Index, typename T> 
typename std::enable_if<BlockSize<T>::vlen != 0, 
     void>::type 
ugemm(Index kc, T alpha, const T *A, const T *B, T beta, 
     T *C, Index incRowC, Index incColC) 
{ 
    typedef T vx __attribute__((vector_size (BlockSize<T>::rwidth/8))); 

    static constexpr Index vlen = BlockSize<T>::vlen; 
    static constexpr Index MR = BlockSize<T>::MR; 
    static constexpr Index NR = BlockSize<T>::NR/vlen; 

    A = (const T*) __builtin_assume_aligned (A, BlockSize<T>::align); 
    B = (const T*) __builtin_assume_aligned (B, BlockSize<T>::align); 

    vx P[MR*NR] = {}; 

    for (Index l=0; l<kc; ++l) { 
     const vx *b = (const vx *)B; 
     for (Index i=0; i<MR; ++i) { 
      for (Index j=0; j<NR; ++j) { 
       P[i*NR+j] += A[i]*b[j]; 
      } 
     } 
     A += MR; 
     B += vlen*NR; 
    } 

    if (alpha!=T(1)) { 
     for (Index i=0; i<MR; ++i) { 
      for (Index j=0; j<NR; ++j) { 
       P[i*NR+j] *= alpha; 
      } 
     } 
    } 

    if (beta!=T(0)) { 
     for (Index i=0; i<MR; ++i) { 
      for (Index j=0; j<NR; ++j) { 
       const T *p = (const T *) &P[i*NR+j]; 
       for (Index j1=0; j1<vlen; ++j1) { 
        C[i*incRowC+(j*vlen+j1)*incColC] *= beta; 
        C[i*incRowC+(j*vlen+j1)*incColC] += p[j1]; 
       } 
      } 
     } 
    } else { 
     for (Index i=0; i<MR; ++i) { 
      for (Index j=0; j<NR; ++j) { 
       const T *p = (const T *) &P[i*NR+j]; 
       for (Index j1=0; j1<vlen; ++j1) { 
        C[i*incRowC+(j*vlen+j1)*incColC] = p[j1]; 
       } 
      } 
     } 
    } 
} 

//-- Macro Kernel -------------------------------------------------------------- 
template <typename Index, typename T, typename Beta, typename TC> 
void 
mgemm(Index mc, Index nc, Index kc, 
     T alpha, 
     const T *A, const T *B, 
     Beta beta, 
     TC *C, Index incRowC, Index incColC) 
{ 
    const Index MR = BlockSize<T>::MR; 
    const Index NR = BlockSize<T>::NR; 
    const Index mp = (mc+MR-1)/MR; 
    const Index np = (nc+NR-1)/NR; 
    const Index mr_ = mc % MR; 
    const Index nr_ = nc % NR; 

    T C_[MR*NR]; 

    #pragma omp parallel for 
    for (Index j=0; j<np; ++j) { 
     const Index nr = (j!=np-1 || nr_==0) ? NR : nr_; 

     for (Index i=0; i<mp; ++i) { 
      const Index mr = (i!=mp-1 || mr_==0) ? MR : mr_; 

      if (mr==MR && nr==NR) { 
       ugemm(kc, alpha, 
         &A[i*kc*MR], &B[j*kc*NR], 
         beta, 
         &C[i*MR*incRowC+j*NR*incColC], 
         incRowC, incColC); 
      } else { 
       ugemm(kc, alpha, 
         &A[i*kc*MR], &B[j*kc*NR], 
         T(0), 
         C_, Index(1), MR); 
       gescal(mr, nr, beta, 
         &C[i*MR*incRowC+j*NR*incColC], 
         incRowC, incColC); 
       geaxpy(mr, nr, T(1), C_, Index(1), MR, 
         &C[i*MR*incRowC+j*NR*incColC], 
         incRowC, incColC); 
      } 
     } 
    } 
} 
//-- Packing blocks ------------------------------------------------------------ 
template <typename Index, typename TA, typename T> 
void 
pack_A(Index mc, Index kc, 
     const TA *A, Index incRowA, Index incColA, 
     T *p) 
{ 
    Index MR = BlockSize<T>::MR; 
    Index mp = (mc+MR-1)/MR; 

    for (Index j=0; j<kc; ++j) { 
     for (Index l=0; l<mp; ++l) { 
      for (Index i0=0; i0<MR; ++i0) { 
       Index i = l*MR + i0; 
       Index nu = l*MR*kc + j*MR + i0; 
       p[nu] = (i<mc) ? A[i*incRowA+j*incColA] 
           : T(0); 
      } 
     } 
    } 
} 

template <typename Index, typename TB, typename T> 
void 
pack_B(Index kc, Index nc, 
     const TB *B, Index incRowB, Index incColB, 
     T *p) 
{ 
    Index NR = BlockSize<T>::NR; 
    Index np = (nc+NR-1)/NR; 

    for (Index l=0; l<np; ++l) { 
     for (Index j0=0; j0<NR; ++j0) { 
      for (Index i=0; i<kc; ++i) { 
       Index j = l*NR+j0; 
       Index nu = l*NR*kc + i*NR + j0; 
       p[nu] = (j<nc) ? B[i*incRowB+j*incColB] 
           : T(0); 
      } 
     } 
    } 
} 
//-- Frame routine ------------------------------------------------------------- 
template <typename Index, typename Alpha, 
     typename TA, typename TB, 
     typename Beta, 
     typename TC> 
void 
gemm(Index m, Index n, Index k, 
    Alpha alpha, 
    const TA *A, Index incRowA, Index incColA, 
    const TB *B, Index incRowB, Index incColB, 
    Beta beta, 
    TC *C, Index incRowC, Index incColC) 
{ 
    typedef typename std::common_type<Alpha, TA, TB>::type T; 

    const Index MC = BlockSize<T>::MC; 
    const Index NC = BlockSize<T>::NC; 
    const Index MR = BlockSize<T>::MR; 
    const Index NR = BlockSize<T>::NR; 

    const Index KC = BlockSize<T>::KC; 
    const Index mb = (m+MC-1)/MC; 
    const Index nb = (n+NC-1)/NC; 
    const Index kb = (k+KC-1)/KC; 
    const Index mc_ = m % MC; 
    const Index nc_ = n % NC; 
    const Index kc_ = k % KC; 

    T *A_ = (T*) malloc_(BlockSize<T>::align, sizeof(T)*(MC*KC+MR)); 
    T *B_ = (T*) malloc_(BlockSize<T>::align, sizeof(T)*(KC*NC+NR)); 

    if (alpha==Alpha(0) || k==0) { 
     gescal(m, n, beta, C, incRowC, incColC); 
     return; 
    } 

    for (Index j=0; j<nb; ++j) { 
     Index nc = (j!=nb-1 || nc_==0) ? NC : nc_; 

     for (Index l=0; l<kb; ++l) { 
      Index kc = (l!=kb-1 || kc_==0) ? KC : kc_; 
      Beta beta_ = (l==0) ? beta : Beta(1); 

      pack_B(kc, nc, 
        &B[l*KC*incRowB+j*NC*incColB], 
        incRowB, incColB, 
        B_); 

      for (Index i=0; i<mb; ++i) { 
       Index mc = (i!=mb-1 || mc_==0) ? MC : mc_; 

       pack_A(mc, kc, 
         &A[i*MC*incRowA+l*KC*incColA], 
         incRowA, incColA, 
         A_); 

       mgemm(mc, nc, kc, 
         T(alpha), A_, B_, beta_, 
         &C[i*MC*incRowC+j*NC*incColC], 
         incRowC, incColC); 
      } 
     } 
    } 
    free_(A_); 
    free_(B_); 
} 

//------------------------------------------------------------------------------ 

void myprod(double *c, const double* a, const double* b, int N) { 
    gemm(N, N, N, 1.0, a, 1, N, b, 1, N, 0.0, c, 1, N); 
} 

int main(int argc, char* argv[]) { 
    int N = atoi(argv[1]); 
    int tries = 4; 
    int rep = std::max<int>(1,10000000/N/N/N); 

    Eigen::MatrixXd a_E = Eigen::MatrixXd::Random(N,N); 
    Eigen::MatrixXd b_E = Eigen::MatrixXd::Random(N,N); 
    Eigen::MatrixXd c_E(N,N); 

    Eigen::BenchTimer t1, t2; 

    BENCH(t1, tries, rep, c_E.noalias() = a_E*b_E); 
    BENCH(t2, tries, rep, myprod(c_E.data(), a_E.data(), b_E.data(), N)); 

    std::cout << "Time taken by Eigen is: " << t1.best() << "\n"; 
    std::cout << "Time taken by for-loop is: " << t2.best() << "\n\n"; 
} 
+0

Vielen Dank! Ihre Webseite ist sehr nützlich! Interessant, dass du Eigen besiegen kannst! Klären Sie die Eigenheiten von BLAS? Könntest du mich auch wissen lassen - für welche Flagge ist mfma? – Leg

+0

FMA steht für [fusioned-multiply-addition] (https://en.wikipedia.org/wiki/Multiply-accumulate_operation#Fused_multiply.E2.80.93add). Es wird von Intel beginnend mit [Haswell] (https://en.wikipedia.org/wiki/Haswell_ (Mikroarchitektur)) unterstützt. Mit -fma erstellt der Compiler Anweisungen dafür. –

+0

Und nein, Eigen verlässt sich nicht auf BLAS. –

2

Es gibt zwei einfache Optimierungen, die ich beraten kann.

1) Vectorize es. Es wäre besser, wenn Sie es mit Inline-Assembly vektorisieren oder Assembly Proc schreiben, aber Sie können Compiler intrinsics auch verwenden. Sie können sogar den Compiler veranlassen, die Schleife zu vektorisieren, aber es ist manchmal schwierig, die richtige Schleife zu schreiben, um vom Compiler vektorisiert zu werden.

2) Machen Sie es parallel. Versuchen Sie es mit OpenMP.

+0

Vectorisiert der Compiler den Code nicht bereits? – Leg

+0

@Leg, wenn Sie mit '-O3' kompilieren, sollte GCC' c [count] + = a [count1] * b [count2] 'vektorisieren. –

2

Ihr Code wird vom Compiler bereits gut vektorisiert. Der Schlüssel für höhere Leistung ist die hierarchische Blockierung, um die Verwendung von Registern und die unterschiedliche Cache-Ebene zu optimieren. Das partielle Loop-Abrolling ist ebenfalls entscheidend, um das Befehls-Pipelining zu verbessern. Die Leistung von Eigens Produkt zu erreichen, erfordert viel Aufwand und Tuning.

Es sollte auch beachtet werden, dass Ihr Benchmark leicht verzerrt und nicht vollständig zuverlässig ist. Hier ist eine verlässlichere Version (Sie benötigen komplette Eigens Quellen bench/BenchTimer.h zu bekommen):

#include<iostream> 
#include<Eigen/Dense> 
#include<bench/BenchTimer.h> 

void myprod(double *c, const double* a, const double* b, int N) { 
    int count = 0; 
    int count1, count2; 
    for (int j=0; j<N; ++j) { 
     count1 = j*N; 
     for (int k=0; k<N; ++k) { 
      c[count] = a[count1]*b[k]; 
      ++count; 
     } 
    } 

    for (int j=0; j<N; ++j) { 
     count2 = N; 
     for (int l=1; l<N; ++l) { 
      count = j*N; 
      count1 = count+l; 
      for (int k=0; k<N; ++k) { 
       c[count]+=a[count1]*b[count2]; 
       ++count; 
       ++count2; 
      } 
     } 
    } 
} 

int main(int argc, char* argv[]) { 
    int N = atoi(argv[1]); 
    int tries = 4; 
    int rep = std::max<int>(1,10000000/N/N/N); 

    Eigen::MatrixXd a_E = Eigen::MatrixXd::Random(N,N); 
    Eigen::MatrixXd b_E = Eigen::MatrixXd::Random(N,N); 
    Eigen::MatrixXd c_E(N,N); 

    Eigen::BenchTimer t1, t2; 

    BENCH(t1, tries, rep, c_E.noalias() = a_E*b_E); 
    BENCH(t2, tries, rep, myprod(c_E.data(), a_E.data(), b_E.data(), N)); 

    std::cout << "\nTime taken by Eigen is: " << t1.best() << "\n"; 
    std::cout << "\nTime taken by for-loop is: " << t2.best() << "\n"; 
} 

Kompilieren mit 3.3-beta1 und FMA freigegeben (-mfma), dann wird der Spalt viel größer, fast eine Größenordnung für N=2000.

+0

Danke wie immer! Ich habe meinen Rechner mit clang ++ auf ElCapitan ausprobiert und hier sind die Timings für den Fall, dass es dich interessiert. Für eine 2000 von 2000 mat-mat Produkt Zeit von Eigen gemacht ist: 1,08311 Zeit genommen durch for-Schleife: 3,37003 – Leg

+1

Könnten Sie lassen Sie mich wissen, was die -mfma Flagge für ist? – Leg

+0

Ich bekomme das gleiche mit for-Schleife (3,36s) aber nur 0,36s mit Eigen. Ich benutze auch OSX und klingle mit '-O3 -DNDEBUG -mfma' und Eigen 3.3-beta1. '-mfma' aktiviert AVX + FMA Befehlssätze. FMA steht für fusioned-multiply-add und ergibt eine zusätzliche x1.7-Beschleunigung. – ggael