Singing Voice Synthesis
Singing Voice Synthesis
This is a template of SVS recipe for ESPnet2.
Table of Contents
- Recipe flow
- How to run
- About data directory
- Score preparation - Case 1: phoneme annotation and standardized score - Case 2: phoneme annotation only
- Supported text cleaner
- Supported text frontend
- Supported Models
Recipe flow
SVS recipe consists of 9 stages.
1. Database-dependent data preparation
Data preparation stage.
It calls local/data.sh
to creates Kaldi-style data directories but with additional score.scp
and label
in data/
for training, validation, and evaluation sets.
See also:
2. Wav dump / Embedding preparation
If you specify --feats_type raw
option, this is a wav dumping stage which reformats wav.scp
in data directories.
Else, if you specify --feats_type fbank
option or --feats_type stft
option, this is a feature extracting stage (to be updated).
Additionally, speaker ID embedding and language ID embedding preparation will be performed in this stage if you specify --use_sid true
and --use_lid true
options. Note that this processing assume that utt2spk
or utt2lang
are correctly created in stage 1, please be careful.
3. Filtering
Filtering stage.
Processing stage to remove long and short utterances from the training and validation sets. You can change the threshold values via --min_wav_duration
and --max_wav_duration
.
Empty text will also be removed.
4. Token list generation
Token list generation stage. It generates token list (dictionary) from srctexts
. You can change the tokenization type via --token_type
option. token_type=phn
are supported. If --cleaner
option is specified, the input text will be cleaned with the specified cleaner. If token_type=phn
, the input text will be converted with G2P module specified by --g2p
option.
See also:
Data preparation will end in stage 4. You can skip data preparation (stage 1 ~ stage 4) via --skip_data_prep
option.
5. SVS statistics collection
Statistics calculation stage. It collects the shape information of the input and output and calculates statistics for feature normalization (mean and variance over training and validation sets).
In this stage, you can set --write_collected_feats true
to store statistics of pitch and feats.
6. SVS training
SVS model training stage. You can change the training setting via --train_config
and --train_args
options.
See also:
Training process will end in stage 6. You can skip training process (stage 5 ~ stage 6) via --skip_train
option.
7. SVS inference
SVS model decoding stage. You can change the decoding setting via --inference_config
and --inference_args
. Compatible vocoder can be trained and loaded.
See also:
8. Objective evaluation
Evaluation stage. It conducts four objective evaluations. See also:
9. Model packing
Packing stage. It packs the trained model files.
How to run
Here, we show the procedure to run the recipe using egs2/ofuton_p_utagoe_db/svs1
.
Move on the recipe directory.
$ cd egs2/ofuton_p_utagoe_db/svs1
Modify OFUTON
variable in db.sh
if you want to change the download directory.
$ vim db.sh
Modify cmd.sh
and conf/*.conf
if you want to use job scheduler. See the detail in using job scheduling system.
$ vim cmd.sh
Run run.sh
, which conducts all of the stages explained above.
$ ./run.sh
As a default, we train Naive_RNN (conf/train.yaml
) with feats_type=raw
+ token_type=phn
.
Then, you can get the following directories in the recipe directory.
├── data/ # Kaldi-style data directory
│ ├── dev/ # validation set
│ ├── eval/ # evaluation set
│ ├── tr_no_dev/ # training set
│ └── token_list/ # token list (directory)
│ └── phn_none_jp/ # token list
├── dump/ # feature dump directory
│ └── raw/
│ ├── org/
│ │ ├── tr_no_dev/ # training set before filtering
│ │ └── dev/ # validation set before filtering
│ ├── srctexts # text to create token list
│ ├── eval/ # evaluation set
│ ├── dev/ # validation set after filtering
│ └── tr_no_dev/ # training set after filtering
└── exp/ # experiment directory
├── svs_stats_raw_phn_none_jp # statistics
│ ├── logdir/ # statistics calculation log directory
│ ├── train/ # train statistics
│ ├── valid/ # valid statistics
└── svs_train_raw_phn_none_jp # model
├── tensorboard/ # tensorboard log
├── images/ # plot of training curves
├── valid/ # valid results
├── decode_train.loss.best/ # decoded results
│ ├── dev/ # validation set
│ └── eval/ # evaluation set
│ ├── norm/ # generated features
│ ├── denorm/ # generated denormalized features
│ ├── MCD_res/ # mel-cepstral distortion
│ ├── VUV_res/ # voiced/unvoiced error rate
│ ├── SEMITONE_res/ # semitone accuracy
│ ├── F0_res/ # log-F0 RMSE
│ ├── wav/ # generated wav via vocoder
│ ├── log/ # log directory
│ ├── feats_type # feature type
│ └── speech_shape # shape info of generated features
├── config.yaml # config used for the training
├── train.log # training log
├── *epoch.pth # model parameter file
├── checkpoint.pth # model + optimizer + scheduler parameter file
├── latest.pth # symlink to latest model parameter
├── *.ave_2best.pth # model averaged parameters
└── *.best.pth # symlink to the best model parameter loss
In decoding, you can see vocoder training to set vocoder.
For the first time, we recommend performing each stage step-by-step via --stage
and --stop_stage
options.
$ ./run.sh --stage 1 --stop_stage 1
$ ./run.sh --stage 2 --stop_stage 2
...
$ ./run.sh --stage 7 --stop_stage 7
This might help you understand each stage's processing and directory structure.
Naive_RNN training
First, complete the data preparation:
$ ./run.sh \
--stage 1 \
--stop_stage 4 \
# for sample_rate 24000 hz
$ ./run.sh \
--fs 24000 \
--n_shift 300 \
--win_length 1200 \
--stage 1 \
--stop_stage 4 \
Warning: Please note that there is different setting in fs
, n_shift
and win_length
in different model. The window shift n_shift
and window length win_lenght
are adapted to the sample rate fs
.
Second, check "train_config" (default conf/train.yaml
), "score_feats_extract" (frame level in RNN) and modify "vocoder_file" with your own vocoder path.
$ ./run.sh --stage 5 \
--train_config conf/tuning/train_naive_rnn.yaml \
--score_feats_extract frame_score_feats \
--pitch_extract dio \
--vocoder_file ${your vocoder path} \
Naive_RNN_DP training
First, complete the data preparation:
$ ./run.sh \
--stage 1 \
--stop_stage 4 \
# for sample_rate 24000 hz
$ ./run.sh \
--fs 24000 \
--n_shift 300 \
--win_length 1200 \
--stage 1 \
--stop_stage 4 \
Warning: Please note that there is different setting in fs
, n_shift
and win_length
in different model. The window shift n_shift
and window length win_lenght
are adapted to the sample rate fs
.
Second, check "train_config" (default conf/train.yaml
), "score_feats_extract" (syllable level in RNN_DP) and modify "vocoder_file" with your own vocoder path.
$ ./run.sh --stage 5 \
--train_config conf/tuning/train_naive_rnn.yaml \
--score_feats_extract syllable_score_feats \
--pitch_extract dio \
--vocoder_file ${your vocoder path} \
XiaoiceSing training
First, complete the data preparation:
$ ./run.sh \
--stage 1 \
--stop_stage 4 \
# for sample_rate 24000 hz
$ ./run.sh \
--fs 24000 \
--n_shift 300 \
--win_length 1200 \
--stage 1 \
--stop_stage 4 \
Warning: Please note that there is different setting in fs
, n_shift
and win_length
in different model. The window shift n_shift
and window length win_lenght
are adapted to the sample rate fs
.
Second, check "train_config" (default conf/train.yaml
), "score_feats_extract" (syllable level in XiaoiceSing) and modify "vocoder_file" with your own vocoder path.
$ ./run.sh --stage 5 \
--train_config conf/tuning/train_naive_rnn.yaml \
--score_feats_extract syllable_score_feats \
--pitch_extract dio \
--vocoder_file ${your vocoder path} \
Diffsinger training
First, complete the data preparation:
$ ./run.sh \
--stage 1 \
--stop_stage 4 \
# for sample_rate 24000 hz
$ ./run.sh \
--fs 24000 \
--n_shift 300 \
--win_length 1200 \
--stage 1 \
--stop_stage 4 \
To train Diffsinger, you need to train a XiaoiceSing first(see XiaoiceSing training). And load pretrain model of XiaoiceSing as Diffsinger:FFTSinger.you can see details of --pretrained_model
here.
$ ./run.sh \
--stage 5 \
--train_config conf/tuning/train_diffsinger.yaml \
--inference_config conf/tuning/decode_diffsinger.yaml \
--score_feats_extract syllable_score_feats \
--pitch_extract dio \
--expdir exp/diffsinger \
--inference_model latest.pth \
--vocoder_file ${your vocoder path} \
--pretrained_model ${your pretrained model path} \
--use_feats_minmax true \
# for example
$ --pretrained_model /exp/xiaoice-2-24-250k/500epoch.pth:svs:svs.fftsinger \
VISinger (1+2) training
The VISinger / VISinger 2 configs are hard coded for 22.05 khz or 44.1 khz and use different feature extraction method. (Note that you can use any feature extraction method but the default method is fbank
.) If you want to use it with 24 khz or 16 khz dataset, please be careful about these points.
First, check fs
(Sampling Rate) and complete the data preparation:
$ ./run.sh \
--fs 44100 \
--n_shift 512 \
--win_length 2048 \
--stage 1 \
--stop_stage 4 \
Second, check "train_config" (default conf/train.yaml
, you can also use --train_config ./conf/tuning/train_visinger2.yaml
to train VISinger 2), "score_feats_extract" (syllable level in VISinger), "svs_task" (gan_svs in VISinger).
# Single speaker 44100 hz case
./run.sh \
--stage 5 \
--fs 44100 \
--n_fft 2048 \
--n_shift 512 \
--win_length 2048 \
--svs_task gan_svs \
--pitch_extract dio \
--feats_extract fbank \
--feats_normalize none \
--score_feats_extract syllable_score_feats \
--train_config ./conf/tuning/train_visinger.yaml \
--inference_config conf/tuning/decode_vits.yaml \
--inference_model latest.pth \
--write_collected_feats true
VISinger 2 Plus training
VISinger 2 Plus has been accepted at SLT 2024. To use the pretrained model, you can download it from here. The model was trained using the OpenCpop dataset. Alternatively, you can choose the multi-singer ACE-OpenCpop dataset by downloading the model from here.
To train the model, you can select multiple configurations to train on either the ACESinger or Opencpop dataset. The default setting in train_visinger2_plus_hubert.yaml
uses hubert_large_ll60k
as the pretrained SSL model. To switch the SSL model, you can modify the configuration by uncommenting the lines for MERT or Chinese Hubert.
./run.sh \
--stage 1 \
--stop_stage 8 \
--ngpu 1 \
--fs 24000 \
--n_fft 2048 \
--n_shift 480 \
--win_length 2048 \
--dumpdir dump/24k \
--expdir exp/24k \
--svs_task gan_svs \
--feats_extract fbank \
--feats_normalize none \
--train_config ./conf/tuning/train_visinger2_plus_hubert.yaml \
--inference_config ./conf/tuning/decode_vits.yaml \
--inference_model latest.pth \
--write_collected_feats false
Singing Tacotron training
First, complete the data preparation:
$ ./run.sh \
--stage 1 \
--stop_stage 4 \
# for sample_rate 24000 hz
$ ./run.sh \
--fs 24000 \
--n_shift 300 \
--win_length 1200 \
--stage 1 \
--stop_stage 4 \
Warning: Please note that there is different setting in fs
, n_shift
and win_length
in different model. The window shift n_shift
and window length win_lenght
are adapted to the sample rate fs
.
Second, check "train_config" (default conf/train.yaml
), "score_feats_extract" (syllable level in Singing Tacotron) and modify "vocoder_file" with your own vocoder path.
$ ./run.sh --stage 5 \
--train_config conf/tuning/train_singing_tacotron.yaml \
--inference_config conf/tuning/decode_singing_tacotron.yaml \
--score_feats_extract syllable_score_feats \
--vocoder_file ${your vocoder path} \
Multi-speaker model with speaker ID embedding training
First, you need to run from the stage 2 and 3 with --use_sid true
to extract speaker ID.
$ ./run.sh --stage 2 --stop_stage 3 --use_sid true
You can find the speaker ID file in dump/raw/*/utt2sid
. Note that you need to correctly create utt2spk
in data prep stage to generate utt2sid
. Then, you can run the training with the config which has spks: #spks
in svs_conf
.
# e.g.
svs_conf:
spks: 5 # Number of speakers
Please run the training from stage 6.
$ ./run.sh --stage 6 --use_sid true --train_config /path/to/your_multi_spk_config.yaml
Multi-language model with language ID embedding training
First, you need to run from the stage 2 and 3 with --use_lid true
to extract speaker ID.
$ ./run.sh --stage 2 --stop_stage 3 --use_lid true
You can find the speaker ID file in dump/raw/*/utt2lid
. Note that you need to additionally create utt2lang
file in stage 1 to generate utt2lid
. Then, you can run the training with the config which has langs: #langs
in svs_conf
.
# e.g.
svs_conf:
langs: 4 # Number of languages
Please run the training from stage 6.
$ ./run.sh --stage 6 --use_lid true --train_config /path/to/your_multi_lang_config.yaml
Of course you can further combine with speaker ID embedding. If you want to use both sid and lid, the process should be like this:
$ ./run.sh --stage 2 --stop_stage 3 --use_lid true --use_sid true
Make your config.
# e.g.
svs_conf:
langs: 4 # Number of languages
spks: 5 # Number of speakers
Please run the training from stage 6.
$ ./run.sh --stage 6 --use_lid true --use_sid true --train_config /path/to/your_multi_spk_multi_lang_config.yaml
Vocoder training
If your --vocoder_file
is set to none, Griffin-Lim will be used. You can also train corresponding vocoder using kan-bayashi/ParallelWaveGAN..
Pretrained vocoder is like follows:
*_hifigan.v1
├── checkpoint-xxxxxxsteps.pkl
├── config.yml
└── stats.h5
# Use the vocoder trained by `parallel_wavegan` repo manually
$ ./run.sh --stage 7 --vocoder_file /path/to/checkpoint-xxxxxxsteps.pkl --inference_tag decode_with_my_vocoder
Evaluation
We provide four objective evaluation metrics:
- Mel-cepstral distortion (MCD)
- Logarithmic rooted mean square error of the fundamental frequency (log-F0 RMSE)
- Semitone accuracy (Semitone ACC)
- Voiced / unvoiced error rate (VUV_E)
- Word/character error rate (WER/CER, optional executated by users)
For MCD, we apply dynamic time-warping (DTW) to match the length difference between ground-truth singing and generated singing.
Here we show the example command to calculate objective metrics:
cd egs2/<recipe_name>/svs1
. ./path.sh
# Evaluate MCD & log-F0 RMSE & Semitone ACC & VUV Error Rate
./pyscripts/utils/evaluate_*.py \
exp/<model_dir_name>/<decode_dir_name>/eval/wav/gen_wavdir_or_wavscp.scp \
dump/raw/eval/gt_wavdir_or_wavscp.scp
# Evaluate CER
./scripts/utils/evaluate_asr.sh \
--model_tag <asr_model_tag> \
--nj 1 \
--inference_args "--beam_size 10 --ctc_weight 0.4 --lm_weight 0.0" \
--gt_text "dump/raw/eval1/text" \
exp/<model_dir_name>/<decode_dir_name>/eval1/wav/wav.scp \
exp/<model_dir_name>/<decode_dir_name>/asr_results
# Since ASR model does not use punctuation, it is better to remove punctuations if it contains
./scripts/utils/remove_punctuation.pl < dump/raw/eval1/text > dump/raw/eval1/text.no_punc
./scripts/utils/evaluate_asr.sh \
--model_tag <asr_model_tag> \
--nj 1 \
--inference_args "--beam_size 10 --ctc_weight 0.4 --lm_weight 0.0" \
--gt_text "dump/raw/eval1/text.no_punc" \
exp/<model_dir_name>/<decode_dir_name>/eval1/wav/wav.scp \
exp/<model_dir_name>/<decode_dir_name>/asr_results
# You can also use openai whisper for evaluation
./scripts/utils/evaluate_asr.sh \
--whisper_tag base \
--nj 1 \
--gt_text "dump/raw/eval1/text" \
exp/<model_dir_name>/<decode_dir_name>/eval1/wav/wav.scp \
exp/<model_dir_name>/<decode_dir_name>/asr_results
While these objective metrics can estimate the quality of synthesized singing, it is still difficult to fully determine human perceptual quality from these values, especially with high-fidelity generated singing. Therefore, we recommend performing the subjective evaluation (eg. MOS) if you want to check perceptual quality in detail.
You can refer this page to launch web-based subjective evaluation system with webMUSHRA.
About data directory
Each directory of training set, development set, and evaluation set, has same directory structure. See also https://github.com/espnet/espnet/tree/master/egs2/TEMPLATE#about-kaldi-style-data-directory about Kaldi data structure. We recommend you running ofuton_p_utagoe_db
recipe and checking the contents of data/
by yourself.
cd egs/ofuton_p_utagoe_db/svs1
./run.sh
Directory structure
data/ ├── tr_no_dev/ # Training set directory │ ├── text # The transcription │ ├── label # Specifying start and end time of the transcription │ ├── score.scp # Score file path │ ├── wav.scp # Wave file path │ ├── utt2spk # A file mapping utterance-id to speaker-id │ ├── spk2utt # A file mapping speaker-id to utterance-id │ ├── segments # [Option] Specifying start and end time of each utterance │ └── (utt2lang) # A file mapping utterance-id to language type (only for multilingual) ├── dev/ │ ... ├── eval/ │ ... └── token_list/ # token list directory ...
text
formatuttidA <transcription> uttidB <transcription> ...
label
formatuttidA (startA1, endA1, phA1) (startA2, endA2, phA1) ... uttidB (startB1, endB1, phB1) (startB2, endB2, phB2) ... ...
score.scp
formatkey1 /some/path/score.json key2 /some/path/score.json ...
Note that for databases without explicit score or MusicXML, we will provide rule-based automatic music transcription to extract related music information in the future.
wav.scp
formatuttidA /path/to/uttidA.wav uttidB /path/to/uttidB.wav ...
utt2spk
formatuttidA speakerA uttidB speakerB uttidC speakerA uttidD speakerB ...
spk2utt
formatspeakerA uttidA uttidC ... speakerB uttidB uttidD ... ...
Note that
spk2utt
file can be generated byutt2spk
, andutt2spk
can be generated byspk2utt
, so it's enough to create either one of them.utils/utt2spk_to_spk2utt.pl data/tr_no_dev/utt2spk > data/tr_no_dev/spk2utt utils/spk2utt_to_utt2spk.pl data/tr_no_dev/spk2utt > data/tr_no_dev/utt2spk
If your corpus doesn't include speaker information, give the same speaker id as the utterance id to satisfy the directory format, otherwise give the same speaker id for all utterances (Actually we don't use speaker information for asr recipe now).
uttidA uttidA uttidB uttidB ...
OR
uttidA dummy uttidB dummy ...
[Option]
segments
formatIf the audio data is originally long recording, about > ~1 hour, and each audio file includes multiple utterances in each section, you need to create
segments
file to specify the start time and end time of each utterance. The format is<utterance_id> <wav_id> <start_time> <end_time>
.ofuton_0000000000000000hato_0007 ofuton_0000000000000000hato 33.470 38.013 ...
Note that if using
segments
,wav.scp
has<wav_id>
which corresponds to thesegments
instead ofutterance_id
.ofuton_0000000000000000hato /path/to/ofuton_0000000000000000hato.wav ...
utt2lang
formatuttidA languageA uttidB languageB uttidC languageA uttidD lagnuageB ...
Note that
utt2lang
file is only generated for multilingual dataset (see in recipeegs/multilingual_four
).
Once you complete creating the data directory, it's better to check it by utils/validate_data_dir.sh
.
utils/validate_data_dir.sh --no-feats data/tr_no_dev
utils/validate_data_dir.sh --no-feats data/dev
utils/validate_data_dir.sh --no-feats data/test
Score preparation
To prepare a new recipe, we first split songs into segments via --silence
option if no official segmentation provided.
Then, we transfer the raw data into score.json
, where situations can be categorized into two cases depending on the annotation:
Case 1: phoneme annotation and standardized score
If the phonemes and notes are aligned in time domain, convert the raw data directly. (eg. Opencpop)
If the phoneme annotation are misaligned with notes in time domain, align phonemes (from
label
) and note-lyric pairs (frommusicXML
) through g2p. (eg. Ofuton)We also offer some automatic fixes for missing silences in the dataset. During the stage1, when you encounter errors such as "Lyrics are longer than phones" or "Phones are longer than lyrics", the scripts will auto-generated the fixing code. You may need to put the code into the
get_error_dict
method inegs2/[dataset name]/svs1/local/prep_segments.py
. Noted that depending on the suggested input_type, you may want to copy it into either thehts
orxml
's error_dict. (For more information, please check namine or natsume
Specially, the note-lyric pairs can be rebuilt through other melody files, like MIDI
, if there's something wrong with the note duration. (eg. Natsume)
Case 2: phoneme annotation only
To be updated.
Problems you might meet
During stage 1, which involves data preparation, you may encounter ValueError
problems that typically indicate errors in the annotation. To address these issues, it is necessary to manually review the raw data in the corresponding sections and make the necessary corrections. While other toolkits and open-source codebases may not impose such requirements or checks, we have found that investing time to resolve these errors significantly enhances the quality of the singing voice synthesizer.
Note that modifications can be made to the raw data locally or through the processing data flow at stage 1. For the convenience of open source, we recommend using the latter.
- To make changes to the raw data, you can use toolkits like music21, miditoolkit, or MuseScore.
- To process in the data flow, you can use score readers and writers provided. Examples can be found in functioin
make_segment
fromegs2/{natsume, ameboshi, pjs}/svs1/local/{prep_segments.py, prep_segments_from_xml.py}/
.
Below are some common errors to watch out for:
1. Wrong segmentation point
- Add pauses or directly split between adjacent lyrics.
- Remove pauses and assign the duration to correct phoneme.
2. Wrong lyric / midi annotation
- Replace with correct one.
- Add missing one and reassign adjacent duration.
- Remove redundant one and reassign adjacent duration.
3. Different lyric-phoneme pairs against the given g2p
- Use a
customed_dic
of syllable-phoneme pairs as following:# e.g. # In Japanese dataset ofuton, the output of "ヴぁ" from pyopenjtalk is different from raw data "v a" > pyopenjtalk.g2p("ヴぁ") v u a # Add the following lyric-phoneme pair to customed_dic ヴぁ v_a
- Specify
--g2p none
and store the lyric-phoneme pairs intoscore.json
, especially for polyphone problem in Mandarin.# e.g. # In Mandarin dataset Opencpop, the pronounce the second "重" should be "chong". > pypinyin.pinyin("情意深重爱恨两重", style=Style.NORMAL) [['qing'], ['shen'], ['yi'], ['zhong'], ['ai'], ['hen'], ['liang'], ['zhong']]
4. Special marks in MusicXML
- Breath:
breath mark
in note.articulations: usually appears at the end of the sentence. In some situations,breath mark
doesn't take effect in its belonging note. Please handle them under local/.br
in note.lyric. (solved in XMLReader)- Special note with a fixed special pitch. (solved in XMLReader)
- Staccato: In some situations, there is a break when
staccato
occurs in note.articulations. We let users to decide whether to perform segmentation under local/.
Supported text cleaner
You can change via --cleaner
option in svs.sh
.
none
: No text cleaner.
You can see the code example from here.
Supported text frontend
You can change via --g2p
option in svs.sh
.
none
: Just separate by space- e.g.:
HH AH0 L OW1 <space> W ER1 L D
->[HH, AH0, L, OW1, <space>, W, ER1, L D]
- e.g.:
pyopenjtalk
: r9y9/pyopenjtalk- e.g.
こ、こんにちは
->[k, o, pau, k, o, N, n, i, ch, i, w, a]
- e.g.
You can see the code example from here.
Supported Models
You can train the following models by changing *.yaml
config for --train_config
option in run.sh
.
You can find example configs of the above models in egs/ofuton_p_utagoe_db/svs1/conf/tuning
.