Training Parler-TTS
TL;DR: After having followed the installation steps, you can reproduce the Parler-TTS Mini v0.1 training recipe with the following command line:
accelerate launch ./training/run_parler_tts_training.py ./helpers/training_configs/starting_point_0.01.json
This sub-folder contains all the information to train or fine-tune your own Parler-TTS model. It consists of:
You can also follow this fine-tuning guide on a mono-speaker dataset example.
1. Architecture
At the moment, Parler-TTS architecture is a carbon copy of the MusicGen architecture and can be decomposed into three distinct stages:
- Text encoder: maps the text descriptions to a sequence of hidden-state representations. Parler-TTS uses a frozen text encoder initialised entirely from Flan-T5
- Parler-TTS decoder: a language model (LM) that auto-regressively generates audio tokens (or codes) conditional on the encoder hidden-state representations
- Audio codec: used to recover the audio waveform from the audio tokens predicted by the decoder. We use the DAC model from Descript, although other codec models, such as EnCodec, can also be used
Parler-TTS however introduces some small tweaks:
- The text description is passed through the text encoder and used in the cross-attention layers of the decoder.
- The text prompt is simply passed through an embedding layer and concatenated to the decoder input hidden states.
- The audio encoder used is DAC instead of Encodec, as it exhibits better quality.
2. Getting started
To get started, you need to follow a few steps:
- Install the requirements.
- Find or initialize the model you'll train on.
- Find and/or annotate the dataset you'll train your model on.
Requirements
The Parler-TTS code is written in PyTorch and Accelerate. It uses some additional requirements, like wandb, especially for logging and evaluation.
To install the package for training, you need to clone the repository from source...
git clone https://github.com/huggingface/parler-tts.git
cd parler-tts
... And then install the requirements:
pip install -e .[train]
Optionally, you can create a wandb account and login to it by following this guide. wandb
allows for better tracking of the experiments metrics and losses.
You also have the option to configure Accelerate by running the following command. Note that you should set the number of GPUs you wish to use for training, and also the data type (dtype) to your preferred dtype for training/inference (e.g. bfloat16
on A100 GPUs, float16
on V100 GPUs, etc.):
accelerate config
Lastly, you can link you Hugging Face account so that you can push model repositories on the Hub. This will allow you to save your trained models on the Hub so that you can share them with the community. Run the command:
git config --global credential.helper store
huggingface-cli login
And then enter an authentication token from https://huggingface.co/settings/tokens. Create a new token if you do not have one already. You should make sure that this token has "write" privileges.
Initialize a model from scratch or use a pre-trained one.
Depending on your compute resources and your dataset, you need to choose between fine-tuning a pre-trained model and training a new model from scratch.
In that sense, we released a 600M checkpoint trained on 10.5K hours of annotated data under the repository id: parler-tts/parler_tts_mini_v0.1
, that you can fine-tune for your own use-case.
You can also train you own model from scratch. You can find here examples on how to initialize a model from scratch. For example, you can initialize a dummy model with:
python helpers/model_init_scripts/init_dummy_model.py ./parler-tts-untrained-dummy --text_model "google-t5/t5-small" --audio_model "parler-tts/dac_44khZ_8kbps"
In the rest of this guide, and to reproduce the Parler-TTS Mini v0.1 training recipe, we'll use a 600M parameters model that we'll initialize with:
python helpers/model_init_scripts/init_model_600M.py ./parler-tts-untrained-600M --text_model "google/flan-t5-base" --audio_model "parler-tts/dac_44khZ_8kbps"
Create or find datasets
To train your own Parler-TTS, you need datasets with 3 main features:
- speech data
- text transcription of the speech data
- conditionning text description - that you can create using Data-Speech, a library that allows you to annotate the speaker and utterance characteristics with natural language description.
Note that we made the choice to use description of the main speech characteristics (speaker pitch, speaking rate, level of noise, etc.) but that you are free to use any handmade or generated text description that makes sense.
To train Parler-TTS Mini v0.1, we used:
- The full LibriTTS-R dataset, a 1K hours high-quality speech dataset.
- A 10K hours subset of Multilingual LibriSpeech.
Both datasets have been annotated using the Data-Speech recipe, respectively here and here.
3. Training
The script run_parler_tts_training.py
is an end-to-end script that:
- load dataset(s) and merge them to the annotation dataset(s) if necessary
- pre-compute audio tokens
- train Parler-TTS
To train Parler-TTS Mini v0.1, we roughly used:
accelerate launch ./training/run_parler_tts_training.py \
--model_name_or_path "./parler-tts-untrained-600M/parler-tts-untrained-600M/" \
--feature_extractor_name "parler-tts/dac_44khZ_8kbps" \
--description_tokenizer_name "google/flan-t5-base" \
--prompt_tokenizer_name "google/flan-t5-base" \
--report_to "wandb" \
--overwrite_output_dir true \
--train_dataset_name "blabble-io/libritts_r+blabble-io/libritts_r+blabble-io/libritts_r+parler-tts/mls_eng_10k" \
--train_metadata_dataset_name "parler-tts/libritts_r_tags_tagged_10k_generated+parler-tts/libritts_r_tags_tagged_10k_generated+parler-tts/libritts_r_tags_tagged_10k_generated+parler-tts/mls-eng-10k-tags_tagged_10k_generated" \
--train_dataset_config_name "clean+clean+other+default" \
--train_split_name "train.clean.360+train.clean.100+train.other.500+train" \
--eval_dataset_name "blabble-io/libritts_r+parler-tts/mls_eng_10k" \
--eval_metadata_dataset_name "parler-tts/libritts_r_tags_tagged_10k_generated+parler-tts/mls-eng-10k-tags_tagged_10k_generated" \
--eval_dataset_config_name "other+default" \
--eval_split_name "test.other+test" \
--target_audio_column_name "audio" \
--description_column_name "text_description" \
--prompt_column_name "text" \
--max_duration_in_seconds 30 \
--min_duration_in_seconds 2.0 \
--max_text_length 400 \
--add_audio_samples_to_wandb true \
--id_column_name "id" \
--preprocessing_num_workers 8 \
--do_train true \
--num_train_epochs 40 \
--gradient_accumulation_steps 8 \
--gradient_checkpointing false \
--per_device_train_batch_size 3 \
--learning_rate 0.00095 \
--adam_beta1 0.9 \
--adam_beta2 0.99 \
--weight_decay 0.01 \
--lr_scheduler_type "constant_with_warmup" \
--warmup_steps 20000 \
--logging_steps 1000 \
--freeze_text_encoder true \
--do_eval true \
--predict_with_generate true \
--include_inputs_for_metrics true \
--evaluation_strategy steps \
--eval_steps 10000 \
--save_steps 10000 \
--per_device_eval_batch_size 12 \
--audio_encoder_per_device_batch_size 20 \
--dtype "bfloat16" \
--seed 456 \
--output_dir "./output_dir_training/" \
--temporary_save_to_disk "./audio_code_tmp/" \
--save_to_disk "./tmp_dataset_audio/" \
--max_eval_samples 96 \
--dataloader_num_workers 8 \
--group_by_length true
In particular, note how multiple training datasets, metadataset, configurations and splits can be loaded by separating the dataset arguments by + symbols:
"train_dataset_name": "blabble-io/libritts_r+blabble-io/libritts_r+blabble-io/libritts_r+parler-tts/mls_eng_10k",
"train_metadata_dataset_name": "parler-tts/libritts_r_tags_tagged_10k_generated+parler-tts/libritts_r_tags_tagged_10k_generated+parler-tts/libritts_r_tags_tagged_10k_generated+parler-tts/mls-eng-10k-tags_tagged_10k_generated",
"train_dataset_config_name": "clean+clean+other+default",
"train_split_name": "train.clean.360+train.clean.100+train.other.500+train",
Additionally, you can also write a JSON config file. Here, starting_point_0.01.json contains the exact same hyper-parameters than above and can be launched like that:
accelerate launch ./training/run_parler_tts_training.py ./helpers/training_configs/starting_point_0.01.json
Training logs will be reported to wandb, provided that you passed --report_to "wandb"
to the arguments. An example of what a training log from the above training looks like can be found here.
Starting training a new model from scratch can easily be overwhelming, so here's what training looked like for v0.1: logs
Scaling to multiple GPUs using distributed data parallelism (DDP) is trivial: simply run accelerate config
and select the multi-GPU option, specifying the IDs of the GPUs you wish to use. The above script can then be run using DDP with no code changes. In our case, we used a node of 8 H100 80GB to train Parler-TTS v0.1 for around 4 days.
There are a few other noteworthy arguments:
train_metadata_dataset_name
andeval_metadata_dataset_name
specify, if necessary, the names of the dataset(s) that contain(s) the conditionning text descriptions. For example, this dataset resulting from the Data-Speech annotation process is saved without the audio column, as it's costly to write and push audio data, so it needs to be concatenated back to the original LibriTTS-R dataset.- As noted above, the script pre-computes audio tokens as computing audio codes is costly and only needs to be done once, since we're freezing the audio encoder.
audio_encoder_per_device_batch_size
is used to precise the per devie batch size for this pre-processing step. - Additionnally, when scaling up the training data and iterating on the hyper-parameters or the model architecture, we might want to avoid recomputing the audio tokens at each training run. That's why we introduced two additional parameters,
save_to_disk
andtemporary_save_to_disk
that serves as temporary buffers to save intermediary datasets. Note that processed data is made of text and audio tokens which are much more memory efficient, so the additional required space is negligible. predict_with_generate
andadd_audio_samples_to_wandb
are required to store generated audios and to compute WER and CLAP similarity.freeze_text_encoder
: which allows to freeze the text encoder, to save compute resources.
And finally, two additional comments:
lr_scheduler_stype
: defines the learning rate schedule, one ofconstant_with_warmup
orcosine
. When experimenting with a training set-up or training for very few epochs, usingconstant_with_warmup
is typically beneficial, since the learning rate remains high over the short training run. When performing longer training runs, using acosine
schedule shoud give better results.dtype
: data type (dtype) in which the model computation should be performed. Note that this only controls the dtype of the computations (forward and backward pass), and not the dtype of the parameters or optimiser states.
Fine-tuning is as easy as modifying
model_name_or_path
to a pre-trained model. For example:--model_name_or_path parler-tts/parler_tts_mini_v0.1
.