-
Notifications
You must be signed in to change notification settings - Fork 21
/
Copy pathtiny_flash_attn_triton.py
244 lines (226 loc) · 9.48 KB
/
tiny_flash_attn_triton.py
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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
# https://github.com/openai/triton/blob/main/python/tutorials/06-fused-attention.py
#
# https://github.com/kyegomez/FlashAttention20Triton
from torch import float32
import torch
import time
import triton
import triton.language as tl
def flash_attn_triton(q, k, v, causal=True, sm_scale=1):
# shape constraints
Lq, Lk, Lv = q.shape[-1], k.shape[-1], v.shape[-1]
assert Lq == Lk and Lk == Lv
assert Lk in {16, 32, 64, 128}
o = torch.empty_like(q)
BLOCK_M = 128
BLOCK_N = 64
# NOTE: 对于flash attention 2, 外层循环的q可以并行处理, 因此每个thread需要计算正确的offset
# 一个q, k, v的shape往往是(bs, head, seqlen, dim)
# 对于(bs, head)中的每个元素都分配一个thread
# 对于seqlen / BLOCK_M个的q分块, 每个分块再分配一个thread
grid = (triton.cdiv(q.shape[2], BLOCK_M), q.shape[0] * q.shape[1], 1)
# NOTE:
# L.shape = (bs * head, seqlen)
# L记录了所有的分母和mi(m_i + tl.math.log2(l_i)), 用于后续的backward
L = torch.empty((q.shape[0] * q.shape[1], q.shape[2]), device=q.device, dtype=torch.float32)
# 设置适当的wrap以提升性能
num_warps = 4 if Lk <= 64 else 8
_fwd_kernel[grid](
q, k, v, sm_scale,
L,
o,
q.stride(0), q.stride(1), q.stride(2), q.stride(3),
k.stride(0), k.stride(1), k.stride(2), k.stride(3),
v.stride(0), v.stride(1), v.stride(2), v.stride(3),
o.stride(0), o.stride(1), o.stride(2), o.stride(3),
q.shape[0], q.shape[1], q.shape[2],
BLOCK_M=BLOCK_M, BLOCK_N=BLOCK_N, DIM=Lk,
IS_CAUSAL=causal,
num_warps=num_warps,
num_stages=4)
return o
@triton.jit
def _fwd_kernel(
Q, K, V, sm_scale,
# L记录了所有的分母和mi, 用于后续的backward
L,
O,
stride_q_bs, stride_q_head, stride_q_seqlen, stride_q_dim,
stride_k_bs, stride_k_head, stride_k_seqlen, stride_k_dim,
stride_v_bs, stride_v_head, stride_v_seqlen, stride_v_dim,
stride_o_bs, stride_o_head, stride_o_seqlen, stride_o_dim,
BS, HEAD, SEQLEN,
# BLOCK_M用于做Q的分块
BLOCK_M: tl.constexpr,
DIM: tl.constexpr,
# BLOCK_N用于做K和V的分块
BLOCK_N: tl.constexpr,
IS_CAUSAL: tl.constexpr,
):
# grid = (cdiv(seqlen, BLOCK_M), bs * head)
# triton.language.program_id(axis) axis is The axis of the 3D launch grid
# Q分块的起始地址
start_m = tl.program_id(0)
# 跳过(bs, head)的偏移
off_bs_head = tl.program_id(1)
# NOTE:
# base = off_bs_head * stride_q_head找到正确的(bs, head)位置
# strides: 步长, advance时直接使用步数, 会自动根据步长计算跳过的元素
# offsets表示parent block (seqlen, dim)中怎么偏移来获取小块
# block_shape=(BLOCK_M, DIM)表示parent block的shape
# order表示用什么顺序读取存储来构造所需的shape
qkv_base_offset = off_bs_head * stride_q_head
Q_block_ptr = tl.make_block_ptr(
# base offset to skip to the right (bs, head)
base=Q + qkv_base_offset,
# the shape of parent
shape=(SEQLEN, DIM),
strides=(stride_q_seqlen, stride_q_dim),
# offset of the block inside of parent block
offsets=(start_m * BLOCK_M, 0),
block_shape=(BLOCK_M, DIM),
order=(1, 0),
)
K_block_ptr = tl.make_block_ptr(
# base offset to skip to the right (bs, head)
base=K + qkv_base_offset,
# the shape of parent
# NOTE: make_block_ptr读入时将K转置了
shape=(DIM, SEQLEN),
strides=(stride_k_dim, stride_k_seqlen),
# 每个Q需要遍历整个的k和v
offsets=(0, 0),
# K根据BLOCK_N分块
block_shape=(DIM, BLOCK_N),
# 读入K的转置
order=(0, 1),
)
V_block_ptr = tl.make_block_ptr(
# base offset to skip to the right (bs, head)
base=V + qkv_base_offset,
# the shape of parent
shape=(SEQLEN, DIM),
strides=(stride_k_seqlen, stride_v_dim),
# 每个Q需要遍历整个的k和v
offsets=(0, 0),
# K根据BLOCK_N分块
block_shape=(BLOCK_N, DIM),
order=(1, 0),
)
# initialize offsets
# NOTE: BLOCK_M表示Q的分块大小, BLOCK_N表示k, v的分块大小
off_m = start_m * BLOCK_M + tl.arange(0, BLOCK_M)
off_n = tl.arange(0, BLOCK_N)
# initialize pointers
# NOTE: 一次处理一个(BLOCK_M, dim)的q, 而max和分母的sum都只需要一维, 即(BLOCK_M, 1)
max = tl.zeros([BLOCK_M], dtype=tl.float32) - float('inf')
# 分母累加的sum, 每行的sum是一样的, 所以只需要一维然后广播即可
denom = tl.zeros([BLOCK_M], dtype=tl.float32)
out_buffer = tl.zeros([BLOCK_M, DIM], dtype=tl.float32)
# NOTE:
# scale sm_scale by log_2(e) and use
# 2^x instead of exp in the loop because CSE and LICM
# don't work as expected with `exp` in the loop
# CSE(common subexpression elimination), LICM(loop invariant code motion)是编译器里的东西
qk_scale = sm_scale * 1.44269504
# load q: stay in SRAM throughout
q = tl.load(Q_block_ptr)
q = (q * qk_scale).to(tl.float16)
# loop over k, v and update accumulator
lo = 0
# NOTE:: CAUSAL就是常说的不能看到后面的文本的自回归模型
hi = (start_m + 1) * BLOCK_M if IS_CAUSAL else SEQLEN
# NOTE:
# 当前q和0..seqlen的kv算attention
# 每次批处理BLOCK_N个k, v(即k, v以BLOCK_N分块)
for start_n in range(lo, hi, BLOCK_N):
k = tl.load(K_block_ptr)
v = tl.load(V_block_ptr)
# compute qk
# NOTE: q.shape = (BLOCK_M, dim), k.shape(已转置) = (dim, BLOCK_N)
qk = tl.zeros([BLOCK_M, BLOCK_N], dtype=tl.float32)
if IS_CAUSAL:
qk = tl.where(off_m[:, None] >= (start_n + off_n[None, :]), qk, float("-inf"))
# NOTE: 执行矩阵乘法(matrix product), k在make_block_ptr时已经转置
# qk init as zero
qk += tl.dot(q, k)
# compute scaling constant
# NOTE:
# max.shape = [BLOCK_M], aka [BLOCK_M, 1]
# qk.shape = [BLOCK_M, BLOCK_N]
# tl.max(block, axis)
# tl.maximum(block, block)
max_new = tl.maximum(max, tl.max(qk, 1))
# 保存exp的值, 节省exp操作
alpha = tl.math.exp2(max - max_new)
# NOTE:
# nume = e^{x - max(x)}
# max.shape = [BLOCK_M], max_new[:, None]扩展成[BLOCK_M, 1]来做广播操作
nume = tl.math.exp2(qk - max_new[:, None])
# scale and update acc
# NOTE: 利用广播来快速构建scale用于更新分母
out_scale = denom * 0 + alpha
# NOTE:
# out_scale.shape = l_i.shape = [BLOCK_M]
# out_scale[:, None]扩展成[BLOCK_M, 1]来做广播操作
# out_buffer = old_out * scale来更新分子
out_buffer *= out_scale[:, None]
out_buffer += tl.dot(nume.to(tl.float16), v)
# update max and denominator
denom = denom * alpha + tl.sum(nume, 1)
max = max_new
# update k v pointer
# NOTE: 计算下一个k, v的分块
# 因为k已经转置(dim, seqlen), 所以算下一批seq的k时是增加k的第二个维度
K_block_ptr = tl.advance(K_block_ptr, (0, BLOCK_N))
V_block_ptr = tl.advance(V_block_ptr, (BLOCK_N, 0))
# write back l and m for backward
# 最后统一更新output buffer, 除上完整的分母
out_buffer = out_buffer / denom[:, None]
# NOTE: 将分母和mi保存到L中, 用于后续的backward
# L.shape = (bs * head, seqlen), 因为每一行的分母和mi是相同的
# off_bs_head = bs * head
l_ptr = L + off_bs_head * SEQLEN + off_m
# write [BLOCK_M] of data to L
tl.store(l_ptr, max + tl.math.log2(denom))
# write back O
O_block_ptr = tl.make_block_ptr(
base=O + qkv_base_offset,
shape=(SEQLEN, DIM),
strides=(stride_o_seqlen, stride_o_dim),
offsets=(start_m * BLOCK_M, 0),
block_shape=(BLOCK_M, DIM),
order=(1, 0),
)
tl.store(O_block_ptr, out_buffer.to(tl.float16))
def ref_attn(q, k, v, causal=True, sm_scale=1):
SEQLEN = q.shape[-2]
M = torch.tril(torch.ones((SEQLEN, SEQLEN), device="cuda"))
p = torch.matmul(q, k.transpose(2, 3)) * sm_scale
if causal:
p[:, :, M == 0] = float("-inf")
p = torch.softmax(p.float(), dim=-1).half()
ref_out = torch.matmul(p, v)
return ref_out
def causal_test(BS, HEAD, SEQLEN, DIM, causal):
dtype = torch.float16
torch.manual_seed(20)
q = (torch.empty((BS, HEAD, SEQLEN, DIM), dtype=dtype, device="cuda").normal_(mean=0.0, std=0.5).requires_grad_())
k = (torch.empty((BS, HEAD, SEQLEN, DIM), dtype=dtype, device="cuda").normal_(mean=0.0, std=0.5).requires_grad_())
v = (torch.empty((BS, HEAD, SEQLEN, DIM), dtype=dtype, device="cuda").normal_(mean=0.0, std=0.5).requires_grad_())
sm_scale = 0.5
# reference implementation
time_ref = time.time()
ref_out = ref_attn(q, k, v, causal=causal, sm_scale=sm_scale)
time_ref = time.time() - time_ref
# triton implementation
time_tri = time.time()
tri_out = flash_attn_triton(q, k, v, causal=causal, sm_scale=sm_scale).half()
time_tri = time.time() - time_tri
# compare
assert torch.allclose(ref_out, tri_out, atol=1e-2, rtol=0)
print("causal = {} ref time: {:.4f} ms, tri time: {:.4f}".format(causal, time_ref * 1000, time_tri * 1000))
def test_attention():
BS, HEAD, SEQLEN, DIM = 1, 2, 1024, 64
causal_test(BS, HEAD, SEQLEN, DIM, causal=False)
causal_test(BS, HEAD, SEQLEN, DIM, causal=True)