84 lines
2.2 KiB
C
84 lines
2.2 KiB
C
/*--------------------------------------------------------------------------*\
|
|
|
|
FILE........: sd.c
|
|
AUTHOR......: David Rowe
|
|
DATE CREATED: 20/7/93
|
|
|
|
Function to determine spectral distortion between two sets of LPCs.
|
|
|
|
\*--------------------------------------------------------------------------*/
|
|
|
|
/*
|
|
Copyright (C) 2009 David Rowe
|
|
|
|
All rights reserved.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License version 2, as
|
|
published by the Free Software Foundation. This program 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 Lesser General Public License
|
|
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#define MAX_N 2048 /* maximum DFT size */
|
|
|
|
#include <math.h>
|
|
#include "four1.h"
|
|
#include "comp.h"
|
|
#include "sd.h"
|
|
|
|
/*---------------------------------------------------------------------------*\
|
|
|
|
FUNCTION....: spectral_dist()
|
|
|
|
AUTHOR......: David Rowe
|
|
DATE CREATED: 20/7/93
|
|
|
|
This function returns the soectral distoertion between two
|
|
sets of LPCs.
|
|
|
|
\*---------------------------------------------------------------------------*/
|
|
|
|
float spectral_dist(float ak1[], float ak2[], int p, int n)
|
|
/* float ak1[]; unquantised set of p+1 LPCs */
|
|
/* float ak2[]; quantised set of p+1 LPCs */
|
|
/* int p; LP order */
|
|
/* int n; DFT size to use for SD calculations (power of 2) */
|
|
{
|
|
COMP A1[MAX_N]; /* DFT of ak1[] */
|
|
COMP A2[MAX_N]; /* DFT of ak2[] */
|
|
float P1,P2; /* power of current bin */
|
|
float sd;
|
|
int i;
|
|
|
|
for(i=0; i<n; i++) {
|
|
A1[i].real = 0.0;
|
|
A1[i].imag = 0.0;
|
|
A2[i].real = 0.0;
|
|
A2[i].imag = 0.0;
|
|
}
|
|
|
|
for(i=0; i<p+1; i++) {
|
|
A1[i].real = ak1[i];
|
|
A2[i].real = ak2[i];
|
|
}
|
|
|
|
four1(&A1[-1].imag,n,-1);
|
|
four1(&A2[-1].imag,n,-1);
|
|
|
|
sd = 0.0;
|
|
for(i=0; i<n; i++) {
|
|
P1 = A1[i].real*A1[i].real + A1[i].imag*A1[i].imag;
|
|
P2 = A2[i].real*A2[i].real + A2[i].imag*A2[i].imag;
|
|
sd += pow(log10(P2/P1),2.0);
|
|
}
|
|
sd = 10.0*sqrt(sd/n); /* sd in dB */
|
|
|
|
return(sd);
|
|
}
|