outputs.py 3.69 KB
Newer Older
1
# SPDX-License-Identifier: Apache-2.0
2
# SPDX-FileCopyrightText: Copyright contributors to the vLLM project
3

4
from dataclasses import dataclass
5
from typing import NamedTuple, Optional
6
7
8
9

import torch


10
class LogprobsLists(NamedTuple):
11

12
    # [num_reqs, max_num_logprobs + 1]
13
    logprob_token_ids: list[list[int]]
14
    # [num_reqs, max_num_logprobs + 1]
15
    logprobs: list[list[float]]
16
    # [num_reqs]
17
    sampled_token_ranks: list[int]
18
19
20
21
22
23
24
25
26
27

    def slice(self, start: int, end: int):
        return LogprobsLists(
            self.logprob_token_ids[start:end],
            self.logprobs[start:end],
            self.sampled_token_ranks[start:end],
        )


class LogprobsTensors(NamedTuple):
28
29

    # [num_reqs, max_num_logprobs + 1]
30
    logprob_token_ids: torch.Tensor
31
    # [num_reqs, max_num_logprobs + 1]
32
33
34
    logprobs: torch.Tensor
    # [num_reqs]
    selected_token_ranks: torch.Tensor
35

36
37
38
39
40
41
42
    def tolists(self):
        return LogprobsLists(
            self.logprob_token_ids.tolist(),
            self.logprobs.tolist(),
            self.selected_token_ranks.tolist(),
        )

43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
    @staticmethod
    def empty_cpu(num_positions: int,
                  num_tokens_per_position: int) -> "LogprobsTensors":
        """Create empty LogprobsTensors on CPU."""

        logprob_token_ids = torch.empty(
            (num_positions, num_tokens_per_position),
            dtype=torch.int32,
            device="cpu")
        logprobs = torch.empty_like(logprob_token_ids, dtype=torch.float32)
        selected_token_ranks = torch.empty(num_positions,
                                           dtype=torch.int32,
                                           device="cpu")
        return LogprobsTensors(
            logprob_token_ids=logprob_token_ids,
            logprobs=logprobs,
            selected_token_ranks=selected_token_ranks,
        )

62
63
64
65

@dataclass
class SamplerOutput:

66
67
68
    # [num_reqs, max_num_generated_tokens]
    # Different requests can have different number of generated tokens.
    # All requests are padded to max_num_generated_tokens.
69
    # PLACEHOLDER_TOKEN_ID (-1 by default) is used for padding.
70
71
    sampled_token_ids: torch.Tensor
    logprobs_tensors: Optional[LogprobsTensors]
72
73


74
# ModelRunnerOutput is serialized and sent to the scheduler process.
75
# This is expensive for torch.Tensor so prefer to use list instead.
76
77
78
79
@dataclass
class ModelRunnerOutput:

    # [num_reqs]
80
    req_ids: list[str]
81
    # req_id -> index
82
    req_id_to_index: dict[str, int]
83

84
85
86
87
    # num_reqs x num_generated_tokens
    # num_generated_tokens is the number of tokens
    # generated in the current step. It can be different for
    # each request due to speculative/jump decoding.
88
    sampled_token_ids: list[list[int]]
89

90
    # num_reqs x num_spec_tokens
91
    spec_token_ids: Optional[list[list[int]]]
92

93
94
    # [num_reqs, max_num_logprobs + 1]
    # [num_reqs, max_num_logprobs + 1]
95
96
97
98
99
100
101
    # [num_reqs]
    logprobs: Optional[LogprobsLists]

    # req_id -> (token_ids, logprobs, ranks)
    # [prompt_len, num_prompt_logprobs]
    # [prompt_len, num_prompt_logprobs]
    # [prompt_len]
102
    prompt_logprobs_dict: dict[str, Optional[LogprobsTensors]]
103

Robert Shaw's avatar
Robert Shaw committed
104
105
106
107
108
109
110
111
112
113
114
115
116
    # [req_ids]
    finished_sending: Optional[set[str]] = None
    finished_recving: Optional[set[str]] = None


EMPTY_MODEL_RUNNER_OUTPUT = ModelRunnerOutput(req_ids=[],
                                              req_id_to_index={},
                                              sampled_token_ids=[],
                                              spec_token_ids=None,
                                              logprobs=None,
                                              prompt_logprobs_dict={},
                                              finished_sending=None,
                                              finished_recving=None)