mha_fwd.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. /******************************************************************************
  2. * Copyright (c) 2024, Tri Dao.
  3. ******************************************************************************/
  4. #include "flash_common.hpp"
  5. #include "fmha_fwd.hpp"
  6. #include "mask.hpp"
  7. fmha_fwd_traits get_ck_fmha_fwd_traits(const mask_info &mask,
  8. std::string dtype,
  9. int head_size,
  10. bool has_dropout,
  11. bool has_lse,
  12. bool enable_alibi)
  13. {
  14. return fmha_fwd_traits{head_size,
  15. head_size,
  16. dtype,
  17. false, // is_group_mode
  18. true, // is_v_rowmajor
  19. mask.type,
  20. enable_alibi ? bias_enum::alibi : bias_enum::no_bias,
  21. has_lse,
  22. has_dropout,
  23. false}; // do_fp8_static_quant
  24. }
  25. fmha_fwd_args get_ck_fmha_fwd_args(bool has_lse,
  26. bool has_dropout_randval,
  27. const mask_info &mask,
  28. // sizes
  29. const int b,
  30. const int seqlen_q,
  31. const int seqlen_k,
  32. const int h,
  33. const int h_k,
  34. const int d,
  35. // device pointers
  36. const at::Tensor q,
  37. const at::Tensor k,
  38. const at::Tensor v,
  39. c10::optional<at::Tensor> &alibi_slopes_,
  40. at::Tensor out,
  41. at::Tensor softmax_lse,
  42. at::Tensor dropout_randval,
  43. float softmax_scale,
  44. float p_dropout,
  45. uint64_t drop_seed,
  46. uint64_t drop_offset)
  47. {
  48. // q: (batch_size, seqlen_q, nheads, d)
  49. // k: (batch_size, seqlen_k, nheads_k, d)
  50. // v: (batch_size, seqlen_k, nheads_k, d)
  51. // o: (batch_size, seqlen_q, nheads, d)
  52. // alibi_slopes:(batch_size, nheads) or (nhead)
  53. // lse: (batch_size, nheads, seqlen_q)
  54. // randval: (batch_size, nheads, seqlen_q, seqlen_k)
  55. ck_tile::index_t stride_q = q.stride(1);
  56. ck_tile::index_t stride_k = k.stride(1);
  57. ck_tile::index_t stride_v = v.stride(1);
  58. ck_tile::index_t stride_o = out.stride(1);
  59. ck_tile::index_t stride_randval = has_dropout_randval ? dropout_randval.stride(2) : 0;
  60. ck_tile::index_t nhead_stride_q = q.stride(2);
  61. ck_tile::index_t nhead_stride_k = k.stride(2);
  62. ck_tile::index_t nhead_stride_v = v.stride(2);
  63. ck_tile::index_t nhead_stride_o = out.stride(2);
  64. ck_tile::index_t nhead_stride_lse = has_lse ? softmax_lse.stride(1) : 0;
  65. ck_tile::index_t nhead_stride_randval = has_dropout_randval ? dropout_randval.stride(1) : 0;
  66. ck_tile::index_t batch_stride_q = q.stride(0);
  67. ck_tile::index_t batch_stride_k = k.stride(0);
  68. ck_tile::index_t batch_stride_v = v.stride(0);
  69. ck_tile::index_t batch_stride_o = out.stride(0);
  70. ck_tile::index_t batch_stride_lse = has_lse ? softmax_lse.stride(0) : 0;
  71. ck_tile::index_t batch_stride_randval = has_dropout_randval ? dropout_randval.stride(0) : 0;
  72. void *alibi_slopes_ptr = nullptr;
  73. ck_tile::index_t stride_alibi_slopes = 0;
  74. if (alibi_slopes_.has_value()) {
  75. auto alibi_slopes = alibi_slopes_.value();
  76. CHECK_DEVICE(alibi_slopes);
  77. TORCH_CHECK(alibi_slopes.stride(-1) == 1, "ALiBi slopes tensor must have contiguous last dimension");
  78. TORCH_CHECK(alibi_slopes.sizes() == torch::IntArrayRef({h}) || alibi_slopes.sizes() == torch::IntArrayRef({b, h}));
  79. alibi_slopes_ptr = alibi_slopes.data_ptr();
  80. stride_alibi_slopes = alibi_slopes.dim() == 2 ? alibi_slopes.stride(0) : 0;
  81. }
  82. return fmha_fwd_args{q.data_ptr(),
  83. k.data_ptr(),
  84. v.data_ptr(),
  85. alibi_slopes_ptr, // bias
  86. has_dropout_randval ? dropout_randval.data_ptr() : nullptr,
  87. nullptr, // lse_acc
  88. nullptr, // o_acc
  89. has_lse ? softmax_lse.data_ptr() : nullptr,
  90. out.data_ptr(),
  91. nullptr, // seqstart_q
  92. nullptr, // seqstart_k
  93. nullptr,
  94. seqlen_q,
  95. seqlen_k,
  96. b,
  97. seqlen_q, // max_seqlen_q
  98. d, // hdim_q
  99. d, // hdim_v
  100. h, // nhead
  101. h_k, // nhead_k
  102. 1, // num_splits
  103. softmax_scale, // scale_s
  104. 1, // scale_p
  105. 1, // scale_o
  106. stride_q,
  107. stride_k,
  108. stride_v,
  109. stride_alibi_slopes,
  110. stride_randval,
  111. 0, // stride_o_acc,
  112. stride_o,
  113. nhead_stride_q,
  114. nhead_stride_k,
  115. nhead_stride_v,
  116. 0, // nhead_stride_bias, FA without bias
  117. nhead_stride_randval,
  118. nhead_stride_lse,
  119. 0, // nhead_stride_lse_acc
  120. 0, // nhead_stride_o_acc
  121. nhead_stride_o,
  122. batch_stride_q,
  123. batch_stride_k,
  124. batch_stride_v,
  125. 0, // batch_stride_bias, FA without bias
  126. batch_stride_randval,
  127. batch_stride_lse,
  128. 0, // batch_stride_lse_acc
  129. 0, // batch_stride_o_acc
  130. batch_stride_o,
  131. 0, // split_stride_lse_acc
  132. 0, // split_stride_o_acc
  133. mask.left,
  134. mask.right,
  135. static_cast<ck_tile::index_t>(mask.type),
  136. p_dropout,
  137. has_dropout_randval,
  138. {drop_seed, drop_offset}};
  139. }
  140. std::vector<at::Tensor>
  141. mha_fwd(at::Tensor &q, // batch_size x seqlen_q x num_heads x head_size
  142. const at::Tensor &k, // batch_size x seqlen_k x num_heads_k x head_size
  143. const at::Tensor &v, // batch_size x seqlen_k x num_heads_k x head_size
  144. c10::optional<at::Tensor> &out_, // batch_size x seqlen_q x num_heads x head_size
  145. c10::optional<at::Tensor> &alibi_slopes_, // num_heads or batch_size x num_heads
  146. const float p_dropout,
  147. const float softmax_scale,
  148. bool is_causal,
  149. int window_size_left,
  150. int window_size_right,
  151. const float /*softcap*/,
  152. const bool return_dropout_randval,
  153. c10::optional<at::Generator> gen_)
  154. {
  155. auto q_dtype = q.dtype();
  156. TORCH_CHECK(q_dtype == torch::kFloat16 || q_dtype == torch::kBFloat16,
  157. "FlashAttention only support fp16 and bf16 data type");
  158. TORCH_CHECK(k.dtype() == q_dtype, "query and key must have the same dtype");
  159. TORCH_CHECK(v.dtype() == q_dtype, "query and value must have the same dtype");
  160. std::string q_dtype_str = q_dtype == torch::kFloat16 ? "fp16" : "bf16";
  161. CHECK_DEVICE(q); CHECK_DEVICE(k); CHECK_DEVICE(v);
  162. TORCH_CHECK(q.stride(-1) == 1, "Input tensor must have contiguous last dimension");
  163. TORCH_CHECK(k.stride(-1) == 1, "Input tensor must have contiguous last dimension");
  164. TORCH_CHECK(v.stride(-1) == 1, "Input tensor must have contiguous last dimension");
  165. const auto sizes = q.sizes();
  166. const int batch_size = sizes[0];
  167. int seqlen_q = sizes[1];
  168. int num_heads = sizes[2];
  169. const int head_size_og = sizes[3];
  170. const int seqlen_k = k.size(1);
  171. const int num_heads_k = k.size(2);
  172. TORCH_CHECK(batch_size > 0, "batch size must be positive");
  173. TORCH_CHECK(head_size_og <= 256, "CK only supports head dimension at most 256");
  174. TORCH_CHECK(num_heads % num_heads_k == 0, "Number of heads in key/value must divide number of heads in query");
  175. if (window_size_left >= seqlen_k) { window_size_left = -1; }
  176. if (window_size_right >= seqlen_k) { window_size_right = -1; }
  177. // causal=true is the same as causal=false in this case
  178. if (seqlen_q == 1 && !alibi_slopes_.has_value()) { is_causal = false; }
  179. mask_info mask;
  180. if (is_causal) {
  181. // Causal is the special case where window_size_right == 0 and window_size_left < 0.
  182. window_size_right = 0;
  183. std::string mask_identify = "b:" + std::to_string(window_size_left) + "," + "0";
  184. mask = mask_info::decode(mask_identify, seqlen_q, seqlen_k); // casual
  185. }
  186. else if (window_size_left == -1 && window_size_right == -1) {
  187. mask = mask_info::decode("0", seqlen_q, seqlen_k); // no mask
  188. }
  189. else {
  190. // Local is the more general case where window_size_right >= 0 or window_size_left >= 0.
  191. std::string mask_identify = "b:" + std::to_string(window_size_left) + "," + std::to_string(window_size_right);
  192. mask = mask_info::decode(mask_identify, seqlen_q, seqlen_k); // local
  193. }
  194. // Faster to transpose q from (b, 1, (nheads_kv ngroups), d) to (b, ngroups, nheads_kv, d) in this case
  195. // H/t Daniel Haziza
  196. const int seqlenq_ngroups_swapped = seqlen_q == 1 && num_heads > num_heads_k && window_size_left < 0 && window_size_right < 0 && p_dropout == 0.f && head_size_og % 8 == 0 && !alibi_slopes_.has_value();
  197. const int ngroups = num_heads / num_heads_k;
  198. if (seqlenq_ngroups_swapped) {
  199. q = q.reshape({batch_size, num_heads_k, ngroups, head_size_og}).transpose(1, 2);
  200. seqlen_q = ngroups;
  201. num_heads = num_heads_k;
  202. }
  203. CHECK_SHAPE(q, batch_size, seqlen_q, num_heads, head_size_og);
  204. CHECK_SHAPE(k, batch_size, seqlen_k, num_heads_k, head_size_og);
  205. CHECK_SHAPE(v, batch_size, seqlen_k, num_heads_k, head_size_og);
  206. at::Tensor q_padded, k_padded, v_padded;
  207. if (head_size_og % 8 != 0) {
  208. q_padded = torch::nn::functional::pad(q, torch::nn::functional::PadFuncOptions({0, 8 - head_size_og % 8}));
  209. k_padded = torch::nn::functional::pad(k, torch::nn::functional::PadFuncOptions({0, 8 - head_size_og % 8}));
  210. v_padded = torch::nn::functional::pad(v, torch::nn::functional::PadFuncOptions({0, 8 - head_size_og % 8}));
  211. }
  212. else {
  213. q_padded = q;
  214. k_padded = k;
  215. v_padded = v;
  216. }
  217. at::Tensor out;
  218. if (out_.has_value()) {
  219. out = out_.value();
  220. TORCH_CHECK(out.dtype() == q_dtype, "Output must have the same dtype as inputs");
  221. CHECK_DEVICE(out);
  222. TORCH_CHECK(out.stride(-1) == 1, "Output tensor must have contiguous last dimension");
  223. CHECK_SHAPE(out, batch_size, sizes[1], sizes[2], head_size_og);
  224. if (seqlenq_ngroups_swapped) {
  225. out = out.reshape({batch_size, num_heads_k, ngroups, head_size_og}).transpose(1, 2);
  226. }
  227. if (head_size_og % 8 != 0) { out = torch::empty_like(q_padded); }
  228. }
  229. else {
  230. out = torch::empty_like(q_padded);
  231. }
  232. auto round_multiple = [](int x, int m) { return (x + m - 1) / m * m; };
  233. const int head_size_8x = round_multiple(head_size_og, 8);
  234. // Otherwise the kernel will be launched from cuda:0 device
  235. // Cast to char to avoid compiler warning about narrowing
  236. at::cuda::CUDAGuard device_guard{(char)q.get_device()};
  237. auto opts = q.options();
  238. bool has_lse = true;
  239. bool has_dropout = p_dropout > 0.0f;
  240. at::Tensor softmax_lse;
  241. // TODO - check gradient, only training require lse
  242. softmax_lse = torch::empty({batch_size, num_heads, seqlen_q}, opts.dtype(torch::kFloat32));
  243. at::Tensor p;
  244. if (return_dropout_randval) {
  245. TORCH_CHECK(has_dropout, "return_dropout_randval require p_dropout > 0");
  246. p = torch::empty({batch_size, num_heads, seqlen_q, seqlen_k}, opts.dtype(torch::kUInt8));
  247. }
  248. uint64_t drop_seed = 1, drop_offset = 0;
  249. int64_t counter_offset = batch_size * num_heads * ck_tile::get_warp_size();
  250. auto options = torch::TensorOptions().dtype(torch::kFloat32).device(torch::kCUDA);
  251. auto rng_state = torch::empty({2}, options.dtype(torch::kInt64));
  252. if (p_dropout > 0.0) {
  253. auto gen = at::get_generator_or_default<at::CUDAGeneratorImpl>(
  254. gen_, at::cuda::detail::getDefaultCUDAGenerator());
  255. // See Note [Acquire lock when using random generators]
  256. std::lock_guard<std::mutex> lock(gen->mutex_);
  257. auto philox_args = gen->philox_cuda_state(counter_offset);
  258. std::tie(drop_seed, drop_offset) = flash::unpack(philox_args);
  259. }
  260. rng_state[0] = *(reinterpret_cast<int64_t*>(&drop_seed));
  261. rng_state[1] = *(reinterpret_cast<int64_t*>(&drop_offset));
  262. if (seqlen_k > 0) {
  263. auto stream = at::cuda::getCurrentHIPStream().stream();
  264. ck_tile::stream_config stream_config{stream};
  265. auto traits =
  266. get_ck_fmha_fwd_traits(mask, q_dtype_str, head_size_8x, has_dropout, has_lse, alibi_slopes_.has_value());
  267. auto args =
  268. get_ck_fmha_fwd_args(
  269. has_lse,
  270. return_dropout_randval,
  271. mask,
  272. batch_size,
  273. seqlen_q,
  274. seqlen_k,
  275. num_heads,
  276. num_heads_k,
  277. head_size_8x,
  278. q_padded,
  279. k_padded,
  280. v_padded,
  281. alibi_slopes_,
  282. out,
  283. softmax_lse,
  284. p,
  285. softmax_scale,
  286. p_dropout,
  287. drop_seed,
  288. drop_offset);
  289. fmha_fwd(traits, args, stream_config);
  290. }
  291. else {
  292. // If seqlen_k == 0, then we have an empty tensor. We need to set the output to 0.
  293. out.zero_();
  294. softmax_lse.fill_(std::numeric_limits<float>::infinity());
  295. }
  296. at::Tensor out_padded = out;
  297. if (head_size_og % 8 != 0) {
  298. out = out.index({"...", torch::indexing::Slice(torch::indexing::None, head_size_og)});
  299. if (out_.has_value()) { out_.value().copy_(out); }
  300. }
  301. if (seqlenq_ngroups_swapped) {
  302. out = out.transpose(1, 2).reshape({batch_size, 1, num_heads_k * seqlen_q, head_size_og});
  303. out_padded = out_padded.transpose(1, 2).reshape({batch_size, 1, num_heads_k * seqlen_q, head_size_og});
  304. q_padded = q_padded.transpose(1, 2).reshape({batch_size, 1, num_heads_k * seqlen_q, head_size_og});
  305. softmax_lse = softmax_lse.reshape({batch_size, num_heads_k * seqlen_q, 1});
  306. }
  307. return {out, q_padded, k_padded, v_padded, out_padded, softmax_lse, p, rng_state};
  308. }