processor.py 7.08 KB
Newer Older
1
import time
2
from typing import Mapping, Optional, Union
3

4
from vllm.config import CacheConfig, LoRAConfig, ModelConfig
5
6
7
from vllm.inputs import (INPUT_REGISTRY, InputRegistry, ProcessorInputs,
                         PromptType, SingletonInputsAdapter)
from vllm.inputs.parse import is_encoder_decoder_inputs
8
9
from vllm.inputs.preprocess import InputPreprocessor
from vllm.lora.request import LoRARequest
10
11
from vllm.multimodal import (MULTIMODAL_REGISTRY, MultiModalKwargs,
                             MultiModalRegistry)
12
13
14
from vllm.pooling_params import PoolingParams
from vllm.prompt_adapter.request import PromptAdapterRequest
from vllm.sampling_params import SamplingParams
15
from vllm.transformers_utils.tokenizer_group import BaseTokenizerGroup
16
from vllm.v1.engine import EngineCoreRequest
17
from vllm.v1.engine.mm_input_mapper import MMHasher, MMInputMapperClient
18
19
20
21
22
23
24


class Processor:

    def __init__(
        self,
        model_config: ModelConfig,
25
        cache_config: CacheConfig,
26
        lora_config: Optional[LoRAConfig],
27
        tokenizer: BaseTokenizerGroup,
28
        input_registry: InputRegistry = INPUT_REGISTRY,
29
        mm_registry: MultiModalRegistry = MULTIMODAL_REGISTRY,
30
31
32
33
34
35
    ):

        self.model_config = model_config
        self.lora_config = lora_config
        self.tokenizer = tokenizer

36
37
        self.generation_config_fields = model_config.try_get_generation_config(
        )
38
        self.input_preprocessor = InputPreprocessor(model_config,
39
40
                                                    self.tokenizer,
                                                    mm_registry)
41
42
43
        self.input_processor = input_registry.create_input_processor(
            model_config)

44
        # Multi-modal (huggingface) input mapper
45
46
47
        self.mm_input_mapper_client = MMInputMapperClient(model_config)

        # Multi-modal hasher (for images)
48
        self.use_hash = (not model_config.disable_mm_preprocessor_cache) or \
49
50
            cache_config.enable_prefix_caching
        self.mm_hasher = MMHasher()
51

52
53
54
55
56
    def process_inputs(
        self,
        request_id: str,
        prompt: PromptType,
        params: Union[SamplingParams, PoolingParams],
57
        arrival_time: Optional[float] = None,
58
59
60
61
        lora_request: Optional[LoRARequest] = None,
        trace_headers: Optional[Mapping[str, str]] = None,
        prompt_adapter_request: Optional[PromptAdapterRequest] = None,
        priority: int = 0,
62
    ) -> EngineCoreRequest:
63

64
        # TODO(woosuk): Support pooling models.
65
66
67
68
69
70
71
72
73
74
75
        # TODO(woosuk): Check max_logprobs
        # TODO(woosuk): Support encoder-decoder models.

        if lora_request is not None and not self.lora_config:
            raise ValueError(f"Got lora_request {lora_request} but LoRA is "
                             "not enabled!")
        if arrival_time is None:
            arrival_time = time.time()
        assert priority == 0, "vLLM V1 does not support priority at the moment."
        assert trace_headers is None, "vLLM V1 does not support tracing yet."

76
77
        # Compute MM hashes (if enabled)
        mm_hashes = None
78
        if self.use_hash:
79
            mm_hashes = self.mm_hasher.hash_prompt_mm_data(prompt)
80

81
82
83
84
85
86
87
88
89
90
91
        # Process inputs.
        preprocessed_inputs = self.input_preprocessor.preprocess(
            prompt,
            request_id=request_id,
            lora_request=lora_request,
            prompt_adapter_request=prompt_adapter_request,
        )
        processed_inputs = self.input_processor(preprocessed_inputs)
        self._validate_model_inputs(processed_inputs)
        eos_token_id = self.input_preprocessor.get_eos_token_id(lora_request)

92
93
94
95
96
97
98
99
100
101
102
103
104
        if is_encoder_decoder_inputs(processed_inputs):
            decoder_inputs = SingletonInputsAdapter(
                processed_inputs["decoder"])
            encoder_inputs = SingletonInputsAdapter(
                processed_inputs["encoder"])
        else:
            decoder_inputs = SingletonInputsAdapter(processed_inputs)
            encoder_inputs = None

        # TODO: Impl encoder-decoder
        if encoder_inputs is not None:
            raise NotImplementedError

105
106
107
108
109
110
        assert isinstance(params, SamplingParams)
        # TODO: can we avoid cloning here in multiproc case
        sampling_params = params.clone()
        sampling_params.update_from_generation_config(
            self.generation_config_fields, eos_token_id)

111
112
        # For merged preprocessor, mm_data is already mm_inputs
        precomputed_mm_inputs = None
113
114
115
116
117
118
119
120
121
122
123
        decoder_mm_data = decoder_inputs.multi_modal_data
        if isinstance(decoder_mm_data, MultiModalKwargs):
            # The output of merged multi-modal processor (`decoder_mm_data`)
            # contains the kwargs for all items from all modalities.
            # This code separates them so that there is one set of kwargs
            # per item per modality.
            precomputed_mm_inputs = [
                MultiModalKwargs.from_items([item])
                for modality in decoder_mm_data.modalities
                for item in decoder_mm_data.get_items(modality)
            ]
124
125
126

        # Apply MM mapper
        mm_inputs = None
127
        if len(decoder_mm_data) > 0:
128
            mm_inputs = self.mm_input_mapper_client.process_inputs(
129
130
131
132
133
                decoder_mm_data,
                mm_hashes,
                decoder_inputs.mm_processor_kwargs,
                precomputed_mm_inputs,
            )
134

135
        return EngineCoreRequest(
136
137
138
            request_id,
            decoder_inputs.prompt,
            decoder_inputs.prompt_token_ids,
139
            mm_inputs,
140
            mm_hashes,
141
142
143
144
145
146
            decoder_inputs.multi_modal_placeholders,
            sampling_params,
            eos_token_id,
            arrival_time,
            lora_request,
        )
147

148
149
150
151
152
153
154
155
156
157
158
    def _validate_model_inputs(self, inputs: ProcessorInputs):
        if is_encoder_decoder_inputs(inputs):
            # For encoder-decoder multimodal models, the max_prompt_len
            # restricts the decoder prompt length
            prompt_inputs = inputs["decoder" if self.model_config.
                                   is_multimodal_model else "encoder"]
        else:
            prompt_inputs = inputs

        prompt_ids = SingletonInputsAdapter(prompt_inputs).prompt_token_ids

159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
        if prompt_ids is None or len(prompt_ids) == 0:
            raise ValueError("Prompt cannot be empty")

        if self.model_config.is_multimodal_model:
            max_prompt_len = self.model_config.max_model_len

            if len(prompt_ids) > max_prompt_len:
                raise ValueError(
                    f"The prompt (total length {len(prompt_ids)}) is too long "
                    f"to fit into the model (context length {max_prompt_len}). "
                    "Make sure that `max_model_len` is no smaller than the "
                    "number of text tokens plus multimodal tokens. For image "
                    "inputs, the number of image tokens depends on the number "
                    "of images, and possibly their aspect ratios as well.")

174
175
176
            # TODO: Find out how many placeholder tokens are there so we can
            # check that chunked prefill does not truncate them
            # max_batch_len = self.scheduler_config.max_num_batched_tokens