forked from pytorch/pytorch
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathNativeMultiheadAttnKernel.cpp
111 lines (92 loc) · 3.63 KB
/
NativeMultiheadAttnKernel.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
#define TORCH_ASSERT_ONLY_METHOD_OPERATORS
#include <ATen/core/Tensor.h>
#include <ATen/Dispatch.h>
#include <ATen/OpMathType.h>
#include <ATen/Parallel.h>
#include <ATen/TensorIterator.h>
#include <ATen/cpu/vec/vec.h>
#include <ATen/cpu/vec/functional.h>
#include <ATen/native/cpu/utils.h>
#include <ATen/native/transformers/attention.h>
#include <c10/util/irange.h>
namespace at::native {
namespace {
template <typename scalar_t>
void cpu_transform_bias_rescale_qkv(
scalar_t* q_k_v_data,
const scalar_t* qkv_data,
const scalar_t* qkv_bias_data,
int64_t B,
int64_t T,
int64_t D,
int64_t num_head) {
int64_t dim_per_head = D / num_head;
// shapes and strides:
// qkv : {B, T, 3, num_head, dim_per_head}
// qkv_bias : {3, num_head, dim_per_head}
// q_k_v : {3, B, num_head, T, dim_per_head}
//
int64_t i_strideB = T * 3 * D;
int64_t i_strideT = 3 * D;
int64_t o_stride = B * num_head * T * dim_per_head;
// inv_sqrt_dim_per_head in accumulate type
using acc_t = at::opmath_type<scalar_t>;
using Vec = vec::Vectorized<acc_t>;
const acc_t s = 1.0 / std::sqrt(static_cast<acc_t>(dim_per_head));
// parallel on {B, num_head, T}
int64_t grain_size = std::max(at::internal::GRAIN_SIZE / (3 * dim_per_head), (int64_t)1);
at::parallel_for(0, B * num_head * T, grain_size, [&](int64_t begin, int64_t end) {
int64_t b{0}, nh{0}, t{0};
data_index_init(begin, b, B, nh, num_head, t, T);
for (const auto i : c10::irange(begin, end)) {
const scalar_t* q_in_ptr = qkv_data + b * i_strideB + t * i_strideT + 0 * D + nh * dim_per_head;
const scalar_t* k_in_ptr = qkv_data + b * i_strideB + t * i_strideT + 1 * D + nh * dim_per_head;
const scalar_t* v_in_ptr = qkv_data + b * i_strideB + t * i_strideT + 2 * D + nh * dim_per_head;
const scalar_t* q_bias_ptr = qkv_bias_data + 0 * D + nh * dim_per_head;
const scalar_t* k_bias_ptr = qkv_bias_data + 1 * D + nh * dim_per_head;
const scalar_t* v_bias_ptr = qkv_bias_data + 2 * D + nh * dim_per_head;
// we can use global index i here for output
scalar_t* q_out_ptr = q_k_v_data + 0 * o_stride + i * dim_per_head;
scalar_t* k_out_ptr = q_k_v_data + 1 * o_stride + i * dim_per_head;
scalar_t* v_out_ptr = q_k_v_data + 2 * o_stride + i * dim_per_head;
// q = (q + bias) * inv_sqrt_dim_per_head
vec::map2<scalar_t>(
[s](Vec q, Vec q_bias) { return (q + q_bias) * Vec(s); },
q_out_ptr, q_in_ptr, q_bias_ptr, dim_per_head);
// k = k + bias
vec::map2<scalar_t>([](Vec k, Vec k_bias) { return k + k_bias; },
k_out_ptr, k_in_ptr, k_bias_ptr, dim_per_head);
// v = v + bias
vec::map2<scalar_t>([](Vec v, Vec v_bias) { return v + v_bias; },
v_out_ptr, v_in_ptr, v_bias_ptr, dim_per_head);
// move to the next index
data_index_step(b, B, nh, num_head, t, T);
}
});
}
void transform_bias_rescale_qkv_kernel_impl(
at::ScalarType type,
void* _q_k_v,
const void* _qkv,
const void* _qkv_bias,
int64_t B,
int64_t T,
int64_t D,
int64_t num_head) {
AT_DISPATCH_FLOATING_TYPES_AND2(kHalf, kBFloat16, type, "transform_bias_rescale_qkv", [&] {
scalar_t* q_k_v = static_cast<scalar_t*>(_q_k_v);
const scalar_t* qkv = static_cast<const scalar_t*>(_qkv);
const scalar_t* qkv_bias = static_cast<const scalar_t*>(_qkv_bias);
cpu_transform_bias_rescale_qkv<scalar_t>(
q_k_v,
qkv,
qkv_bias,
B,
T,
D,
num_head);
});
}
} // anonymous namespace
REGISTER_DISPATCH(transform_bias_rescale_qkv_stub, &transform_bias_rescale_qkv_kernel_impl);
} // at::native