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

4
5
6
7
8
# Adapted from
# https://github.com/fmmoret/vllm/blob/fm-support-lora-on-quantized-models/tests/lora/test_llama.py
from dataclasses import dataclass

import pytest
9
import os
10
11

import vllm
12
from vllm.distributed import cleanup_dist_env_and_memory
13
14
from vllm.lora.request import LoRARequest

15
from vllm.platforms import current_platform
16
from ..utils import models_path_prefix
17
18
19
20
21
22
23
24


@dataclass
class ModelWithQuantization:
    model_path: str
    quantization: str


25
MODELS: list[ModelWithQuantization]
26
#AWQ quantization is currently not supported in ROCm.
27
if current_platform.is_rocm():
28
29
    MODELS = [
        ModelWithQuantization(
30
            model_path=os.path.join(models_path_prefix, "TheBloke/TinyLlama-1.1B-Chat-v0.3-GPTQ"),
31
            quantization="gptq"),
32
33
34
35
    ]
else:
    MODELS = [
        ModelWithQuantization(
36
            model_path=os.path.join(models_path_prefix, "TheBloke/TinyLlama-1.1B-Chat-v0.3-AWQ"),
37
            quantization="awq"),
38
        ModelWithQuantization(
39
            model_path=os.path.join(models_path_prefix, "TheBloke/TinyLlama-1.1B-Chat-v0.3-GPTQ"),
40
            quantization="gptq"),
41
    ]
42
43


44
45
46
def do_sample(llm: vllm.LLM,
              lora_path: str,
              lora_id: int,
47
              max_tokens: int = 256) -> list[str]:
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
    raw_prompts = [
        "Give me an orange-ish brown color",
        "Give me a neon pink color",
    ]

    def format_prompt_tuples(prompt):
        return f"<|im_start|>user\n{prompt}<|im_end|>\n<|im_start|>assistant\n"

    prompts = [format_prompt_tuples(p) for p in raw_prompts]

    sampling_params = vllm.SamplingParams(temperature=0,
                                          max_tokens=max_tokens,
                                          stop=["<|im_end|>"])
    outputs = llm.generate(
        prompts,
        sampling_params,
        lora_request=LoRARequest(str(lora_id), lora_id, lora_path)
        if lora_id else None)
    # Print the outputs.
67
    generated_texts: list[str] = []
68
69
70
71
72
73
74
75
76
    for output in outputs:
        prompt = output.prompt
        generated_text = output.outputs[0].text
        generated_texts.append(generated_text)
        print(f"Prompt: {prompt!r}, Generated text: {generated_text!r}")
    return generated_texts


@pytest.mark.parametrize("model", MODELS)
77
def test_quant_model_lora(tinyllama_lora_files, model):
78

79
80
81
82
83
84
85
86
    llm = vllm.LLM(
        model=model.model_path,
        enable_lora=True,
        max_num_seqs=16,
        max_loras=4,
        max_model_len=400,
        gpu_memory_utilization=0.2,  #avoid OOM
        quantization=model.quantization,
87
        trust_remote_code=True,
88
89
        enable_chunked_prefill=True,
        tokenizer=tinyllama_lora_files)
90
91
92
93
94
95

    if model.quantization is None:
        expected_lora_output = [
            "#ff8050",
            "#ff8080",
        ]
96
    elif model.quantization == "awq":
97
98
99
100
        expected_lora_output = [
            "#f07700: A v",
            "#f00000: A v",
        ]
101
    elif model.quantization == "gptq":
102
103
104
105
106
107
108
109
        expected_lora_output = [
            "#f08800: This is",
            "#f07788 \n#",
        ]

    def expect_match(output, expected_output):
        # HACK: GPTQ lora outputs are just incredibly unstable.
        # Assert that the outputs changed.
110
        if (model.quantization == "gptq"
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
                and expected_output is expected_lora_output):
            for i, o in enumerate(output):
                assert o.startswith(
                    '#'), f"Expected example {i} to start with # but got {o}"
            return
        assert output == expected_output

    max_tokens = 10

    print("lora adapter created")
    print("lora 1")
    output = do_sample(llm,
                       tinyllama_lora_files,
                       lora_id=1,
                       max_tokens=max_tokens)
    expect_match(output, expected_lora_output)

    print("lora 2")
    output = do_sample(llm,
                       tinyllama_lora_files,
                       lora_id=2,
                       max_tokens=max_tokens)
    expect_match(output, expected_lora_output)

    print("removing lora")

    del llm
138
    cleanup_dist_env_and_memory()
139
140
141


@pytest.mark.parametrize("model", MODELS)
142
143
144
145
def test_quant_model_tp_equality(tinyllama_lora_files, num_gpus_available,
                                 model):
    if num_gpus_available < 2:
        pytest.skip(f"Not enough GPUs for tensor parallelism {2}")
146
    if model.quantization == "gptq":
147
        pytest.skip("GPTQ lora outputs are just incredibly unstable")
148
149
150
151
152
153
154
    llm_tp1 = vllm.LLM(
        model=model.model_path,
        enable_lora=True,
        max_num_seqs=16,
        max_loras=4,
        gpu_memory_utilization=0.2,  #avoid OOM
        quantization=model.quantization,
155
156
        trust_remote_code=True,
        enable_chunked_prefill=True)
157
158
159
    output_tp1 = do_sample(llm_tp1, tinyllama_lora_files, lora_id=1)

    del llm_tp1
160
    cleanup_dist_env_and_memory()
161

162
163
164
165
166
167
168
    llm_tp2 = vllm.LLM(
        model=model.model_path,
        enable_lora=True,
        max_num_seqs=16,
        max_loras=4,
        tensor_parallel_size=2,
        gpu_memory_utilization=0.2,  #avoid OOM
169
170
        quantization=model.quantization,
        enable_chunked_prefill=True)
171
172
173
    output_tp2 = do_sample(llm_tp2, tinyllama_lora_files, lora_id=1)

    del llm_tp2
174
    cleanup_dist_env_and_memory()
175
176

    assert output_tp1 == output_tp2