The code base for project Layer-Condensed KV Cache, a new variant of transformer decoders in which queries of all layers are paired with keys and values of just the top layer. It reduces the memory and computation cost, reduces the number of parameters, significantly improves the inference throughput with comparable or better task performance. The paper "Layer-Condensed KV Cache for Efficient Inference of Large Language Models" was accepted to ACL 2024 main conference.
This work is inspired by Probabilistic Transformer, where we consider the stacking layer structure of a transformer as an iterative process of improving token representation.
- [24/12/08] We release the main branch, with a general framework for Cross-Layer KV Sharing. A illustrative post can be found on PaperWeekly (in Chinese). See the published branch for the old version of the code.
- [24/10/18] Our new empirical study "A Systematic Study of Cross-Layer KV Sharing for Efficient LLM Inference" has released on arXiv. A new configuration has been found to be more efficient than the original LCKV.
- [24/05/28] This code base now also supports Cross-Layer Attention (CLA). The idea is similar, but they 1) divide the transformer layers into small groups with 2-4 layers in each group; 2) pairs the queries of all the layers with the keys and values of the bottom layer in each group. See details in their paper "Reducing Transformer Key-Value Cache Size with Cross-Layer Attention".
- [24/05/20] LCKV initial paper and code release.
- [24/05/12] Our paper was accepted to ACL 2024 main conference.
- [24/02/14] Our paper "Layer-Condensed KV Cache for Efficient Inference of Large Language Models" was submitted to ARR February 2024 cycle.
We have released a series of pre-trained models described in our paper on HuggingFace. There is no need to clone this repo if you just want to use the pre-trained models. Load the model with the following code:
# Use a pipeline as a high-level helper
from transformers import pipeline
pipe = pipeline("text-generation", model="whynlp/tinyllama-lckv-w2-ft-100b", trust_remote_code=True)
# Load model directly
from transformers import AutoModelForCausalLM
model = AutoModelForCausalLM.from_pretrained("whynlp/tinyllama-lckv-w2-ft-100b", trust_remote_code=True)
See more models on the HuggingFace model hub. Note that these models are for research purposes only and may not be suitable for production.
Model | Paper Section | Dev ppl. | Common-sense Reasoning |
---|---|---|---|
whynlp/tinyllama-lckv-w10-ft-250b | -- | 7.939 | 50.86 |
whynlp/tinyllama-lckv-w2-ft-100b | Appendix C.1, Table 7 (line 5) | 8.514 | 49.55 |
whynlp/tinyllama-lckv-w10-100b | Section 3.2, Table 2 (line 3) | 9.265 | 46.84 |
whynlp/tinyllama-lckv-w2-100b | Section 3.2, Table 2 (line 2) | 9.746 | 45.45 |
You may install the dependencies with the following commands:
conda install pytorch pytorch-cuda=12.1 -c pytorch -c nvidia
pip install -r requirements.txt
where the CUDA version is set to 12.1
. For other CUDA versions, please refer to installation instructions of PyTorch. See Trouble shooting for more details.
Our implementation is based on HuggingFace transformers
. We register a new model lckv-llama
that supports the Layer-Condensed KV Cache. It inherits from the llama
model and adds support for the Layer-Condensed KV Cache.
Note
It is difficult to support the Layer-Condensed KV Cache for a variety of models with a small amount of code. This is because the Layer-Condensed KV Cache requires to modify the attention mechanism and training recipe of the transformer decoder. Currently, we only implemented the Layer-Condensed KV Cache for the llama
model, and it is possible to extend it to other models with similar structures.
import models # register the lckv-llama model
from transformers import AutoModelForCausalLM, AutoTokenizer
tokenizer = AutoTokenizer.from_pretrained("huggyllama/llama-7b")
model = AutoModelForCausalLM.from_config(config="configs/tinyllama_lckv.json")
and now you have a randomly initialized model with the Layer-Condensed KV Cache.
To accelerate the training and inference of the model, one could apply the liger kernel supported by transformers
library. The provided training script run_clm.py
has already activated the liger kernel. See more details here.
We provide some sample configuration files in the configs
folder. The config settings are defined in models/configuration_lckv.py. You may refer to this file for more details.
from models import LCKVLlamaConfig
# we have prepared a sample configuration file
config = LCKVLlamaConfig.from_pretrained("configs/tinyllama_lckv.json")
# below is the LCKV config. you may modify the configuration as you like
config.forward_passes = 7 # m in the paper
config.backward_passes = 2 # b in the paper
config.layer_types = "0_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_21" # for each layer, which layer to attend to
# we also support this
config.layer_types = "0_10_10_10_10_10_10_10_10_10_10_10_10_10_10_10_10_10_10_10_10_21" # the sandwich-middle configuration
config.layer_types = "0_1_2_3_4_5_6_7_8_9_10_11_12_13_14_15_16_17_18_19_20_21" # Llama config
config.layer_types = "0_0_2_2_4_4_6_6_8_8_10_10_12_12_14_14_16_16_18_18_20_20" # CLA config
config.sliding_window = 1024 # the window size for the sliding window attention
config.layer_types = "0s_1s_2s_3s_4s_5s_6s_7s_8s_9s_10s_11_11_11_11_11_11_11_11_11_11_11" # YOCO config, 's' is for sliding window
config.sliding_window = 1024 # the window size for the sliding window attention
config.layer_types = "0_1s_1s_3s_3s_3s_0_7s_7s_9s_9s_9s_12_13s_13s_15s_15s_15s_12_19s_19s_19s" # MixAttention (Pairs) config
# we also support sequential training / inference, which will process the tokens one by one
# corresponding to LCKV paper Figure 2(a)
config.use_sequential = True
accelerate launch run_clm.py \
--config_name configs/tinyllama_lckv.json \
--config_overrides forward_passes=7,backward_passes=2,layer_types=0_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_20_21 \
...
With the above configurations, you can create CLA, YOCO or any configurations in Cross-Layer KV Sharing or MixAttention without changing the code. The only thing you need to do is to write the correct layer_types
in the configuration file.
We use the same training script as the original transformers
library. You may refer to the official documentation for more details.
We provide a training script run_clm.sh
for training a 50M parameter model on the wikitext-103
dataset. You may run the script with:
bash run_clm.sh
See the script for more details. For pretraining on SlimPajama, please follow the instructions in tinyllama-zh and replace the dataset with SlimPajama.
We may initialize our LCKV model from a pretrained model. Most parts of the model structure are consistent with the standard transformer model and we can directly inherit the weights. For the KV weights
Simply add --model_name_or_path
to the training script:
accelerate launch run_clm.py \
--model_name_or_path TinyLlama/TinyLlama-1.1B-intermediate-step-1195k-token-2.5T \
--config configs/tinyllama_lckv.json \
...
See the script run_clm.sh
for more details.
Following MLKV, we may average the weights from multiple layers to initialize the KV weights. We provide a script convert_pretrained.py
to convert the pretrained model to the LCKV model. You may run the following command:
python convert_pretrained.py --model_name_or_path TinyLlama/TinyLlama-1.1B-intermediate-step-1195k-token-2.5T --config_name configs/tinyllama_lckv.json --output_dir outputs/tinyllama-converted
The KV weights of each layer will be the average from the all the layers attends to it. For example,
# the CLA / MLKV config
config.layer_types = "0_0_2_2_4_4_6_6"
# then layer 0 will have the average KV weights from layer 0 and 1 in the pretrained model
# layer 2 will have the average KV weights from layer 2 and 3 in the pretrained model
# the LCKV config
config.layer_types = "0_6_6_6_6_6_6_7"
# then layer 0 will inherit the KV weights from layer 0 in the pretrained model
# layer 6 will have the average KV weights from layer 1, 2, 3, 4, 5, 6 in the pretrained model
# layer 7 will inherit the KV weights from layer 7 in the pretrained model
then, use the converted model to initialize the LCKV model:
accelerate launch run_clm.py \
--model_name_or_path outputs/tinyllama-converted \
...
Our experiments show that such an initialization strategy can effectively improve the performance of the model in most cases.
We use the same inference script as the original transformers
library. To perform inference, you may run the following command:
bash run_generation.sh
You may get responses from the trained model given any prompts. See the script for more details.
We integrate our model with StreamingLLM. To perform streaming inference, you may run the following command:
bash run_streaming.sh
See the script for more details. The run_generation.py
script also supports streaming inference with the --sink_cache
flag.
The generation script also supports sliding window attention inference. If the model is trained with sliding window attention, the generation script will automatically use the sliding window attention for inference.
We use LM-Harness to evaluate the model. You may run the following command:
python test_harness.py --model_name_or_path ...
with the path to the model checkpoint. Run python test_harness.py --help
for more details.
To test the latency of the model, you may run the following command:
python test_latency.py
Warning
This section is currently experimental and may not work as expected.
We provide a script run_sft.sh
for supervised instruction fine-tuning. The code is consistent with the official trl
library from HuggingFace. You may run the script with:
bash run_sft.sh
See the script for more details.
To chat with the fine-tuned model, you may run the following command:
python chat.py --model_name_or_path outputs/llamatiny-sft-test
It will load the fine-tuned model and you can chat with it.
We mostly follow that of transformers
. Run the following command to check the code style:
# Use `pip install ruff` to install ruff if it is not available
ruff check models
See more details in pyproject.toml
.
Behavior:
Runtime error.
ImportError: /home/.../flash_attn_2_cuda.cpython-38-x86_64-linux-gnu.so: undefined symbol: _ZN2at4_ops9_pad_enum4callERKNS_6TensorEN3c108ArrayRefINS5_6SymIntEEElNS5_...
Solution:
pip uninstall flash-attn
FLASH_ATTENTION_FORCE_BUILD=TRUE pip install flash-attn
The cuda version may affect the installation of:
Please make sure to install the correct version of the packages (so long as they are consistent, the code would work). Also make sure that nvcc
is installed and available in the path.
Our experiment environment uses CUDA 12.1
and you may install with
conda install pytorch==2.5.0 pytorch-cuda=12.1 -c pytorch -c nvidia
pip install -r requirements.txt
Behavior: Model inference with sequential update will produce different outputs with parallel update.
This is due to the precision issues. We find that using bfloat16
, the down projection in Llama MLP will produce different results when inference with different number of tokens.
- Is it possible to integrate the LCKV with MQA / GQA?
Yes. The fact is that we have already done this in our experiments. Tinyllama uses 32 attention heads and 4 KV heads. We follow the same setting in our experiments. If you want to experiment with different settings, you may modify the num_attention_heads
and num_key_value_heads
in the configuration file.