mirror of https://github.com/hpcaitech/ColossalAI
181 lines
6.3 KiB
Python
181 lines
6.3 KiB
Python
import enum
|
|
|
|
import torch
|
|
import torch.nn as nn
|
|
|
|
from colossalai.kernel.op_builder.scaled_masked_softmax import ScaledMaskedSoftmaxBuilder
|
|
from colossalai.kernel.op_builder.scaled_upper_triangle_masked_softmax import ScaledUpperTrainglemaskedSoftmaxBuilder
|
|
|
|
try:
|
|
from colossalai._C import scaled_masked_softmax, scaled_upper_triang_masked_softmax
|
|
except ImportError:
|
|
scaled_masked_softmax = None
|
|
scaled_upper_triang_masked_softmax = None
|
|
|
|
|
|
class AttnMaskType(enum.Enum):
|
|
padding = 1
|
|
causal = 2
|
|
|
|
|
|
class ScaledUpperTriangMaskedSoftmax(torch.autograd.Function):
|
|
"""
|
|
Fused operation which performs following three operations in sequence
|
|
|
|
1. Scale the tensor.
|
|
2. Apply upper triangular mask (typically used in gpt models).
|
|
3. Perform softmax.
|
|
"""
|
|
|
|
@staticmethod
|
|
def forward(ctx, inputs, scale):
|
|
global scaled_upper_triang_masked_softmax
|
|
if scaled_upper_triang_masked_softmax:
|
|
scaled_upper_triang_masked_softmax = ScaledUpperTrainglemaskedSoftmaxBuilder().load()
|
|
|
|
scale_t = torch.tensor([scale])
|
|
softmax_results = scaled_upper_triang_masked_softmax.forward(inputs, scale_t[0])
|
|
|
|
ctx.save_for_backward(softmax_results, scale_t)
|
|
return softmax_results
|
|
|
|
@staticmethod
|
|
def backward(ctx, output_grads):
|
|
softmax_results, scale_t = ctx.saved_tensors
|
|
input_grads = scaled_upper_triang_masked_softmax.backward(output_grads, softmax_results, scale_t[0])
|
|
|
|
return input_grads, None
|
|
|
|
|
|
class ScaledMaskedSoftmax(torch.autograd.Function):
|
|
"""
|
|
Fused operation which performs following three operations in sequence
|
|
|
|
1. Scale the tensor.
|
|
2. Apply the mask.
|
|
3. Perform softmax.
|
|
"""
|
|
|
|
@staticmethod
|
|
def forward(ctx, inputs, mask, scale):
|
|
scale_t = torch.tensor([scale])
|
|
|
|
# build and load kernel if not pre-built
|
|
global scaled_masked_softmax
|
|
if scaled_masked_softmax is None:
|
|
scaled_masked_softmax = ScaledMaskedSoftmaxBuilder().load()
|
|
|
|
softmax_results = scaled_masked_softmax.forward(inputs, mask, scale_t[0])
|
|
ctx.save_for_backward(softmax_results, scale_t)
|
|
return softmax_results
|
|
|
|
@staticmethod
|
|
def backward(ctx, output_grads):
|
|
softmax_results, scale_t = ctx.saved_tensors
|
|
|
|
input_grads = scaled_masked_softmax.backward(output_grads, softmax_results, scale_t[0])
|
|
return input_grads, None, None, None
|
|
|
|
|
|
class FusedScaleMaskSoftmax(nn.Module):
|
|
"""
|
|
Fused operation: scaling + mask + softmax
|
|
|
|
Arguments:
|
|
input_in_fp16: Flag to indicate if input in fp16 data format.
|
|
input_in_bf16: Flag to indicate if input in bf16 data format.
|
|
attn_mask_type: Attention mask type (pad or causal)
|
|
scaled_masked_softmax_fusion: Flag to indicate user want to use softmax fusion
|
|
mask_func: Mask function to be applied.
|
|
softmax_in_fp32: If True, softmax in performed at fp32 precision.
|
|
scale: Scaling factor used in input tensor scaling.
|
|
"""
|
|
|
|
def __init__(
|
|
self,
|
|
input_in_fp16,
|
|
input_in_bf16,
|
|
attn_mask_type,
|
|
scaled_masked_softmax_fusion,
|
|
mask_func,
|
|
softmax_in_fp32,
|
|
scale,
|
|
):
|
|
super(FusedScaleMaskSoftmax, self).__init__()
|
|
self.input_in_fp16 = input_in_fp16
|
|
self.input_in_bf16 = input_in_bf16
|
|
assert not (self.input_in_fp16
|
|
and self.input_in_bf16), "both fp16 and bf16 flags cannot be active at the same time."
|
|
self.input_in_float16 = self.input_in_fp16 or self.input_in_bf16
|
|
self.attn_mask_type = attn_mask_type
|
|
self.scaled_masked_softmax_fusion = scaled_masked_softmax_fusion
|
|
self.mask_func = mask_func
|
|
self.softmax_in_fp32 = softmax_in_fp32
|
|
self.scale = scale
|
|
assert (self.scale is None or softmax_in_fp32), "softmax should be in fp32 when scaled"
|
|
|
|
def forward(self, input, mask):
|
|
# [b, np, sq, sk]
|
|
assert input.dim() == 4
|
|
|
|
if self.is_kernel_available(mask, *input.size()):
|
|
return self.forward_fused_softmax(input, mask)
|
|
else:
|
|
return self.forward_torch_softmax(input, mask)
|
|
|
|
def is_kernel_available(self, mask, b, np, sq, sk):
|
|
attn_batches = b * np
|
|
|
|
if (self.scaled_masked_softmax_fusion # user want to fuse
|
|
and self.input_in_float16 # input must be fp16
|
|
and mask is not None # mask tensor must not be None
|
|
and 16 < sk <= 2048 # sk must be 16 ~ 2048
|
|
and sq % 4 == 0 # sq must be divisor of 4
|
|
and attn_batches % 4 == 0 # np * b must be divisor of 4
|
|
):
|
|
if 0 <= sk <= 2048:
|
|
batch_per_block = self.get_batch_per_block(sq, sk, b, np)
|
|
|
|
if self.attn_mask_type == AttnMaskType.causal:
|
|
if attn_batches % batch_per_block == 0:
|
|
return True
|
|
else:
|
|
if sq % batch_per_block == 0:
|
|
return True
|
|
return False
|
|
|
|
def forward_fused_softmax(self, input, mask):
|
|
b, np, sq, sk = input.size()
|
|
scale = self.scale if self.scale is not None else 1.0
|
|
|
|
if self.attn_mask_type == AttnMaskType.causal:
|
|
assert sq == sk, "causal mask is only for self attention"
|
|
|
|
# input is 3D tensor (attn_batches, sq, sk)
|
|
input = input.view(-1, sq, sk)
|
|
probs = ScaledUpperTriangMaskedSoftmax.apply(input, scale)
|
|
return probs.view(b, np, sq, sk)
|
|
else:
|
|
# input is 4D tensor (b, np, sq, sk)
|
|
return ScaledMaskedSoftmax.apply(input, mask, scale)
|
|
|
|
def forward_torch_softmax(self, input, mask):
|
|
if self.input_in_float16 and self.softmax_in_fp32:
|
|
input = input.float()
|
|
|
|
if self.scale is not None:
|
|
input = input * self.scale
|
|
mask_output = self.mask_func(input, mask) if mask is not None else input
|
|
probs = torch.nn.Softmax(dim=-1)(mask_output)
|
|
|
|
if self.input_in_float16 and self.softmax_in_fp32:
|
|
if self.input_in_fp16:
|
|
probs = probs.half()
|
|
else:
|
|
probs = probs.bfloat16()
|
|
|
|
return probs
|
|
|
|
def get_batch_per_block(self, sq, sk, b, np):
|
|
return scaled_masked_softmax.get_batch_per_block(sq, sk, b, np)
|