vocal_parallel_embedding.py 4.83 KB
Newer Older
Jee Jee Li's avatar
Jee Jee Li committed
1
2
3
4
5
6
7
8
# SPDX-License-Identifier: Apache-2.0
# SPDX-FileCopyrightText: Copyright contributors to the vLLM project

import torch
import torch.nn as nn
import torch.nn.functional as F
from transformers import PretrainedConfig

9
from vllm.config.lora import LoRAConfig
10
from vllm.model_executor.layers.vocab_parallel_embedding import VocabParallelEmbedding
Jee Jee Li's avatar
Jee Jee Li committed
11
12
13
14
15
16
17
18
19
from vllm.platforms import current_platform

from .base import BaseLayerWithLoRA


class VocabParallelEmbeddingWithLoRA(BaseLayerWithLoRA):
    def __init__(self, base_layer: VocabParallelEmbedding) -> None:
        super().__init__()
        self.base_layer = base_layer
20
21
        self.embeddings_slice: tuple[int, int] | None
        self.embeddings_weights: torch.Tensor | None
Jee Jee Li's avatar
Jee Jee Li committed
22
23

    def create_lora_weights(
24
25
26
        self,
        max_loras: int,
        lora_config: LoRAConfig,
27
        model_config: PretrainedConfig | None = None,
28
    ) -> None:
Jee Jee Li's avatar
Jee Jee Li committed
29
30
31
        if self.base_layer.num_added_embeddings_per_partition > 0:
            # We can start adding lora weights
            self.embeddings_weights = self.base_layer.weight.data[
32
                self.base_layer.num_org_embeddings_per_partition : self.base_layer.num_org_embeddings_per_partition  # noqa: E501
33
34
                + self.base_layer.num_added_embeddings_per_partition
            ]
Jee Jee Li's avatar
Jee Jee Li committed
35
            self.embeddings_slice = (
36
37
38
39
40
                self.base_layer.shard_indices.added_vocab_start_index
                - self.base_layer.org_vocab_size,
                self.base_layer.shard_indices.added_vocab_end_index
                - self.base_layer.org_vocab_size,
            )
Jee Jee Li's avatar
Jee Jee Li committed
41
            self.base_layer.weight.data[
42
43
                self.base_layer.num_org_embeddings_per_partition :
            ].fill_(0)
Jee Jee Li's avatar
Jee Jee Li committed
44
45
46
47
48
49
50
        else:
            self.embeddings_slice = None
            self.embeddings_weights = None

        self.lora_a_stacked = torch.zeros(
            (
                max_loras,
51
                self.base_layer.org_vocab_size,
Jee Jee Li's avatar
Jee Jee Li committed
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
                lora_config.max_lora_rank,
            ),
            dtype=lora_config.lora_dtype,
            device=self.base_layer.weight.device,
        )
        self.lora_b_stacked = torch.zeros(
            (
                max_loras,
                1,
                self.base_layer.embedding_dim,
                lora_config.max_lora_rank,
            ),
            dtype=lora_config.lora_dtype,
            device=self.base_layer.weight.device,
        )
        self.lora_a_stacked_2d = self.lora_a_stacked.view(
            self.lora_a_stacked.shape[0] * self.lora_a_stacked.shape[1],
            self.lora_a_stacked.shape[2],
        )

    def reset_lora(self, index: int):
        self.lora_a_stacked[index] = 0
        self.lora_b_stacked[index] = 0

    def set_lora(
        self,
        index: int,
79
80
        lora_a: torch.Tensor | list[torch.Tensor],
        lora_b: torch.Tensor | list[torch.Tensor],
Jee Jee Li's avatar
Jee Jee Li committed
81
    ):
82
83
        assert isinstance(lora_a, torch.Tensor)
        assert isinstance(lora_b, torch.Tensor)
Jee Jee Li's avatar
Jee Jee Li committed
84
        self.reset_lora(index)
85
86
        # NOTE self.lora_a_stacked is row-major, and lora_a is col-major,
        # so we need transpose here
87

88
89
90
91
92
93
        self.lora_a_stacked[index, : lora_a.shape[1], : lora_a.shape[0]].copy_(
            lora_a.T, non_blocking=True
        )
        self.lora_b_stacked[index, 0, : lora_b.shape[0], : lora_b.shape[1]].copy_(
            lora_b, non_blocking=True
        )
Jee Jee Li's avatar
Jee Jee Li committed
94
95
96
97
98
99
100
101
102
103
104

    def forward(self, x: torch.Tensor) -> torch.Tensor:
        # NB: Don't use torch.narrow here. torch.narrow triggers some
        # Dynamic Shape specialization in torch.compile
        num_tokens = x.shape[0]
        indices_1 = self.punica_wrapper._embeddings_indices[1][:num_tokens]

        full_lora_a_embeddings = F.embedding(
            x + indices_1,
            self.lora_a_stacked_2d,
        )
105
        full_output = self.base_layer.forward(x)
Jee Jee Li's avatar
Jee Jee Li committed
106
107
108
109

        full_output_org = full_output
        if full_output.ndim == 3:
            full_output = full_output.view(
110
111
                full_output.shape[0] * full_output.shape[1], -1
            )
Jee Jee Li's avatar
Jee Jee Li committed
112
113
        if full_lora_a_embeddings.ndim == 3:
            full_lora_a_embeddings = full_lora_a_embeddings.view(
114
                full_lora_a_embeddings.shape[0] * full_lora_a_embeddings.shape[1],
Jee Jee Li's avatar
Jee Jee Li committed
115
116
117
                -1,
            )

118
        lora_output: torch.Tensor | None = self.punica_wrapper.add_lora_embedding(
119
120
            full_output, full_lora_a_embeddings, self.lora_b_stacked, add_input=True
        )
Jee Jee Li's avatar
Jee Jee Li committed
121
122
123
124
125
126
127
128
129
130
131
132

        if not current_platform.can_update_inplace():
            full_output = lora_output

        return full_output.view_as(full_output_org)

    @classmethod
    def can_replace_layer(
        cls,
        source_layer: nn.Module,
        lora_config: LoRAConfig,
        packed_modules_list: list,
133
        model_config: PretrainedConfig | None = None,
Jee Jee Li's avatar
Jee Jee Li committed
134
135
136
137
138
139
    ) -> bool:
        return type(source_layer) is VocabParallelEmbedding

    @property
    def weight(self):
        return self.base_layer.weight