123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474 |
- import torch
- from torch.nn.parameter import Parameter
- from typing import List, Dict, Any, Optional, TypeVar, NamedTuple
- from contextlib import suppress
- from aphrodite.modeling.layers.linear import (LinearMethodBase,
- set_weight_attrs)
- from aphrodite.quantization.base_config import (QuantizationConfig)
- from aphrodite.modeling.layers.linear import (ColumnParallelLinear,
- QKVParallelLinear,
- RowParallelLinear)
- HAS_QUANTS = False
- with suppress(ImportError):
- from aphrodite._quant_C import quant_ops as ops
- HAS_QUANTS = True
- class BitsandBytesConfig(QuantizationConfig):
- """Config class for BitsandBytes.
- Reference: https://arxiv.org/abs/2208.07339
- """
- def __init__(
- self,
- weight_bits: int,
- group_size: int,
- zero_point: bool,
- from_float: bool,
- quant_mode: str, # llm_int8, smoothquant, weight_only
- ) -> None:
- if not HAS_QUANTS:
- raise ImportError("Could not find the quantization kernels.")
- self.weight_bits = weight_bits
- self.group_size = group_size
- self.zero_point = zero_point
- self.from_float = from_float
- self.quant_mode = quant_mode
- if quant_mode == "weight_only" and self.weight_bits != 4:
- raise ValueError(
- "Currently, only 4-bit weight quantization is supported for "
- f"BNB weight_only, but got {self.weight_bits} bits.")
- if quant_mode in ["llm_int8", "smoothquant"] and self.weight_bits != 8:
- raise ValueError(
- "Currently, only 8-bit weight quantization is supported for "
- "BNB llm_int8 or smoothquant, "
- f"but got {self.weight_bits} bits.")
- self.pack_factor = 32 // self.weight_bits
- def __repr__(self) -> str:
- return (f"BitsandBytesConfig(weight_bits={self.weight_bits}, "
- f"group_size={self.group_size}, "
- f"zero_point={self.zero_point}, "
- f"from_float={self.from_float}, "
- f"quant_mode={self.quant_mode})")
- def get_name(self) -> str:
- return "bitsandbytes"
- def get_supported_act_dtypes(self) -> List[torch.dtype]:
- return [torch.half, torch.bfloat16]
- def get_min_capability(self) -> int:
- # The BitsandBytes kernel only supports Ampere or newer GPUs.
- return 75
- def merge_weight(self) -> bool:
- return True
- def rope_style(self) -> Optional[bool]:
- return None
- def quant_vocab(self) -> List[bool]:
- return [False, False]
- def support_fused_moe(self) -> bool:
- return False
- @staticmethod
- def get_config_filenames() -> List[str]:
- return [
- "quant_config.json",
- "quantize_config.json",
- ]
- @classmethod
- def from_config(cls, config: Dict[str, Any]) -> "BitsandBytesConfig":
- weight_bits = cls.get_from_keys(config, ["w_bit", "bits"])
- group_size = cls.get_from_keys(config, ["q_group_size", "group_size"])
- zero_point = cls.get_from_keys(config, ["zero_point"])
- try:
- from_float = cls.get_from_keys(config, ["from_float"])
- except Exception:
- from_float = False
- try:
- quant_mode = cls.get_from_keys(config, ["quant_mode"])
- except Exception:
- quant_mode = "weight_only"
- return cls(weight_bits, group_size, zero_point, from_float, quant_mode)
- def get_linear_method(self) -> "BNBLinearMethod":
- return BNBLinearMethod(self)
- def get_scaled_act_names(self) -> List[str]:
- return ["gelu", "gelu_fast", "gelu_new", "gelu_pytorch_tanh"]
- class BNBLinearMethod(LinearMethodBase):
- """Linear method for BitsandBytes.
- Args:
- quant_config: The BitsandBytes quantization config.
- """
- def __init__(self, quant_config: BitsandBytesConfig):
- self.quant_config = quant_config
- def create_weights(self, input_size_per_partition: int,
- output_partition_sizes: List[int], input_size: int,
- output_size: int,
- params_dtype: torch.dtype) -> Dict[str, Any]:
- if self.quant_config.quant_mode == "weight_only" and \
- input_size_per_partition % self.quant_config.group_size != 0:
- raise ValueError(
- "The input size is not aligned with the quantized "
- "weight shape. This can be caused by too large "
- "tensor parallel size.")
- output_size_per_partition = sum(output_partition_sizes)
- if self.quant_config.quant_mode == "weight_only" and \
- output_size_per_partition % self.quant_config.pack_factor != 0:
- raise ValueError(
- "The output size is not aligned with the quantized "
- "weight shape. This can be caused by too large "
- "tensor parallel size.")
- if self.quant_config.quant_mode == "weight_only" and \
- not self.quant_config.from_float:
- qweight = Parameter(
- torch.empty(
- input_size_per_partition,
- output_size_per_partition // self.quant_config.pack_factor,
- dtype=torch.int32,
- ),
- requires_grad=False,
- )
- set_weight_attrs(
- qweight, {
- "input_dim": 0,
- "output_dim": 1,
- "packed_dim": 1,
- "pack_factor": self.quant_config.pack_factor,
- })
- qzeros = Parameter(
- torch.empty(
- input_size_per_partition // self.quant_config.group_size,
- output_size_per_partition // self.quant_config.pack_factor,
- dtype=torch.int32,
- ),
- requires_grad=False,
- )
- set_weight_attrs(
- qzeros, {
- "input_dim": 0,
- "output_dim": 1,
- "packed_dim": 1,
- "pack_factor": self.quant_config.pack_factor,
- })
- scales = Parameter(
- torch.empty(
- input_size_per_partition // self.quant_config.group_size,
- output_size_per_partition,
- dtype=params_dtype,
- ),
- requires_grad=False,
- )
- set_weight_attrs(scales, {
- "input_dim": 0,
- "output_dim": 1,
- })
- return {
- "qweight": qweight,
- "qzeros": qzeros,
- "scales": scales,
- }
- else:
- weight = Parameter(torch.empty(output_size_per_partition,
- input_size_per_partition,
- dtype=params_dtype),
- requires_grad=False)
- set_weight_attrs(weight, {"input_dim": 1, "output_dim": 0})
- return {"weight": weight}
- def apply_weights(self,
- weights: Dict[str, torch.Tensor],
- x: torch.Tensor,
- bias: Optional[torch.Tensor] = None) -> torch.Tensor:
- if self.quant_config.quant_mode == "weight_only":
- qweight = weights["qweight"].data
- scales_zeros = weights["scales_zeros"].data
- pack_factor = self.quant_config.pack_factor
- out_shape = (x.shape[:-1] + (qweight.shape[-1] * pack_factor, ))
- reshaped_x = x.reshape(-1, x.shape[-1])
- out = ops.autoquant_s4_f16_gemm(reshaped_x, qweight, scales_zeros)
- if bias is not None:
- out = out + bias
- return out.reshape(out_shape)
- else:
- weight = weights["weight"]
- state = weights["state"]
- if weight.CB is not None:
- state.CB = weight.CB
- state.SCB = weight.SCB
- weight.CB = None
- weight.SCB = None
- import bitsandbytes as bnb
- out = bnb.matmul(x, weight, bias=bias, state=state)
- if not state.has_fp16_weights and \
- state.CB is not None and state.CxB is not None:
- # we converted 8-bit row major to turing/ampere format
- # in the first inference pass
- # we no longer need the row-major weight
- del state.CB
- weight.data = state.CxB
- return out
- def apply_moe_weights(self, w1: Dict[str,
- torch.Tensor], w2: Dict[str,
- torch.Tensor],
- x: torch.Tensor, gating_output: torch.Tensor,
- topk: int, renormalize: bool) -> torch.Tensor:
- raise NotImplementedError
- T = TypeVar("T", bound="torch.nn.Module")
- class QParams(NamedTuple):
- """A class to hold the quantization parameters."""
- scales: torch.Tensor
- zero_points: Optional[torch.Tensor]
- @torch.no_grad()
- def cal_qparams_per_group_minmax(w: torch.Tensor,
- n_bits: int = 4,
- group_size: int = 128):
- """Calculate quantization parameters for each group using min and max
- values."""
- outc, inc = w.shape
- assert inc >= group_size, \
- 'Input channels should be greater than or equal to group_size.'
- assert inc % group_size == 0, \
- 'Input channels should be divisible by group_size.'
- w_group_wise = w.reshape(outc, -1, group_size)
- w_min = w_group_wise.min(dim=-1, keepdim=True)[0]
- w_max = w_group_wise.max(dim=-1, keepdim=True)[0]
- q_max = 2**n_bits - 1
- q_min = 0
- scales = (w_max - w_min)
- scales = scales.clamp_(min=1e-5).div_(q_max)
- # zero_points = (-w_min / scales).round().clamp(q_min, q_max)
- zero_points = (-torch.round(w_min / scales)).clamp_(q_min, q_max)
- return QParams(scales=scales, zero_points=zero_points)
- def convert_s4(qw: torch.Tensor,
- qz: torch.Tensor,
- s: torch.Tensor,
- group_size: int = 128):
- assert qw.is_contiguous()
- assert qz.is_contiguous()
- assert s.is_contiguous()
- _qw = torch.zeros_like(qw)
- _sz = torch.zeros_like(s, dtype=torch.int32) # half2
- _ws = torch.zeros_like(s)
- ops.autoquant_convert_s4_k_m8(_qw, _sz, _ws, qw, s, qz,
- qw.size(-1) * 8, qw.size(0), group_size)
- return _qw, _sz
- def tp_m_s4(x: torch.Tensor, tp: int = 1):
- return x.view(x.size(0) // 32, tp, -1, 128).permute(0, 2, 3,
- 1).contiguous()
- def quant(weight: torch.Tensor,
- qparams: Optional[QParams] = None) -> torch.Tensor:
- """Perform fake quantization on the given weight tensor.
- Args:
- weight (torch.Tensor): The weight tensor with shape
- (out_features, in_features).
- qparams (Optional[QParams]): A namedtuple containing 'scales'
- and 'zero_points'.
- Returns:
- torch.Tensor: The fake quantized weight tensor.
- """
- if qparams is None:
- qparams = cal_qparams_per_group_minmax(weight)
- scales = qparams.scales
- zero_points = qparams.zero_points
- out_c, in_c = weight.shape
- # Reshape the weights if using per_group quantization
- # per tensor scales shape: [1]
- # per channel scales shape: [out_c, 1]
- # per group scales shape: [out_c, in_c//group_size, 1]
- if len(scales.shape) > 2:
- # scales shape: [out_c, in_c//group_size, 1]
- weight = weight.reshape(out_c, scales.shape[1], -1)
- if zero_points is None:
- real_qweight = (weight / scales).round()
- else:
- real_qweight = ((weight + (scales * zero_points)) / scales).round()
- if len(scales.shape) > 2:
- real_qweight = real_qweight.reshape(out_c, in_c)
- return real_qweight.to(torch.int32)
- # core quantization method (simulated quantization)
- def quantize_tensor(
- weight,
- n_bits=4,
- group_size=128,
- ):
- pack_num = 32 // n_bits
- pack_order = [0, 2, 4, 6, 1, 3, 5, 7]
- org_weight_shape = weight.shape
- out_features = org_weight_shape[0]
- in_features = org_weight_shape[1]
- qparams = cal_qparams_per_group_minmax(weight, n_bits)
- i32_w = quant(weight, qparams)
- i32_w = i32_w.t().contiguous()
- w_pack_oc = out_features // (32 // n_bits)
- w_inc = in_features
- pack_int_w = torch.zeros((w_inc, w_pack_oc),
- dtype=torch.int32,
- device=weight.device)
- for col in range(pack_int_w.shape[1]):
- for i in range(pack_num):
- pack_int_w_col = i32_w[:, col * pack_num + pack_order[i]]
- pack_int_w[:, col] |= pack_int_w_col << (i * n_bits)
- qweight = pack_int_w
- scales = qparams.scales.squeeze(-1).t().contiguous()
- if qparams.zero_points is not None:
- zeros = qparams.zero_points.to(torch.int32)
- zeros = zeros.squeeze(-1).t().contiguous()
- z_inc = in_features // group_size
- z_oc = out_features // (32 // n_bits)
- pack_int_zeros = torch.zeros((z_inc, z_oc),
- dtype=torch.int32,
- device=weight.device)
- for col in range(pack_int_zeros.shape[1]):
- for i in range(pack_num):
- qzero_col = zeros[:, col * pack_num + pack_order[i]]
- pack_int_zeros[:, col] |= qzero_col << (i * n_bits)
- qzeros = pack_int_zeros
- return qweight, scales, qzeros
- def replace_quant_params(model,
- quant_config,
- modules_to_not_convert="lm_head"):
- """
- modules_to_not_convert (`str`, *optional*, defaults to `lm_head`):
- Name of the module to not convert in `Linear8bitLt`.
- In practice we keep the `lm_head` in full precision
- for numerical stability reasons.
- """
- if not isinstance(modules_to_not_convert, list):
- modules_to_not_convert = [modules_to_not_convert]
- for name, module in model.named_children():
- if len(list(module.children())) > 0:
- replace_quant_params(module, quant_config, modules_to_not_convert)
- if isinstance(
- module,
- (ColumnParallelLinear, QKVParallelLinear, RowParallelLinear)) \
- and name not in modules_to_not_convert:
- if quant_config.from_float:
- module.linear_weights.pop("weight")
- param = module._parameters["weight"]
- if quant_config.quant_mode in ("llm_int8", "smoothquant"):
- import bitsandbytes as bnb
- new_value = bnb.nn.Int8Params(param.data,
- requires_grad=False,
- has_fp16_weights=False)
- state = bnb.MatmulLtState()
- if quant_config.quant_mode == "smoothquant":
- state.threshold = 0.0
- else:
- state.threshold = 6.0
- state.has_fp16_weights = False
- state.memory_efficient_backward = False
- state.use_pool = True
- module._parameters["weight"] = new_value
- module.linear_weights["weight"] = new_value
- module.linear_weights["state"] = state
- set_weight_attrs(
- new_value, {
- "input_dim": 0,
- "output_dim": 1,
- "packed_dim": 1,
- "pack_factor": quant_config.pack_factor,
- })
- del param
- torch.cuda.empty_cache()
- elif quant_config.quant_mode == "weight_only":
- data_fp = param.cuda()
- _qweight, _scales, _qzeros = quantize_tensor(
- data_fp, n_bits=4, group_size=128)
- qweight, scales_zeros = convert_s4(_qweight, _qzeros,
- _scales)
- torch.cuda.synchronize()
- param_qweight = Parameter(qweight, requires_grad=False)
- param_scales_zeros = Parameter(scales_zeros,
- requires_grad=False)
- module.register_parameter("qweight", param_qweight)
- module.register_parameter("scales_zeros",
- param_scales_zeros)
- set_weight_attrs(
- param_qweight, {
- "input_dim": 0,
- "output_dim": 1,
- "packed_dim": 1,
- "pack_factor": quant_config.pack_factor,
- })
- set_weight_attrs(param_scales_zeros, {
- "input_dim": 0,
- "output_dim": 1,
- })
- module.linear_weights["qweight"] = param_qweight
- module.linear_weights["scales_zeros"] = param_scales_zeros
- del _qzeros
- del _scales
- del param
- delattr(module, "weight")
- torch.cuda.empty_cache()
- else: # load packed int4 weight
- module.linear_weights.pop("qweight")
- module.linear_weights.pop("qzeros")
- module.linear_weights.pop("scales")
- _qweight = module._parameters["qweight"]
- _qzeros = module._parameters["qzeros"]
- _scales = module._parameters["scales"]
- qweight, scales_zeros = convert_s4(_qweight.data, _qzeros.data,
- _scales.data)
- param_qweight = Parameter(qweight, requires_grad=False)
- param_scales_zeros = Parameter(scales_zeros,
- requires_grad=False)
- del _qweight
- del _qzeros
- del _scales
- delattr(module, "qweight")
- delattr(module, "qzeros")
- delattr(module, "scales")
- module.register_parameter("qweight", param_qweight)
- module.register_parameter("scales_zeros", param_scales_zeros)
- set_weight_attrs(
- param_qweight, {
- "input_dim": 0,
- "output_dim": 1,
- "packed_dim": 1,
- "pack_factor": quant_config.pack_factor,
- })
- set_weight_attrs(param_scales_zeros, {
- "input_dim": 0,
- "output_dim": 1,
- })
- module.linear_weights["qweight"] = param_qweight
- module.linear_weights["scales_zeros"] = param_scales_zeros
- torch.cuda.synchronize()
- torch.cuda.empty_cache()
|