# Copyright (c) 2022 PaddlePaddle Authors. All Rights Reserved. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. import unittest import librosa import numpy as np import paddle import paddleaudio from .base import FeatTest from paddleaudio.functional.window import get_window class TestLibrosa(FeatTest): def initParmas(self): self.n_fft = 512 self.hop_length = 128 self.n_mels = 40 self.n_mfcc = 20 self.fmin = 0.0 self.window_str = 'hann' self.pad_mode = 'reflect' self.top_db = 80.0 def test_stft(self): if len(self.waveform.shape) == 2: # (C, T) self.waveform = self.waveform.squeeze( 0) # 1D input for librosa.feature.melspectrogram feature_librosa = librosa.core.stft( y=self.waveform, n_fft=self.n_fft, hop_length=self.hop_length, win_length=None, window=self.window_str, center=True, dtype=None, pad_mode=self.pad_mode, ) x = paddle.to_tensor(self.waveform).unsqueeze(0) window = get_window(self.window_str, self.n_fft, dtype=x.dtype) feature_paddle = paddle.signal.stft( x=x, n_fft=self.n_fft, hop_length=self.hop_length, win_length=None, window=window, center=True, pad_mode=self.pad_mode, normalized=False, onesided=True, ).squeeze(0) np.testing.assert_array_almost_equal( feature_librosa, feature_paddle, decimal=5) def test_istft(self): if len(self.waveform.shape) == 2: # (C, T) self.waveform = self.waveform.squeeze( 0) # 1D input for librosa.feature.melspectrogram # Get stft result from librosa. stft_matrix = librosa.core.stft( y=self.waveform, n_fft=self.n_fft, hop_length=self.hop_length, win_length=None, window=self.window_str, center=True, pad_mode=self.pad_mode, ) feature_librosa = librosa.core.istft( stft_matrix=stft_matrix, hop_length=self.hop_length, win_length=None, window=self.window_str, center=True, dtype=None, length=None, ) x = paddle.to_tensor(stft_matrix).unsqueeze(0) window = get_window( self.window_str, self.n_fft, dtype=paddle.to_tensor(self.waveform).dtype) feature_paddle = paddle.signal.istft( x=x, n_fft=self.n_fft, hop_length=self.hop_length, win_length=None, window=window, center=True, normalized=False, onesided=True, length=None, return_complex=False, ).squeeze(0) np.testing.assert_array_almost_equal( feature_librosa, feature_paddle, decimal=5) def test_mel(self): feature_librosa = librosa.filters.mel( sr=self.sr, n_fft=self.n_fft, n_mels=self.n_mels, fmin=self.fmin, fmax=None, htk=False, norm='slaney', dtype=self.waveform.dtype, ) feature_compliance = paddleaudio.compliance.librosa.compute_fbank_matrix( sr=self.sr, n_fft=self.n_fft, n_mels=self.n_mels, fmin=self.fmin, fmax=None, htk=False, norm='slaney', dtype=self.waveform.dtype, ) x = paddle.to_tensor(self.waveform) feature_functional = paddleaudio.functional.compute_fbank_matrix( sr=self.sr, n_fft=self.n_fft, n_mels=self.n_mels, f_min=self.fmin, f_max=None, htk=False, norm='slaney', dtype=x.dtype, ) np.testing.assert_array_almost_equal(feature_librosa, feature_compliance) np.testing.assert_array_almost_equal(feature_librosa, feature_functional) def test_melspect(self): if len(self.waveform.shape) == 2: # (C, T) self.waveform = self.waveform.squeeze( 0) # 1D input for librosa.feature.melspectrogram # librosa: feature_librosa = librosa.feature.melspectrogram( y=self.waveform, sr=self.sr, n_fft=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, fmin=self.fmin) # paddleaudio.compliance.librosa: feature_compliance = paddleaudio.compliance.librosa.melspectrogram( x=self.waveform, sr=self.sr, window_size=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, fmin=self.fmin, to_db=False) # paddleaudio.features.layer x = paddle.to_tensor( self.waveform, dtype=paddle.float64).unsqueeze(0) # Add batch dim. feature_extractor = paddleaudio.features.MelSpectrogram( sr=self.sr, n_fft=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, f_min=self.fmin, dtype=x.dtype) feature_layer = feature_extractor(x).squeeze(0).numpy() np.testing.assert_array_almost_equal( feature_librosa, feature_compliance, decimal=5) np.testing.assert_array_almost_equal( feature_librosa, feature_layer, decimal=5) def test_log_melspect(self): if len(self.waveform.shape) == 2: # (C, T) self.waveform = self.waveform.squeeze( 0) # 1D input for librosa.feature.melspectrogram # librosa: feature_librosa = librosa.feature.melspectrogram( y=self.waveform, sr=self.sr, n_fft=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, fmin=self.fmin) feature_librosa = librosa.power_to_db(feature_librosa, top_db=None) # paddleaudio.compliance.librosa: feature_compliance = paddleaudio.compliance.librosa.melspectrogram( x=self.waveform, sr=self.sr, window_size=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, fmin=self.fmin) # paddleaudio.features.layer x = paddle.to_tensor( self.waveform, dtype=paddle.float64).unsqueeze(0) # Add batch dim. feature_extractor = paddleaudio.features.LogMelSpectrogram( sr=self.sr, n_fft=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, f_min=self.fmin, dtype=x.dtype) feature_layer = feature_extractor(x).squeeze(0).numpy() np.testing.assert_array_almost_equal( feature_librosa, feature_compliance, decimal=5) np.testing.assert_array_almost_equal( feature_librosa, feature_layer, decimal=4) def test_mfcc(self): if len(self.waveform.shape) == 2: # (C, T) self.waveform = self.waveform.squeeze( 0) # 1D input for librosa.feature.melspectrogram # librosa: feature_librosa = librosa.feature.mfcc( y=self.waveform, sr=self.sr, S=None, n_mfcc=self.n_mfcc, dct_type=2, norm='ortho', lifter=0, n_fft=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, fmin=self.fmin) # paddleaudio.compliance.librosa: feature_compliance = paddleaudio.compliance.librosa.mfcc( x=self.waveform, sr=self.sr, n_mfcc=self.n_mfcc, dct_type=2, norm='ortho', lifter=0, window_size=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, fmin=self.fmin, top_db=self.top_db) # paddleaudio.features.layer x = paddle.to_tensor( self.waveform, dtype=paddle.float64).unsqueeze(0) # Add batch dim. feature_extractor = paddleaudio.features.MFCC( sr=self.sr, n_mfcc=self.n_mfcc, n_fft=self.n_fft, hop_length=self.hop_length, n_mels=self.n_mels, f_min=self.fmin, top_db=self.top_db, dtype=x.dtype) feature_layer = feature_extractor(x).squeeze(0).numpy() np.testing.assert_array_almost_equal( feature_librosa, feature_compliance, decimal=4) np.testing.assert_array_almost_equal( feature_librosa, feature_layer, decimal=4) if __name__ == '__main__': unittest.main()