2025-04-07 10:56:12 +08:00
# SPDX-License-Identifier: Apache-2.0
2025-04-22 09:59:09 +08:00
# Copyright (c) 2025 Huawei Technologies Co., Ltd. All Rights Reserved.
2025-04-07 10:56:12 +08:00
# Copyright 2023 The vLLM team.
# Copyright 2023 DeepSeek-AI and the HuggingFace Inc. team. All rights reserved.
#
# This code is based on EleutherAI's GPT-NeoX library and the GPT-NeoX
# and OPT implementations in this library. It has been modified from its
# original forms to accommodate minor architectural differences compared
# to GPT-NeoX and OPT used by the Meta AI team that trained the model.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
2025-04-19 17:38:18 +08:00
# # Adapted from
# # vllm-project/vllm/blob/main/vllm/model_executor/models/deepseek_v2.py
# # https://github.com/huggingface/transformers/blob/v4.28.0/src/transformers/models/llama/modeling_llama.py
# # vllm-project/vllm/vllm/model_executor/models/deepseek_v2.py
# """Inference-only DeepseekV2/DeepseekV3 model."""
2025-06-28 00:12:43 +08:00
from typing import Any , Callable , Dict , Iterable , List , Optional , Tuple , Union
2025-04-07 10:56:12 +08:00
import torch
2025-05-01 13:51:42 +08:00
import torch_npu
2025-04-07 10:56:12 +08:00
from torch import nn
from transformers import PretrainedConfig
2025-04-23 16:23:25 +08:00
from vllm . attention import Attention , AttentionMetadata
2025-07-26 17:15:47 +08:00
from vllm . config import ( CacheConfig , ModelConfig , VllmConfig ,
get_current_vllm_config )
2025-06-25 19:56:49 +08:00
from vllm . distributed import ( get_pp_group , get_tensor_model_parallel_rank ,
2025-04-19 17:38:18 +08:00
get_tensor_model_parallel_world_size ,
2025-06-25 19:56:49 +08:00
get_tp_group , split_tensor_along_last_dim ,
tensor_model_parallel_all_gather ,
tensor_model_parallel_all_reduce ,
tensor_model_parallel_reduce_scatter )
2025-07-21 09:08:04 +08:00
from vllm . distributed . parallel_state import get_dp_group , get_ep_group
2025-04-19 17:38:18 +08:00
from vllm . forward_context import get_forward_context
2025-05-01 13:51:42 +08:00
from vllm . model_executor . layers . activation import SiluAndMul
2025-04-07 10:56:12 +08:00
from vllm . model_executor . layers . layernorm import RMSNorm
2025-04-19 17:38:18 +08:00
from vllm . model_executor . layers . linear import ( ColumnParallelLinear ,
2025-05-01 13:51:42 +08:00
MergedColumnParallelLinear ,
2025-04-19 17:38:18 +08:00
ReplicatedLinear ,
2025-05-01 13:51:42 +08:00
RowParallelLinear ,
UnquantizedLinearMethod )
2025-04-07 10:56:12 +08:00
from vllm . model_executor . layers . logits_processor import LogitsProcessor
from vllm . model_executor . layers . quantization import QuantizationConfig
2025-04-19 17:38:18 +08:00
from vllm . model_executor . layers . rotary_embedding import get_rope
2025-04-07 10:56:12 +08:00
from vllm . model_executor . layers . sampler import get_sampler
from vllm . model_executor . layers . vocab_parallel_embedding import (
ParallelLMHead , VocabParallelEmbedding )
2025-06-28 00:12:43 +08:00
from vllm . model_executor . model_loader . weight_utils import (
default_weight_loader , maybe_remap_kv_scale_name )
2025-04-19 17:38:18 +08:00
from vllm . model_executor . models . deepseek_v2 import \
[refactor] Refactoring AscendFusedMoE (#1229)
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
This PR is used for resolved [issue
1147](https://github.com/vllm-project/vllm-ascend/issues/1147)
1. Move fused_moe code into one file `fused_moe.py`.
2. Integrate branch conditions into function `get_fused_moe_state`.
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
1. This PR has removed the env `VLLM_ENABLE_MC2`, because I think this
env is useless, we can make judgments based on the current scenario
without this env, it will only increase complexity.
2. This PR has removed the env `USING_LCCL_COM`, because this env has
already expired.
3. `additional_config.expert_tensor_parallel_size` has already expired,
and now we also use parameter `enable_expert_parallel`, consistent with
the vLLM.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
Signed-off-by: zzzzwwjj <1183291235@qq.com>
2025-06-17 17:49:03 +08:00
DeepseekV2ForCausalLM # noqa: E501
2025-04-19 17:38:18 +08:00
from vllm . model_executor . models . deepseek_v2 import \
[refactor] Refactoring AscendFusedMoE (#1229)
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
This PR is used for resolved [issue
1147](https://github.com/vllm-project/vllm-ascend/issues/1147)
1. Move fused_moe code into one file `fused_moe.py`.
2. Integrate branch conditions into function `get_fused_moe_state`.
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
1. This PR has removed the env `VLLM_ENABLE_MC2`, because I think this
env is useless, we can make judgments based on the current scenario
without this env, it will only increase complexity.
2. This PR has removed the env `USING_LCCL_COM`, because this env has
already expired.
3. `additional_config.expert_tensor_parallel_size` has already expired,
and now we also use parameter `enable_expert_parallel`, consistent with
the vLLM.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
Signed-off-by: zzzzwwjj <1183291235@qq.com>
2025-06-17 17:49:03 +08:00
yarn_get_mscale # noqa: E501
2025-06-28 00:12:43 +08:00
from vllm . model_executor . models . deepseek_v2 import (
DeepseekV2Attention , DeepseekV2DecoderLayer , DeepseekV2MLAAttention ,
get_spec_layer_idx_from_weight_name )
2025-04-07 10:56:12 +08:00
from vllm . model_executor . models . utils import (
2025-06-28 00:12:43 +08:00
PPMissingLayer , is_pp_missing_parameter ,
make_empty_intermediate_tensors_factory , make_layers , maybe_prefix )
2025-04-07 10:56:12 +08:00
from vllm . sequence import IntermediateTensors
2025-06-05 16:28:01 +08:00
from vllm_ascend . ascend_config import get_ascend_config
2025-04-19 17:38:18 +08:00
from vllm_ascend . ops . fused_moe import AscendFusedMoE
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
from vllm_ascend . quantization . quant_config import AscendLinearMethod
2025-05-01 13:51:42 +08:00
from vllm_ascend . quantization . w8a8_dynamic import AscendW8A8DynamicLinearMethod
2025-07-11 08:51:17 +08:00
from vllm_ascend . utils import dispose_tensor , npu_prefetch
2025-05-01 13:51:42 +08:00
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
class CustomDeepseekV2SiluAndMul ( SiluAndMul ) :
def __init__ ( self ,
* ,
weight_scale : Optional [ Callable [ [ ] , torch . Tensor ] ] = None ) :
super ( ) . __init__ ( )
self . weight_scale = weight_scale
def forward_oot ( self , x : Union [ torch . Tensor , Tuple [ torch . Tensor ,
torch . Tensor ] ] ) :
if isinstance ( x , tuple ) :
assert self . weight_scale is not None
# For AscendW8A8DynamicLinearMethod:
# a dynamic scale is passed along with the quantized value.
quantized_x , dynamic_scale = x
return torch_npu . npu_dequant_swiglu_quant (
x = quantized_x ,
weight_scale = self . weight_scale ( ) ,
activation_scale = dynamic_scale ,
activate_left = True ,
quant_mode = 1 )
else :
return super ( ) . forward_oot ( x )
class CustomDeepseekV2MergedReplicatedLinear ( ReplicatedLinear ) :
def __init__ (
self ,
input_size : int ,
output_sizes : list [ int ] ,
bias : bool = True ,
quant_config : Optional [ QuantizationConfig ] = None ,
prefix : str = " " ,
) :
self . output_sizes = output_sizes
super ( ) . __init__ ( input_size ,
sum ( output_sizes ) ,
bias = bias ,
quant_config = quant_config ,
prefix = prefix )
def weight_loader ( self , param : torch . nn . Parameter ,
loaded_weight : torch . Tensor , loaded_shard_id : int ) :
# With no support for GGUF format yet.
assert not getattr ( param , " is_gguf_weight " , False )
assert not getattr ( param , " is_gguf_weight_type " , False )
assert loaded_shard_id < len ( self . output_sizes )
shard_offset = sum ( self . output_sizes [ : loaded_shard_id ] )
shard_size = self . output_sizes [ loaded_shard_id ]
shard = param . data . narrow ( param . output_dim , shard_offset , shard_size )
assert shard . size ( ) == loaded_weight . size ( ) , (
f " Tried to load weights of size { loaded_weight . size ( ) } "
f " to a parameter shard of id { loaded_shard_id } size { shard . size ( ) } "
)
shard . copy_ ( loaded_weight )
2025-06-25 19:56:49 +08:00
class CustomDeepseekV2RowParallelLinearReplaceAllreduce ( RowParallelLinear ) :
def forward (
self ,
input_ ,
is_prefill = True
) - > Union [ torch . Tensor , tuple [ torch . Tensor , Optional [ nn . Parameter ] ] ] :
if self . input_is_parallel :
input_parallel = input_
else :
tp_rank = get_tensor_model_parallel_rank ( )
splitted_input = split_tensor_along_last_dim (
input_ , num_partitions = self . tp_size )
input_parallel = splitted_input [ tp_rank ] . contiguous ( )
# Matrix multiply.
assert self . quant_method is not None
# Only fuse bias add into GEMM for rank 0 (this ensures that
# bias will not get added more than once in TP>1 case)
bias_ = None if ( self . tp_rank > 0 or self . skip_bias_add ) else self . bias
output_parallel = self . quant_method . apply ( self ,
input_parallel ,
bias = bias_ )
if self . reduce_results and self . tp_size > 1 :
if not is_prefill and output_parallel . shape [ 0 ] % self . tp_size == 0 :
output = tensor_model_parallel_reduce_scatter ( output_parallel ,
dim = 0 )
else :
output = tensor_model_parallel_all_reduce ( output_parallel )
else :
output = output_parallel
output_bias = self . bias if self . skip_bias_add else None
if not self . return_bias :
return output
return output , output_bias
class CustomDeepseekV2RowParallelLinear ( RowParallelLinear ) :
def forward (
self ,
input_ ,
is_prefill = True
) - > Union [ torch . Tensor , tuple [ torch . Tensor , Optional [ nn . Parameter ] ] ] :
if self . input_is_parallel :
input_parallel = input_
else :
tp_rank = get_tensor_model_parallel_rank ( )
splitted_input = split_tensor_along_last_dim (
input_ , num_partitions = self . tp_size )
input_parallel = splitted_input [ tp_rank ] . contiguous ( )
# Matrix multiply.
assert self . quant_method is not None
# Only fuse bias add into GEMM for rank 0 (this ensures that
# bias will not get added more than once in TP>1 case)
bias_ = None if ( self . tp_rank > 0 or self . skip_bias_add ) else self . bias
output_parallel = self . quant_method . apply ( self ,
input_parallel ,
bias = bias_ )
if self . reduce_results and self . tp_size > 1 :
output = tensor_model_parallel_all_reduce ( output_parallel )
else :
output = output_parallel
output_bias = self . bias if self . skip_bias_add else None
if not self . return_bias :
return output
return output , output_bias
2025-05-01 13:51:42 +08:00
class CustomDeepseekV2MLP ( nn . Module ) :
def __init__ (
self ,
hidden_size : int ,
intermediate_size : int ,
hidden_act : str ,
quant_config : Optional [ QuantizationConfig ] = None ,
reduce_results : bool = True ,
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
force_replicate : bool = False ,
2025-05-01 13:51:42 +08:00
prefix : str = " " ,
) - > None :
super ( ) . __init__ ( )
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
if not force_replicate :
self . gate_up_proj = MergedColumnParallelLinear (
hidden_size , [ intermediate_size ] * 2 ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .gate_up_proj " )
self . down_proj = RowParallelLinear ( intermediate_size ,
hidden_size ,
bias = False ,
quant_config = quant_config ,
reduce_results = reduce_results ,
prefix = f " { prefix } .down_proj " )
else :
self . gate_up_proj = CustomDeepseekV2MergedReplicatedLinear (
hidden_size , [ intermediate_size ] * 2 ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .gate_up_proj " )
self . down_proj = ReplicatedLinear ( intermediate_size ,
hidden_size ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .down_proj " )
2025-05-01 13:51:42 +08:00
if hidden_act != " silu " :
raise ValueError ( f " Unsupported activation: { hidden_act } . "
" Only silu is supported for now. " )
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
quant_method = self . gate_up_proj . quant_method
if isinstance ( quant_method , UnquantizedLinearMethod ) :
self . act_fn = CustomDeepseekV2SiluAndMul ( )
elif ( isinstance ( quant_method , AscendLinearMethod ) and isinstance (
quant_method . quant_method , AscendW8A8DynamicLinearMethod ) ) :
# TODO(sdmyzlp): Currently preserved as before:
# 1. The only quantization supported for silu is W8A8Dynamic
# 2. Output dtype of gate_up/down is fixed to be int32/bfloat16
#
# Maybe one can implement a better and more general configuration
# scheme, e.g. by somehow passing around the tweaked `quant_config`
self . act_fn = CustomDeepseekV2SiluAndMul (
# Use lazy binding, for `weight_scale_fp32` is accessible
# only after `process_weights_after_loading`.
weight_scale = lambda : self . gate_up_proj . weight_scale_fp32 )
# To be consumed by AscendW8A8DynamicLinearMethod.apply()
self . gate_up_proj . _ascend_quant_config = {
" output_dtype " : torch . int32 ,
" pertoken_scale " : False ,
" return_scale " : True ,
}
self . down_proj . _ascend_quant_config = {
" output_dtype " : torch . bfloat16 ,
" pertoken_scale " : True ,
" return_scale " : False ,
}
else :
raise NotImplementedError (
f " Quantization with [ { type ( quant_method ) } ] is NOT supported " )
2025-05-01 13:51:42 +08:00
def forward ( self , x ) :
gate_up , _ = self . gate_up_proj ( x )
x = self . act_fn ( gate_up )
x , _ = self . down_proj ( x )
return x
2025-04-19 17:38:18 +08:00
2025-04-07 10:56:12 +08:00
2025-04-19 17:38:18 +08:00
class CustomDeepseekV2MoE ( nn . Module ) :
top_k : int
2025-04-07 10:56:12 +08:00
def __init__ (
self ,
config : PretrainedConfig ,
quant_config : Optional [ QuantizationConfig ] = None ,
prefix : str = " " ,
) :
2025-04-19 17:38:18 +08:00
super ( ) . __init__ ( )
2025-04-07 10:56:12 +08:00
self . tp_size = get_tensor_model_parallel_world_size ( )
self . routed_scaling_factor = config . routed_scaling_factor
self . n_shared_experts = config . n_shared_experts
2025-04-19 17:38:18 +08:00
if self . tp_size > config . n_routed_experts :
raise ValueError (
f " Tensor parallel size { self . tp_size } is greater than "
f " the number of experts { config . n_routed_experts } . " )
2025-04-07 10:56:12 +08:00
if config . hidden_act != " silu " :
raise ValueError ( f " Unsupported activation: { config . hidden_act } . "
" Only silu is supported for now. " )
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
ascend_config = get_ascend_config ( )
self . torchair_graph_enabled = ascend_config . torchair_graph_config . enabled
self . enable_multistream_moe = \
2025-08-01 09:24:11 +08:00
ascend_config . torchair_graph_config . enable_multistream_moe and \
self . torchair_graph_enabled
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
2025-04-07 10:56:12 +08:00
self . gate = ReplicatedLinear ( config . hidden_size ,
config . n_routed_experts ,
bias = False ,
quant_config = None ,
prefix = f " { prefix } .gate " )
if config . topk_method == " noaux_tc " :
self . gate . e_score_correction_bias = nn . Parameter (
torch . empty ( config . n_routed_experts ) )
else :
self . gate . e_score_correction_bias = None
2025-04-19 17:38:18 +08:00
self . experts = AscendFusedMoE (
2025-04-07 10:56:12 +08:00
num_experts = config . n_routed_experts ,
top_k = config . num_experts_per_tok ,
hidden_size = config . hidden_size ,
intermediate_size = config . moe_intermediate_size ,
reduce_results = False ,
renormalize = config . norm_topk_prob ,
quant_config = quant_config ,
use_grouped_topk = True ,
num_expert_group = config . n_group ,
topk_group = config . topk_group ,
prefix = f " { prefix } .experts " ,
scoring_func = config . scoring_func ,
e_score_correction_bias = self . gate . e_score_correction_bias )
if config . n_shared_experts is not None :
2025-07-10 12:07:05 +08:00
self . all_reduce_merge = self . experts . all_reduce_merge
reduce_results = not self . all_reduce_merge
2025-04-07 10:56:12 +08:00
intermediate_size = ( config . moe_intermediate_size *
config . n_shared_experts )
2025-05-01 13:51:42 +08:00
self . shared_experts = CustomDeepseekV2MLP (
2025-04-07 10:56:12 +08:00
hidden_size = config . hidden_size ,
intermediate_size = intermediate_size ,
hidden_act = config . hidden_act ,
quant_config = quant_config ,
2025-07-10 12:07:05 +08:00
reduce_results = reduce_results ,
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
force_replicate = self . enable_multistream_moe ,
2025-04-07 10:56:12 +08:00
prefix = f " { prefix } .shared_experts " ,
)
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
else :
self . shared_experts = None # type: ignore
2025-04-19 17:38:18 +08:00
CustomDeepseekV2MoE . top_k = config . num_experts_per_tok
self . dp_size = get_dp_group ( ) . world_size
self . tp_group = get_tp_group ( ) . device_group
2025-05-15 09:19:55 +08:00
self . tp_rank = get_tp_group ( ) . rank_in_group
2025-06-07 21:11:36 +08:00
self . ep_group = get_ep_group ( )
2025-07-26 17:15:47 +08:00
self . kv_consumer = None
transfer_config = get_current_vllm_config ( ) . kv_transfer_config
if transfer_config is not None :
self . kv_consumer = transfer_config . kv_role == " kv_consumer "
2025-04-19 17:38:18 +08:00
2025-06-04 18:31:41 +08:00
self . params_dtype = torch . get_default_dtype ( )
2025-07-11 08:53:17 +08:00
self . rm_router_logits = self . experts . rm_router_logits
2025-06-04 18:31:41 +08:00
2025-06-25 19:56:49 +08:00
def forward ( self ,
hidden_states : torch . Tensor ,
attn_metadata : Optional [ AttentionMetadata ] = None ,
replace_allreduce : bool = False ) - > torch . Tensor :
2025-07-28 14:06:20 +08:00
forward_context = get_forward_context ( )
2025-05-15 09:19:55 +08:00
# when profile runs, force experts to load balanced tokens
# to avoid high memory consumption on a single rank.
2025-07-28 14:06:20 +08:00
enable_force_load_balance = forward_context . in_profile_run
is_prefill = forward_context . with_prefill
2025-07-26 17:15:47 +08:00
# If this node is kv_consumer, we force the moe always runs in decode path to make sure
# the behaviour aligned between dummy_run and normal model_execute.
if self . kv_consumer :
is_prefill = False
enable_force_load_balance = False
2025-04-19 17:38:18 +08:00
# router_logits: (num_tokens, n_experts)
2025-07-11 08:53:17 +08:00
router_logits = None
2025-07-29 23:53:19 +08:00
if not self . rm_router_logits and not self . enable_multistream_moe :
2025-07-11 08:53:17 +08:00
router_logits , _ = self . gate ( hidden_states )
2025-04-29 17:06:19 +08:00
Support multistream of shared experts in FusedMoE (#997)
Contains on #1111 for completeness.
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
Implement multi-stream parallelism for MoE layers with shared experts,
where computation of shared experts will be overlapped with expert token
dispatch and combine. Also, when multi-stream is enabled, weights of
shared experts will be force to replicate across all cards, regardless
of any tensor parallelism configurations, to avoid AllReduce operations.
With the expected overlaping being:
```
| shared gate_up | shared act | | shared down |
| dispatch | routed gate_up, act, down | combine |
```
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
No.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
Tested on 1x16 910 node, with tailored 2 layer DSKv2.
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
---------
Signed-off-by: sdmyzlp <lrwei2@petalmail.com>
2025-06-11 09:18:38 +08:00
experts_hidden_states = self . experts (
2025-06-04 18:31:41 +08:00
hidden_states = hidden_states ,
2025-04-19 17:38:18 +08:00
router_logits = router_logits ,
is_prefill = is_prefill ,
2025-05-15 09:19:55 +08:00
top_k = CustomDeepseekV2MoE . top_k ,
enable_force_load_balance = enable_force_load_balance ,
[refactor] Refactoring AscendFusedMoE (#1229)
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
This PR is used for resolved [issue
1147](https://github.com/vllm-project/vllm-ascend/issues/1147)
1. Move fused_moe code into one file `fused_moe.py`.
2. Integrate branch conditions into function `get_fused_moe_state`.
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
1. This PR has removed the env `VLLM_ENABLE_MC2`, because I think this
env is useless, we can make judgments based on the current scenario
without this env, it will only increase complexity.
2. This PR has removed the env `USING_LCCL_COM`, because this env has
already expired.
3. `additional_config.expert_tensor_parallel_size` has already expired,
and now we also use parameter `enable_expert_parallel`, consistent with
the vLLM.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
Signed-off-by: zzzzwwjj <1183291235@qq.com>
2025-06-17 17:49:03 +08:00
shared_experts = self . shared_experts ,
2025-07-11 08:53:17 +08:00
gate = self . gate ,
2025-06-25 19:56:49 +08:00
replace_allreduce = replace_allreduce )
2025-06-05 23:39:38 +08:00
[refactor] Refactoring AscendFusedMoE (#1229)
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
This PR is used for resolved [issue
1147](https://github.com/vllm-project/vllm-ascend/issues/1147)
1. Move fused_moe code into one file `fused_moe.py`.
2. Integrate branch conditions into function `get_fused_moe_state`.
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
1. This PR has removed the env `VLLM_ENABLE_MC2`, because I think this
env is useless, we can make judgments based on the current scenario
without this env, it will only increase complexity.
2. This PR has removed the env `USING_LCCL_COM`, because this env has
already expired.
3. `additional_config.expert_tensor_parallel_size` has already expired,
and now we also use parameter `enable_expert_parallel`, consistent with
the vLLM.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
Signed-off-by: zzzzwwjj <1183291235@qq.com>
2025-06-17 17:49:03 +08:00
hidden_states = (
experts_hidden_states [ 0 ] * self . routed_scaling_factor +
experts_hidden_states [ 1 ] )
2025-07-10 12:07:05 +08:00
if self . all_reduce_merge :
# When all_reduce_merge is in progress, shared_experts does not do all_reduce in mlp, but waits until shared_experts+router_experts are completed before doing all_reduce
hidden_states = tensor_model_parallel_all_reduce ( hidden_states )
[refactor] Refactoring AscendFusedMoE (#1229)
<!-- Thanks for sending a pull request!
BEFORE SUBMITTING, PLEASE READ
https://docs.vllm.ai/en/latest/contributing/overview.html
-->
### What this PR does / why we need it?
This PR is used for resolved [issue
1147](https://github.com/vllm-project/vllm-ascend/issues/1147)
1. Move fused_moe code into one file `fused_moe.py`.
2. Integrate branch conditions into function `get_fused_moe_state`.
<!--
- Please clarify what changes you are proposing. The purpose of this
section is to outline the changes and how this PR fixes the issue.
If possible, please consider writing useful notes for better and faster
reviews in your PR.
- Please clarify why the changes are needed. For instance, the use case
and bug description.
- Fixes #
-->
### Does this PR introduce _any_ user-facing change?
1. This PR has removed the env `VLLM_ENABLE_MC2`, because I think this
env is useless, we can make judgments based on the current scenario
without this env, it will only increase complexity.
2. This PR has removed the env `USING_LCCL_COM`, because this env has
already expired.
3. `additional_config.expert_tensor_parallel_size` has already expired,
and now we also use parameter `enable_expert_parallel`, consistent with
the vLLM.
<!--
Note that it means *any* user-facing change including all aspects such
as API, interface or other behavior changes.
Documentation-only updates are not considered user-facing changes.
-->
### How was this patch tested?
<!--
CI passed with new added/existing test.
If it was tested in a way different from regular unit tests, please
clarify how you tested step by step, ideally copy and paste-able, so
that other reviewers can test and check, and descendants can verify in
the future.
If tests were not added, please describe why they were not added and/or
why it was difficult to add.
-->
Signed-off-by: zzzzwwjj <1183291235@qq.com>
2025-06-17 17:49:03 +08:00
return hidden_states
2025-04-19 17:38:18 +08:00
class CustomDeepseekV2MLAAttention ( DeepseekV2MLAAttention ) :
def __init__ (
self ,
config : PretrainedConfig ,
hidden_size : int ,
num_heads : int ,
qk_nope_head_dim : int ,
qk_rope_head_dim : int ,
v_head_dim : int ,
q_lora_rank : Optional [ int ] ,
kv_lora_rank : int ,
rope_theta : float = 10000 ,
rope_scaling : Optional [ Dict [ str , Any ] ] = None ,
max_position_embeddings : int = 8192 ,
cache_config : Optional [ CacheConfig ] = None ,
quant_config : Optional [ QuantizationConfig ] = None ,
prefix : str = " " ,
) - > None :
nn . Module . __init__ ( self )
self . hidden_size = hidden_size
self . qk_nope_head_dim = qk_nope_head_dim
self . qk_rope_head_dim = qk_rope_head_dim
self . qk_head_dim = qk_nope_head_dim + qk_rope_head_dim
self . v_head_dim = v_head_dim
self . q_lora_rank = q_lora_rank
self . kv_lora_rank = kv_lora_rank
self . num_heads = num_heads
tp_size = get_tensor_model_parallel_world_size ( )
assert num_heads % tp_size == 0
self . num_local_heads = num_heads / / tp_size
self . scaling = self . qk_head_dim * * - 0.5
self . rope_theta = rope_theta
self . max_position_embeddings = max_position_embeddings
2025-06-25 19:56:49 +08:00
self . prefix = prefix
self . debug_layer_idx = int ( self . prefix . split ( " . " ) [ - 2 ] )
ascend_config = get_ascend_config ( )
self . torchair_graph_enabled = ascend_config . torchair_graph_config . enabled
self . enable_multistream_mla = \
ascend_config . torchair_graph_config . enable_multistream_mla
2025-04-19 17:38:18 +08:00
if self . q_lora_rank is not None :
self . q_a_proj = ReplicatedLinear ( self . hidden_size ,
self . q_lora_rank ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .q_a_proj " )
self . q_a_layernorm = RMSNorm ( self . q_lora_rank ,
eps = config . rms_norm_eps )
self . q_b_proj = ColumnParallelLinear ( q_lora_rank ,
self . num_heads *
self . qk_head_dim ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .q_b_proj " )
else :
self . q_proj = ColumnParallelLinear ( self . hidden_size ,
self . num_heads *
self . qk_head_dim ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .q_proj " )
self . kv_a_proj_with_mqa = ReplicatedLinear (
self . hidden_size ,
self . kv_lora_rank + self . qk_rope_head_dim ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .kv_a_proj_with_mqa " )
self . kv_a_layernorm = RMSNorm ( self . kv_lora_rank ,
eps = config . rms_norm_eps )
self . kv_b_proj = ColumnParallelLinear (
self . kv_lora_rank ,
self . num_heads * ( self . qk_nope_head_dim + self . v_head_dim ) ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .kv_b_proj " )
2025-06-25 19:56:49 +08:00
if ( config . n_routed_experts is not None
and self . debug_layer_idx > = config . first_k_dense_replace
and self . debug_layer_idx % config . moe_layer_freq == 0 and
ascend_config . torchair_graph_config . enable_multistream_moe ) :
self . o_proj = CustomDeepseekV2RowParallelLinearReplaceAllreduce (
self . num_heads * self . v_head_dim ,
self . hidden_size ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .o_proj " )
else :
self . o_proj = CustomDeepseekV2RowParallelLinear (
self . num_heads * self . v_head_dim ,
self . hidden_size ,
bias = False ,
quant_config = quant_config ,
prefix = f " { prefix } .o_proj " )
2025-04-19 17:38:18 +08:00
if rope_scaling :
rope_scaling [ " rope_type " ] = ' deepseek_yarn '
self . rotary_emb = get_rope ( qk_rope_head_dim ,
rotary_dim = qk_rope_head_dim ,
max_position = max_position_embeddings ,
base = rope_theta ,
rope_scaling = rope_scaling ,
is_neox_style = False )
if rope_scaling :
mscale_all_dim = rope_scaling . get ( " mscale_all_dim " , False )
scaling_factor = rope_scaling [ " factor " ]
mscale = yarn_get_mscale ( scaling_factor , float ( mscale_all_dim ) )
self . scaling = self . scaling * mscale * mscale
# In the MLA backend, kv_cache includes both k_c and
# pe (i.e. decoupled position embeddings). In particular,
# the concat_and_cache_mla op requires
# k_c.size(1) + k_pe.size(1) == kv_cache.size(2)
# i.e.
# kv_lora_rank + qk_rope_head_dim == head_size
self . mla_attn = Attention (
num_heads = self . num_local_heads ,
head_size = self . kv_lora_rank + self . qk_rope_head_dim ,
scale = self . scaling ,
num_kv_heads = 1 ,
cache_config = cache_config ,
quant_config = quant_config ,
prefix = f " { prefix } .attn " ,
use_mla = True ,
# MLA Args
q_lora_rank = self . q_lora_rank ,
kv_lora_rank = self . kv_lora_rank ,
qk_nope_head_dim = self . qk_nope_head_dim ,
qk_rope_head_dim = self . qk_rope_head_dim ,
qk_head_dim = self . qk_head_dim ,
v_head_dim = self . v_head_dim ,
rotary_emb = self . rotary_emb ,
q_proj = self . q_proj if self . q_lora_rank is None else self . q_b_proj ,
kv_a_proj_with_mqa = self . kv_a_proj_with_mqa ,
kv_a_layernorm = self . kv_a_layernorm ,
kv_b_proj = self . kv_b_proj ,
o_proj = self . o_proj ,
)
2025-04-23 16:23:25 +08:00
def forward (
self ,
positions : torch . Tensor ,
hidden_states : torch . Tensor ,
kv_cache : Optional [ torch . Tensor ] = None ,
attn_metadata : Optional [ AttentionMetadata ] = None ) - > torch . Tensor :
2025-07-28 14:06:20 +08:00
forward_context = get_forward_context ( )
2025-06-26 09:32:07 +08:00
enable_multistream_mla = ( self . enable_multistream_mla
and attn_metadata is not None
2025-07-28 14:06:20 +08:00
and not forward_context . with_prefill
2025-06-26 09:32:07 +08:00
and attn_metadata . num_decodes > 0 )
forward_kwargs = { " enable_multistream_mla " : enable_multistream_mla }
2025-04-19 17:38:18 +08:00
if self . q_lora_rank is not None :
2025-07-11 08:51:17 +08:00
npu_prefetch ( self . q_a_proj . weight ,
hidden_states ,
enabled = enable_multistream_mla )
2025-04-19 17:38:18 +08:00
ckq = self . q_a_proj ( hidden_states ) [ 0 ]
2025-07-11 08:51:17 +08:00
hidden_states_or_q_c = self . q_a_layernorm ( ckq )
forward_kwargs [ ' ckq ' ] = ckq
2025-04-19 17:38:18 +08:00
else :
hidden_states_or_q_c = hidden_states
2025-06-05 16:28:01 +08:00
if self . torchair_graph_enabled :
2025-07-15 11:52:16 +08:00
output_shape = hidden_states . shape
output = torch . empty ( output_shape ,
dtype = hidden_states_or_q_c . dtype ,
device = hidden_states_or_q_c . device )
forward_kwargs [ ' output ' ] = output
2025-05-12 19:14:07 +08:00
output = self . mla_attn . impl . forward ( self . mla_attn ,
hidden_states_or_q_c ,
hidden_states , None , kv_cache ,
attn_metadata ,
* * forward_kwargs )
2025-07-15 11:52:16 +08:00
output = output . view ( - 1 , output_shape [ - 1 ] )
2025-05-12 19:14:07 +08:00
return output
2025-04-19 17:38:18 +08:00
else :
2025-04-23 16:23:25 +08:00
kv_c , k_pe = self . kv_a_proj_with_mqa ( hidden_states ) [ 0 ] . split (
[ self . kv_lora_rank , self . qk_rope_head_dim ] , dim = - 1 )
kv_c_normed = self . kv_a_layernorm ( kv_c . contiguous ( ) )
return self . mla_attn ( hidden_states_or_q_c ,
kv_c_normed ,
k_pe ,
output_shape = hidden_states . shape )
2025-04-19 17:38:18 +08:00
2025-04-07 10:56:12 +08:00
class CustomDeepseekV2DecoderLayer ( DeepseekV2DecoderLayer ) :
def __init__ (
self ,
config : PretrainedConfig ,
prefix : str ,
model_config : ModelConfig ,
cache_config : Optional [ CacheConfig ] = None ,
quant_config : Optional [ QuantizationConfig ] = None ,
) - > None :
nn . Module . __init__ ( self )
self . hidden_size = config . hidden_size
rope_theta = getattr ( config , " rope_theta " , 10000 )
rope_scaling = getattr ( config , " rope_scaling " , None )
max_position_embeddings = getattr ( config , " max_position_embeddings " ,
8192 )
# DecoderLayers are created with `make_layers` which passes the prefix
# with the layer's index.
layer_idx = int ( prefix . split ( sep = ' . ' ) [ - 1 ] )
2025-04-17 11:56:05 +08:00
self . layer_idx = layer_idx
2025-06-25 19:56:49 +08:00
self . layers = config . num_hidden_layers
self . tp_size = get_tensor_model_parallel_world_size ( )
self . tp_rank = get_tp_group ( ) . rank_in_group
ascend_config = get_ascend_config ( )
2025-04-19 17:38:18 +08:00
# TODO: enable mla in vllm-ascend
2025-04-07 10:56:12 +08:00
if model_config . use_mla :
2025-04-19 17:38:18 +08:00
attn_cls = CustomDeepseekV2MLAAttention
2025-04-07 10:56:12 +08:00
else :
attn_cls = DeepseekV2Attention
self . self_attn = attn_cls (
config = config ,
hidden_size = self . hidden_size ,
num_heads = config . num_attention_heads ,
qk_nope_head_dim = config . qk_nope_head_dim ,
qk_rope_head_dim = config . qk_rope_head_dim ,
v_head_dim = config . v_head_dim ,
q_lora_rank = config . q_lora_rank
if hasattr ( config , " q_lora_rank " ) else None ,
kv_lora_rank = config . kv_lora_rank ,
rope_theta = rope_theta ,
rope_scaling = rope_scaling ,
max_position_embeddings = max_position_embeddings ,
cache_config = cache_config ,
quant_config = quant_config ,
prefix = f " { prefix } .self_attn " ,
)
if ( config . n_routed_experts is not None
and layer_idx > = config . first_k_dense_replace
and layer_idx % config . moe_layer_freq == 0 ) :
self . mlp = CustomDeepseekV2MoE (
config = config ,
quant_config = quant_config ,
prefix = f " { prefix } .mlp " ,
)
2025-06-25 19:56:49 +08:00
self . mla_moe_communication = ascend_config . torchair_graph_config . enable_multistream_moe \
2025-07-15 11:52:16 +08:00
and model_config . use_mla and self . tp_size > 1
2025-04-07 10:56:12 +08:00
else :
2025-05-01 13:51:42 +08:00
self . mlp = CustomDeepseekV2MLP (
2025-04-07 10:56:12 +08:00
hidden_size = config . hidden_size ,
intermediate_size = config . intermediate_size ,
hidden_act = config . hidden_act ,
quant_config = quant_config ,
prefix = f " { prefix } .mlp " ,
)
2025-06-25 19:56:49 +08:00
self . mla_moe_communication = False
2025-04-07 10:56:12 +08:00
self . input_layernorm = RMSNorm ( config . hidden_size ,
eps = config . rms_norm_eps )
self . post_attention_layernorm = RMSNorm ( config . hidden_size ,
eps = config . rms_norm_eps )
2025-04-17 11:56:05 +08:00
self . routed_scaling_factor = config . routed_scaling_factor
2025-06-25 19:56:49 +08:00
self . first_k_dense_replace = config . first_k_dense_replace
2025-04-07 10:56:12 +08:00
2025-04-23 16:23:25 +08:00
def forward (
self ,
positions : torch . Tensor ,
hidden_states : torch . Tensor ,
residual : Optional [ torch . Tensor ] ,
kv_cache : Optional [ torch . Tensor ] = None ,
attn_metadata : Optional [ AttentionMetadata ] = None ,
2025-06-25 19:56:49 +08:00
replace_allreduce : bool = False ,
2025-04-23 16:23:25 +08:00
) - > torch . Tensor :
# Self Attention
2025-06-25 19:56:49 +08:00
if attn_metadata is not None and attn_metadata . num_decodes > 0 :
mla_moe_communication = self . mla_moe_communication and replace_allreduce
else :
mla_moe_communication = False
2025-04-23 16:23:25 +08:00
if residual is None :
residual = hidden_states
hidden_states = self . input_layernorm ( hidden_states )
else :
2025-05-29 11:48:26 +08:00
previous_hidden_states , previous_residual = hidden_states , residual
2025-04-23 16:23:25 +08:00
hidden_states , residual = self . input_layernorm (
hidden_states , residual )
2025-05-29 11:48:26 +08:00
# Dispose hidden_states and residual from the previous layer
# to save npu memory because they're no longer used.
dispose_tensor ( previous_hidden_states )
dispose_tensor ( previous_residual )
2025-06-25 19:56:49 +08:00
if mla_moe_communication and self . layer_idx > self . first_k_dense_replace :
hidden_states = tensor_model_parallel_all_gather ( hidden_states ,
dim = 0 )
2025-05-29 11:48:26 +08:00
2025-04-23 16:23:25 +08:00
hidden_states = self . self_attn (
positions = positions ,
hidden_states = hidden_states ,
kv_cache = kv_cache ,
attn_metadata = attn_metadata ,
)
2025-06-25 19:56:49 +08:00
if mla_moe_communication and residual . shape [ 0 ] != hidden_states . shape [
0 ] :
chunk_hidden_states = torch . tensor_split ( residual ,
self . tp_size ,
dim = 0 )
residual = chunk_hidden_states [ self . tp_rank ]
2025-04-23 16:23:25 +08:00
if hidden_states . dtype == torch . float16 :
# Fix FP16 overflow
# We scale both hidden_states and residual before
# rmsnorm, and rmsnorm result would not affect by scale.
hidden_states * = 1. / self . routed_scaling_factor
if self . layer_idx == 0 :
# The residual is shared by all layers, we only scale it on
# first layer.
residual * = 1. / self . routed_scaling_factor
# Fully Connected
hidden_states , residual = self . post_attention_layernorm (
hidden_states , residual )
2025-05-31 06:03:03 +08:00
if isinstance ( self . mlp , CustomDeepseekV2MoE ) :
2025-06-25 19:56:49 +08:00
hidden_states = self . mlp ( hidden_states ,
attn_metadata ,
replace_allreduce = mla_moe_communication )
2025-05-31 06:03:03 +08:00
else :
hidden_states = self . mlp ( hidden_states )
2025-04-23 16:23:25 +08:00
2025-05-01 13:51:42 +08:00
if isinstance (
self . mlp ,
CustomDeepseekV2MLP ) and hidden_states . dtype == torch . float16 :
2025-04-23 16:23:25 +08:00
# Fix FP16 overflow
# Scaling the DeepseekV2MLP output, it is the input of
# input_layernorm of next decoder layer.
# The scaling of DeepseekV2MOE output would be done in the forward
# of DeepseekV2MOE
hidden_states * = 1. / self . routed_scaling_factor
2025-06-25 19:56:49 +08:00
if mla_moe_communication and self . layer_idx == self . layers - 1 :
hidden_states = tensor_model_parallel_all_gather ( hidden_states ,
dim = 0 )
residual = tensor_model_parallel_all_gather ( residual , dim = 0 )
2025-04-23 16:23:25 +08:00
return hidden_states , residual
2025-04-07 10:56:12 +08:00
class CustomDeepseekV2Model ( nn . Module ) :
fall_back_to_pt_during_load = False
def __init__ ( self , * , vllm_config : VllmConfig , prefix : str = " " ) :
super ( ) . __init__ ( )
config = vllm_config . model_config . hf_config
model_config = vllm_config . model_config
cache_config = vllm_config . cache_config
quant_config = vllm_config . quant_config
2025-04-19 17:38:18 +08:00
self . padding_idx = config . pad_token_id
2025-04-07 10:56:12 +08:00
self . vocab_size = config . vocab_size
2025-06-25 19:56:49 +08:00
self . tp_size = get_tensor_model_parallel_world_size ( )
2025-04-07 10:56:12 +08:00
if get_pp_group ( ) . is_first_rank :
self . embed_tokens = VocabParallelEmbedding (
config . vocab_size ,
config . hidden_size ,
quant_config = quant_config ,
prefix = f " { prefix } .embed_tokens " )
else :
self . embed_tokens = PPMissingLayer ( )
self . start_layer , self . end_layer , self . layers = make_layers (
config . num_hidden_layers ,
lambda prefix : CustomDeepseekV2DecoderLayer (
config ,
prefix ,
model_config = model_config ,
cache_config = cache_config ,
quant_config = quant_config ,
) ,
prefix = f " { prefix } .layers " )
if get_pp_group ( ) . is_last_rank :
self . norm = RMSNorm ( config . hidden_size , eps = config . rms_norm_eps )
else :
self . norm = PPMissingLayer ( )
self . make_empty_intermediate_tensors = (
make_empty_intermediate_tensors_factory (
[ " hidden_states " , " residual " ] , config . hidden_size ) )
def get_input_embeddings ( self , input_ids : torch . Tensor ) - > torch . Tensor :
return self . embed_tokens ( input_ids )
def forward (
self ,
input_ids : torch . Tensor ,
positions : torch . Tensor ,
2025-04-23 16:23:25 +08:00
kv_caches : Optional [ List [ torch . Tensor ] ] = None ,
attn_metadata : Optional [ AttentionMetadata ] = None ,
intermediate_tensors : Optional [ IntermediateTensors ] = None ,
2025-04-07 10:56:12 +08:00
inputs_embeds : Optional [ torch . Tensor ] = None ,
) - > Union [ torch . Tensor , IntermediateTensors ] :
if get_pp_group ( ) . is_first_rank :
if inputs_embeds is not None :
hidden_states = inputs_embeds
else :
hidden_states = self . get_input_embeddings ( input_ids )
residual = None
else :
assert intermediate_tensors is not None
hidden_states = intermediate_tensors [ " hidden_states " ]
residual = intermediate_tensors [ " residual " ]
2025-06-25 19:56:49 +08:00
replace_allreduce = hidden_states . shape [ 0 ] % self . tp_size == 0
2025-04-23 16:23:25 +08:00
for i in range ( self . start_layer , self . end_layer ) :
layer = self . layers [ i ]
hidden_states , residual = layer (
2025-06-25 19:56:49 +08:00
positions ,
hidden_states ,
residual ,
2025-04-23 16:23:25 +08:00
kv_caches [ i -
self . start_layer ] if kv_caches is not None else None ,
2025-06-25 19:56:49 +08:00
attn_metadata ,
replace_allreduce = replace_allreduce )
2025-04-07 10:56:12 +08:00
if not get_pp_group ( ) . is_last_rank :
return IntermediateTensors ( {
" hidden_states " : hidden_states ,
" residual " : residual
} )
hidden_states , _ = self . norm ( hidden_states , residual )
return hidden_states
class CustomDeepseekV2ForCausalLM ( DeepseekV2ForCausalLM ) :
# add `packed_modules_mapping` in `DeepseekV2ForCausalLM` to support weight merging
packed_modules_mapping = {
" gate_up_proj " : [ " gate_proj " , " up_proj " ] ,
" experts " :
[ " experts.0.gate_proj " , " experts.0.up_proj " , " experts.0.down_proj " ]
}
def __init__ ( self , * , vllm_config : VllmConfig , prefix : str = " " ) :
nn . Module . __init__ ( self )
config = vllm_config . model_config . hf_config
quant_config = vllm_config . quant_config
self . config = config
self . quant_config = quant_config
self . model = CustomDeepseekV2Model ( vllm_config = vllm_config ,
prefix = maybe_prefix (
prefix , " model " ) )
2025-04-17 11:56:05 +08:00
if get_pp_group ( ) . is_last_rank :
self . lm_head = ParallelLMHead ( config . vocab_size ,
config . hidden_size ,
2025-07-29 18:51:57 +08:00
quant_config = quant_config ,
prefix = maybe_prefix (
prefix , " lm_head " ) )
2025-04-17 11:56:05 +08:00
else :
self . lm_head = PPMissingLayer ( )
2025-04-07 10:56:12 +08:00
self . logits_processor = LogitsProcessor ( config . vocab_size )
self . sampler = get_sampler ( )
self . make_empty_intermediate_tensors = (
self . model . make_empty_intermediate_tensors )
2025-06-28 00:12:43 +08:00
# NOTE: This `load_weights` is mainly copied from
# https://github.com/vllm-project/vllm/commit/07b8fae219b1fff51ef115c38c44b51395be5bb5
# to fix CI, and it is different from the implementation in main
# TODO: support eplb style load_weights
def load_weights ( self , weights : Iterable [ tuple [ str ,
torch . Tensor ] ] ) - > set [ str ] :
""" """
stacked_params_mapping = [
# (param_name, shard_name, shard_id)
( " gate_up_proj " , " gate_proj " , 0 ) ,
( " gate_up_proj " , " up_proj " , 1 ) ,
]
# Params for weights, fp8 weight scales, fp8 activation scales
# (param_name, weight_name, expert_id, shard_id)
expert_params_mapping = AscendFusedMoE . make_expert_params_mapping (
ckpt_gate_proj_name = " gate_proj " ,
ckpt_down_proj_name = " down_proj " ,
ckpt_up_proj_name = " up_proj " ,
num_experts = self . config . n_routed_experts )
params_dict = dict ( self . named_parameters ( ) )
loaded_params : set [ str ] = set ( )
for name , loaded_weight in weights :
if " rotary_emb.inv_freq " in name :
continue
spec_layer = get_spec_layer_idx_from_weight_name ( self . config , name )
if spec_layer is not None :
continue # skip spec decode layers for main model
for ( param_name , weight_name , shard_id ) in stacked_params_mapping :
# Skip non-stacked layers and experts (experts handled below).
if weight_name not in name :
continue
# We have mlp.experts[0].gate_proj in the checkpoint.
# Since we handle the experts below in expert_params_mapping,
# we need to skip here BEFORE we update the name, otherwise
# name will be updated to mlp.experts[0].gate_up_proj, which
# will then be updated below in expert_params_mapping
# for mlp.experts[0].gate_gate_up_proj, which breaks load.
if ( ( " mlp.experts. " in name ) and name not in params_dict ) :
continue
name = name . replace ( weight_name , param_name )
# Skip loading extra bias for GPTQ models.
if name . endswith ( " .bias " ) and name not in params_dict :
continue
if is_pp_missing_parameter ( name , self ) :
continue
param = params_dict [ name ]
weight_loader = param . weight_loader
weight_loader ( param , loaded_weight , shard_id )
break
else :
for mapping in expert_params_mapping :
param_name , weight_name , expert_id , shard_id = mapping
if weight_name not in name :
continue
name = name . replace ( weight_name , param_name )
if is_pp_missing_parameter ( name , self ) :
continue
param = params_dict [ name ]
weight_loader = param . weight_loader
2025-07-09 08:52:24 +08:00
weight_loader ( param ,
loaded_weight ,
name ,
shard_id = shard_id ,
expert_id = expert_id ,
return_success = False )
2025-06-28 00:12:43 +08:00
break
else :
# Skip loading extra bias for GPTQ models.
if name . endswith ( " .bias " ) and name not in params_dict :
continue
# Remapping the name of FP8 kv-scale.
name = maybe_remap_kv_scale_name ( name , params_dict )
if name is None :
continue
if is_pp_missing_parameter ( name , self ) :
continue
param = params_dict [ name ]
weight_loader = getattr ( param , " weight_loader " ,
default_weight_loader )
weight_loader ( param , loaded_weight )
loaded_params . add ( name )
return loaded_params
2025-04-23 16:23:25 +08:00
def forward (
self ,
input_ids : torch . Tensor ,
positions : torch . Tensor ,
kv_caches : Optional [ List [ torch . Tensor ] ] = None ,
attn_metadata : Optional [ AttentionMetadata ] = None ,
intermediate_tensors : Optional [ IntermediateTensors ] = None ,
inputs_embeds : Optional [ torch . Tensor ] = None ,
) - > Union [ torch . Tensor , IntermediateTensors ] :
hidden_states = self . model ( input_ids , positions , kv_caches ,
attn_metadata , intermediate_tensors ,
inputs_embeds )
return hidden_states