// copied from https://github.com/ggerganov/llama.cpp/blob/b2899/ggml-cuda/mmq.cu template static __device__ __forceinline__ void mul_mat_q( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const block_q_t * x = (const block_q_t *) vx; const block_q8_1 * y = (const block_q8_1 *) vy; const int blocks_per_row_x = ncols_x / qk; const int blocks_per_col_y = nrows_y / QK8_1; const int blocks_per_warp = WARP_SIZE / qi; const int & ncols_dst = ncols_y; const int row_dst_0 = blockIdx.x*mmq_y; const int & row_x_0 = row_dst_0; const int col_dst_0 = blockIdx.y*mmq_x; const int & col_y_0 = col_dst_0; int * tile_x_ql = nullptr; half2 * tile_x_dm = nullptr; int * tile_x_qh = nullptr; int * tile_x_sc = nullptr; allocate_tiles(&tile_x_ql, &tile_x_dm, &tile_x_qh, &tile_x_sc); __shared__ int tile_y_qs[mmq_x * WARP_SIZE]; __shared__ half2 tile_y_ds[mmq_x * WARP_SIZE/QI8_1]; float sum[mmq_y/WARP_SIZE][mmq_x/nwarps] = {{0.0f}}; for (int ib0 = 0; ib0 < blocks_per_row_x; ib0 += blocks_per_warp) { load_tiles(x + row_x_0*blocks_per_row_x + ib0, tile_x_ql, tile_x_dm, tile_x_qh, tile_x_sc, threadIdx.y, nrows_x-row_x_0-1, threadIdx.x, blocks_per_row_x); #pragma unroll for (int ir = 0; ir < qr; ++ir) { const int kqs = ir*WARP_SIZE + threadIdx.x; const int kbxd = kqs / QI8_1; #pragma unroll for (int i = 0; i < mmq_x; i += nwarps) { const int col_y_eff = min(col_y_0 + threadIdx.y + i, ncols_y-1); // to prevent out-of-bounds memory accesses const block_q8_1 * by0 = &y[col_y_eff*blocks_per_col_y + ib0 * (qk/QK8_1) + kbxd]; const int index_y = (threadIdx.y + i) * WARP_SIZE + kqs % WARP_SIZE; tile_y_qs[index_y] = get_int_from_int8_aligned(by0->qs, threadIdx.x % QI8_1); } #pragma unroll for (int ids0 = 0; ids0 < mmq_x; ids0 += nwarps * QI8_1) { const int ids = (ids0 + threadIdx.y * QI8_1 + threadIdx.x / (WARP_SIZE/QI8_1)) % mmq_x; const int kby = threadIdx.x % (WARP_SIZE/QI8_1); const int col_y_eff = min(col_y_0 + ids, ncols_y-1); // if the sum is not needed it's faster to transform the scale to f32 ahead of time const half2 * dsi_src = &y[col_y_eff*blocks_per_col_y + ib0 * (qk/QK8_1) + ir*(WARP_SIZE/QI8_1) + kby].ds; half2 * dsi_dst = &tile_y_ds[ids * (WARP_SIZE/QI8_1) + kby]; if (need_sum) { *dsi_dst = *dsi_src; } else { float * dfi_dst = (float *) dsi_dst; *dfi_dst = __low2float(*dsi_src); } } __syncthreads(); // #pragma unroll // unrolling this loop causes too much register pressure for (int k = ir*WARP_SIZE/qr; k < (ir+1)*WARP_SIZE/qr; k += vdr) { #pragma unroll for (int j = 0; j < mmq_x; j += nwarps) { #pragma unroll for (int i = 0; i < mmq_y; i += WARP_SIZE) { sum[i/WARP_SIZE][j/nwarps] += vec_dot( tile_x_ql, tile_x_dm, tile_x_qh, tile_x_sc, tile_y_qs, tile_y_ds, threadIdx.x + i, threadIdx.y + j, k); } } } __syncthreads(); } } #pragma unroll for (int j = 0; j < mmq_x; j += nwarps) { const int col_dst = col_dst_0 + j + threadIdx.y; if (col_dst >= ncols_dst) { return; } #pragma unroll for (int i = 0; i < mmq_y; i += WARP_SIZE) { const int row_dst = row_dst_0 + threadIdx.x + i; if (row_dst >= nrows_dst) { continue; } dst[col_dst*nrows_dst + row_dst] = __float2half(sum[i/WARP_SIZE][j/nwarps]); } } } #if defined(USE_ROCM) #define MMQ_X_Q4_0 64 #define MMQ_Y_Q4_0 128 #define NWARPS_Q4_0 8 #else #define MMQ_X_Q4_0 4 #define MMQ_Y_Q4_0 32 #define NWARPS_Q4_0 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q4_0, 2) #endif mul_mat_q4_0( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q4_0; const int mmq_y = MMQ_Y_Q4_0; const int nwarps = NWARPS_Q4_0; mul_mat_q, load_tiles_q4_0, VDR_Q4_0_Q8_1_MMQ, vec_dot_q4_0_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q4_0_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { int mmq_x = MMQ_X_Q4_0; int mmq_y = MMQ_Y_Q4_0; int nwarps = NWARPS_Q4_0; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q4_0<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q4_0<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q4_1 64 #define MMQ_Y_Q4_1 128 #define NWARPS_Q4_1 8 #else #define MMQ_X_Q4_1 4 #define MMQ_Y_Q4_1 32 #define NWARPS_Q4_1 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q4_1, 2) #endif mul_mat_q4_1( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q4_1; const int mmq_y = MMQ_Y_Q4_1; const int nwarps = NWARPS_Q4_1; mul_mat_q, load_tiles_q4_1, VDR_Q4_1_Q8_1_MMQ, vec_dot_q4_1_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q4_1_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { int mmq_x = MMQ_X_Q4_1; int mmq_y = MMQ_Y_Q4_1; int nwarps = NWARPS_Q4_1; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q4_1<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q4_1<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q5_0 64 #define MMQ_Y_Q5_0 128 #define NWARPS_Q5_0 8 #else #define MMQ_X_Q5_0 4 #define MMQ_Y_Q5_0 32 #define NWARPS_Q5_0 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q5_0, 2) #endif mul_mat_q5_0( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q5_0; const int mmq_y = MMQ_Y_Q5_0; const int nwarps = NWARPS_Q5_0; mul_mat_q, load_tiles_q5_0, VDR_Q5_0_Q8_1_MMQ, vec_dot_q5_0_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q5_0_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { const int mmq_x = MMQ_X_Q5_0; const int mmq_y = MMQ_Y_Q5_0; const int nwarps = NWARPS_Q5_0; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q5_0<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q5_0<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q5_1 64 #define MMQ_Y_Q5_1 128 #define NWARPS_Q5_1 8 #else #define MMQ_X_Q5_1 4 #define MMQ_Y_Q5_1 32 #define NWARPS_Q5_1 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q5_1, 2) #endif mul_mat_q5_1( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q5_1; const int mmq_y = MMQ_Y_Q5_1; const int nwarps = NWARPS_Q5_1; mul_mat_q, load_tiles_q5_1, VDR_Q5_1_Q8_1_MMQ, vec_dot_q5_1_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q5_1_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { const int mmq_x = MMQ_X_Q5_1; const int mmq_y = MMQ_Y_Q5_1; const int nwarps = NWARPS_Q5_1; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q5_1<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q5_1<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q8_0 64 #define MMQ_Y_Q8_0 128 #define NWARPS_Q8_0 8 #else #define MMQ_X_Q8_0 4 #define MMQ_Y_Q8_0 32 #define NWARPS_Q8_0 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q8_0, 2) #endif mul_mat_q8_0( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q8_0; const int mmq_y = MMQ_Y_Q8_0; const int nwarps = NWARPS_Q8_0; mul_mat_q, load_tiles_q8_0, VDR_Q8_0_Q8_1_MMQ, vec_dot_q8_0_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q8_0_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { const int mmq_x = MMQ_X_Q8_0; const int mmq_y = MMQ_Y_Q8_0; const int nwarps = NWARPS_Q8_0; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q8_0<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q8_0<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q2_K 64 #define MMQ_Y_Q2_K 128 #define NWARPS_Q2_K 8 #else #define MMQ_X_Q2_K 4 #define MMQ_Y_Q2_K 32 #define NWARPS_Q2_K 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q2_K, 2) #endif mul_mat_q2_K( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q2_K; const int mmq_y = MMQ_Y_Q2_K; const int nwarps = NWARPS_Q2_K; mul_mat_q, load_tiles_q2_K, VDR_Q2_K_Q8_1_MMQ, vec_dot_q2_K_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q2_K_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { const int mmq_x = MMQ_X_Q2_K; const int mmq_y = MMQ_Y_Q2_K; const int nwarps = NWARPS_Q2_K; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q2_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q2_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q3_K 64 #define MMQ_Y_Q3_K 128 #define NWARPS_Q3_K 8 #else #define MMQ_X_Q3_K 4 #define MMQ_Y_Q3_K 32 #define NWARPS_Q3_K 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q3_K, 2) #endif mul_mat_q3_K( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q3_K; const int mmq_y = MMQ_Y_Q3_K; const int nwarps = NWARPS_Q3_K; mul_mat_q, load_tiles_q3_K, VDR_Q3_K_Q8_1_MMQ, vec_dot_q3_K_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q3_K_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { const int mmq_x = MMQ_X_Q3_K; const int mmq_y = MMQ_Y_Q3_K; const int nwarps = NWARPS_Q3_K; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q3_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q3_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q4_K 64 #define MMQ_Y_Q4_K 128 #define NWARPS_Q4_K 8 #else #define MMQ_X_Q4_K 4 #define MMQ_Y_Q4_K 32 #define NWARPS_Q4_K 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q4_K, 2) #endif mul_mat_q4_K( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q4_K; const int mmq_y = MMQ_Y_Q4_K; const int nwarps = NWARPS_Q4_K; mul_mat_q, load_tiles_q4_K, VDR_Q4_K_Q8_1_MMQ, vec_dot_q4_K_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q4_K_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { const int mmq_x = MMQ_X_Q4_K; const int mmq_y = MMQ_Y_Q4_K; const int nwarps = NWARPS_Q4_K; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q4_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q4_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q5_K 64 #define MMQ_Y_Q5_K 128 #define NWARPS_Q5_K 8 #else #define MMQ_X_Q5_K 4 #define MMQ_Y_Q5_K 32 #define NWARPS_Q5_K 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q5_K, 2) #endif mul_mat_q5_K( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q5_K; const int mmq_y = MMQ_Y_Q5_K; const int nwarps = NWARPS_Q5_K; mul_mat_q, load_tiles_q5_K, VDR_Q5_K_Q8_1_MMQ, vec_dot_q5_K_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q5_K_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { const int mmq_x = MMQ_X_Q5_K; const int mmq_y = MMQ_Y_Q5_K; const int nwarps = NWARPS_Q5_K; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q5_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q5_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } } #if defined(USE_ROCM) #define MMQ_X_Q6_K 64 #define MMQ_Y_Q6_K 128 #define NWARPS_Q6_K 8 #else #define MMQ_X_Q6_K 4 #define MMQ_Y_Q6_K 32 #define NWARPS_Q6_K 4 #endif template static __global__ void #if defined(USE_ROCM) __launch_bounds__(WARP_SIZE*NWARPS_Q6_K, 2) #endif mul_mat_q6_K( const void * __restrict__ vx, const void * __restrict__ vy, half * __restrict__ dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst) { const int mmq_x = MMQ_X_Q6_K; const int mmq_y = MMQ_Y_Q6_K; const int nwarps = NWARPS_Q6_K; mul_mat_q, load_tiles_q6_K, VDR_Q6_K_Q8_1_MMQ, vec_dot_q6_K_q8_1_mul_mat> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } static void ggml_mul_mat_q6_K_q8_1_cuda( const void * vx, const void * vy, half * dst, const int ncols_x, const int nrows_x, const int ncols_y, const int nrows_y, const int nrows_dst, cudaStream_t stream) { const int mmq_x = MMQ_X_Q6_K; const int mmq_y = MMQ_Y_Q6_K; const int nwarps = NWARPS_Q6_K; const int block_num_x = (nrows_x + mmq_y - 1) / mmq_y; const int block_num_y = (ncols_y + mmq_x - 1) / mmq_x; const dim3 block_nums(block_num_x, block_num_y, 1); const dim3 block_dims(WARP_SIZE, nwarps, 1); if (nrows_x % mmq_y == 0) { const bool need_check = false; mul_mat_q6_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } else { const bool need_check = true; mul_mat_q6_K<<>> (vx, vy, dst, ncols_x, nrows_x, ncols_y, nrows_y, nrows_dst); } }