|
@@ -13,16 +13,24 @@ from funasr.export.models.modules.multihead_att import \
|
|
|
from funasr.export.utils.torch_function import MakePadMask, subsequent_mask
|
|
from funasr.export.utils.torch_function import MakePadMask, subsequent_mask
|
|
|
|
|
|
|
|
class XformerDecoder(nn.Module):
|
|
class XformerDecoder(nn.Module):
|
|
|
- def __init__(self, model, max_seq_len=512, **kwargs):
|
|
|
|
|
|
|
+ def __init__(self,
|
|
|
|
|
+ model,
|
|
|
|
|
+ max_seq_len = 512,
|
|
|
|
|
+ model_name = 'decoder',
|
|
|
|
|
+ onnx: bool = True,):
|
|
|
super().__init__()
|
|
super().__init__()
|
|
|
self.embed = Embedding(model.embed, max_seq_len)
|
|
self.embed = Embedding(model.embed, max_seq_len)
|
|
|
self.model = model
|
|
self.model = model
|
|
|
- self.make_pad_mask = MakePadMask(max_seq_len, flip=False)
|
|
|
|
|
|
|
+ if onnx:
|
|
|
|
|
+ self.make_pad_mask = MakePadMask(max_seq_len, flip=False)
|
|
|
|
|
+ else:
|
|
|
|
|
+ self.make_pad_mask = subsequent_mask(max_seq_len, flip=False)
|
|
|
|
|
+
|
|
|
if isinstance(self.model.decoders[0].self_attn, MultiHeadedAttention):
|
|
if isinstance(self.model.decoders[0].self_attn, MultiHeadedAttention):
|
|
|
self.num_heads = self.model.decoders[0].self_attn.h
|
|
self.num_heads = self.model.decoders[0].self_attn.h
|
|
|
self.hidden_size = self.model.decoders[0].self_attn.linear_out.out_features
|
|
self.hidden_size = self.model.decoders[0].self_attn.linear_out.out_features
|
|
|
|
|
|
|
|
- # replace multihead attention module into customized module.
|
|
|
|
|
|
|
+ # replace multi-head attention module into customized module.
|
|
|
for i, d in enumerate(self.model.decoders):
|
|
for i, d in enumerate(self.model.decoders):
|
|
|
# d is DecoderLayer
|
|
# d is DecoderLayer
|
|
|
if isinstance(d.self_attn, MultiHeadedAttention):
|
|
if isinstance(d.self_attn, MultiHeadedAttention):
|
|
@@ -31,17 +39,23 @@ class XformerDecoder(nn.Module):
|
|
|
d.src_attn = OnnxMultiHeadedAttention(d.src_attn)
|
|
d.src_attn = OnnxMultiHeadedAttention(d.src_attn)
|
|
|
self.model.decoders[i] = OnnxDecoderLayer(d)
|
|
self.model.decoders[i] = OnnxDecoderLayer(d)
|
|
|
|
|
|
|
|
- self.model_name = "xformer_decoder"
|
|
|
|
|
|
|
+ self.model_name = model_name
|
|
|
|
|
|
|
|
def prepare_mask(self, mask):
|
|
def prepare_mask(self, mask):
|
|
|
|
|
+ mask_3d_btd = mask[:, :, None]
|
|
|
if len(mask.shape) == 2:
|
|
if len(mask.shape) == 2:
|
|
|
- mask = mask[:, None, None, :]
|
|
|
|
|
|
|
+ mask_4d_bhlt = 1 - mask[:, None, None, :]
|
|
|
elif len(mask.shape) == 3:
|
|
elif len(mask.shape) == 3:
|
|
|
- mask = mask[:, None, :]
|
|
|
|
|
- mask = 1 - mask
|
|
|
|
|
- return mask * -10000.0
|
|
|
|
|
|
|
+ mask_4d_bhlt = 1 - mask[:, None, :]
|
|
|
|
|
+
|
|
|
|
|
+ mask_4d_bhlt = mask_4d_bhlt * -10000.0
|
|
|
|
|
+ return mask_3d_btd, mask_4d_bhlt
|
|
|
|
|
+
|
|
|
|
|
+ def forward(self,
|
|
|
|
|
+ tgt,
|
|
|
|
|
+ memory,
|
|
|
|
|
+ cache):
|
|
|
|
|
|
|
|
- def forward(self, tgt, memory, cache):
|
|
|
|
|
mask = subsequent_mask(tgt.size(-1)).unsqueeze(0) # (B, T)
|
|
mask = subsequent_mask(tgt.size(-1)).unsqueeze(0) # (B, T)
|
|
|
|
|
|
|
|
x = self.embed(tgt)
|
|
x = self.embed(tgt)
|
|
@@ -63,33 +77,37 @@ class XformerDecoder(nn.Module):
|
|
|
|
|
|
|
|
def get_dummy_inputs(self, enc_size):
|
|
def get_dummy_inputs(self, enc_size):
|
|
|
tgt = torch.LongTensor([0]).unsqueeze(0)
|
|
tgt = torch.LongTensor([0]).unsqueeze(0)
|
|
|
- enc_out = torch.randn(1, 100, enc_size)
|
|
|
|
|
|
|
+ memory = torch.randn(1, 100, enc_size)
|
|
|
|
|
+ cache_num = len(self.model.decoders)
|
|
|
cache = [
|
|
cache = [
|
|
|
torch.zeros((1, 1, self.model.decoders[0].size))
|
|
torch.zeros((1, 1, self.model.decoders[0].size))
|
|
|
- for _ in range(len(self.model.decoders))
|
|
|
|
|
|
|
+ for _ in range(cache_num)
|
|
|
]
|
|
]
|
|
|
- return (tgt, enc_out, cache)
|
|
|
|
|
|
|
+ return (tgt, memory, cache)
|
|
|
|
|
|
|
|
def is_optimizable(self):
|
|
def is_optimizable(self):
|
|
|
return True
|
|
return True
|
|
|
|
|
|
|
|
def get_input_names(self):
|
|
def get_input_names(self):
|
|
|
|
|
+ cache_num = len(self.model.decoders)
|
|
|
return ["tgt", "memory"] + [
|
|
return ["tgt", "memory"] + [
|
|
|
- "cache_%d" % i for i in range(len(self.model.decoders))
|
|
|
|
|
|
|
+ "cache_%d" % i for i in range(cache_num)
|
|
|
]
|
|
]
|
|
|
|
|
|
|
|
def get_output_names(self):
|
|
def get_output_names(self):
|
|
|
- return ["y"] + ["out_cache_%d" % i for i in range(len(self.model.decoders))]
|
|
|
|
|
|
|
+ cache_num = len(self.model.decoders)
|
|
|
|
|
+ return ["y"] + ["out_cache_%d" % i for i in range(cache_num)]
|
|
|
|
|
|
|
|
def get_dynamic_axes(self):
|
|
def get_dynamic_axes(self):
|
|
|
ret = {
|
|
ret = {
|
|
|
"tgt": {0: "tgt_batch", 1: "tgt_length"},
|
|
"tgt": {0: "tgt_batch", 1: "tgt_length"},
|
|
|
"memory": {0: "memory_batch", 1: "memory_length"},
|
|
"memory": {0: "memory_batch", 1: "memory_length"},
|
|
|
}
|
|
}
|
|
|
|
|
+ cache_num = len(self.model.decoders)
|
|
|
ret.update(
|
|
ret.update(
|
|
|
{
|
|
{
|
|
|
- "cache_%d" % d: {0: "cache_%d_batch" % d, 1: "cache_%d_length" % d}
|
|
|
|
|
- for d in range(len(self.model.decoders))
|
|
|
|
|
|
|
+ "cache_%d" % d: {0: "cache_%d_batch" % d, 2: "cache_%d_length" % d}
|
|
|
|
|
+ for d in range(cache_num)
|
|
|
}
|
|
}
|
|
|
)
|
|
)
|
|
|
return ret
|
|
return ret
|