forked from auspicious3000/contentvec
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy path02_create_contentvec_dict.py
104 lines (85 loc) · 3.82 KB
/
02_create_contentvec_dict.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
import argparse
from resemblyzer import VoiceEncoder, preprocess_wav
import torch
from os.path import join, exists
from tqdm import tqdm
import librosa
import pickle
import torch.multiprocessing as mp
from torchfcpe import spawn_bundled_infer_model
def extract_embedding(filepath, encoder):
wav = preprocess_wav(filepath)
file_embedding = encoder.embed_utterance(wav)
embedding = torch.tensor(file_embedding)
return embedding
def process_files(rank, filenames, root_folder, device_id, return_dict):
torch.cuda.set_device(device_id)
encoder = VoiceEncoder()
fcpe = spawn_bundled_infer_model(device=device_id)
def get_f0_with_fcpe(filepath):
audio, sr = librosa.load(filepath, sr=None, mono=True)
_audio = torch.from_numpy(audio).to(device_id).unsqueeze(0)
f0 = fcpe(_audio, sr=sr, decoder_mode="local_argmax", threshold=0.006)
f0 = f0.squeeze().cpu().numpy()
f0_p = f0[f0 > 0]
return f0_p.min(), f0_p.max(), f0_p.mean()
speaker_dict = {}
for filepath in tqdm(filenames, position=rank):
speaker_id = str(filepath)
filepath = join(root_folder, filepath)
if not exists(filepath):
print(f"file {filepath} doesn't exist!")
continue
embedding = extract_embedding(filepath, encoder=encoder)
try:
f0_min, f0_max, f0_mean = get_f0_with_fcpe(filepath)
except Exception as e:
print(f"Error: {filepath}: {e}")
continue
speaker_dict[speaker_id] = embedding.numpy(), (f0_min, f0_max, f0_mean)
return_dict[rank] = speaker_dict
def parallel_process(filenames, root_folder, num_processes):
mp.set_start_method('spawn', force=True)
manager = mp.Manager()
return_dict = manager.dict()
num_devices = torch.cuda.device_count()
chunk_size = len(filenames) // num_processes
processes = []
for i in range(num_processes):
start = i * chunk_size
end = None if i == num_processes - 1 else (i + 1) * chunk_size
file_chunk = filenames[start:end]
device_id = i % num_devices
p = mp.Process(target=process_files, args=(i, file_chunk, root_folder, device_id, return_dict))
p.start()
processes.append(p)
for p in processes:
p.join()
speaker_dict = {}
for rank, part_dict in return_dict.items():
speaker_dict.update(part_dict)
return speaker_dict
def generate_list_dict_from_list(filelist_train, filelist_val, root_folder, num_processes):
speaker_dict = {'train': {}, 'valid': {}}
speaker_dict['valid'] = parallel_process(filelist_val, root_folder, num_processes)
speaker_dict['train'] = parallel_process(filelist_train, root_folder, num_processes)
return speaker_dict
if __name__ == "__main__":
parser = argparse.ArgumentParser()
parser.add_argument('-t', '--input_train', type=str, default="data/00_filelist/train.tsv")
parser.add_argument('-v', '--input_val', type=str, default="data/00_filelist/valid.tsv")
parser.add_argument('-d', '--dataset_dir', type=str, default="dataset_raw")
parser.add_argument('-o', '--output', type=str, default='data/01_spk2info.dict')
parser.add_argument('-n', '--num_process', type=int, default=5)
args = parser.parse_args()
with open(args.input_train, "r", encoding='utf-8') as file:
data = file.readlines()[1:]
filelist_train = [line.split("\t")[0] for line in data]
with open(args.input_val, "r", encoding='utf-8') as file:
data = file.readlines()[1:]
filelist_val = [line.split("\t")[0] for line in data]
speaker_list_dict = generate_list_dict_from_list(filelist_train, filelist_val, args.dataset_dir, args.num_process)
del filelist_train
del filelist_val
with open(args.output, 'wb') as file:
pickle.dump(speaker_list_dict, file)