benchmark_latency.py 8.83 KB
Newer Older
1
"""Benchmark the latency of processing a single batch of requests."""
2
import argparse
3
import json
4
import time
5
6
from pathlib import Path
from typing import Optional
7
8
9

import numpy as np
import torch
10
from tqdm import tqdm
11

Woosuk Kwon's avatar
Woosuk Kwon committed
12
from vllm import LLM, SamplingParams
13
from vllm.model_executor.layers.quantization import QUANTIZATION_METHODS
14
15
16


def main(args: argparse.Namespace):
17
18
19
    print(args)

    # NOTE(woosuk): If the request cannot be processed in a single batch,
Zhuohan Li's avatar
Zhuohan Li committed
20
    # the engine will automatically process the request in multiple batches.
21
    llm = LLM(model=args.model,
22
23
              speculative_model=args.speculative_model,
              num_speculative_tokens=args.num_speculative_tokens,
24
25
26
27
28
29
30
              tokenizer=args.tokenizer,
              quantization=args.quantization,
              tensor_parallel_size=args.tensor_parallel_size,
              trust_remote_code=args.trust_remote_code,
              dtype=args.dtype,
              enforce_eager=args.enforce_eager,
              kv_cache_dtype=args.kv_cache_dtype,
31
              quantization_param_path=args.quantization_param_path,
32
33
              device=args.device,
              ray_workers_use_nsight=args.ray_workers_use_nsight,
34
              use_v2_block_manager=args.use_v2_block_manager,
35
36
37
              enable_chunked_prefill=args.enable_chunked_prefill,
              download_dir=args.download_dir,
              block_size=args.block_size)
38

Woosuk Kwon's avatar
Woosuk Kwon committed
39
40
41
42
43
    sampling_params = SamplingParams(
        n=args.n,
        temperature=0.0 if args.use_beam_search else 1.0,
        top_p=1.0,
        use_beam_search=args.use_beam_search,
44
        ignore_eos=True,
Woosuk Kwon's avatar
Woosuk Kwon committed
45
46
        max_tokens=args.output_len,
    )
47
    print(sampling_params)
48
49
50
51
    dummy_prompt_token_ids = np.random.randint(10000,
                                               size=(args.batch_size,
                                                     args.input_len))
    dummy_prompt_token_ids = dummy_prompt_token_ids.tolist()
52

53
54
55
56
57
58
59
60
61
    def run_to_completion(profile_dir: Optional[str] = None):
        if profile_dir:
            with torch.profiler.profile(
                    activities=[
                        torch.profiler.ProfilerActivity.CPU,
                        torch.profiler.ProfilerActivity.CUDA,
                    ],
                    on_trace_ready=torch.profiler.tensorboard_trace_handler(
                        str(profile_dir))) as p:
62
63
64
65
66
67
68
69
70
71
72
73
                llm.generate(prompt_token_ids=dummy_prompt_token_ids,
                             sampling_params=sampling_params,
                             use_tqdm=False)
            print(p.key_averages())
        else:
            start_time = time.perf_counter()
            llm.generate(prompt_token_ids=dummy_prompt_token_ids,
                         sampling_params=sampling_params,
                         use_tqdm=False)
            end_time = time.perf_counter()
            latency = end_time - start_time
            return latency
74

75
    print("Warming up...")
76
77
    for _ in tqdm(range(args.num_iters_warmup), desc="Warmup iterations"):
        run_to_completion(profile_dir=None)
78

79
    if args.profile:
80
81
        profile_dir = args.profile_result_dir
        if not profile_dir:
82
83
84
            profile_dir = Path(
                "."
            ) / "vllm_benchmark_result" / f"latency_result_{time.time()}"
85
        print(f"Profiling (results will be saved to '{profile_dir}')...")
86
        run_to_completion(profile_dir=profile_dir)
87
88
        return

89
90
    # Benchmark.
    latencies = []
91
    for _ in tqdm(range(args.num_iters), desc="Profiling iterations"):
92
        latencies.append(run_to_completion(profile_dir=None))
93
94
95
    latencies = np.array(latencies)
    percentages = [10, 25, 50, 75, 90]
    percentiles = np.percentile(latencies, percentages)
96
    print(f'Avg latency: {np.mean(latencies)} seconds')
97
98
    for percentage, percentile in zip(percentages, percentiles):
        print(f'{percentage}% percentile latency: {percentile} seconds')
99

100
101
102
103
104
105
106
107
108
109
    # Output JSON results if specified
    if args.output_json:
        results = {
            "avg_latency": np.mean(latencies),
            "latencies": latencies.tolist(),
            "percentiles": dict(zip(percentages, percentiles.tolist())),
        }
        with open(args.output_json, "w") as f:
            json.dump(results, f, indent=4)

110
111

if __name__ == '__main__':
112
    parser = argparse.ArgumentParser(
113
        description='Benchmark the latency of processing a single batch of '
114
        'requests till completion.')
115
    parser.add_argument('--model', type=str, default='facebook/opt-125m')
116
117
    parser.add_argument('--speculative-model', type=str, default=None)
    parser.add_argument('--num-speculative-tokens', type=int, default=None)
118
    parser.add_argument('--tokenizer', type=str, default=None)
119
120
    parser.add_argument('--quantization',
                        '-q',
121
                        choices=[*QUANTIZATION_METHODS, None],
122
                        default=None)
123
    parser.add_argument('--tensor-parallel-size', '-tp', type=int, default=1)
124
125
126
    parser.add_argument('--input-len', type=int, default=32)
    parser.add_argument('--output-len', type=int, default=128)
    parser.add_argument('--batch-size', type=int, default=8)
127
128
129
    parser.add_argument('--n',
                        type=int,
                        default=1,
130
                        help='Number of generated sequences per prompt.')
131
    parser.add_argument('--use-beam-search', action='store_true')
132
133
134
135
    parser.add_argument('--num-iters-warmup',
                        type=int,
                        default=10,
                        help='Number of iterations to run for warmup.')
136
137
    parser.add_argument('--num-iters',
                        type=int,
138
                        default=30,
139
                        help='Number of iterations to run.')
140
141
    parser.add_argument('--trust-remote-code',
                        action='store_true',
142
                        help='trust remote code from huggingface')
143
144
145
146
147
148
149
150
151
    parser.add_argument(
        '--dtype',
        type=str,
        default='auto',
        choices=['auto', 'half', 'float16', 'bfloat16', 'float', 'float32'],
        help='data type for model weights and activations. '
        'The "auto" option will use FP16 precision '
        'for FP32 and FP16 models, and BF16 precision '
        'for BF16 models.')
152
153
154
    parser.add_argument('--enforce-eager',
                        action='store_true',
                        help='enforce eager mode and disable CUDA graph')
155
156
157
    parser.add_argument(
        "--kv-cache-dtype",
        type=str,
158
        choices=['auto', 'fp8'],
159
160
        default='auto',
        help=
161
162
        'Data type for kv cache storage. If "auto", will use model data type. '
        'FP8_E5M2 (without scaling) is only supported on cuda version greater '
163
164
        'than 11.8. On ROCm (AMD GPU), FP8_E4M3 is '
        'instead supported for common inference criteria.')
165
166
167
168
169
170
171
172
173
174
    parser.add_argument(
        '--quantization-param-path',
        type=str,
        default=None,
        help='Path to the JSON file containing the KV cache scaling factors. '
        'This should generally be supplied, when KV cache dtype is FP8. '
        'Otherwise, KV cache scaling factors default to 1.0, which may cause '
        'accuracy issues. FP8_E5M2 (without scaling) is only supported on '
        'cuda version greater than 11.8. On ROCm (AMD GPU), FP8_E4M3 is '
        'instead supported for common inference criteria.')
175
176
177
178
    parser.add_argument(
        '--profile',
        action='store_true',
        help='profile the generation process of a single batch')
179
180
181
182
    parser.add_argument(
        '--profile-result-dir',
        type=str,
        default=None,
183
184
        help=('path to save the pytorch profiler output. Can be visualized '
              'with ui.perfetto.dev or Tensorboard.'))
185
186
187
188
    parser.add_argument(
        "--device",
        type=str,
        default="cuda",
189
190
        choices=["cuda", "cpu"],
        help='device type for vLLM execution, supporting CUDA and CPU.')
191
192
193
194
195
196
    parser.add_argument('--block-size',
                        type=int,
                        default=16,
                        help='block size of key/value cache')
    parser.add_argument(
        '--enable-chunked-prefill',
197
        action='store_true',
198
199
        help='If True, the prefill requests can be chunked based on the '
        'max_num_batched_tokens')
200
    parser.add_argument('--use-v2-block-manager', action='store_true')
201
202
203
204
205
    parser.add_argument(
        "--ray-workers-use-nsight",
        action='store_true',
        help="If specified, use nsight to profile ray workers",
    )
206
207
208
209
210
    parser.add_argument('--download-dir',
                        type=str,
                        default=None,
                        help='directory to download and load the weights, '
                        'default to the default cache dir of huggingface')
211
212
213
214
215
    parser.add_argument(
        '--output-json',
        type=str,
        default=None,
        help='Path to save the latency results in JSON format.')
216
217
    args = parser.parse_args()
    main(args)