AWQ: Activation-aware Weight Quantization for LLM Compression and Acceleration [Paper]
Efficient and accurate low-bit weight quantization (INT3/4) for LLMs, supporting instruction-tuned models and multi-modal LMs.
The current release supports:
- AWQ search for accurate quantization.
- Pre-computed AWQ model zoo for LLMs (LLaMA-1&2, OPT, Vicuna, LLaVA; load to generate quantized weights).
- Memory-efficient 4-bit Linear in PyTorch.
- Efficient CUDA kernel implementation for fast inference (support context and decoding stage).
- Examples on 4-bit inference of an instruction-tuned model (Vicuna) and multi-modal LM (LLaVA).
Check out TinyChat, which delievers 30 tokens/second inference performance (3.2x faster than FP16) for the LLaMA-2 chatbot on the resource-constrained NVIDIA Jetson Orin!
It also offers a turn-key solution for on-device inference of LLMs on resource-constrained edge platforms. With TinyChat, it is now possible to run large models on small and low-power devices even without Internet connection.
- [2023/10] AWQ is integrated into NVIDIA TensorRT-LLM
- [2023/09] AWQ is integrated into FastChat, vLLM, HuggingFace TGI, and LMDeploy.
- [2023/09] ⚡ Check out our latest TinyChat, which is ~2x faster than the first release on Orin!
- [2023/09] ⚡ Check out AutoAWQ, a third-party implementation to make AWQ easier to expand to new models, improve inference speed, and integrate into Huggingface.
- [2023/07] 🔥 We released TinyChat, an efficient and lightweight chatbot interface based on AWQ. TinyChat enables efficient LLM inference on both cloud and edge GPUs. LLama-2-chat models are supported! Check out our implementation here.
- [2023/07] 🔥 We added AWQ support and pre-computed search results for Llama-2 models (7B & 13B). Checkout our model zoo here!
- [2023/07] We extended the support for more LLM models including MPT, Falcon, and BLOOM.
- Clone this repository and navigate to AWQ folder
git clone https://github.com/mit-han-lab/llm-awq
cd llm-awq
- Install Package
conda create -n awq python=3.10 -y
conda activate awq
pip install --upgrade pip # enable PEP 660 support
pip install -e .
-
For edge devices like Orin, before running the commands above, please:
- Modify pyproject.toml by commenting out this line.
- Set this line to transformers==4.32.0.
- Manually install precompiled PyTorch binaries (>=2.0.0) from NVIDIA.
- Set the appropriate Python version for conda environment (e.g.,
conda create -n awq python=3.8 -y
for JetPack 5).
- Install efficient W4A16 (4-bit weight, 16-bit activation) CUDA kernel and optimized FP16 kernels (e.g. layernorm, positional encodings).
cd awq/kernels
python setup.py install
We provide pre-computed AWQ search results for multiple model families, including LLaMA, OPT, Vicuna, and LLaVA. To get the pre-computed AWQ search results, run:
# git lfs install # install git lfs if not already
git clone https://huggingface.co/datasets/mit-han-lab/awq-model-zoo awq_cache
The detailed support list:
Models | Sizes | INT4-g128 | INT3-g128 |
---|---|---|---|
LLaMA-2 | 7B/7B-chat/13B/13B-chat | ✅ | ✅ |
LLaMA | 7B/13B/30B/65B | ✅ | ✅ |
OPT | 125m/1.3B/2.7B/6.7B/13B/30B | ✅ | ✅ |
Vicuna-v1.1 | 7B/13B | ✅ | |
LLaVA-v0 | 13B | ✅ |
AWQ can be easily applied to various LMs thanks to its good generalization, including instruction-tuned models and multi-modal LMs. It provides an easy-to-use tool to reduce the serving cost of LLMs.
Here we provide two examples of AWQ application: Vicuna-7B (chatbot) and LLaVA-13B (visual reasoning) under ./examples
directory. AWQ can easily reduce the GPU memory of model serving and speed up token generation. It provides accurate quantization, providing reasoning outputs. You should be able to observe memory savings when running the models with 4-bit weights.
Note that we perform AWQ using only textual calibration data, depsite we are running on multi-modal input. Please refer to ./examples
for details.
We provide several sample script to run AWQ (please refer to ./scripts
). We use OPT-6.7B as an example.
- Perform AWQ search and save search results (we already did it for you):
python -m awq.entry --model_path /PATH/TO/OPT/opt-6.7b \
--w_bit 4 --q_group_size 128 \
--run_awq --dump_awq awq_cache/opt-6.7b-w4-g128.pt
- Evaluate the AWQ quantized model on WikiText-2 (simulated pseudo quantization)
python -m awq.entry --model_path /PATH/TO/OPT/opt-6.7b \
--tasks wikitext \
--w_bit 4 --q_group_size 128 \
--load_awq awq_cache/opt-6.7b-w4-g128.pt \
--q_backend fake
- Generate real quantized weights (INT4)
mkdir quant_cache
python -m awq.entry --model_path /PATH/TO/OPT/opt-6.7b \
--w_bit 4 --q_group_size 128 \
--load_awq awq_cache/opt-6.7b-w4-g128.pt \
--q_backend real --dump_quant quant_cache/opt-6.7b-w4-g128-awq.pt
- Load and evaluate the real quantized model (now you can see smaller gpu memory usage)
python -m awq.entry --model_path /PATH/TO/OPT/opt-6.7b \
--tasks wikitext \
--w_bit 4 --q_group_size 128 \
--load_quant quant_cache/opt-6.7b-w4-g128-awq.pt
If you find AWQ useful or relevant to your research, please kindly cite our paper:
@article{lin2023awq,
title={AWQ: Activation-aware Weight Quantization for LLM Compression and Acceleration},
author={Lin, Ji and Tang, Jiaming and Tang, Haotian and Yang, Shang and Dang, Xingyu and Han, Song},
journal={arXiv},
year={2023}
}
SmoothQuant: Accurate and Efficient Post-Training Quantization for Large Language Models
GPTQ: Accurate Post-training Compression for Generative Pretrained Transformers
LLaVA: Large Language and Vision Assistant
$ python -m awq.entry --model_path /data/opt-1.3b-hf/ --w_bit 8 --run_awq --dump_awq awq_cache/opt-1.3b-w8.pt
Quantization config: {'zero_point': True, 'q_group_size': -1}
* Building model /data/opt-1.3b-hf/
* Split into 60 blocks
Running AWQ...: 0%| | 0/24 [00:00<?, ?it/s]
[('model.decoder.layers.0.self_attn_layer_norm', ('model.decoder.layers.0.self_attn.q_proj', 'model.decoder.layers.0.self_attn.k_proj', 'model.decoder.layers.0.self_attn.v_proj'), tensor([0.9119, 1.3075, 0.8703, ..., 1.0233, 1.0013, 0.9770])), ('model.decoder.layers.0.self_attn.v_proj', ('model.decoder.layers.0.self_attn.out_proj',), tensor([1.3301, 1.2490, 1.2283, ..., 1.1685, 1.2393, 1.2438])), ('model.decoder.layers.0.final_layer_norm', ('model.decoder.layers.0.fc1',), tensor([0.2656, 0.4866, 0.3155, ..., 0.2604, 0.3211, 0.3796])), ('model.decoder.layers.0.fc1', ('model.decoder.layers.0.fc2',), tensor([0.3859, 0.3912, 0.3935, ..., 0.3041, 0.4645, 0.4367]))]
Running AWQ...: 4%|████████▉ | 1/24 [00:25<09:54, 25.84s/it]
[('model.decoder.layers.0.self_attn_layer_norm', ('model.decoder.layers.0.self_attn.q_proj', 'model.decoder.layers.0.self_attn.k_proj', 'model.decoder.layers.0.self_attn.v_proj'), tensor([0.9119, 1.3075, 0.8703, ..., 1.0233, 1.0013, 0.9770])), ('model.decoder.layers.0.self_attn.v_proj', ('model.decoder.layers.0.self_attn.out_proj',), tensor([1.3301, 1.2490, 1.2283, ..., 1.1685, 1.2393, 1.2438])), ('model.decoder.layers.0.final_layer_norm', ('model.decoder.layers.0.fc1',), tensor([0.2656, 0.4866, 0.3155, ..., 0.2604, 0.3211, 0.3796])), ('model.decoder.layers.0.fc1', ('model.decoder.layers.0.fc2',), tensor([0.3859, 0.3912, 0.3935, ..., 0.3041, 0.4645, 0.4367])), ('model.decoder.layers.1.self_attn_layer_norm', ('model.decoder.layers.1.self_attn.q_proj', 'model.decoder.layers.1.self_attn.k_proj', 'model.decoder.layers.1.self_attn.v_proj'), tensor([0.9036, 0.6764, 0.6403, ..., 0.8947, 0.9801, 0.8870])), ('model.decoder.layers.1.self_attn.v_proj', ('model.decoder.layers.1.self_attn.out_proj',), tensor([1.3371, 1.3844, 1.4209, ..., 1.4862, 1.4690, 1.2799])), ('model.decoder.layers.1.final_layer_norm', ('model.decoder.layers.1.fc1',), tensor([0.7314, 0.5508, 0.7057, ..., 0.4870, 0.6115, 0.5066])), ('model.decoder.layers.1.fc1', ('model.decoder.layers.1.fc2',), tensor([0.6070, 1.6315, 0.5499, ..., 1.7474, 0.4939, 1.7453]))]
Running AWQ...: 8%|█████████████████▉ | 2/24 [00:46<08:26, 23.03s/it]
[('model.decoder.layers.0.self_attn_layer_norm', ('model.decoder.layers.0.self_attn.q_proj', 'model.decoder.layers.0.self_attn.k_proj', 'model.decoder.layers.0.self_attn.v_proj'), tensor([0.9119, 1.3075, 0.8703, ..., 1.0233, 1.0013, 0.9770])), ('model.decoder.layers.0.self_attn.v_proj', ('model.decoder.layers.0.self_attn.out_proj',), tensor([1.3301, 1.2490, 1.2283, ..., 1.1685, 1.2393, 1.2438])), ('model.decoder.layers.0.final_layer_norm', ('model.decoder.layers.0.fc1',), tensor([0.2656, 0.4866, 0.3155, ..., 0.2604, 0.3211, 0.3796])), ('model.decoder.layers.0.fc1', ('model.decoder.layers.0.fc2',), tensor([0.3859, 0.3912, 0.3935, ..., 0.3041, 0.4645, 0.4367])), ('model.decoder.layers.1.self_attn_layer_norm', ('model.decoder.layers.1.self_attn.q_proj', 'model.decoder.layers.1.self_attn.k_proj', 'model.decoder.layers.1.self_attn.v_proj'), tensor([0.9036, 0.6764, 0.6403, ..., 0.8947, 0.9801, 0.8870])), ('model.decoder.layers.1.self_attn.v_proj', ('model.decoder.layers.1.self_attn.out_proj',), tensor([1.3371, 1.3844, 1.4209, ..., 1.4862, 1.4690, 1.2799])), ('model.decoder.layers.1.final_layer_norm', ('model.decoder.layers.1.fc1',), tensor([0.7314, 0.5508, 0.7057, ..., 0.4870, 0.6115, 0.5066])), ('model.decoder.layers.1.fc1', ('model.decoder.layers.1.fc2',), tensor([0.6070, 1.6315, 0.5499, ..., 1.7474, 0.4939, 1.7453])), ('model.decoder.layers.2.self_attn_layer_norm', ('model.decoder.layers.2.self_attn.q_proj', 'model.decoder.layers.2.self_attn.k_proj', 'model.decoder.layers.2.self_attn.v_proj'), tensor([0.8909, 0.8273, 0.6141, ..., 0.8716, 0.9316, 0.8075])), ('model.decoder.layers.2.self_attn.v_proj', ('model.decoder.layers.2.self_attn.out_proj',), tensor([1.0159, 0.9391, 1.3082, ..., 0.7308, 1.2391, 1.5200])), ('model.decoder.layers.2.final_layer_norm', ('model.decoder.layers.2.fc1',), tensor([0.5209, 0.4312, 0.6109, ..., 0.4170, 0.4209, 0.4315])), ('model.decoder.layers.2.fc1', ('model.decoder.layers.2.fc2',), tensor([0.4667, 0.4628, 0.5028, ..., 0.4864, 0.4994, 0.4969]))]
Running AWQ...: 12%|██████████████████████████▉ | 3/24 [01:07<07:37, 21.77s/it]
...
AWQ results saved at awq_cache/opt-1.3b-w8.pt