- Introduction
- Quantization Fundamentals
- SmoothQuant and Our Enhancement
- Validated Models
- Example
- Supported Framework Matrix
Quantization is a common compression operation to reduce memory and accelerate inference by converting the floating point matrix to an integer matrix. For large language models (LLMs) with gigantic parameters, the systematic outliers make quantification of activations difficult. SmoothQuant, a training free post-training quantization (PTQ) solution, offline migrates this difficulty from activations to weights with a mathematically equivalent transformation.
Quantization is a common compression operation to reduce memory and accelerate inference; therefore, the difficulty of LLM deployment can be alleviated. Quantization converts the floating point matrix to an integer matrix.
The equation of quantization is as follows:
where
There are several choices of sharing quantization parameters among tensor elements, also called quantization granularity. The coarsest level, per-tensor granularity, is that all elements in the tensor share the same quantization parameters. Finer granularity means sharing quantization parameters per row or per column for 2D matrices and per channel for 3D matrices. Similarly, the finest granularity is that each element has an individual set of quantization parameters.
However, due to the model accuracy and computational consumption, per-tensor or per-channel are usually adopted. In the following part, We will show that per-channel could bring lower quantization loss but has some limitations, that is why normally we use per-channel for weight quantization and per-tensor for activation/input quantization
Suppose the weight tensor is:
import torch
W = torch.Tensor(
[[0.6839, 0.4741, 0.7451],
[0.9301, 0.1742, 0.6835]]
)
According to the formula (1), we need scale
The per-tensor quantization function is:
def quantize(x, num_bits=8):
q_min, q_max = 0, 2. ** num_bits - 1.
scale = (torch.max(x) - torch.min(x)) / (2 ** num_bits - 1)
scale = torch.clip(scale, min=1e-5)
zp = torch.round(0 - (torch.min(x)) / scale)
q_x = x / scale + zp
q_x.clamp_(q_min, q_max).round_()
print(f'scale = {scale}, zp = {zp}')
return q_x, scale, zp
Then we can get the quantized
>>> W_q, scale, zp = quantize(W)
scale = 0.00296431384049356, zp = -59.0
>>> W_q
tensor([[172., 101., 192.],
[255., 0., 172.]])
With the value of scale and zp, we can dequantize the tensor.
def dequantize(q_x, scale, zp):
return scale * (q_x - zp)
>>> W_dq = dequantize(W_dq, 0.001, -50)
>>> W_dq
tensor([[0.1220, 0.0500, 0.1430],
[0.2570, 0.0500, 0.1890]])
>>> loss = torch.nn.MSELoss()(W_dq, W)
>>> loss.item()
0.1983354538679123
>>> W_dq = dequantize(W_q, scale, zp)
>>> W_dq
tensor([[0.6848, 0.4743, 0.7440],
[0.9308, 0.1749, 0.6848]])
>>> loss = torch.nn.MSELoss()(W_dq, W)
>>> loss.item()
7.385297635664756e-07
The difference between
Similarly, the example of per-channel quantization is as follows:
def quantize_per_channel(x, num_bits=8):
q_min, q_max = 0, 2. ** num_bits - 1.
x_tmp = x.detach().reshape(x.shape[0], -1)
scales = x_tmp.max(dim=-1, keepdim=True)[0] / (2 ** num_bits - 1)
zp = torch.round(0 - x_tmp.min(dim=-1, keepdim=True)[0].divide(scales))
q_x = x_tmp.divide(scales) + zp
q_x.clamp_(q_min, q_max).round_()
print(f'scale = {scales}, \n zp = {zp}')
return q_x, scale, zp
def dequantize_per_channel(q_x, scales, zp):
print(q_x, scales, zp)
print(scales * (q_x - zp))
return scales * (q_x - zp)
>>>W_q, scale, zp = quantize_per_channel(W)
scale = tensor([[0.0029],
[0.0036]]),
zp = tensor([[-162.],
[ -48.]])
>>>W_q
tensor([[ 72., 0., 93.],
[207., 0., 139.]])
>>>W_dq = dequantize_per_channel(W_q, scales, zp)
>>>W_dq
tensor([[0.6837, 0.4734, 0.7451],
[0.9301, 0.1751, 0.6821]])
And the loss is
>>> loss = torch.nn.MSELoss()(W_dq, W)
>>> loss.item()
5.637690492221736e-07
Through this example, we can see that per-channel quantization has finer granularity and has lower loss (loss 5.6376e-07 for per-channel quantization and 7.3852e-07 for per-tensor quantization).
For a linear layer in most model,
def quantize_per_tensor_absmax(x, n_bits=8):
scales = x.abs().max()
q_max = 2**(n_bits-1)-1
scales.clamp_(min=1e-5).div_(q_max)
q_x = x / scales
q_x = q_x.clamp_(-q_max, q_max).round_()
return q_x, scales
def dequantize(q_x, scale):
return scale * q_x
Randomly initialize the
>>>W = torch.rand(2, 3, dtype=torch.float32)
>>>X = torch.rand(3, 4, dtype=torch.float32)
>>>W
tensor([[0.0806, 0.7589, 0.6038],
[0.3815, 0.5040, 0.7174]])
>>>X
tensor([[0.5444, 0.5826, 0.7772, 0.5555],
[0.3740, 0.3253, 0.0698, 0.1381],
[0.5972, 0.0086, 0.0737, 0.8298]])
>>>Y = torch.matmul(W, X)
>>>Y
tensor([[0.6883, 0.2991, 0.1601, 0.6506],
[0.8246, 0.3924, 0.3845, 0.8768]])
Quantize weight and activation, matmul(quantize(X), quantize(Y))
>>>W_q, W_scale = quantize_per_tensor_absmax(W)
>>>X_q, X_scale = quantize_per_tensor_absmax(X)
>>>print(f'{W_q}\n{W_scale.item()}')
>>>print(f'{X_q}\n{X_scale.item()}')
tensor([[ 13., 127., 101.],
[ 64., 84., 120.]])
0.0059755356051027775
tensor([[ 83., 89., 119., 85.],
[ 57., 50., 11., 21.],
[ 91., 1., 11., 127.]])
0.006533813662827015
>>>Y_q = torch.matmul(W_q, X_q)
>>>Y_q
tensor([[17509., 7608., 4055., 16599.],
[21020., 10016., 9860., 22444.]])
>>>Y_dq = dequantize(Y, W_scale * X_scale)
>>>Y_dq
tensor([[0.6836, 0.2970, 0.1583, 0.6481],
[0.8207, 0.3911, 0.3850, 0.8763]])
Though per-channel quantization could bring lower quantization error, we could not apply it for activations due to the difficulty of the dequantization. We would prove it in the following image and the zero point of quantization would be ignored for simplicity.
The image on the left presents a normal linear forward with 1x2 input
In the previous subsection, we have explained why per-channel quantization could not be applied for activation, even though it could lead to lower quantization loss. However, the quantization error loss of activation plays an important role in the accuracy loss of model quantization123.
To reduce the quantization loss of activations, lots of methods have been proposed. In the following, we briefly introduce SPIQ1, Outlier Suppression2 and Smoothquant3. All these three methods share a similar idea to migrate the difficulty from activation quantization to weight quantization but differ in how much difficulty to be transferred.
So the first question is how to migrate the difficulty from activation to weights? The solution is straightforward, that is to convert the network to an output equivalent network that is presented in the image below and apply quantization to this equivalent network. The intuition is that each channel of activation could be scaled to make it more quantization-friendly, similar to a fake per-channel activation quantization.
Please note that this conversion will make the quantization of weights more difficult, because the scales attached to weights shown above are per-input-channel, while quantization of weights is per-output-channel or per-tensor.
So the second question is how much difficulty to be migrated, that is how to choose the conversion per-channel scale
SPIQ just adopts the quantization scale of activations as the conversion per-channel scale.
Outlier suppression adopts the scale of the preceding layernorm as the conversion per-channel scale.
Smoothquant introduces a hyperparameter
j is the index of the input channels.
For most of the models such as OPT and BLOOM,
SmoothQuant method aims to split the quantization difficulty of weight and activation by using a fixed-value
Our proposed method consists of 5 major steps:
- Hook input and output values of all layers using register_forward_hook.
- Generate a list of
$\alpha$ values given user-defined$\alpha$ range and step_sizes. - Recalculate smoothing factor given an
$\alpha$ value and adjust parameters(weights and activations). - Perform per-channel quantization_dequantization of weights and per-tensor quantization_dequantization of inputs to predict the layer-wise output corresponding to the given
$\alpha$ value. - Calculate the mean-squared loss with respect to the actual output value, recover the adjusted parameters and save the layer-wise optimal
$\alpha$ values.
Multiple criteria (e.g min, max and mean) are supported to determine the
In our experiments, an
fully automated: the user only needs to pass a model and dataloader
from neural_compressor.adaptor.torch_utils.smooth_quant import TorchSmoothQuant
sq = TorchSmoothQuant(model, dataloader)
sq.transform(alpha) ##alpha could be a float or a string 'auto'
please note that we rely on torch jit to analyze the model. If you are using huggingface model, you could set torchscript to True when loading the model or set the return_dict to False"
support lots of fusing patterns: when applying the conversion per-channel scales, a mul layer needs to be inserted, which will introduce some overhead. The official code fuses this op to the previous layernorm, while we support more fusing patterns, like linear_1->relu->linear_2, which means the scales of linear_1 will be fused to linear_2. All the supported patterns are shown below. Currently we only handle the layer whose scale could be fused, we are trying to support other layers, please stay tuned.
conv2d/linear->relu/leakyrelu/hardtanh->conv2d/linear/layernorm/batchnorm/instancenorm/t5norm/llamanorm/groupnorm/
conv2d/linear->conv2d/linear/layernorm/batchnorm/instancenorm/t5norm/llamanorm/groupnorm
Neural Compressor: 2.1
IPEX (Intel Extension for PyTorch): 2.0
Dataset: lambada
Task: text-generation
alpha [0.4, 0.6] is sweet spot region in SmoothQuant paper
Model\Last token accuracy | FP32 | INT8 (w/o SmoothQuant) | INT8 (w/ SmoothQuant) | INT8 (w/ SmoothQuant auto tuning) |
---|---|---|---|---|
bigscience/bloom-560m | 65.20% | 63.44% | 66.48% (alpha=0.5) | 64.76% (alpha: 95.9% over 0.6, 4.1% in [0.4, 0.6]) |
bigscience/bloom-1b7 | 71.43% | 67.78% | 72.56% (alpha=0.5) | 72.58% (alpha: 55.1% over 0.6, 30.6% in [0.4, 0.6], 14.3% under 0.4) |
bigscience/bloom-3b | 73.97% | 69.99% | 74.02% (alpha=0.5) | 74.16% (alpha: 100% over 0.6) |
bigscience/bloom-7b1 | 77.44% | 75.46% | 77.02%(alpha=0.5) | 77.45% (alpha: 91.8% over 0.6, 4.9% in [0.4, 0.6], 3.3% under 0.4) |
bigscience/bloom-176b | 84.17% | 82.13% | 83.52% (alpha=0.6) | - |
facebook/opt-125m | 63.89% | 63.48% | 63.44% (alpha=0.5) | 64.14% (alpha: 59.4% over 0.6, 8.1% in [0.4, 0.6], 32.4% under 0.4) |
facebook/opt-1.3b | 75.41% | 73.59% | 70.94% (alpha=0.5) | 74.80% (alpha: 69.9% over 0.6, 24.7% in [0.4, 0.6], 5.5% under 0.4) |
facebook/opt-2.7b | 77.79% | 78.57% | 78.60%(alpha=0.5) | 78.25% (alpha: 73.2% over 0.6, 21.6% in [0.4, 0.6], 5.2% under 0.4) |
facebook/opt-6.7b | 81.26% | 76.65% | 81.58%(alpha=0.5) | 81.39% (alpha: 68.0% over 0.6, 26.8% in [0.4, 0.6], 5.2% under 0.4) |
EleutherAI/gpt-j-6B | 79.17% | 78.82% | 78.84%(alpha=0.6) | 79.29% (alpha: 96.4% over 0.6, 3.6% in [0.4, 0.6]) |
User could refer to examples on how to use smooth quant.
recipes = {
"smooth_quant": True,
"smooth_quant_args": {
"alpha": 0.5,
"folding": True,
},
}
conf = PostTrainingQuantConfig(recipes=recipes)
smooth_quant_args description:
"alpha": "auto", a float value or a list of float values. Default is 0.5. "auto" means automatic tuning.
"folding": whether to fold mul into the previous layer, where mul is required to update the input distribution during smoothing.
- True: Fold inserted mul into the previous layer. IPEX will only insert mul for layers can do folding.
- False: Allow inserting mul to update the input distribution and no folding. IPEX (version>=2.1) can fuse inserted mul automatically. For Stock PyTorch, setting folding=False will convert the model to a QDQ model.
To find the best alpha
, users can utilize the auto-tuning feature. Compares to setting the alpha to "auto"
, this tuning process uses the evaluation result on the entire dataset as the metric to find the best alpha
. To use this feature, users need to provide a list of scalars between 0 and 1 for the alpha
item. Here is an example:
import numpy as np
conf = PostTrainingQuantConfig(
quant_level='auto', # quant_level can also be 1
...
recipes={"smooth_quant": True,
"smooth_quant_args": {"alpha": np.arange(0.1, 0.5, 0.05).tolist(),}
...
})
Framework | Alpha | Folding |
---|---|---|
PyTorch | [0-1] / 'auto' | False |
IPEX | [0-1] / 'auto' | True / False(Version>2.1) |
ONNX | [0-1] | True |
Tensorflow | [0-1] | False |
ITEX | [0-1] | False |
Footnotes
-
Yvinec, Edouard, et al. "SPIQ: Data-Free Per-Channel Static Input Quantization." Proceedings of the IEEE/CVF Winter Conference on Applications of Computer Vision. 2023. ↩ ↩2
-
Wei, Xiuying, et al. "Outlier suppression: Pushing the limit of low-bit transformer language models." arXiv preprint arXiv:2209.13325 (2022). ↩ ↩2
-
Xiao, Guangxuan, et al. "Smoothquant: Accurate and efficient post-training quantization for large language models." arXiv preprint arXiv:2211.10438 (2022). ↩ ↩2