#include "common.h" #include "gemm.h" #include "vec.h" #include "vec_pack.h" namespace { // For this cpu kernel, we have some innovations aside from the existing gpu kernels: // 1) Use less parallel loops, i.e. 4 including l2_norm. // 2) Fuse part of l2_norm with the rest of the computation. #define THREAD_BUFFER_ALLOC(dst, base_ptr, offset, type, size) \ type* dst = reinterpret_cast((base_ptr) + (offset)); \ offset += (size); template inline void fill_stub(scalar_t* __restrict__ out, float val, int size) { using Vec = at::vec::Vectorized; constexpr int kVecSize = Vec::size(); const Vec data_vec = Vec(static_cast(val)); int d = 0; #pragma GCC unroll 4 for (; d <= size - kVecSize; d += kVecSize) { data_vec.store(out + d); } if (size - d > 0) { data_vec.store(out + d, size - d); } } template void chunk_gated_delta_rule_kernel_impl( scalar_t* __restrict__ out, // [B, T, HV, EV] float* __restrict__ final_state_data, // [N, HV, EK, EV] const scalar_t* __restrict__ q_orig, // [B, T, HK, EK] const scalar_t* __restrict__ k_orig, // [B, T, HK, EK] const scalar_t* __restrict__ v_orig, // [B, T, HV, EV] const float* __restrict__ g_orig, // [B, T, HV] FP32 const scalar_t* __restrict__ b_orig, // [B, T, HV] const int32_t* __restrict__ cu_seqlens_ptr, // [N + 1] INT32 float* __restrict__ buff, scalar_t* __restrict__ reduced_buff, scalar_t* __restrict__ thread_buff, const int32_t* __restrict__ chunk_offsets_ptr, const int32_t* __restrict__ chunk_indices_ptr, bool use_qk_l2norm_in_kernel, const int64_t& batch_size, const int64_t& global_seq_len, const int64_t& qk_num_head, const int64_t& v_num_head, const int64_t& qk_head_size, const int64_t& v_head_size, const int64_t& qStrideH, const int64_t& qStrideT, const int64_t& kStrideH, const int64_t& kStrideT, const int64_t& vStrideH, const int64_t& vStrideT, const int64_t& oStrideH, const int64_t& oStrideT, const int64_t& global_total_seq_length, const int64_t& global_num_chunk, const int64_t& buff_size_16bit_per_thread, double eps = 1e-5) { int64_t gStrideH = 1; int64_t gStrideT = v_num_head; int64_t bStrideH = 1; int64_t bStrideT = v_num_head; int64_t final_state_StrideN = v_num_head * qk_head_size * v_head_size; int64_t final_state_StrideH = qk_head_size * v_head_size; int64_t final_state_StrideE = v_head_size; int64_t head_group = v_num_head / qk_num_head; float scale = 1.0 / std::sqrt(qk_head_size); using bVec = at::vec::Vectorized; using fVec = at::vec::Vectorized; constexpr int64_t VecSize = bVec::size(); constexpr int64_t fVecSize = fVec::size(); // Data pointers float* g_pad = buff; float* core_attn_out = g_pad + v_num_head * global_total_seq_length; float* decay_mask = core_attn_out + batch_size * v_num_head * global_total_seq_length * v_head_size; float* v_beta_attn = decay_mask + v_num_head * global_total_seq_length * chunk_size; scalar_t* q_pad = reduced_buff; scalar_t* k_pad = q_pad + qk_num_head * global_total_seq_length * qk_head_size; scalar_t* v_pad = k_pad + qk_num_head * global_total_seq_length * qk_head_size; scalar_t* k_beta = v_pad + v_num_head * global_total_seq_length * v_head_size; scalar_t* v_beta = k_beta + v_num_head * global_total_seq_length * qk_head_size; scalar_t* k_cumdecay_reduced = v_beta + v_num_head * global_total_seq_length * v_head_size; scalar_t* q_norm_sum = k_cumdecay_reduced + v_num_head * global_total_seq_length * qk_head_size; scalar_t* k_norm_sum = q_norm_sum + qk_num_head * global_seq_len; if (use_qk_l2norm_in_kernel) { at::parallel_for(0, qk_num_head * global_seq_len, 0, [&](int64_t begin, int64_t end) { int64_t h_qk = 0, l = 0; data_index_init(begin, h_qk, qk_num_head, l, global_seq_len); for (int64_t i = begin; i < end; ++i) { auto q_norm_sum_ptr = q_norm_sum + h_qk * global_seq_len + l; auto k_norm_sum_ptr = k_norm_sum + h_qk * global_seq_len + l; float sum_q = float(0); float sum_k = float(0); fVec sum_q_fvec = fVec(float(0)); fVec sum_k_fvec = fVec(float(0)); int64_t q_offset = l * qStrideT + h_qk * qStrideH; int64_t k_offset = l * qStrideT + h_qk * qStrideH; int64_t d; for (d = 0; d <= qk_head_size - VecSize; d += VecSize) { bVec q_bvec = bVec::loadu(q_orig + q_offset + d); fVec q_fvec0, q_fvec1; std::tie(q_fvec0, q_fvec1) = at::vec::convert_to_float(q_bvec); sum_q_fvec += q_fvec0 * q_fvec0; sum_q_fvec += q_fvec1 * q_fvec1; bVec k_bvec = bVec::loadu(k_orig + k_offset + d); fVec k_fvec0, k_fvec1; std::tie(k_fvec0, k_fvec1) = at::vec::convert_to_float(k_bvec); sum_k_fvec += k_fvec0 * k_fvec0; sum_k_fvec += k_fvec1 * k_fvec1; } sum_q += vec_reduce_sum(sum_q_fvec); sum_k += vec_reduce_sum(sum_k_fvec); q_norm_sum_ptr[0] = static_cast(float(1) / std::sqrt(sum_q + eps)); k_norm_sum_ptr[0] = static_cast(float(1) / std::sqrt(sum_k + eps)); data_index_step(h_qk, qk_num_head, l, global_seq_len); } }); } // query = query * scale // k_beta = key * beta.unsqueeze(-1) // v_beta = value * beta.unsqueeze(-1) // Padding for q/k/v/beta at::parallel_for(0, qk_num_head * global_num_chunk, 1, [&](int64_t begin, int64_t end) { int ompIdx = at::get_thread_num(); int64_t h_qk = 0, c = 0; data_index_init(begin, h_qk, qk_num_head, c, global_num_chunk); for ([[maybe_unused]] auto z : c10::irange(begin, end)) { int64_t ib = chunk_indices_ptr[c * 2]; // idx_batch int64_t ic = chunk_indices_ptr[c * 2 + 1]; // idx_chunk int64_t l_orig = cu_seqlens_ptr[ib] + ic * chunk_size; int64_t l = c * chunk_size; bool is_tail = (c + 1 == chunk_offsets_ptr[ib + 1]); int64_t seq_len = cu_seqlens_ptr[ib + 1] - cu_seqlens_ptr[ib]; int64_t real_chunk_size = is_tail ? seq_len - ic * chunk_size : chunk_size; auto q_orig_ptr = q_orig + h_qk * qStrideH + l_orig * qStrideT; auto k_orig_ptr = k_orig + h_qk * kStrideH + l_orig * kStrideT; auto v_orig_ptr = v_orig + l_orig * vStrideT; auto b_orig_ptr = b_orig + l_orig * bStrideT; auto q_pad_ptr = q_pad + h_qk * global_total_seq_length * qk_head_size + l * qk_head_size; auto k_pad_ptr = k_pad + h_qk * global_total_seq_length * qk_head_size + l * qk_head_size; auto v_pad_ptr = v_pad + l * v_head_size; auto k_beta_ptr = k_beta + l * qk_head_size; auto v_beta_ptr = v_beta + l * v_head_size; for (int64_t j = 0; j < real_chunk_size; j++) { auto curr_q_orig = q_orig_ptr + j * qStrideT; auto curr_k_orig = k_orig_ptr + j * kStrideT; auto curr_q_pad = q_pad_ptr + j * qk_head_size; auto curr_k_pad = k_pad_ptr + j * qk_head_size; auto q_scale = use_qk_l2norm_in_kernel ? *(q_norm_sum + h_qk * global_seq_len + l_orig + j) : static_cast(1); auto k_scale = use_qk_l2norm_in_kernel ? *(k_norm_sum + h_qk * global_seq_len + l_orig + j) : static_cast(1); auto q_scale_vec = bVec(q_scale); auto k_scale_vec = bVec(k_scale); int64_t i = 0; scalar_t scale_reduced = static_cast(scale); auto vec_scale_reduced = bVec(scale_reduced); for (; i < fVecSize * (qk_head_size / fVecSize); i += fVecSize) { auto tmp0 = bVec::loadu(curr_q_orig + i, fVecSize); auto tmp1 = tmp0 * q_scale_vec * vec_scale_reduced; tmp1.store(curr_q_pad + i, fVecSize); auto tmp3 = bVec::loadu(curr_k_orig + i, fVecSize); auto tmp4 = tmp3 * k_scale_vec; tmp4.store(curr_k_pad + i, fVecSize); } for (auto hi = 0; hi < head_group; hi++) { int64_t h = h_qk * head_group + hi; auto curr_v_orig = v_orig_ptr + h * vStrideH + j * vStrideT; auto curr_b_orig = b_orig_ptr + h * bStrideH + j * bStrideT; scalar_t b_orig_val_reduced = *(curr_b_orig); auto curr_v_pad = v_pad_ptr + h * global_total_seq_length * v_head_size + j * v_head_size; auto curr_k_beta = k_beta_ptr + h * global_total_seq_length * qk_head_size + j * qk_head_size; auto curr_v_beta = v_beta_ptr + h * global_total_seq_length * v_head_size + j * v_head_size; // query = query * scale // k_beta = key * beta.unsqueeze(-1) int64_t i = 0; auto vec_b_reduced = bVec(b_orig_val_reduced); for (; i < fVecSize * (qk_head_size / fVecSize); i += fVecSize) { auto tmp0 = bVec::loadu(curr_k_orig + i, fVecSize); auto tmp2 = tmp0 * k_scale_vec * vec_b_reduced; tmp2.store(curr_k_beta + i, fVecSize); } // v_beta = value * beta.unsqueeze(-1) i = 0; for (; i < VecSize * (v_head_size / VecSize); i += VecSize) { auto tmp3 = bVec::loadu(curr_v_orig + i); tmp3.store(curr_v_pad + i); auto tmp5 = tmp3 * vec_b_reduced; tmp5.store(curr_v_beta + i); } } } for (int64_t j = real_chunk_size; j < chunk_size; j++) { auto curr_q_pad = q_pad_ptr + j * qk_head_size; auto curr_k_pad = k_pad_ptr + j * qk_head_size; int64_t i = 0; auto vec_zero = bVec(0.0); for (; i < VecSize * (qk_head_size / VecSize); i += VecSize) { vec_zero.store(curr_q_pad + i); vec_zero.store(curr_k_pad + i); } for (auto hi = 0; hi < head_group; hi++) { int64_t h = h_qk * head_group + hi; auto curr_v_pad = v_pad_ptr + h * global_total_seq_length * v_head_size + j * v_head_size; auto curr_k_beta = k_beta_ptr + h * global_total_seq_length * qk_head_size + j * qk_head_size; auto curr_v_beta = v_beta_ptr + h * global_total_seq_length * v_head_size + j * v_head_size; int64_t i = 0; for (; i < VecSize * (qk_head_size / VecSize); i += VecSize) { vec_zero.store(curr_k_beta + i); } i = 0; for (; i < VecSize * (v_head_size / VecSize); i += VecSize) { vec_zero.store(curr_v_pad + i); vec_zero.store(curr_v_beta + i); } } } // Move to the next query data_index_step(h_qk, qk_num_head, c, global_num_chunk); } }); at::parallel_for(0, v_num_head * global_num_chunk, 1, [&](int64_t begin, int64_t end) { int64_t h = 0, c = 0; data_index_init(begin, h, v_num_head, c, global_num_chunk); int ompIdx = at::get_thread_num(); int64_t offset = 0; scalar_t* thread_buff_ptr = thread_buff + ompIdx * buff_size_16bit_per_thread; THREAD_BUFFER_ALLOC(k_transpose, thread_buff_ptr, offset, scalar_t, qk_head_size * chunk_size); THREAD_BUFFER_ALLOC(v_pack, thread_buff_ptr, offset, scalar_t, chunk_size * v_head_size); THREAD_BUFFER_ALLOC(k_beta_g, thread_buff_ptr, offset, scalar_t, chunk_size * qk_head_size); THREAD_BUFFER_ALLOC(k_beta_g_pack, thread_buff_ptr, offset, scalar_t, chunk_size * qk_head_size); THREAD_BUFFER_ALLOC(curr_attn, thread_buff_ptr, offset, float, chunk_size* chunk_size * 2); THREAD_BUFFER_ALLOC(curr_attn_reduced, thread_buff_ptr, offset, scalar_t, chunk_size * chunk_size); THREAD_BUFFER_ALLOC(k_cumdecay, thread_buff_ptr, offset, float, chunk_size* qk_head_size * 2); THREAD_BUFFER_ALLOC(row, thread_buff_ptr, offset, float, chunk_size * 2); THREAD_BUFFER_ALLOC(updated, thread_buff_ptr, offset, float, chunk_size * 2); for ([[maybe_unused]] auto z : c10::irange(begin, end)) { int64_t ib = chunk_indices_ptr[c * 2]; // idx_batch int64_t ic = chunk_indices_ptr[c * 2 + 1]; // idx_chunk int64_t l_orig = cu_seqlens_ptr[ib] + ic * chunk_size; int64_t seq_len = cu_seqlens_ptr[ib + 1] - cu_seqlens_ptr[ib]; int64_t h_qk = h / head_group; auto curr_g_orig = g_orig + h * gStrideH + l_orig * gStrideT; auto curr_g_pad = g_pad + h * global_total_seq_length + c * chunk_size; auto curr_decay_mask = decay_mask + h * global_total_seq_length * chunk_size + c * chunk_size * chunk_size; auto curr_k_pad = k_pad + h_qk * global_total_seq_length * qk_head_size + c * chunk_size * qk_head_size; auto curr_k_beta = k_beta + h * global_total_seq_length * qk_head_size + c * chunk_size * qk_head_size; auto curr_k_cumdecay_reduced = k_cumdecay_reduced + h * global_total_seq_length * qk_head_size + c * chunk_size * qk_head_size; auto curr_v_beta = v_beta + h * global_total_seq_length * v_head_size + c * chunk_size * v_head_size; auto curr_value = v_beta_attn + h * global_total_seq_length * v_head_size + c * chunk_size * v_head_size; float acc_val = 0; for (int64_t i = 0; i < chunk_size; i++) { // Padding for g // g = g.cumsum(dim=-1) // g: [B, HV, num_chunk, chunk_size] if (ic * chunk_size + i < seq_len) { acc_val += curr_g_orig[i * gStrideT]; } curr_g_pad[i] = acc_val; // decay_mask = ((g.unsqueeze(-1) - g.unsqueeze(-2)).tril().exp().float()).tril() // decay_mask: [B, HV, num_chunk, chunk_size, chunk_size] float curr_g_pad_i = static_cast(curr_g_pad[i]); auto vec_curr_g_pad_i = fVec(curr_g_pad_i); int64_t j = 0; int64_t len = i + 1; for (; j < fVecSize * (len / fVecSize); j += fVecSize) { auto tmp0 = fVec::loadu(curr_g_pad + j); auto tmp1 = vec_curr_g_pad_i - tmp0; auto tmp2 = tmp1.exp_u20(); tmp2.store(curr_decay_mask + i * chunk_size + j); } if (j < len) { auto tmp0 = fVec::loadu(curr_g_pad + j, len - j); auto tmp1 = vec_curr_g_pad_i - tmp0; auto tmp2 = tmp1.exp_u20(); tmp2.store(curr_decay_mask + i * chunk_size + j, len - j); } } // attn = k_beta @ key.transpose(-1, -2) // attn: [B, HV, num_chunk, chunk_size, chunk_size] // transpose and pack for key pack_vnni( /* dst */ k_transpose, /* src */ curr_k_pad, /* N */ chunk_size, /* K */ qk_head_size, /* ld_src */ qk_head_size, /* ld_dst */ chunk_size); // k_beta @ key.transpose(-1, -2) at::native::cpublas::brgemm( /* M */ chunk_size, /* N */ chunk_size, /* K */ qk_head_size, /* lda */ qk_head_size, /* ldb */ chunk_size, /* ldc */ chunk_size, /* add_C */ false, /* A */ curr_k_beta, /* B */ k_transpose, /* C */ curr_attn); // attn = attn * decay_mask for (int64_t m = 0; m < chunk_size; m++) { at::vec::map2( [](fVec x, fVec y) { return fVec(0) - x * y; }, curr_attn + m * chunk_size, curr_attn + m * chunk_size, curr_decay_mask + m * chunk_size, chunk_size); } // chunk decay // attn: [B, HV, num_chunk, chunk_size, chunk_size] // mask = torch.triu(torch.ones(chunk_size, chunk_size, dtype=torch.bool, device=query.device), diagonal=0) // attn = -attn.masked_fill(mask, 0) // attn[..., i, :i] = row + (row.unsqueeze(-1) * sub).sum(-2) [B, HV, num_chunk, i] // attn = attn + torch.eye(chunk_size, dtype=attn.dtype, device=attn.device) // attn = -attn.masked_fill(mask, 0) for (int i = 0; i < chunk_size; i++) { const auto vec_zero = fVec(0); int64_t len = chunk_size - i; int64_t front = len % fVecSize; int64_t j = i; // first masked vec for alignment if (front > 0) { vec_zero.store(curr_attn + i * chunk_size + j, front); j += front; } for (; j < fVecSize * (chunk_size / fVecSize); j += fVecSize) { vec_zero.store(curr_attn + i * chunk_size + j); } } for (int i = 1; i < chunk_size; i++) { // row = attn[..., i, :i] [B, HK, num_chunk, i] int64_t j = 0; int64_t len = i; for (; j < fVecSize * (len / fVecSize); j += fVecSize) { auto tmp0 = fVec::loadu(curr_attn + i * chunk_size + j); tmp0.store(row + j); } if (j < len) { auto tmp0 = fVec::loadu(curr_attn + i * chunk_size + j, len - j); tmp0.store(row + j, len - j); } // (row.unsqueeze(-1) * sub).sum(-2) fill_stub(updated, 0, i); for (int k = 0; k < i; k++) { float row_k = row[k]; auto vec_row_k = fVec(row_k); int64_t j = 0; int64_t len = i; for (; j < fVecSize * (len / fVecSize); j += fVecSize) { auto tmp0 = fVec::loadu(curr_attn + k * chunk_size + j); auto tmp1 = vec_row_k * tmp0; auto tmp2 = fVec::loadu(updated + j); auto tmp3 = tmp1 + tmp2; tmp3.store(updated + j); } if (j < len) { auto tmp0 = fVec::loadu(curr_attn + k * chunk_size + j, len - j); auto tmp1 = vec_row_k * tmp0; auto tmp2 = fVec::loadu(updated + j); auto tmp3 = tmp1 + tmp2; tmp3.store(updated + j, len - j); } } // attn[..., i, :i] = row + sum(...) j = 0; len = i; for (; j < fVecSize * (len / fVecSize); j += fVecSize) { auto tmp0 = fVec::loadu(row + j); auto tmp1 = fVec::loadu(updated + j); auto tmp2 = tmp0 + tmp1; tmp2.store(curr_attn + i * chunk_size + j); } if (j < len) { auto tmp0 = fVec::loadu(row + j, len - j); auto tmp1 = fVec::loadu(updated + j, len - j); auto tmp2 = tmp0 + tmp1; tmp2.store(curr_attn + i * chunk_size + j, len - j); } } for (int i = 0; i < chunk_size; i++) { curr_attn[i * chunk_size + i] += 1.0f; at::vec::map( [](fVec x) { return x; }, curr_attn_reduced + i * chunk_size, curr_attn + i * chunk_size, chunk_size); } // v_beta_attn = attn @ v_beta // k_cumdecay = attn @ (k_beta * g.exp().unsqueeze(-1)) // v_beta_attn: [B, HV, num_chunk, chunk_size, EV] // k_beta_g = k_beta * g: [B, HV, num_chunk, chunk_size, EK] // k_cumdecay: [B, HV, num_chunk, chunk_size, EK] // pack for value pack_vnni2( /* dst */ v_pack, /* src */ curr_v_beta, /* N */ chunk_size, /* K */ v_head_size, /* ld_src */ v_head_size, /* ld_dst */ v_head_size); // value = attn @ v_beta at::native::cpublas::brgemm( /* M */ chunk_size, /* N */ v_head_size, /* K */ chunk_size, /* lda */ chunk_size, /* ldb */ v_head_size, /* ldc */ v_head_size, /* add_C */ false, /* A */ curr_attn_reduced, /* B */ v_pack, /* C */ curr_value); // k_beta_g = k_beta * g.exp().unsqueeze(-1) for (int64_t j = 0; j < chunk_size; j++) { int64_t i = 0; float g_exp = std::exp(curr_g_pad[j]); scalar_t g_exp_reduced = static_cast(g_exp); auto vec_g_exp_reduced = bVec(g_exp_reduced); for (; i < VecSize * (qk_head_size / VecSize); i += VecSize) { auto tmp0 = bVec::loadu(curr_k_beta + j * qk_head_size + i); auto tmp1 = tmp0 * vec_g_exp_reduced; tmp1.store(k_beta_g + j * qk_head_size + i); } } // pack for k_beta_g pack_vnni2( /* dst */ k_beta_g_pack, /* src */ k_beta_g, /* N */ chunk_size, /* K */ qk_head_size, /* ld_src */ qk_head_size, /* ld_dst */ qk_head_size); // k_cumdecay = attn @ k_beta_g at::native::cpublas::brgemm( /* M */ chunk_size, /* N */ qk_head_size, /* K */ chunk_size, /* lda */ chunk_size, /* ldb */ qk_head_size, /* ldc */ qk_head_size, /* add_C */ false, /* A */ curr_attn_reduced, /* B */ k_beta_g_pack, /* C */ k_cumdecay); for (int i = 0; i < chunk_size; i++) { at::vec::map( [](fVec x) { return x; }, curr_k_cumdecay_reduced + i * qk_head_size, k_cumdecay + i * qk_head_size, qk_head_size); } // Move to the next query data_index_step(h, v_num_head, c, global_num_chunk); } }); // for each chunk at::parallel_for(0, batch_size * v_num_head, 1, [&](int64_t begin, int64_t end) { int64_t b = 0, h = 0; data_index_init(begin, b, batch_size, h, v_num_head); int ompIdx = at::get_thread_num(); int64_t offset = /* k_transpose */ qk_head_size * chunk_size + /* v_pack */ chunk_size * v_head_size + /* k_beta_g */ chunk_size * qk_head_size + /* k_beta_g_pack */ chunk_size * qk_head_size + /* attn */ chunk_size * chunk_size * 2 + /* attn_reduced */ chunk_size * chunk_size + /* k_cumdecay */ chunk_size * qk_head_size * 2 + /* row */ chunk_size * 2 + /* updated */ chunk_size * 2; scalar_t* thread_buff_ptr = thread_buff + ompIdx * buff_size_16bit_per_thread; THREAD_BUFFER_ALLOC( curr_last_recurrent_state_reduced, thread_buff_ptr, offset, scalar_t, qk_head_size * v_head_size); THREAD_BUFFER_ALLOC( curr_last_recurrent_state_pack_reduced, thread_buff_ptr, offset, scalar_t, qk_head_size * v_head_size); THREAD_BUFFER_ALLOC(k_transpose_i, thread_buff_ptr, offset, scalar_t, qk_head_size * chunk_size); THREAD_BUFFER_ALLOC(attn_i, thread_buff_ptr, offset, float, chunk_size* chunk_size * 2); THREAD_BUFFER_ALLOC(attn_i_reduced, thread_buff_ptr, offset, scalar_t, chunk_size * chunk_size); THREAD_BUFFER_ALLOC(v_prime, thread_buff_ptr, offset, float, chunk_size* v_head_size * 2); THREAD_BUFFER_ALLOC(v_prime_reduced, thread_buff_ptr, offset, scalar_t, chunk_size * v_head_size); THREAD_BUFFER_ALLOC(v_prime_pack_reduced, thread_buff_ptr, offset, scalar_t, chunk_size * v_head_size); THREAD_BUFFER_ALLOC(qg, thread_buff_ptr, offset, scalar_t, chunk_size * qk_head_size); THREAD_BUFFER_ALLOC(attn_inter, thread_buff_ptr, offset, float, chunk_size* v_head_size * 2); THREAD_BUFFER_ALLOC(kg, thread_buff_ptr, offset, scalar_t, chunk_size * qk_head_size); THREAD_BUFFER_ALLOC(kg_transpose, thread_buff_ptr, offset, scalar_t, qk_head_size * chunk_size); THREAD_BUFFER_ALLOC(kgv, thread_buff_ptr, offset, float, qk_head_size* v_head_size * 2); for ([[maybe_unused]] auto z : c10::irange(begin, end)) { int64_t start_q = cu_seqlens_ptr[b]; int64_t seq_len = cu_seqlens_ptr[b + 1] - start_q; int64_t num_chunk = chunk_offsets_ptr[b + 1] - chunk_offsets_ptr[b]; int64_t chunk_offset = chunk_offsets_ptr[b]; int64_t len_offset = chunk_offset * chunk_size; int64_t h_qk = h / head_group; auto out_ptr = out + start_q * oStrideT; auto curr_q = q_pad + len_offset * qk_head_size + h_qk * global_total_seq_length * qk_head_size; // [num_chunk, chunk_size, EK] auto curr_k = k_pad + len_offset * qk_head_size + h_qk * global_total_seq_length * qk_head_size; // [num_chunk, chunk_size, EK] auto curr_v = v_beta_attn + h * global_total_seq_length * v_head_size; // [num_chunk, chunk_size, EV] auto curr_decay_mask = decay_mask + h * global_total_seq_length * chunk_size; // [num_chunk, chunk_size, chunk_size] auto curr_k_cumdecay_reduced = k_cumdecay_reduced + h * global_total_seq_length * qk_head_size; // [num_chunk, chunk_size, EK] auto curr_last_recurrent_state = final_state_data + b * final_state_StrideN + h * final_state_StrideH; // [EK, EV] auto curr_g_pad = g_pad + len_offset + h * global_total_seq_length; // [num_chunk, chunk_size] auto curr_core_attn_out = core_attn_out + len_offset * v_head_size + h * global_total_seq_length * v_head_size; // [num_chunk, chunk_size, EV] for (int64_t c = 0; c < num_chunk; c++) { for (int i = 0; i < qk_head_size; i++) { at::vec::map( [](fVec x) { return x; }, curr_last_recurrent_state_reduced + i * v_head_size, curr_last_recurrent_state + i * v_head_size, v_head_size); } auto q_i = curr_q + c * chunk_size * qk_head_size; // [chunk_size, EK] auto k_i = curr_k + c * chunk_size * qk_head_size; // [chunk_size, EK] auto v_i = curr_v + (chunk_offset + c) * chunk_size * v_head_size; // [chunk_size, EV] auto decay_mask_i = curr_decay_mask + (chunk_offset + c) * chunk_size * chunk_size; // [chunk_size, chunk_size] auto k_cumdecay_i_reduced = curr_k_cumdecay_reduced + (chunk_offset + c) * chunk_size * qk_head_size; // [chunk_size, EK] auto g_pad_i = curr_g_pad + c * chunk_size; // [chunk_size] auto core_attn_out_i = curr_core_attn_out + c * chunk_size * v_head_size; // [chunk_size, EV] // attn_i = (q_i @ k_i.transpose(-1, -2) * decay_mask[:, :, i]).masked_fill_(mask, 0) // k_transpose_i = k_i.transpose(-1, -2) pack_vnni( /* dst */ k_transpose_i, /* src */ k_i, /* N */ chunk_size, /* K */ qk_head_size, /* ld_src */ qk_head_size, /* ld_dst */ chunk_size); // attn_i = q_i @ k_transpose_i at::native::cpublas::brgemm( /* M */ chunk_size, /* N */ chunk_size, /* K */ qk_head_size, /* lda */ qk_head_size, /* ldb */ chunk_size, /* ldc */ chunk_size, /* add_C */ false, /* A */ q_i, /* B */ k_transpose_i, /* C */ attn_i); // attn_i = attn_i * decay_mask_i for (int64_t m = 0; m < chunk_size; m++) { auto attn_i_m = attn_i + m * chunk_size; auto attn_i_reduced_m = attn_i_reduced + m * chunk_size; auto decay_mask_i_m = decay_mask_i + m * chunk_size; int64_t n = 0; for (; n < fVecSize * (chunk_size / fVecSize); n += fVecSize) { auto tmp0 = fVec::loadu(attn_i_m + n); auto tmp1 = fVec::loadu(decay_mask_i_m + n); auto tmp2 = tmp0 * tmp1; auto tmp3 = at::vec::convert(tmp2); tmp3.store(attn_i_reduced_m + n, fVecSize); } if (n < chunk_size) { auto tmp0 = fVec::loadu(attn_i_m + n, chunk_size - n); auto tmp1 = fVec::loadu(decay_mask_i_m + n, chunk_size - n); auto tmp2 = tmp0 * tmp1; auto tmp3 = at::vec::convert(tmp2); tmp3.store(attn_i_reduced_m + n, chunk_size - n); } } // mask = torch.triu(torch.ones(chunk_size, chunk_size, dtype=torch.bool, device=query.device), diagonal=1) // attn_i = attn_i.masked_fill_(mask, 0) for (int i = 0; i < chunk_size - 1; i++) { const auto vec_zero = bVec(0); int64_t len = chunk_size - i - 1; int64_t front = len % VecSize; int64_t j = i + 1; // first masked vec for alignment if (front > 0) { vec_zero.store(attn_i_reduced + i * chunk_size + j, front); j += front; } for (; j < VecSize * (chunk_size / VecSize); j += VecSize) { vec_zero.store(attn_i_reduced + i * chunk_size + j); } } // pack for curr_last_recurrent_state pack_vnni2( /* dst */ curr_last_recurrent_state_pack_reduced, /* src */ curr_last_recurrent_state_reduced, /* N */ qk_head_size, /* K */ v_head_size, /* ld_src */ v_head_size, /* ld_dst */ v_head_size); // v_prime = k_cumdecay_i @ curr_last_recurrent_state: [chunk_size, EV] // k_cumdecay_i: [chunk_size, EK] // curr_last_recurrent_state: [EK, EV] at::native::cpublas::brgemm( /* M */ chunk_size, /* N */ v_head_size, /* K */ qk_head_size, /* lda */ qk_head_size, /* ldb */ v_head_size, /* ldc */ v_head_size, /* add_C */ false, /* A */ k_cumdecay_i_reduced, /* B */ curr_last_recurrent_state_pack_reduced, /* C */ v_prime); // v_new = v_prime = v_i - v_prime // v_i: [chunk_size, EV] for (int64_t m = 0; m < chunk_size; m++) { int64_t i = 0; for (; i < fVecSize * (v_head_size / fVecSize); i += fVecSize) { auto tmp0 = fVec::loadu(v_i + m * v_head_size + i); auto tmp1 = fVec::loadu(v_prime + m * v_head_size + i); auto tmp2 = tmp0 - tmp1; auto tmp3 = at::vec::convert(tmp2); tmp3.store(v_prime_reduced + m * v_head_size + i, fVecSize); } } // attn_inter = (q_i * g[:, :, i, :, None].exp()) @ last_recurrent_state // qg = q_i * g[:, :, i, :, None].exp(): [chunk_size, EK] // q_i: [chunk_size, EK] // g[:, :, i, :, None]: [chunk_size, 1] for (int64_t m = 0; m < chunk_size; m++) { auto g_pad_i_m = g_pad_i + m; auto g_exp = std::exp(*g_pad_i_m); int64_t i = 0; scalar_t g_exp_reduced = static_cast(g_exp); auto vec_g_exp_reduced = bVec(g_exp_reduced); for (; i < VecSize * (qk_head_size / VecSize); i += VecSize) { auto tmp0 = bVec::loadu(q_i + m * qk_head_size + i); auto tmp2 = tmp0 * vec_g_exp_reduced; tmp2.store(qg + m * qk_head_size + i); } } // attn_inter = qg @ curr_last_recurrent_state: [chunk_size, EV] // curr_last_recurrent_state: [EK, EV] at::native::cpublas::brgemm( /* M */ chunk_size, /* N */ v_head_size, /* K */ qk_head_size, /* lda */ qk_head_size, /* ldb */ v_head_size, /* ldc */ v_head_size, /* add_C */ false, /* A */ qg, /* B */ curr_last_recurrent_state_pack_reduced, /* C */ attn_inter); // core_attn_out[:, :, i] = attn_inter + attn_i @ v_new // pack for v_prime pack_vnni2( /* dst */ v_prime_pack_reduced, /* src */ v_prime_reduced, /* N */ chunk_size, /* K */ v_head_size, /* ld_src */ v_head_size, /* ld_dst */ v_head_size); // attn_inter = attn_inter + attn_i @ v_new: [chunk_size, EV] // attn_i: [chunk_size, chunk_size] // v_new: [chunk_size, EV] at::native::cpublas::brgemm( /* M */ chunk_size, /* N */ v_head_size, /* K */ chunk_size, /* lda */ chunk_size, /* ldb */ v_head_size, /* ldc */ v_head_size, /* add_C */ true, /* A */ attn_i_reduced, /* B */ v_prime_pack_reduced, /* C */ attn_inter); // core_attn_out[:, :, i] = attn_inter for (int64_t m = 0; m < chunk_size; m++) { at::vec::map( [](fVec x) { return x; }, core_attn_out_i + m * v_head_size, attn_inter + m * v_head_size, v_head_size); } // last_recurrent_state = ( // last_recurrent_state * g[:, :, i, -1, None, None].exp() // + (k_i * (g[:, :, i, -1, None] - g[:, :, i]).exp()[..., None]).transpose(-1, -2) @ v_new // ) // 1) last_recurrent_state * g[:, :, i, -1, None, None].exp() // curr_last_recurrent_state: [EK, EV] // g[:, :, i, -1, None, None]: [1, 1] // last_recurrent_state * g[:, :, i, -1, None, None].exp(): [EK, EV] auto g_pad_i_last = g_pad_i + chunk_size - 1; auto g_exp_last = std::exp(g_pad_i_last[0]); for (int64_t m = 0; m < qk_head_size; m++) { int64_t i = 0; auto vec_g_exp_last = fVec(g_exp_last); for (; i < fVecSize * (v_head_size / fVecSize); i += fVecSize) { auto tmp0 = bVec::loadu(curr_last_recurrent_state_reduced + m * v_head_size + i); auto tmp1 = at::vec::convert(tmp0); auto tmp2 = tmp1 * vec_g_exp_last; tmp2.store(curr_last_recurrent_state + m * v_head_size + i); } if (i < v_head_size) { auto tmp0 = bVec::loadu(curr_last_recurrent_state_reduced + m * v_head_size + i, v_head_size - i); auto tmp1 = at::vec::convert(tmp0); auto tmp2 = tmp1 * vec_g_exp_last; tmp2.store(curr_last_recurrent_state + m * v_head_size + i, v_head_size - i); } } // 2) (k_i * (g[:, :, i, -1, None] - g[:, :, i]).exp()[..., None]).transpose(-1, -2) @ v_new // k_i: [chunk_size, EK] // g[:, :, i, -1, None]: [1] // g[:, :, i]: [chunk_size] // (g[:, :, i, -1, None] - g[:, :, i]).exp()[..., None]: [chunk_size, 1] // kg = k_i * (g[:, :, i, -1, None] - g[:, :, i]).exp()[..., None]: [chunk_size, EK] // (k_i * (g[:, :, i, -1, None] - g[:, :, i]).exp()[..., None]).transpose(-1, -2): [EK, chunk_size] // v_new: [chunk_size, EV] // (k_i * (g[:, :, i, -1, None] - g[:, :, i]).exp()[..., None]).transpose(-1, -2) @ v_new: [EK, EV] // kg = k_i * (g[:, :, i, -1, None] - g[:, :, i]).exp()[..., None] for (int64_t m = 0; m < chunk_size; m++) { auto g_exp = std::exp((g_pad_i_last[0] - g_pad_i[m])); int64_t i = 0; scalar_t g_exp_reduced = static_cast(g_exp); auto vec_g_exp_reduced = bVec(g_exp_reduced); for (; i < VecSize * (qk_head_size / VecSize); i += VecSize) { auto tmp0 = bVec::loadu(k_i + m * qk_head_size + i); auto tmp2 = tmp0 * vec_g_exp_reduced; tmp2.store(kg + m * qk_head_size + i); } } // kg.transpose(-1, -2): [EK, chunk_size] at::native::utils::transpose( /* M */ chunk_size, /* N */ qk_head_size, /* src */ kg, /* ld_src */ qk_head_size, /* dst */ kg_transpose, /* ld_dst */ chunk_size); // kgv = kg.transpose(-1, -2) @ v_new // v_new: [chunk_size, EV] at::native::cpublas::brgemm( /* M */ qk_head_size, /* N */ v_head_size, /* K */ chunk_size, /* lda */ chunk_size, /* ldb */ v_head_size, /* ldc */ v_head_size, /* add_C */ false, /* A */ kg_transpose, /* B */ v_prime_pack_reduced, /* C */ kgv); // last_recurrent_state = 1) + 2) for (int64_t m = 0; m < qk_head_size; m++) { at::vec::map2( [](fVec x, fVec y) { return x + y; }, curr_last_recurrent_state + m * v_head_size, curr_last_recurrent_state + m * v_head_size, kgv + m * v_head_size, v_head_size); } } // core_attn_out -> output // output: [B, T, HV, EV] // core_attn_out: [B, HV, padded_T, EV] auto curr_out = out_ptr + h * oStrideH; for (int64_t m = 0; m < seq_len; m++) { at::vec::map( [](fVec x) { return x; }, curr_out + m * oStrideT, curr_core_attn_out + m * v_head_size, v_head_size); } // Move to the next query data_index_step(b, batch_size, h, v_num_head); } }); } inline float softplus(float x, double threshold = 20.0) { if (x > threshold) return x; else if (x < -threshold) return std::exp(x); else return std::log1p(std::exp(x)); } inline at::vec::Vectorized softplus(const at::vec::Vectorized& x, double threshold = 20.0) { using Vec = at::vec::Vectorized; Vec mask_hi = x > Vec(threshold); Vec mask_lo = x < Vec(-threshold); Vec expx = x.exp_u20(); Vec log1pex = (expx + Vec(1.0f)).log(); return Vec::blendv(Vec::blendv(log1pex, expx, mask_lo), x, mask_hi); } template void fused_sigmoid_gating_delta_rule_update_kernel_impl( const scalar_t* __restrict__ q_ptr, const scalar_t* __restrict__ k_ptr, const scalar_t* __restrict__ v_ptr, const float* __restrict__ A_log_ptr, const scalar_t* __restrict__ a_ptr, const scalar_t* __restrict__ dt_bias_ptr, const scalar_t* __restrict__ b_ptr, const int32_t* __restrict__ indices_ptr, float* __restrict__ state_ptr, scalar_t* __restrict__ o_ptr, float* __restrict__ qk_scale_buf, int64_t seq_len, int64_t batch_size, int64_t num_heads, int64_t head_dim, int64_t v_num_heads, int64_t v_head_dim, int64_t q_strideB, int64_t q_strideS, int64_t q_strideH, int64_t k_strideB, int64_t k_strideS, int64_t k_strideH, int64_t v_strideB, int64_t v_strideS, int64_t v_strideH, bool use_qk_l2norm_in_kernel, double softplus_threshold) { using bVec = at::vec::Vectorized; using fVec = at::vec::Vectorized; constexpr int64_t VecSize = bVec::size(); constexpr int64_t fVecSize = fVec::size(); int64_t group_size = v_num_heads / num_heads; double scale = 1 / std::sqrt(head_dim); fVec scale_vec = fVec(scale); if (use_qk_l2norm_in_kernel) { float eps = 1e-5; at::parallel_for(0, batch_size * seq_len * num_heads, 0, [&](int64_t begin, int64_t end) { int64_t bi{0}, si{0}, ni{0}; data_index_init(begin, bi, batch_size, si, seq_len, ni, num_heads); for (int64_t i = begin; i < end; ++i) { float sum_q = float(0); float sum_k = float(0); fVec sum_q_fvec = fVec(float(0)); fVec sum_k_fvec = fVec(float(0)); int64_t q_offset = bi * q_strideB + si * q_strideS + ni * q_strideH; int64_t k_offset = bi * k_strideB + si * k_strideS + ni * k_strideH; int64_t q_scale_offset = bi * seq_len * num_heads + si * num_heads + ni; int64_t k_scale_offset = q_scale_offset + batch_size * seq_len * num_heads; int64_t d; #pragma GCC unroll 4 for (d = 0; d <= head_dim - VecSize; d += VecSize) { bVec q_bvec = bVec::loadu(q_ptr + q_offset + d); fVec q_fvec0, q_fvec1; std::tie(q_fvec0, q_fvec1) = at::vec::convert_to_float(q_bvec); sum_q_fvec += q_fvec0 * q_fvec0; sum_q_fvec += q_fvec1 * q_fvec1; bVec k_bvec = bVec::loadu(k_ptr + k_offset + d); fVec k_fvec0, k_fvec1; std::tie(k_fvec0, k_fvec1) = at::vec::convert_to_float(k_bvec); sum_k_fvec += k_fvec0 * k_fvec0; sum_k_fvec += k_fvec1 * k_fvec1; } #pragma GCC unroll 4 for (; d < head_dim; ++d) { float q_val = static_cast(q_ptr[q_offset + d]); sum_q += q_val * q_val; float k_val = static_cast(k_ptr[k_offset + d]); sum_k += k_val * k_val; } sum_q += vec_reduce_sum(sum_q_fvec); sum_k += vec_reduce_sum(sum_k_fvec); qk_scale_buf[q_scale_offset] = float(1) / std::sqrt(sum_q + eps); qk_scale_buf[k_scale_offset] = float(1) / std::sqrt(sum_k + eps); data_index_step(bi, batch_size, si, seq_len, ni, num_heads); } }); } at::parallel_for(0, batch_size * seq_len * v_num_heads, 0, [&](int64_t begin, int64_t end) { int64_t bi{0}, si{0}, ni{0}; data_index_init(begin, bi, batch_size, si, seq_len, ni, v_num_heads); for (int64_t i = begin; i < end; ++i) { int64_t cache_index = indices_ptr[bi]; int64_t state_offset = (cache_index * v_num_heads + ni) * head_dim * v_head_dim; float g_val = -std::exp(A_log_ptr[ni]) * softplus(float(a_ptr[bi * v_num_heads + ni]) + float(dt_bias_ptr[ni]), softplus_threshold); float g_val_exp = std::exp(g_val); fVec g_val_exp_vec = fVec(g_val_exp); int64_t q_offset = si * q_strideS + bi * q_strideB + (ni / group_size) * q_strideH; int64_t k_offset = si * k_strideS + bi * k_strideB + (ni / group_size) * k_strideH; int64_t q_scale_offset = bi * seq_len * num_heads + si * num_heads + (ni / group_size); int64_t k_scale_offset = q_scale_offset + batch_size * seq_len * num_heads; float q_scale = use_qk_l2norm_in_kernel ? qk_scale_buf[q_scale_offset] : 1.0f; float k_scale = use_qk_l2norm_in_kernel ? qk_scale_buf[k_scale_offset] : 1.0f; int64_t v_offset = si * v_strideS + bi * v_strideB + ni * v_strideH; int64_t o_offset = ((bi * seq_len + si) * v_num_heads + ni) * v_head_dim; float beta_val = 1 / (1 + std::exp(-b_ptr[ni])); fVec beta_vec = fVec(beta_val); int64_t dvi = 0; for (; dvi <= v_head_dim - VecSize; dvi += VecSize) { fVec kv_mem_vec0 = fVec(float(0)); fVec kv_mem_vec1 = fVec(float(0)); for (int di = 0; di < head_dim; ++di) { fVec k_val_vec = fVec(k_ptr[k_offset + di] * k_scale); fVec state_vec0 = fVec::loadu(state_ptr + state_offset + di * v_head_dim + dvi); fVec state_vec1 = fVec::loadu(state_ptr + state_offset + di * v_head_dim + dvi + fVecSize); kv_mem_vec0 = kv_mem_vec0 + state_vec0 * g_val_exp_vec * k_val_vec; kv_mem_vec1 = kv_mem_vec1 + state_vec1 * g_val_exp_vec * k_val_vec; } bVec v_bvec = bVec::loadu(v_ptr + v_offset + dvi); fVec v_vec0, v_vec1; std::tie(v_vec0, v_vec1) = at::vec::convert_to_float(v_bvec); fVec dt_vec0 = (v_vec0 - kv_mem_vec0) * beta_vec; fVec dt_vec1 = (v_vec1 - kv_mem_vec1) * beta_vec; fVec o_vec0 = fVec(float(0)); fVec o_vec1 = fVec(float(0)); for (int di = 0; di < head_dim; ++di) { fVec q_vec = fVec(q_ptr[q_offset + di] * q_scale); fVec k_vec = fVec(k_ptr[k_offset + di] * k_scale); fVec state_vec0 = fVec::loadu(state_ptr + state_offset + di * v_head_dim + dvi); fVec state_vec1 = fVec::loadu(state_ptr + state_offset + di * v_head_dim + dvi + fVecSize); state_vec0 = state_vec0 * g_val_exp_vec + k_vec * dt_vec0; state_vec1 = state_vec1 * g_val_exp_vec + k_vec * dt_vec1; o_vec0 = o_vec0 + state_vec0 * q_vec * scale_vec; o_vec1 = o_vec1 + state_vec1 * q_vec * scale_vec; state_vec0.store(state_ptr + state_offset + di * v_head_dim + dvi); state_vec1.store(state_ptr + state_offset + di * v_head_dim + dvi + fVecSize); } bVec o_vec = at::vec::convert_from_float(o_vec0, o_vec1); o_vec.store(o_ptr + o_offset + dvi); } for (; dvi < v_head_dim; ++dvi) { float kv_mem_val = 0; for (int di = 0; di < head_dim; ++di) { float k_val = k_ptr[k_offset + di] * k_scale; state_ptr[state_offset + di * v_head_dim + dvi] *= g_val_exp; kv_mem_val += state_ptr[state_offset + di * v_head_dim + dvi] * k_val; } float v_val = v_ptr[v_offset + dvi]; float dt_val = (v_val - kv_mem_val) * beta_val; float o_val = 0; for (int di = 0; di < head_dim; ++di) { float q_val = q_ptr[q_offset + di] * q_scale; float k_val = k_ptr[k_offset + di] * k_scale; state_ptr[state_offset + di * v_head_dim + dvi] += k_val * dt_val; o_val += state_ptr[state_offset + di * v_head_dim + dvi] * q_val * scale; } o_ptr[o_offset + dvi] = o_val; } data_index_step(bi, batch_size, si, seq_len, ni, v_num_heads); } }); } template void fused_gdn_gating_kernel_impl( float* __restrict__ A_log, const scalar_t* __restrict__ a, const scalar_t* __restrict__ b, const scalar_t* __restrict__ dt_bias, float* __restrict__ out, scalar_t* __restrict__ beta, int64_t batch, int64_t num_heads) { using bVec = at::vec::Vectorized; using fVec = at::vec::Vectorized; constexpr int vec_size = bVec::size(); constexpr int fvec_size = fVec::size(); const fVec neg_one(-1.0f); const fVec one(1.0f); at::parallel_for(0, batch, 0, [&](int64_t begin, int64_t end) { for (int64_t i = begin; i < end; ++i) { int64_t j = 0; for (; j < num_heads - (num_heads % vec_size); j += vec_size) { fVec A_log_vec0 = fVec::loadu(A_log + j); fVec A_log_vec1 = fVec::loadu(A_log + j + fvec_size); bVec dt_bias_vec = bVec::loadu(dt_bias + j); bVec a_bvec = bVec::loadu(a + i * num_heads + j); bVec b_bvec = bVec::loadu(b + i * num_heads + j); fVec a0, a1, dt_bias_vec0, dt_bias_vec1, b0, b1; std::tie(a0, a1) = at::vec::convert_to_float(a_bvec); std::tie(b0, b1) = at::vec::convert_to_float(b_bvec); std::tie(dt_bias_vec0, dt_bias_vec1) = at::vec::convert_to_float(dt_bias_vec); fVec g0 = neg_one * A_log_vec0.exp_u20() * softplus(a0 + dt_bias_vec0); fVec g1 = neg_one * A_log_vec1.exp_u20() * softplus(a1 + dt_bias_vec1); fVec beta0 = one / (one + (neg_one * b0).exp_u20()); fVec beta1 = one / (one + (neg_one * b1).exp_u20()); g0.store(out + i * num_heads + j); g1.store(out + i * num_heads + j + fvec_size); bVec beta_vec = at::vec::convert_from_float(beta0, beta1); beta_vec.store(beta + i * num_heads + j); } for (; j < num_heads; ++j) { out[i * num_heads + j] = -std::exp(A_log[j]) * softplus(float(a[i * num_heads + j]) + float(dt_bias[j])); beta[i * num_heads + j] = 1 / (1 + std::exp(-b[i * num_heads + j])); } } }); } } // anonymous namespace template inline void CHECK_INPUT_SHAPE_DTYPE(const at::Tensor& tensor, const int64_t& dim, const at::IntArrayRef& sizes, at::ScalarType st) { TORCH_CHECK(tensor.sizes() == sizes, "Input tensor shape mismatch: expected ", sizes, ", got ", tensor.sizes()); TORCH_CHECK(tensor.dtype() == st, "Input tensor dtype mismatch"); CHECK_DIM(dim, tensor); if (is_last_dim_contiguous) { CHECK_LAST_DIM_CONTIGUOUS_INPUT(tensor); } else { CHECK_CONTIGUOUS(tensor); } } // query: [B, T, HK, EK] // key: [B, T, HK, EK] // value: [B, T, HV, EV] // g: [B, T, HV] FP32 // beta: [B, T, HV] // initial_state: [N, HV, EK, EV] FP32 // output_final_state: bool // cu_seqlens: [N + 1] INT32 // head_first: bool // use_qk_l2norm_in_kernel: bool std::tuple chunk_gated_delta_rule_cpu( const at::Tensor& query, const at::Tensor& key, const at::Tensor& value, const at::Tensor& g, const at::Tensor& beta, const at::Tensor& initial_state, bool output_final_state, const at::Tensor& cu_seqlens, bool head_first, bool use_qk_l2norm_in_kernel, double eps = 1e-5) { RECORD_FUNCTION( "sgl-kernel::chunk_gated_delta_rule_cpu", std::vector({query, key, value, g, beta, initial_state})); TORCH_CHECK(head_first == false, "chunk_gated_delta_rule_cpu does not support head first"); int64_t B = query.size(0); int64_t global_seq_len = query.size(1); int64_t qk_num_head = query.size(2); int64_t qk_head_size = query.size(3); int64_t v_num_head = value.size(2); int64_t v_head_size = value.size(3); int64_t batch_size = initial_state.size(0); CHECK_EQ(B, 1); TORCH_CHECK(v_num_head % qk_num_head == 0, "expect v_num_head multiple of qk_num_head."); TORCH_CHECK(qk_head_size % 32 == 0, "expect qk_head_size to be multiples of 32."); TORCH_CHECK(v_head_size % 32 == 0, "expect v_head_size to be multiples of 32."); CHECK_INPUT_SHAPE_DTYPE(query, 4, {B, global_seq_len, qk_num_head, qk_head_size}, at::kBFloat16); CHECK_INPUT_SHAPE_DTYPE(key, 4, {B, global_seq_len, qk_num_head, qk_head_size}, at::kBFloat16); CHECK_INPUT_SHAPE_DTYPE(value, 4, {B, global_seq_len, v_num_head, v_head_size}, at::kBFloat16); CHECK_INPUT_SHAPE_DTYPE(g, 3, {B, global_seq_len, v_num_head}, at::kFloat); CHECK_INPUT_SHAPE_DTYPE(beta, 3, {B, global_seq_len, v_num_head}, at::kBFloat16); CHECK_INPUT_SHAPE_DTYPE(cu_seqlens, 1, {batch_size + 1}, at::kInt); CHECK_INPUT_SHAPE_DTYPE(initial_state, 4, {batch_size, v_num_head, qk_head_size, v_head_size}, at::kFloat); at::Tensor output = at::empty_like(value, value.options()); // [B, T, HV, EV] at::Tensor final_state = initial_state.to(at::kFloat); // [N, HV, EK, EV] // Strides int64_t qStrideH = query.stride(2); int64_t qStrideT = query.stride(1); int64_t kStrideH = key.stride(2); int64_t kStrideT = key.stride(1); int64_t vStrideH = value.stride(2); int64_t vStrideT = value.stride(1); int64_t oStrideH = output.stride(2); int64_t oStrideT = output.stride(1); constexpr int64_t chunk_size = 64; // Deduce the global chunks // e.g. cu_seqlens: [0, 5, 13, 16], chunk_size = 4 // chunk_offsets: [0, 2, 4, 5] // chunk_indices (batch_id, local_chunk_id): [[0, 0], [0, 1], [1, 0], [1, 1], [2, 0]] at::Tensor chunk_offsets = at::empty(batch_size + 1, cu_seqlens.options()); auto chunk_offsets_ptr = chunk_offsets.data_ptr(); chunk_offsets_ptr[0] = 0; int32_t* cu_seqlens_ptr = cu_seqlens.data_ptr(); int64_t s = 0; int64_t e = 0; int64_t s_pad = 0; int64_t e_pad = 0; for (int64_t b = 0; b < batch_size; b++) { e = cu_seqlens_ptr[b + 1]; int64_t seq_len = e - s; int64_t pad_size = (chunk_size - seq_len % chunk_size) % chunk_size; int64_t total_seq_length = seq_len + pad_size; e_pad = s_pad + total_seq_length; chunk_offsets[b + 1] = e_pad / chunk_size; s = e; s_pad = e_pad; } int64_t global_total_seq_length = e_pad; int64_t global_num_chunk = chunk_offsets_ptr[batch_size]; at::Tensor chunk_indices = at::empty(global_num_chunk * 2, cu_seqlens.options()); auto chunk_indices_ptr = chunk_indices.data_ptr(); int64_t curr_c = 0; for (int64_t b = 0; b < batch_size; b++) { int64_t batch_chunk_num = chunk_offsets_ptr[b + 1] - chunk_offsets_ptr[b]; for (int64_t c = 0; c < batch_chunk_num; c++) { chunk_indices_ptr[curr_c * 2] = b; chunk_indices_ptr[curr_c * 2 + 1] = c; curr_c += 1; } } // Allocate buffer int64_t buff_size = v_num_head * global_total_seq_length // g_pad_data + batch_size * v_num_head * global_total_seq_length * v_head_size // core_attn + v_num_head * global_total_seq_length * chunk_size // decay_mask + v_num_head * global_total_seq_length * v_head_size; // v_beta_attn at::Tensor buff_data = at::empty({buff_size}, query.options().dtype(at::kFloat)); int64_t reduced_buff_size = qk_num_head * global_total_seq_length * qk_head_size // q_pad_data + qk_num_head * global_total_seq_length * qk_head_size // k_pad_data + v_num_head * global_total_seq_length * v_head_size // v_pad_data + v_num_head * global_total_seq_length * qk_head_size // k_beta_data + v_num_head * global_total_seq_length * v_head_size // v_beta_data + v_num_head * global_total_seq_length * qk_head_size // k_cumdecay_reduced + qk_num_head * global_seq_len // q_norm_sum + qk_num_head * global_seq_len; // k_norm_sum at::Tensor reduced_buff_data = at::empty({reduced_buff_size}, query.options()); int64_t num_thread = at::get_num_threads(); int64_t buff_size_16bit_per_thread = /* k_transpose */ qk_head_size * chunk_size + /* v_pack */ chunk_size * v_head_size + /* k_beta_g */ chunk_size * qk_head_size + /* k_beta_g_pack */ chunk_size * qk_head_size + /* attn */ chunk_size * chunk_size * 2 + /* attn_reduced */ chunk_size * chunk_size + /* k_cumdecay */ chunk_size * qk_head_size * 2 + /* row */ chunk_size * 2 + /* updated */ chunk_size * 2 + /* curr_last_recurrent_state_reduced */ qk_head_size * v_head_size + /* curr_last_recurrent_state_pack_reduced */ qk_head_size * v_head_size + /* k_transpose_i */ qk_head_size * chunk_size + /* attn_i */ chunk_size * chunk_size * 2 + /* attn_i_reduced */ chunk_size * chunk_size + /* v_prime */ chunk_size * v_head_size * 2 + /* v_prime_reduced */ chunk_size * v_head_size + /* v_prime_pack_reduced */ chunk_size * v_head_size + /* qg */ chunk_size * qk_head_size + /* attn_inter */ chunk_size * v_head_size * 2 + /* kg */ chunk_size * qk_head_size + /* kg_transpose */ qk_head_size * chunk_size + /* kgv */ qk_head_size * v_head_size * 2; at::Tensor thread_buff_data = at::empty({num_thread, buff_size_16bit_per_thread}, query.options()); AT_DISPATCH_REDUCED_FLOATING_TYPES(query.scalar_type(), "chunk_gated_delta_rule_kernel", [&] { chunk_gated_delta_rule_kernel_impl( output.data_ptr(), final_state.data_ptr(), query.data_ptr(), key.data_ptr(), value.data_ptr(), g.data_ptr(), beta.data_ptr(), cu_seqlens_ptr, buff_data.data_ptr(), reduced_buff_data.data_ptr(), thread_buff_data.data_ptr(), chunk_offsets_ptr, chunk_indices_ptr, use_qk_l2norm_in_kernel, batch_size, global_seq_len, qk_num_head, v_num_head, qk_head_size, v_head_size, qStrideH, qStrideT, kStrideH, kStrideT, vStrideH, vStrideT, oStrideH, oStrideT, global_total_seq_length, global_num_chunk, buff_size_16bit_per_thread, eps); }); return std::make_tuple(std::move(output), std::move(final_state)); } // A_log: [v_num_heads] // dt_bias: [v_num_heads] // query: [seq_len, batch_size, num_heads, head_dim] // key: [seq_len, batch_size, num_heads, head_dim] // value: [seq_len, batch_size, v_num_heads, v_head_dim] // a: [batch_size, v_num_heads] // b: [batch_size, v_num_heads] // initial_state_source:[num_tokens, v_num_heads, head_dim, v_head_dim] // initial_state_indices: [batch_size] // cu_seqlens: [batch_size + 1] at::Tensor fused_sigmoid_gating_delta_rule_update_cpu( const at::Tensor& A_log, const at::Tensor& dt_bias, const at::Tensor& q, const at::Tensor& k, const at::Tensor& v, const at::Tensor& a, const at::Tensor& b, at::Tensor& initial_state_source, const at::Tensor& initial_state_indices, const at::Tensor& cu_seqlens, bool use_qk_l2norm_in_kernel, double softplus_beta = 1.0, double softplus_threshold = 20.0) { RECORD_FUNCTION( "sgl-kernel::fused_sigmoid_gating_delta_rule_update_cpu", std::vector( {A_log, dt_bias, q, k, v, a, b, initial_state_source, initial_state_indices, cu_seqlens})); CHECK_DIM(4, q); CHECK_DIM(4, v); CHECK_LAST_DIM_CONTIGUOUS_INPUT(q); int64_t seq_len = q.size(0); int64_t batch_size = q.size(1); int64_t num_heads = q.size(2); int64_t head_dim = q.size(3); int64_t v_num_heads = v.size(2); int64_t v_head_dim = v.size(3); CHECK_INPUT_SHAPE_DTYPE(k, {seq_len, batch_size, num_heads, head_dim}, q.scalar_type()); CHECK_INPUT_SHAPE_DTYPE(v, {seq_len, batch_size, v_num_heads, v_head_dim}, q.scalar_type()); CHECK_INPUT_SHAPE_DTYPE(A_log, {v_num_heads}, at::kFloat); CHECK_INPUT_SHAPE_DTYPE(a, {batch_size, v_num_heads}, q.scalar_type()); CHECK_INPUT_SHAPE_DTYPE(dt_bias, {v_num_heads}, q.scalar_type()); CHECK_INPUT_SHAPE_DTYPE(b, {batch_size, v_num_heads}, q.scalar_type()); CHECK_INPUT_SHAPE_DTYPE(initial_state_indices, {batch_size}, at::kInt); CHECK_INPUT_SHAPE_DTYPE(cu_seqlens, {batch_size + 1}, at::kInt); CHECK_INPUT_SHAPE_DTYPE( initial_state_source, {initial_state_source.size(0), v_num_heads, head_dim, v_head_dim}, at::kFloat); CHECK(initial_state_source.size(0) >= batch_size); CHECK_EQ(v_num_heads % num_heads, 0); int64_t q_strideB = q.stride(1); int64_t q_strideS = q.stride(0); int64_t q_strideH = q.stride(2); int64_t k_strideB = k.stride(1); int64_t k_strideS = k.stride(0); int64_t k_strideH = k.stride(2); int64_t v_strideB = v.stride(1); int64_t v_strideS = v.stride(0); int64_t v_strideH = v.stride(2); at::Tensor core_attn_out = at::empty({batch_size, seq_len, v_num_heads, v_head_dim}, q.options()); at::Tensor qk_scale_buf = at::empty({2 * batch_size, seq_len, num_heads}, at::kFloat); AT_DISPATCH_REDUCED_FLOATING_TYPES(q.scalar_type(), "fused_sigmoid_gating_delta_rule_update_kernel_impl", [&] { fused_sigmoid_gating_delta_rule_update_kernel_impl( q.data_ptr(), k.data_ptr(), v.data_ptr(), A_log.data_ptr(), a.data_ptr(), dt_bias.data_ptr(), b.data_ptr(), initial_state_indices.data_ptr(), initial_state_source.data_ptr(), core_attn_out.data_ptr(), qk_scale_buf.data_ptr(), seq_len, batch_size, num_heads, head_dim, v_num_heads, v_head_dim, q_strideB, q_strideS, q_strideH, k_strideB, k_strideS, k_strideH, v_strideB, v_strideS, v_strideH, use_qk_l2norm_in_kernel, softplus_threshold); }); return core_attn_out; } // A_log: [num_v_heads] // a: [batch, num_v_heads] // b: [batch, num_v_heads] // dt_bias: [num_v_heads] // -A_log.float().exp() * F.softplus(a.float() + dt_bias) std::tuple fused_gdn_gating_cpu(const at::Tensor& A_log, const at::Tensor& a, const at::Tensor& b, const at::Tensor& dt_bias) { RECORD_FUNCTION("sgl-kernel::fused_gdn_gating_cpu", std::vector({A_log, a, b, dt_bias})); CHECK_DIM(1, A_log); CHECK_DIM(2, a); CHECK_DIM(2, b); CHECK_DIM(1, dt_bias); CHECK_CONTIGUOUS(a); CHECK_EQ(A_log.size(0), a.size(1)); CHECK_EQ(A_log.size(0), dt_bias.size(0)); int batch = a.size(0); int num_heads = a.size(1); CHECK_EQ(b.size(0), batch); CHECK_EQ(b.size(1), num_heads); at::Tensor out = at::empty({1, batch, num_heads}, a.options().dtype(at::kFloat)); at::Tensor beta = at::empty({1, batch, num_heads}, b.options()); AT_DISPATCH_REDUCED_FLOATING_TYPES(a.scalar_type(), "fused_gdn_gating_kernel", [&] { fused_gdn_gating_kernel_impl( A_log.data_ptr(), a.data_ptr(), b.data_ptr(), dt_bias.data_ptr(), out.data_ptr(), beta.data_ptr(), batch, num_heads); }); return std::make_tuple(out, beta); }