Skip to content

vllm.model_executor.layers.quantization.compressed_tensors.schemes.compressed_tensors_w4a8_int

W4A8_SUPPORTED_BITS module-attribute

W4A8_SUPPORTED_BITS = list(keys())

W4A8_SUPPORTED_TYPES_MAP module-attribute

W4A8_SUPPORTED_TYPES_MAP = {4: int4}

__all__ module-attribute

__all__ = ['CompressedTensorsW4A8Int']

logger module-attribute

logger = init_logger(__name__)

CompressedTensorsW4A8Int

Bases: CompressedTensorsScheme

Source code in vllm/model_executor/layers/quantization/compressed_tensors/schemes/compressed_tensors_w4a8_int.py
class CompressedTensorsW4A8Int(CompressedTensorsScheme):
    _kernel_backends_being_used: set[str] = set()

    def __init__(self,
                 strategy: str,
                 num_bits: int,
                 group_size: Optional[int] = None,
                 is_static_input_scheme: bool = False,
                 input_symmetric: bool = True):
        self.strategy = strategy
        self.group_size = -1 if group_size is None else group_size
        self.is_static_input_scheme = is_static_input_scheme
        self.input_symmetric = input_symmetric

        if num_bits not in W4A8_SUPPORTED_TYPES_MAP:
            raise ValueError(
                f"Unsupported num_bits = {num_bits}."
                f"Supported num_bits = {W4A8_SUPPORTED_TYPES_MAP.keys()}")
        self.quant_type = W4A8_SUPPORTED_TYPES_MAP[num_bits]

    @classmethod
    def get_min_capability(cls) -> int:
        return 1

    def create_weights(self, layer: torch.nn.Module, output_size: int,
                       input_size: int, output_partition_sizes: list[int],
                       input_size_per_partition: int,
                       params_dtype: torch.dtype, weight_loader: Callable,
                       **kwargs):
        output_size_per_partition = sum(output_partition_sizes)
        row_parallel = (input_size != input_size_per_partition)

        # Compute effective group_size
        if self.group_size == -1:
            effective_group_size = (input_size_per_partition
                                    if row_parallel else input_size)
        else:
            effective_group_size = self.group_size

        # Ensure group_size divides input_size_per_partition
        assert input_size_per_partition % effective_group_size == 0, (
            f"input_size_per_partition {input_size_per_partition}"
            f" not divisible by group_size {effective_group_size}")

        # Determine scale partitioning
        is_channelwise = (self.group_size == -1)
        repeat_scales = (is_channelwise and row_parallel)
        partition_scales = not repeat_scales

        mp_linear_kernel_config = MPLinearLayerConfig(
            full_weight_shape=(input_size, output_size),
            partition_weight_shape=(input_size_per_partition,
                                    output_size_per_partition),
            weight_type=self.quant_type,
            act_type=params_dtype,
            group_size=effective_group_size,
            zero_points=False,
            has_g_idx=False,
        )

        kernel_type = choose_mp_linear_kernel(mp_linear_kernel_config)
        if kernel_type.__name__ not in self._kernel_backends_being_used:
            logger.info("Using %s for CompressedTensorsW4A8Int",
                        kernel_type.__name__)
            self._kernel_backends_being_used.add(kernel_type.__name__)

        scales_and_zp_size = input_size_per_partition // effective_group_size

        weight = ModelWeightParameter(data=torch.empty(
            output_size_per_partition,
            input_size_per_partition,
            dtype=torch.int8),
                                      input_dim=1,
                                      output_dim=0,
                                      weight_loader=weight_loader)
        layer.register_parameter("weight", weight)

        weight_scale_args = {
            "weight_loader":
            weight_loader,
            "data":
            torch.empty(output_size_per_partition,
                        scales_and_zp_size,
                        dtype=params_dtype)
        }

        if partition_scales:
            weight_scale = GroupQuantScaleParameter(output_dim=0,
                                                    input_dim=1,
                                                    **weight_scale_args)
        else:
            weight_scale = ChannelQuantScaleParameter(output_dim=0,
                                                      **weight_scale_args)

        layer.register_parameter("weight_packed", weight)
        layer.register_parameter("weight_scale", weight_scale)

        self.kernel = kernel_type(mp_linear_kernel_config,
                                  w_q_param_name="weight_packed",
                                  w_s_param_name="weight_scale",
                                  w_zp_param_name=None,
                                  w_gidx_param_name=None)

    def process_weights_after_loading(self, layer: torch.nn.Module) -> None:
        self.kernel.process_weights_after_loading(layer)

    def apply_weights(self, layer: torch.nn.Module, x: torch.Tensor,
                      bias: Optional[torch.Tensor]) -> torch.Tensor:
        return self.kernel.apply_weights(layer, x, bias)

_kernel_backends_being_used class-attribute instance-attribute

_kernel_backends_being_used: set[str] = set()

group_size instance-attribute

group_size = -1 if group_size is None else group_size

input_symmetric instance-attribute

input_symmetric = input_symmetric

is_static_input_scheme instance-attribute

is_static_input_scheme = is_static_input_scheme

quant_type instance-attribute

quant_type = W4A8_SUPPORTED_TYPES_MAP[num_bits]

strategy instance-attribute

strategy = strategy

__init__

__init__(
    strategy: str,
    num_bits: int,
    group_size: Optional[int] = None,
    is_static_input_scheme: bool = False,
    input_symmetric: bool = True,
)
Source code in vllm/model_executor/layers/quantization/compressed_tensors/schemes/compressed_tensors_w4a8_int.py
def __init__(self,
             strategy: str,
             num_bits: int,
             group_size: Optional[int] = None,
             is_static_input_scheme: bool = False,
             input_symmetric: bool = True):
    self.strategy = strategy
    self.group_size = -1 if group_size is None else group_size
    self.is_static_input_scheme = is_static_input_scheme
    self.input_symmetric = input_symmetric

    if num_bits not in W4A8_SUPPORTED_TYPES_MAP:
        raise ValueError(
            f"Unsupported num_bits = {num_bits}."
            f"Supported num_bits = {W4A8_SUPPORTED_TYPES_MAP.keys()}")
    self.quant_type = W4A8_SUPPORTED_TYPES_MAP[num_bits]

apply_weights

apply_weights(
    layer: Module, x: Tensor, bias: Optional[Tensor]
) -> Tensor
Source code in vllm/model_executor/layers/quantization/compressed_tensors/schemes/compressed_tensors_w4a8_int.py
def apply_weights(self, layer: torch.nn.Module, x: torch.Tensor,
                  bias: Optional[torch.Tensor]) -> torch.Tensor:
    return self.kernel.apply_weights(layer, x, bias)

create_weights

create_weights(
    layer: Module,
    output_size: int,
    input_size: int,
    output_partition_sizes: list[int],
    input_size_per_partition: int,
    params_dtype: dtype,
    weight_loader: Callable,
    **kwargs,
)
Source code in vllm/model_executor/layers/quantization/compressed_tensors/schemes/compressed_tensors_w4a8_int.py
def create_weights(self, layer: torch.nn.Module, output_size: int,
                   input_size: int, output_partition_sizes: list[int],
                   input_size_per_partition: int,
                   params_dtype: torch.dtype, weight_loader: Callable,
                   **kwargs):
    output_size_per_partition = sum(output_partition_sizes)
    row_parallel = (input_size != input_size_per_partition)

    # Compute effective group_size
    if self.group_size == -1:
        effective_group_size = (input_size_per_partition
                                if row_parallel else input_size)
    else:
        effective_group_size = self.group_size

    # Ensure group_size divides input_size_per_partition
    assert input_size_per_partition % effective_group_size == 0, (
        f"input_size_per_partition {input_size_per_partition}"
        f" not divisible by group_size {effective_group_size}")

    # Determine scale partitioning
    is_channelwise = (self.group_size == -1)
    repeat_scales = (is_channelwise and row_parallel)
    partition_scales = not repeat_scales

    mp_linear_kernel_config = MPLinearLayerConfig(
        full_weight_shape=(input_size, output_size),
        partition_weight_shape=(input_size_per_partition,
                                output_size_per_partition),
        weight_type=self.quant_type,
        act_type=params_dtype,
        group_size=effective_group_size,
        zero_points=False,
        has_g_idx=False,
    )

    kernel_type = choose_mp_linear_kernel(mp_linear_kernel_config)
    if kernel_type.__name__ not in self._kernel_backends_being_used:
        logger.info("Using %s for CompressedTensorsW4A8Int",
                    kernel_type.__name__)
        self._kernel_backends_being_used.add(kernel_type.__name__)

    scales_and_zp_size = input_size_per_partition // effective_group_size

    weight = ModelWeightParameter(data=torch.empty(
        output_size_per_partition,
        input_size_per_partition,
        dtype=torch.int8),
                                  input_dim=1,
                                  output_dim=0,
                                  weight_loader=weight_loader)
    layer.register_parameter("weight", weight)

    weight_scale_args = {
        "weight_loader":
        weight_loader,
        "data":
        torch.empty(output_size_per_partition,
                    scales_and_zp_size,
                    dtype=params_dtype)
    }

    if partition_scales:
        weight_scale = GroupQuantScaleParameter(output_dim=0,
                                                input_dim=1,
                                                **weight_scale_args)
    else:
        weight_scale = ChannelQuantScaleParameter(output_dim=0,
                                                  **weight_scale_args)

    layer.register_parameter("weight_packed", weight)
    layer.register_parameter("weight_scale", weight_scale)

    self.kernel = kernel_type(mp_linear_kernel_config,
                              w_q_param_name="weight_packed",
                              w_s_param_name="weight_scale",
                              w_zp_param_name=None,
                              w_gidx_param_name=None)

get_min_capability classmethod

get_min_capability() -> int
Source code in vllm/model_executor/layers/quantization/compressed_tensors/schemes/compressed_tensors_w4a8_int.py
@classmethod
def get_min_capability(cls) -> int:
    return 1

process_weights_after_loading

process_weights_after_loading(layer: Module) -> None
Source code in vllm/model_executor/layers/quantization/compressed_tensors/schemes/compressed_tensors_w4a8_int.py
def process_weights_after_loading(self, layer: torch.nn.Module) -> None:
    self.kernel.process_weights_after_loading(layer)