Skip to content

Latest commit

 

History

History
162 lines (87 loc) · 8.94 KB

File metadata and controls

162 lines (87 loc) · 8.94 KB

How to optimize DGEMM on x86 CPU platforms

General matrix/matrix multiplication (GEMM) is a core routine of many popular algorithms. On modern computing platforms with hierarchical memory architecture, it is typically possible that we can reach near-optimal performance for GEMM. For example, on most x86 CPUs, Intel MKL, as well as other well-known BLAS implementations including OpenBLAS and BLIS, can provide >90% of the peak performance for GEMM. On the GPU side, cuBLAS, provided by NVIDIA, can also provide near-optimal performance for GEMM. Though optimizing serial implementation of GEMM on x86 platforms is never a new topic, a tutorial discussing optimizing GEMM on x86 platforms with AVX512 instructions is missing among existing learning resources online. Additionally, with the increasing on data width compared between AVX512 and its predecessors AVX2, AVX, SSE4 and etc, the gap between the peak computational capability and the memory bandwidth continues growing. This simultaneously gives rise of the requirement on programmers to design more delicate prefetching schemes in order to hide the memory latency. Comparing with existed turials, ours is the first one which not only touches the implementation leaveraging AVX512 instructions, and provides step-wise optimization with prefetching strategies as well. The DGEMM implementation eventually reaches comparable performance to Intel MKL.

Q & A

I warmly welcome questions and discussions through pull requests or my personal email at [email protected]

Hardware platforms and software configurations

  • We require a CPU with the CPU flag avx512f to run all test cases in this tutorial. This can be checked on terminal using the command: lscpu | grep "avx512f".
  • The experimental data shown are collected on an Intel Xeon W-2255 CPU (2 AVX512 units, base frequency 3.7 GHz, turbo boost frequency running AVX512 instructions on a single core: 4.0 GHz). This workstation is equipped with 4X8GB=32GB DRAM at 2933 GHz. The theoretical peak performance on a single core is: 2(FMA)*2(AVX512 Units)*512(data with)/64(bit of a fp64 number)*4 GHz = 128 GFLOPS.
  • We compiled the program with gcc 7.3.0 under Ubuntu 18.04.5 LTS.
  • Intel MKL version: oneMKL 2021.1 beta.

How to run

Just three steps.

  • We first modify the path of MKL in Makefile.
  • Second, type in make to compile. A binary executable dgemm_x86 will be generated.
  • Third, run the binary using ./dgemm_x86 [kernel_number], where kernel_number selects the kernel for benchmark. 0 represents Intel MKL and 1-19 represent 19 kernels demonstrating the optimizing strategies. Here kernel18 is the best serial version while kernel19 is the best parallel version. Both of them reach the performance comparable to / faster than Intel MKL on lastest Intel CPUs.

Related good GEMM tutorials/materials on x86-64 CPUs

Step-wise Optimizations

Here we take the column-major implemetation for DGEMM.

Kernel 1 (naive version)

source code

Kernel 1 is the most naive implementation of DGEMM.

Kernel 2 (register re-use)

source code

Observing the innermost loop in kernel1, C(i,j) is irrelevant to the innermost loop index k, therefore, one can load it into the register before entering the k-loop to avoid unnecessary memory access.

image

Kernel 3 (2x2 register blocking)

source code

Since matrix multiplication is an algorithm at the complexity order of N^3, one should basically re-use the data at register/cache levels to improve the performance. In kernel2, we cached a C element in register but failed to re-use the data on matrix A and B at the register level. Now we update the whole 2x2 block of C matrix by loading a 2x1 slice of A and an 1x2 slice of B. Then we conduct an outer-product rank-1 update on the 2x2 C block. This significantly improves the performance compared to the previous step.

image

Kernel 4 (4x4 register blocking)

source code

Now we more aggresively leverage the register-blocking strategy. We update on an 4x4 C block each time. The performance further improves.

image

Kernel 5 (Kernel 4 + AVX2)

source code

Instead of processing data with scalar instructions, now we process data using Single Instruction Multiple Data (SIMD) instructions. An AVX2 instruction, whose data width is 256-bit, can process 4 fp64 elements at a lane.

image

Kernel 6 (Kernel 5 + loop unrolling x 4)

source code

We unroll the loop by 4 folds. This step slightly improves the performance.

image

Kernel 7 (8x4 kernel + AVX2 + loop unrolling x 4)

source code

We changed the previous kernel from 4x4 to the current 8x4 so that we obtain a better utilization on all 256-bit YMM registers (16 YMMs on SNB/HSW/BRW and 32 YMM/ZMM on SKL/CSL).

image

Kernel 8 (Kernel 7 + cache blocking)

source code

We notice that the performance around 30 GFLOPS cannot be maintained when data cannot be held in cache. Therefore, we introduce cache blocking to maintain the good performance when matrix sizes are small.

image

Kernel 9 (Kernel 8 + packing)

source code

To avoid the TLB miss when accessing the cache blocks, we pack the data blocks into continous memory before loading them. This strategy boosts the performance.

image

Kernel 10 (24x8 kernel + AVX512 + blocking + packing)

source code

We update kernel9 with a larger micro kernel (24x8 instead of 8x4) and wider data width (AVX512 instead of AVX2). All other strategies maintain.

image

Kernel 11 (Kernel 10 + discontinous packing on B)

source code

Previously in kernel10 we pack the matrix B into totally continous memory. Recalling that the L2 cache contains multiple cache line buffers so that the L2 hardware prefetcher can prefetch the data from lower memory hierarchy. In this step, breaking the continous memory access in B can benefit the L2 hw prefetcher so that the memory latency can be further hidden.

image

Kernel 12 (Kernel 11: from instrinsics to inline ASM )

source code

image

Kernel 13 (Kernel 12 + changing the whole macro kernel into inline ASM)

source code

image

Kernel 14 (Kernel 13 + software prefetching on A)

source code

image

Kernel 15 (Kernel 14 + software prefetching on B)

source code

image

Kernel 16 (Kernel 15 + software prefetching on C)

source code

image

Kernel 17 (Kernel 16 + fine-tuned matrix scaling routine on C)

source code

image

Kernel 18 (Kernel 17 fine-grained packing for B to benefit the CPU frequency boosting)

source code

image

Kernel 18 comparison against Intel MKL

image