utils.py 16 KB
Newer Older
1
# SPDX-License-Identifier: Apache-2.0
2
# SPDX-FileCopyrightText: Copyright contributors to the vLLM project
3
import argparse
4
import contextlib
5
import multiprocessing
6
import time
7
import weakref
8
from collections.abc import Callable, Sequence
9
from contextlib import AbstractContextManager
10
from dataclasses import dataclass
11
from multiprocessing import connection
12
from multiprocessing.process import BaseProcess
13
from multiprocessing.queues import Queue
14
15
16
17
18
19
20
21
from typing import (
    TYPE_CHECKING,
    Any,
    Generic,
    TypeVar,
    Union,
    overload,
)
22
23

import torch
24
from torch.autograd.profiler import record_function
25

26
import vllm.envs as envs
27
from vllm.logger import init_logger
28
from vllm.usage.usage_lib import UsageContext, is_usage_stats_enabled, usage_message
29
from vllm.utils.network_utils import get_open_port, get_open_zmq_ipc_path, get_tcp_uri
30
from vllm.utils.system_utils import kill_process_tree
31
from vllm.v1.core.sched.output import SchedulerOutput
32

33
if TYPE_CHECKING:
34
35
    import numpy as np

36
    from vllm.v1.engine.coordinator import DPCoordinator
37
    from vllm.v1.engine.utils import CoreEngineActorManager, CoreEngineProcManager
38

39
logger = init_logger(__name__)
40
41
42
43

T = TypeVar("T")


44
class ConstantList(Generic[T], Sequence):
45
    def __init__(self, x: list[T]) -> None:
46
47
48
        self._x = x

    def append(self, item):
49
        raise TypeError("Cannot append to a constant list")
50
51

    def extend(self, item):
52
        raise TypeError("Cannot extend a constant list")
53
54

    def insert(self, item):
55
        raise TypeError("Cannot insert into a constant list")
56
57

    def pop(self, item):
58
        raise TypeError("Cannot pop from a constant list")
59
60

    def remove(self, item):
61
        raise TypeError("Cannot remove from a constant list")
62
63

    def clear(self):
64
        raise TypeError("Cannot clear a constant list")
65

66
    def index(self, item: T, start: int = 0, stop: int | None = None) -> int:
67
        return self._x.index(item, start, stop if stop is not None else len(self._x))
68
69

    @overload
70
    def __getitem__(self, item: int) -> T: ...
71
72

    @overload
73
    def __getitem__(self, s: slice, /) -> list[T]: ...
74

75
    def __getitem__(self, item: int | slice) -> T | list[T]:
76
77
78
        return self._x[item]

    @overload
79
    def __setitem__(self, item: int, value: T): ...
80
81

    @overload
82
    def __setitem__(self, s: slice, value: T, /): ...
83

84
    def __setitem__(self, item: int | slice, value: T | list[T]):
85
        raise TypeError("Cannot set item in a constant list")
86
87

    def __delitem__(self, item):
88
        raise TypeError("Cannot delete item from a constant list")
89
90
91
92
93
94
95
96
97

    def __iter__(self):
        return iter(self._x)

    def __contains__(self, item):
        return item in self._x

    def __len__(self):
        return len(self._x)
98

99
100
101
    def __repr__(self):
        return f"ConstantList({self._x})"

102
103
104
    def copy(self) -> list[T]:
        return self._x.copy()

105

106
class CpuGpuBuffer:
107
    """Buffer to easily copy tensors between CPU and GPU."""
108
109
110

    def __init__(
        self,
111
        *size: int | torch.SymInt,
112
113
114
        dtype: torch.dtype,
        device: torch.device,
        pin_memory: bool,
115
116
        with_numpy: bool = True,
    ) -> None:
117
        self.cpu = torch.zeros(*size, dtype=dtype, device="cpu", pin_memory=pin_memory)
118
        self.gpu = torch.zeros_like(self.cpu, device=device)
119
120
121
122
123
124
125
126
        self.np: np.ndarray
        # To keep type hints simple (avoiding generics and subclasses), we
        # only conditionally create the numpy array attribute. This can cause
        # AttributeError if `self.np` is accessed when `with_numpy=False`.
        if with_numpy:
            if dtype == torch.bfloat16:
                raise ValueError(
                    "Bfloat16 torch tensors cannot be directly cast to a "
127
128
                    "numpy array, so call CpuGpuBuffer with with_numpy=False"
                )
129
            self.np = self.cpu.numpy()
130

131
    def copy_to_gpu(self, n: int | None = None) -> torch.Tensor:
132
133
134
135
        if n is None:
            return self.gpu.copy_(self.cpu, non_blocking=True)
        return self.gpu[:n].copy_(self.cpu[:n], non_blocking=True)

136
    def copy_to_cpu(self, n: int | None = None) -> torch.Tensor:
137
138
139
140
141
142
143
        """NOTE: Because this method is non-blocking, explicit synchronization
        is needed to ensure the data is copied to CPU."""
        if n is None:
            return self.cpu.copy_(self.gpu, non_blocking=True)
        return self.cpu[:n].copy_(self.gpu[:n], non_blocking=True)


144
def get_engine_client_zmq_addr(local_only: bool, host: str, port: int = 0) -> str:
145
    """Assign a new ZMQ socket address.
Rui Qiao's avatar
Rui Qiao committed
146

147
148
    If local_only is True, participants are colocated and so a unique IPC
    address will be returned.
Rui Qiao's avatar
Rui Qiao committed
149

150
151
    Otherwise, the provided host and port will be used to construct a TCP
    address (port == 0 means assign an available port)."""
Rui Qiao's avatar
Rui Qiao committed
152

153
154
155
156
157
    return (
        get_open_zmq_ipc_path()
        if local_only
        else (get_tcp_uri(host, port or get_open_port()))
    )
Rui Qiao's avatar
Rui Qiao committed
158
159


160
161
class APIServerProcessManager:
    """Manages a group of API server processes.
162

163
164
165
166
167
168
169
170
171
172
173
174
175
    Handles creation, monitoring, and termination of API server worker
    processes. Also monitors extra processes to check if they are healthy.
    """

    def __init__(
        self,
        target_server_fn: Callable,
        listen_address: str,
        sock: Any,
        args: argparse.Namespace,
        num_servers: int,
        input_addresses: list[str],
        output_addresses: list[str],
176
        stats_update_address: str | None = None,
177
        tensor_queue: Queue | None = None,
178
179
    ):
        """Initialize and start API server worker processes.
180

181
182
183
184
185
186
187
188
        Args:
            target_server_fn: Function to call for each API server process
            listen_address: Address to listen for client connections
            sock: Socket for client connections
            args: Command line arguments
            num_servers: Number of API server processes to start
            input_addresses: Input addresses for each API server
            output_addresses: Output addresses for each API server
189
            stats_update_address: Optional stats update address
190
            tensor_queue: Optional tensor IPC queue for sharing MM tensors
191
192
193
194
        """
        self.listen_address = listen_address
        self.sock = sock
        self.args = args
195

196
197
198
199
        # Start API servers
        spawn_context = multiprocessing.get_context("spawn")
        self.processes: list[BaseProcess] = []

200
201
202
        for i, in_addr, out_addr in zip(
            range(num_servers), input_addresses, output_addresses
        ):
203
204
205
            client_config = {
                "input_address": in_addr,
                "output_address": out_addr,
206
                "client_count": num_servers,
207
                "client_index": i,
208
209
210
            }
            if stats_update_address is not None:
                client_config["stats_update_address"] = stats_update_address
211
212
            if tensor_queue is not None:
                client_config["tensor_queue"] = tensor_queue
213

214
215
216
217
218
            proc = spawn_context.Process(
                target=target_server_fn,
                name=f"ApiServer_{i}",
                args=(listen_address, sock, args, client_config),
            )
219
220
221
222
223
224
225
226
227
            self.processes.append(proc)
            proc.start()

        logger.info("Started %d API server processes", len(self.processes))

        # Shutdown only the API server processes on garbage collection
        # The extra processes are managed by their owners
        self._finalizer = weakref.finalize(self, shutdown, self.processes)

228
229
230
231
    def shutdown(self, timeout: float | None = None) -> None:
        """Shutdown API server processes with configurable timeout"""
        if self._finalizer.detach() is not None:
            shutdown(self.processes, timeout=timeout)
232
233
234


def wait_for_completion_or_failure(
235
    api_server_manager: APIServerProcessManager,
236
237
    engine_manager: Union["CoreEngineProcManager", "CoreEngineActorManager"]
    | None = None,
238
    coordinator: "DPCoordinator | None" = None,
239
) -> None:
240
    """Wait for all processes to complete or detect if any fail.
241

242
    Raises an exception if any process exits with a non-zero status.
Rui Qiao's avatar
Rui Qiao committed
243
244
245
246
247
248
249

    Args:
        api_server_manager: The manager for API servers.
        engine_manager: The manager for engine processes.
            If CoreEngineProcManager, it manages local engines;
            if CoreEngineActorManager, it manages all engines.
        coordinator: The coordinator for data parallel.
250
251
    """

252
    from vllm.v1.engine.utils import CoreEngineActorManager, CoreEngineProcManager
253

254
255
256
257
258
    try:
        logger.info("Waiting for API servers to complete ...")
        # Create a mapping of sentinels to their corresponding processes
        # for efficient lookup
        sentinel_to_proc: dict[Any, BaseProcess] = {
259
            proc.sentinel: proc for proc in api_server_manager.processes
260
261
262
263
264
        }

        if coordinator:
            sentinel_to_proc[coordinator.proc.sentinel] = coordinator.proc

Rui Qiao's avatar
Rui Qiao committed
265
266
267
        actor_run_refs = []
        if isinstance(engine_manager, CoreEngineProcManager):
            for proc in engine_manager.processes:
268
                sentinel_to_proc[proc.sentinel] = proc
Rui Qiao's avatar
Rui Qiao committed
269
270
        elif isinstance(engine_manager, CoreEngineActorManager):
            actor_run_refs = engine_manager.get_run_refs()
271
272

        # Check if any process terminates
Rui Qiao's avatar
Rui Qiao committed
273
        while sentinel_to_proc or actor_run_refs:
274
            # Wait for any process to terminate
275
            ready_sentinels: list[Any] = connection.wait(sentinel_to_proc, timeout=5)
276
277
278
279
280
281
282
283
284

            # Process any terminated processes
            for sentinel in ready_sentinels:
                proc = sentinel_to_proc.pop(sentinel)

                # Check if process exited with error
                if proc.exitcode != 0:
                    raise RuntimeError(
                        f"Process {proc.name} (PID: {proc.pid}) "
285
286
                        f"died with exit code {proc.exitcode}"
                    )
Rui Qiao's avatar
Rui Qiao committed
287
288
289

            if actor_run_refs:
                import ray
290

Rui Qiao's avatar
Rui Qiao committed
291
292
                _, actor_run_refs = ray.wait(actor_run_refs, timeout=5)

293
294
295
    except KeyboardInterrupt:
        logger.info("Received KeyboardInterrupt, shutting down API servers...")
    except Exception as e:
296
        logger.exception("Exception occurred while running API servers: %s", str(e))
297
298
299
        raise


Robert Shaw's avatar
Robert Shaw committed
300
# Note(rob): shutdown function cannot be a bound method,
301
# else the gc cannot collect the object.
302
303
304
305
306
307
308
309
310
311
312
313
314
def shutdown(procs: list[BaseProcess], timeout: float | None = None) -> None:
    """Shutdown processes with timeout.

    Args:
        procs: List of processes to shutdown
        timeout: Maximum time in seconds to wait for graceful shutdown
    """
    if timeout is None:
        timeout = 0.0

    # Allow at least 5 seconds for remaining procs to terminate.
    timeout = max(timeout, 5.0)

Robert Shaw's avatar
Robert Shaw committed
315
    # Shutdown the process.
316
317
318
319
    for proc in procs:
        if proc.is_alive():
            proc.terminate()

320
321
    # Allow time for remaining procs to terminate.
    deadline = time.monotonic() + timeout
322
323
324
325
326
327
328
329
    for proc in procs:
        remaining = deadline - time.monotonic()
        if remaining <= 0:
            break
        if proc.is_alive():
            proc.join(remaining)

    for proc in procs:
330
331
        if proc.is_alive() and (pid := proc.pid) is not None:
            kill_process_tree(pid)
Robert Shaw's avatar
Robert Shaw committed
332

333

334
335
336
def copy_slice(
    from_tensor: torch.Tensor, to_tensor: torch.Tensor, length: int
) -> torch.Tensor:
337
338
339
340
341
    """
    Copy the first length elements of a tensor into another tensor in a
    non-blocking manner.

    Used to copy pinned CPU tensor data to pre-allocated GPU tensors.
342
343

    Returns the sliced target tensor.
344
    """
345
    return to_tensor[:length].copy_(from_tensor[:length], non_blocking=True)
346
347


348
def report_usage_stats(
349
350
    vllm_config, usage_context: UsageContext = UsageContext.ENGINE_CONTEXT
) -> None:
351
352
353
354
355
356
357
    """Report usage statistics if enabled."""

    if not is_usage_stats_enabled():
        return

    from vllm.model_executor.model_loader import get_architecture_class_name

358
359
    parallel_config = vllm_config.parallel_config

360
361
362
363
364
    # Prepare KV connector string if applicable
    kv_connector = None
    if vllm_config.kv_transfer_config is not None:
        kv_connector = vllm_config.kv_transfer_config.kv_connector

365
366
367
368
369
    usage_message.report_usage(
        get_architecture_class_name(vllm_config.model_config),
        usage_context,
        extra_kvs={
            # Common configuration
370
371
372
373
            "dtype": str(vllm_config.model_config.dtype),
            "block_size": vllm_config.cache_config.block_size,
            "gpu_memory_utilization": vllm_config.cache_config.gpu_memory_utilization,
            "kv_cache_memory_bytes": vllm_config.cache_config.kv_cache_memory_bytes,
374
            # Quantization
375
376
            "quantization": vllm_config.model_config.quantization,
            "kv_cache_dtype": str(vllm_config.cache_config.cache_dtype),
377
            # Feature flags
378
379
380
            "enable_lora": bool(vllm_config.lora_config),
            "enable_prefix_caching": vllm_config.cache_config.enable_prefix_caching,
            "enforce_eager": vllm_config.model_config.enforce_eager,
381
            "disable_custom_all_reduce": parallel_config.disable_custom_all_reduce,
382
383
384
385
386
387
388
389
390
            # Distributed parallelism settings
            "tensor_parallel_size": parallel_config.tensor_parallel_size,
            "data_parallel_size": parallel_config.data_parallel_size,
            "pipeline_parallel_size": parallel_config.pipeline_parallel_size,
            "enable_expert_parallel": parallel_config.enable_expert_parallel,
            # All2All backend for MoE expert parallel
            "all2all_backend": parallel_config.all2all_backend,
            # KV connector used
            "kv_connector": kv_connector,
391
392
        },
    )
393
394


395
396
397
_PROFILER_FUNC = None


398
def record_function_or_nullcontext(name: str) -> AbstractContextManager:
399
400
401
402
403
404
405
    global _PROFILER_FUNC

    # fast path assume it is set
    if _PROFILER_FUNC is not None:
        return _PROFILER_FUNC(name)

    func = contextlib.nullcontext
406
    if envs.VLLM_CUSTOM_SCOPES_FOR_PROFILING:
407
408
409
        func = record_function
    elif envs.VLLM_NVTX_SCOPES_FOR_PROFILING:
        import nvtx
410

411
412
413
414
        func = nvtx.annotate

    _PROFILER_FUNC = func
    return func(name)
415
416
417
418
419
420
421
422
423
424
425
426


def tensor_data(tensor: torch.Tensor) -> memoryview:
    """Get the raw data of a tensor as a uint8 memoryview, useful for
    serializing and hashing.

    Args:
        tensor: The input tensor.

    Returns:
        A memoryview of the tensor data as uint8.
    """
427
    return tensor.flatten().cpu().contiguous().view(torch.uint8).numpy().data
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477


@dataclass
class IterationDetails:
    num_ctx_requests: int
    num_ctx_tokens: int
    num_generation_requests: int
    num_generation_tokens: int

    def __repr__(self) -> str:
        return f"IterationDetails(num_ctx_requests={self.num_ctx_requests},\
                 num_ctx_tokens={self.num_ctx_tokens}, \
                 num_generation_requests={self.num_generation_requests}, \
                 num_generation_tokens={self.num_generation_tokens})"


def compute_iteration_details(scheduler_output: SchedulerOutput) -> IterationDetails:
    """
    Compute the number of context/generation requests and tokens
    for the current iteration's scheduler output. A requests is regarded
    as a context request if its output tokens are still 0, an extended chunk
    of chunked prefill falls into this category.

    Args:
        scheduler_output: The scheduler output for the current iteration.

    Returns:
        An IterationDetails object containing the number of
        context/generation requests and tokens.
    """
    num_context_requests = 0
    num_context_tokens = 0
    num_generation_requests = 0
    num_generation_tokens = 0
    new_req_ids = {new_req.req_id for new_req in scheduler_output.scheduled_new_reqs}
    for req_id, num_tokens in scheduler_output.num_scheduled_tokens.items():
        if scheduler_output.scheduled_cached_reqs.is_context_phase(req_id) or (
            req_id in new_req_ids
        ):
            num_context_requests += 1
            num_context_tokens += num_tokens
        else:
            num_generation_requests += 1
            num_generation_tokens += num_tokens
    return IterationDetails(
        num_context_requests,
        num_context_tokens,
        num_generation_requests,
        num_generation_tokens,
    )