summaryrefslogtreecommitdiff
path: root/sms/spectralApprox.c
diff options
context:
space:
mode:
Diffstat (limited to 'sms/spectralApprox.c')
-rw-r--r--sms/spectralApprox.c213
1 files changed, 105 insertions, 108 deletions
diff --git a/sms/spectralApprox.c b/sms/spectralApprox.c
index 9e77f7a..c19afbe 100644
--- a/sms/spectralApprox.c
+++ b/sms/spectralApprox.c
@@ -35,121 +35,118 @@
* \param sizeSpec2 size of output envelope
* \param nCoefficients number of coefficients to use in approximation
* \return error code \see SMS_ERRORS (or -1 if the algorithm just messes up,
- it will print an error of its own.
+ * it will print an error of its own.
*/
-int sms_spectralApprox (sfloat *pFSpec1, int sizeSpec1, int sizeSpec1Used,
- sfloat *pFSpec2, int sizeSpec2, int nCoefficients)
+int sms_spectralApprox(sfloat *pFSpec1, int sizeSpec1, int sizeSpec1Used,
+ sfloat *pFSpec2, int sizeSpec2, int nCoefficients,
+ sfloat *envelope)
{
- sfloat fHopSize, fCurrentLoc = 0, fLeft = 0, fRight = 0, fValue = 0,
- fLastLocation, fSizeX, fSpec2Acum=0, fNextHop, fDeltaY, *pFEnvelope;
- int iFirstGood = 0, iLastSample = 0, i, j;
+ sfloat fHopSize, fCurrentLoc = 0, fLeft = 0, fRight = 0, fValue = 0,
+ fLastLocation, fSizeX, fSpec2Acum=0, fNextHop, fDeltaY;
+ int iFirstGood = 0, iLastSample = 0, i, j;
- /* when number of coefficients is smaller than 2 do not approximate */
- if (nCoefficients < 2)
- {
- for (i = 0; i < sizeSpec2; i++)
- pFSpec2[i] = 1;
- return(SMS_OK);
- }
-
- if ((pFEnvelope = (sfloat *) calloc(nCoefficients, sizeof(sfloat))) == NULL)
- return(SMS_MALLOC);
-
- /* calculate the hop size */
- if (sizeSpec1 != sizeSpec1Used)
- fHopSize = (sfloat) sizeSpec1Used / nCoefficients;
- else //why is this here, would be the same as sizeSpec1Used / nCoefficients
- fHopSize = (sfloat) sizeSpec1 / nCoefficients;
- if(nCoefficients > sizeSpec1)
- nCoefficients = sizeSpec1;
+ /* when number of coefficients is smaller than 2 do not approximate */
+ if(nCoefficients < 2)
+ {
+ for (i = 0; i < sizeSpec2; i++)
+ pFSpec2[i] = 1;
+ return SMS_OK;
+ }
+ /* calculate the hop size */
+ if(sizeSpec1 != sizeSpec1Used)
fHopSize = (sfloat) sizeSpec1Used / nCoefficients;
-
- /* approximate by linear interpolation */
- if (fHopSize > 1)
- {
- iFirstGood = 0;
- for (i = 0; i < nCoefficients; i++)
- {
- iLastSample = fLastLocation = fCurrentLoc + fHopSize;
- iLastSample = MIN (sizeSpec1-1, iLastSample);
- if (iLastSample < sizeSpec1-1)
- fRight = pFSpec1[iLastSample] +
- (pFSpec1[iLastSample+1] - pFSpec1[iLastSample]) *
- (fLastLocation - iLastSample);
- else
- fRight = pFSpec1[iLastSample];
- fValue = 0;
- for (j = iFirstGood; j <= iLastSample; j++)
- fValue = MAX (fValue, pFSpec1[j]);
- fValue = MAX (fValue, MAX (fRight, fLeft));
- pFEnvelope[i] = fValue;
- fLeft = fRight;
- fCurrentLoc = fLastLocation;
- iFirstGood = (int) (1+ fCurrentLoc);
- }
- }
- else if (fHopSize == 1)
- {
- for (i = 0; i < nCoefficients; i++)
- pFEnvelope[i] = pFSpec1[i];
- }
- else
- {
- free (pFEnvelope);
- //printf ("SpectralApprox: sizeSpec1 has too many nCoefficients\n"); /* \todo need to increase the frequency? */
- sms_error ("SpectralApprox: sizeSpec1 has too many nCoefficients\n"); /* \todo need to increase the frequency? */
- return -1;
- }
+ else //why is this here, would be the same as sizeSpec1Used / nCoefficients
+ fHopSize = (sfloat) sizeSpec1 / nCoefficients;
+
+ if(nCoefficients > sizeSpec1)
+ nCoefficients = sizeSpec1;
+
+ fHopSize = (sfloat) sizeSpec1Used / nCoefficients;
+
+ /* approximate by linear interpolation */
+ if (fHopSize > 1)
+ {
+ iFirstGood = 0;
+ for(i = 0; i < nCoefficients; i++)
+ {
+ iLastSample = fLastLocation = fCurrentLoc + fHopSize;
+ iLastSample = MIN (sizeSpec1-1, iLastSample);
+ if(iLastSample < sizeSpec1-1)
+ fRight = pFSpec1[iLastSample] +
+ (pFSpec1[iLastSample+1] - pFSpec1[iLastSample]) *
+ (fLastLocation - iLastSample);
+ else
+ fRight = pFSpec1[iLastSample];
+ fValue = 0;
+ for(j = iFirstGood; j <= iLastSample; j++)
+ fValue = MAX (fValue, pFSpec1[j]);
+ fValue = MAX (fValue, MAX (fRight, fLeft));
+ envelope[i] = fValue;
+ fLeft = fRight;
+ fCurrentLoc = fLastLocation;
+ iFirstGood = (int) (1+ fCurrentLoc);
+ }
+ }
+ else if(fHopSize == 1)
+ {
+ for(i = 0; i < nCoefficients; i++)
+ envelope[i] = pFSpec1[i];
+ }
+ else
+ {
+ sms_error("SpectralApprox: sizeSpec1 has too many nCoefficients\n"); /* \todo need to increase the frequency? */
+ return -1;
+ }
+
+ /* Creates Spec2 from Envelope */
+ if(nCoefficients < sizeSpec2)
+ {
+ fSizeX = (sfloat) (sizeSpec2-1) / nCoefficients;
- /* Creates Spec2 from Envelope */
- if (nCoefficients < sizeSpec2)
- {
- fSizeX = (sfloat) (sizeSpec2-1) / nCoefficients;
+ /* the first step */
+ fNextHop = fSizeX / 2;
+ fDeltaY = envelope[0] / fNextHop;
+ fSpec2Acum=pFSpec2[j=0]=0;
+ while(++j < fNextHop)
+ pFSpec2[j] = (fSpec2Acum += fDeltaY);
- /* the first step */
- fNextHop = fSizeX / 2;
- fDeltaY = pFEnvelope[0] / fNextHop;
- fSpec2Acum=pFSpec2[j=0]=0;
- while (++j < fNextHop)
- pFSpec2[j] = (fSpec2Acum += fDeltaY);
-
- /* middle values */
- for (i = 0; i <= nCoefficients-2; ++i)
- {
- fDeltaY = (pFEnvelope[i+1] - pFEnvelope[i]) / fSizeX;
- /* first point of a segment */
- pFSpec2[j] = (fSpec2Acum = (pFEnvelope[i]+(fDeltaY*(j-fNextHop))));
- ++j;
- /* remaining points */
- fNextHop += fSizeX;
- while (j < fNextHop)
- pFSpec2[j++] = (fSpec2Acum += fDeltaY);
- }
+ /* middle values */
+ for(i = 0; i <= nCoefficients-2; ++i)
+ {
+ fDeltaY = (envelope[i+1] - envelope[i]) / fSizeX;
+ /* first point of a segment */
+ pFSpec2[j] = (fSpec2Acum = (envelope[i]+(fDeltaY*(j-fNextHop))));
+ ++j;
+ /* remaining points */
+ fNextHop += fSizeX;
+ while(j < fNextHop)
+ pFSpec2[j++] = (fSpec2Acum += fDeltaY);
+ }
- /* last step */
- fDeltaY = -pFEnvelope[i] * 2 / fSizeX;
- /* first point of the last segment */
- pFSpec2[j] = (fSpec2Acum = (pFEnvelope[i]+(fDeltaY*(j-fNextHop))));
- ++j;
- fNextHop += fSizeX / 2;
- while (j < sizeSpec2-1)
- pFSpec2[j++]=(fSpec2Acum += fDeltaY);
- /* last should be exactly zero */
- pFSpec2[sizeSpec2-1] = .0;
+ /* last step */
+ fDeltaY = -envelope[i] * 2 / fSizeX;
+ /* first point of the last segment */
+ pFSpec2[j] = (fSpec2Acum = (envelope[i]+(fDeltaY*(j-fNextHop))));
+ ++j;
+ fNextHop += fSizeX / 2;
+ while(j < sizeSpec2-1)
+ pFSpec2[j++]=(fSpec2Acum += fDeltaY);
+ /* last should be exactly zero */
+ pFSpec2[sizeSpec2-1] = .0;
+ }
+ else if(nCoefficients == sizeSpec2)
+ {
+ for(i = 0; i < nCoefficients; i++)
+ pFSpec2[i] = envelope[i];
+ }
+ else
+ {
+ sms_error("SpectralApprox: sizeSpec2 has too many nCoefficients\n");
+ return -1;
}
- else if (nCoefficients == sizeSpec2)
- {
- for (i = 0; i < nCoefficients; i++)
- pFSpec2[i] = pFEnvelope[i];
- }
- else
- {
- free (pFEnvelope);
- //printf ("SpectralApprox: sizeSpec2 has too many nCoefficients\n");
- sms_error ("SpectralApprox: sizeSpec2 has too many nCoefficients\n");
- return -1;
- }
- free (pFEnvelope); /* \todo make this a static array */
- return (SMS_OK);
+
+ return SMS_OK;
}
+
+