123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475 |
- /* Spectrum_to_Excitation.cpp
- *
- * Copyright (C) 1992-2011,2014,2015,2016,2017 Paul Boersma
- *
- * This code is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or (at
- * your option) any later version.
- *
- * This code is distributed in the hope that it will be useful, but
- * WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this work. If not, see <http://www.gnu.org/licenses/>.
- */
- #include "Spectrum_to_Excitation.h"
- autoExcitation Spectrum_to_Excitation (Spectrum me, double dbark) {
- try {
- integer nbark = Melder_iround (25.6 / dbark);
- double *re = my z [1], *im = my z [2];
- autoNUMvector <double> auditoryFilter (1, nbark);
- double filterArea = 0.0;
- for (integer i = 1; i <= nbark; i ++) {
- double bark = dbark * (i - nbark/2) + 0.474;
- filterArea += auditoryFilter [i] = pow (10, (1.581 + 0.75 * bark - 1.75 * sqrt (1 + bark * bark)));
- }
- /*for (integer i = 1; i <= nbark; i ++)
- auditoryFilter [i] /= filterArea;*/
- autoNUMvector <double> rFreqs (1, nbark + 1);
- autoNUMvector <integer> iFreqs (1, nbark + 1);
- for (integer i = 1; i <= nbark + 1; i ++) {
- rFreqs [i] = Excitation_barkToHertz (dbark * (i - 1));
- iFreqs [i] = Sampled_xToNearestIndex (me, rFreqs [i]);
- }
- autoNUMvector <double> inSig (1, nbark);
- for (integer i = 1; i <= nbark; i ++) {
- integer low = iFreqs [i], high = iFreqs [i + 1] - 1;
- if (low < 1) low = 1;
- if (high > my nx) high = my nx;
- for (integer j = low; j <= high; j ++) {
- inSig [i] += re [j] * re [j] + im [j] * im [j]; // Pa2 s2
- }
- /* An anti-undersampling correction. */
- if (high >= low)
- inSig [i] *= 2.0 * (rFreqs [i + 1] - rFreqs [i]) / (high - low + 1) * my dx; // Pa2: power density in this band
- }
- /* Convolution with auditory (masking) filter. */
- autoNUMvector <double> outSig (1, 2 * nbark);
- for (integer i = 1; i <= nbark; i ++) {
- for (integer j = 1; j <= nbark; j ++) {
- outSig [i + j] += inSig [i] * auditoryFilter [j];
- }
- }
- autoExcitation thee = Excitation_create (dbark, nbark);
- for (integer i = 1; i <= nbark; i ++) {
- thy z [1] [i] = Excitation_soundPressureToPhon (sqrt (outSig [i + nbark/2]), Sampled_indexToX (thee.get(), i));
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U": not converted to Excitation.");
- }
- }
- /* End of file Spectrum_to_Excitation.cpp */
|