Skip to content

Commit

Permalink
Update KWS, MSNoise, Signalmixer Data Loaders & Evaluation Notebook, …
Browse files Browse the repository at this point in the history
…Add New Scripts for Mixed Signals (analogdevicesinc#299)
  • Loading branch information
EyubogluMerve authored Jun 28, 2024
1 parent bc32ee9 commit ba6c02b
Show file tree
Hide file tree
Showing 11 changed files with 651 additions and 911 deletions.
486 changes: 396 additions & 90 deletions datasets/kws20.py

Large diffs are not rendered by default.

493 changes: 0 additions & 493 deletions datasets/mixedkws.py

This file was deleted.

175 changes: 76 additions & 99 deletions datasets/msnoise.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,11 +43,11 @@ class MSnoise:
Args:
root (string): Root directory of dataset where ``MSnoise/processed/dataset.pt``
exist.
classes(array): List of keywords to be used.
d_type(string): Option for the created dataset. ``train`` or ``test``.
dataset_len(int): Dataset length to be returned.
remove_unknowns (bool, optional): If true, unchosen classes are not gathered as
the unknown class.
classes (array): List of keywords to be used.
d_type (string): Option for the created dataset. ``train`` or ``test``.
dataset_len (int): Dataset length to be returned.
exp_len (int, optional): Expected length of the 1-sec audio samples.
desired_probs (array, optional): Desired probabilities array for each noise type specified.
transform (callable, optional): A function/transform that takes in an PIL image
and returns a transformed version.
quantize (bool, optional): If true, the datasets are prepared and saved as
Expand All @@ -67,19 +67,16 @@ class MSnoise:
'Square': 18, 'SqueakyChair': 19, 'Station': 20, 'TradeShow': 21, 'Traffic': 22,
'Typing': 23, 'VacuumCleaner': 24, 'WasherDryer': 25, 'Washing': 26}

def __init__(self, root, classes, d_type, dataset_len, exp_len=16384, fs=16000,
noise_time_step=0.25, remove_unknowns=False, transform=None,
quantize=False, download=False):
def __init__(self, root, classes, d_type, dataset_len, exp_len=16384, desired_probs=None,
transform=None, quantize=False, download=False):
self.root = root
self.classes = classes
self.d_type = d_type
self.remove_unknowns = remove_unknowns
self.transform = transform

self.dataset_len = dataset_len
self.exp_len = exp_len
self.fs = fs
self.noise_time_step = noise_time_step
self.desired_probs = desired_probs

self.noise_train_folder = os.path.join(self.raw_folder, 'noise_train')
self.noise_test_folder = os.path.join(self.raw_folder, 'noise_test')
Expand All @@ -97,9 +94,6 @@ def __init__(self, root, classes, d_type, dataset_len, exp_len=16384, fs=16000,
# rms values for each sample to be returned
self.rms = np.zeros(self.dataset_len)

self.__filter_dtype()
self.__filter_classes()

@property
def raw_folder(self):
"""Folder for the raw data.
Expand All @@ -117,6 +111,13 @@ def __download(self):
self.__download_raw(self.url_train)
self.__download_raw(self.url_test)

# Fix the naming convention mismatches
for record_name in os.listdir(self.noise_test_folder):
if 'Neighbor' in record_name.split('_')[0]:
rec_pth = f'NeighborSpeaking_{record_name.split("_")[-1]}'
rec_pth = os.path.join(self.noise_test_folder, rec_pth)
os.rename(os.path.join(self.noise_test_folder, record_name), rec_pth)

def __download_raw(self, api_url):
opener = urllib.request.build_opener()
opener.addheaders = [('User-agent', 'Mozilla/5.0')]
Expand Down Expand Up @@ -151,52 +152,6 @@ def __makedir_exist_ok(self, dirpath):
else:
raise

def __filter_dtype(self):

if self.d_type == 'train':
bool_list = [i == 0 for i in self.data_type]
idx_to_select = [i for i, x in enumerate(bool_list) if x]
elif self.d_type == 'test':
bool_list = [i == 1 for i in self.data_type]
idx_to_select = [i for i, x in enumerate(bool_list) if x]
else:
print(f'Unknown data type: {self.d_type}')
return

self.data = [self.data[i] for i in idx_to_select]
self.targets = [self.targets[i] for i in idx_to_select]
self.rms_val = [self.rms_val[i] for i in idx_to_select]
del self.data_type

def __filter_classes(self):
print('\n')
self.targets = np.array(self.targets)
initial_new_class_label = len(self.class_dict)
new_class_label = initial_new_class_label
for c in self.classes:
if c not in self.class_dict:
print(f'Class is not in the data: {c}')
return
# else:
print(f'Class {c}, {self.class_dict[c]}')
bool_list = [self.class_dict[c] == i for i in self.targets]
idx = [i for i, x in enumerate(bool_list) if x]
self.targets[idx] = new_class_label
print(f'{c}: {new_class_label - initial_new_class_label}')
new_class_label += 1

self.targets[(self.targets < initial_new_class_label)] = new_class_label
if self.remove_unknowns:
bool_list = [i != new_class_label for i in self.targets]
idx_to_keep = [i for i, x in enumerate(bool_list) if x]

self.data = [self.data[i] for i in idx_to_keep]
self.targets = [self.targets[i] for i in idx]
self.rms_val = [self.rms_val[i] for i in idx]

self.targets = [target - initial_new_class_label for target in self.targets]
print('\n')

@staticmethod
def quantize_audio(data, num_bits=8):
"""Quantize audio
Expand All @@ -213,13 +168,10 @@ def __len__(self):

def __getitem__(self, index):

rec_num = len(self.data)

rnd_num = np.random.randint(0, rec_num)
rnd_num = np.random.choice(range(len(self.data)), p=self.final_probs)
self.rms[index] = self.rms_val[rnd_num]

rec_len = len(self.data[rnd_num])

max_start_idx = rec_len - self.exp_len
start_idx = np.random.randint(0, max_start_idx)
end_idx = start_idx + self.exp_len
Expand All @@ -237,44 +189,54 @@ def __reshape_audio(self, audio, row_len=128):

return torch.transpose(torch.tensor(audio.reshape((-1, row_len))), 1, 0)

def __gen_datasets(self, exp_len=16384, row_len=128, overlap_ratio=0):
def __gen_datasets(self):

with warnings.catch_warnings():
warnings.simplefilter('error')

# PARAMETERS
overlap = int(np.ceil(row_len * overlap_ratio))
num_rows = int(np.ceil(exp_len / (row_len - overlap)))
data_len = int((num_rows*row_len - (num_rows-1)*overlap))
print(f'data_len: {data_len}')

# Cleaning the duplicate labels
labels = list(self.classes)
train_list = sorted(os.listdir(self.noise_train_folder))
test_list = sorted(os.listdir(self.noise_test_folder))
labels_train = set(sorted({i.split('_')[0] for i in train_list if '_' in i}))
labels_test = set(sorted({i.split('_')[0] for i in test_list if '_' in i}))
labels = labels_train | labels_test
labels_to_remove = set()
for label in labels:
other_labels = labels - {label}
for other_label_name in other_labels:
if label in other_label_name:
labels_to_remove.add(label)
break
labels = labels - labels_to_remove
labels = sorted(labels)
print(f'Labels: {labels}')

# Folders
train_test_folders = [self.noise_train_folder, self.noise_test_folder]

if self.d_type == 'train':
check_label = labels_train
audio_folder = [self.noise_train_folder]
elif self.d_type == 'test':
check_label = labels_test
audio_folder = [self.noise_test_folder]

for label in self.classes:
if label not in check_label:
print(f'Label {label} is not in the MSnoise {self.d_type} dataset.')
labels.remove(label)

print(f'Labels for {self.d_type}: {labels}')

if self.desired_probs is None or len(self.desired_probs) != len(labels):
self.desired_probs = []
print('Each class will be selected using the same probability!')
label_count = len(labels)
for i in range(label_count):
self.desired_probs.append(1/label_count)

elif np.sum(self.desired_probs) != 1:
print('Sum of the probabilities is not 1!\n')
print('Carrying out the normal probability distribution.')
self.desired_probs = self.desired_probs / np.sum(self.desired_probs)

print(f'Desired probabilities for each class: {self.desired_probs}')

self.data_class_count = {}
data_in = []
data_type = []
data_class = []
rms_val = []

for i, label in enumerate(labels):
for folder in train_test_folders:
count = 0
for folder in audio_folder:
for record_name in sorted(os.listdir(folder)):
if record_name.split('_')[0] in label:
record_path = os.path.join(folder, record_name)
Expand All @@ -292,12 +254,24 @@ def __gen_datasets(self, exp_len=16384, row_len=128, overlap_ratio=0):

data_class.append(i)
rms_val.append(np.mean(record**2)**0.5)
count += 1
self.data_class_count[label] = count

noise_dataset = (data_in, data_class, data_type, rms_val)

final_probs = np.zeros(len(data_in))

idx = 0
for i, label in enumerate(labels):
for _ in range(self.data_class_count[label]):
final_probs[idx] = self.desired_probs[i]/self.data_class_count[label]
idx += 1
self.final_probs = final_probs
return noise_dataset


def MSnoise_get_datasets(data, load_train=True, load_test=True):
def MSnoise_get_datasets(data, desired_probs=None, train_len=346338, test_len=11005,
load_train=True, load_test=True):
"""
Load the folded 1D version of MS Scalable Noisy Speech dataset (MS-SNSD)
Expand All @@ -316,22 +290,23 @@ def MSnoise_get_datasets(data, load_train=True, load_test=True):
'Square', 'SqueakyChair', 'Station', 'Traffic',
'Typing', 'VacuumCleaner', 'WasherDryer', 'Washing', 'TradeShow']

remove_unknowns = True
transform = transforms.Compose([
ai8x.normalize(args=args)
])
quantize = True

if load_train:
train_dataset = MSnoise(root=data_dir, classes=classes, d_type='train', dataset_len=11005,
remove_unknowns=remove_unknowns, transform=transform,
train_dataset = MSnoise(root=data_dir, classes=classes, d_type='train',
dataset_len=train_len, desired_probs=desired_probs,
transform=transform,
quantize=quantize, download=True)
else:
train_dataset = None

if load_test:
test_dataset = MSnoise(root=data_dir, classes=classes, d_type='test', dataset_len=11005,
remove_unknowns=remove_unknowns, transform=transform,
test_dataset = MSnoise(root=data_dir, classes=classes, d_type='test',
dataset_len=test_len, desired_probs=desired_probs,
transform=transform,
quantize=quantize, download=True)

if args.truncate_testset:
Expand All @@ -342,7 +317,8 @@ def MSnoise_get_datasets(data, load_train=True, load_test=True):
return train_dataset, test_dataset


def MSnoise_get_unquantized_datasets(data, load_train=True, load_test=True):
def MSnoise_get_unquantized_datasets(data, desired_probs=None, train_len=346338, test_len=11005,
load_train=True, load_test=True):
"""
Load the folded 1D and unquantized version of MS Scalable Noisy Speech dataset (MS-SNSD)
Expand All @@ -360,20 +336,21 @@ def MSnoise_get_unquantized_datasets(data, load_train=True, load_test=True):
'Square', 'SqueakyChair', 'Station', 'Traffic',
'Typing', 'VacuumCleaner', 'WasherDryer', 'Washing', 'TradeShow']

remove_unknowns = True
transform = None
quantize = False

if load_train:
train_dataset = MSnoise(root=data_dir, classes=classes, d_type='train', dataset_len=11005,
remove_unknowns=remove_unknowns, transform=transform,
train_dataset = MSnoise(root=data_dir, classes=classes, d_type='train',
dataset_len=train_len, desired_probs=desired_probs,
transform=transform,
quantize=quantize, download=True)
else:
train_dataset = None

if load_test:
test_dataset = MSnoise(root=data_dir, classes=classes, d_type='test', dataset_len=11005,
remove_unknowns=remove_unknowns, transform=transform,
test_dataset = MSnoise(root=data_dir, classes=classes, d_type='test',
dataset_len=test_len, desired_probs=desired_probs,
transform=transform,
quantize=quantize, download=True)

if args.truncate_testset:
Expand Down
Loading

0 comments on commit ba6c02b

Please sign in to comment.