summaryrefslogtreecommitdiff
path: root/sms/fileIO.c
diff options
context:
space:
mode:
Diffstat (limited to 'sms/fileIO.c')
-rw-r--r--sms/fileIO.c633
1 files changed, 633 insertions, 0 deletions
diff --git a/sms/fileIO.c b/sms/fileIO.c
new file mode 100644
index 0000000..67d750a
--- /dev/null
+++ b/sms/fileIO.c
@@ -0,0 +1,633 @@
+/*
+ * Copyright (c) 2008 MUSIC TECHNOLOGY GROUP (MTG)
+ * UNIVERSITAT POMPEU FABRA
+ *
+ *
+ * This program 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 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 General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ *
+ */
+/*! \file fileIO.c
+ * \brief SMS file input and output
+ */
+
+#include "sms.h"
+
+/*! \brief file identification constant
+ *
+ * constant number that is first within SMS_Header, in order to correctly
+ * identify an SMS file when read.
+ */
+#define SMS_MAGIC 767
+
+static char pChTextString[1000]; /*!< string to store analysis parameters in sms header */
+
+/*! \brief initialize the header structure of an SMS file
+ *
+ * \param pSmsHeader header for SMS file
+ */
+void sms_initHeader (SMS_Header *pSmsHeader)
+{
+ pSmsHeader->iSmsMagic = SMS_MAGIC;
+ pSmsHeader->iHeadBSize = sizeof(SMS_Header);
+ pSmsHeader->nFrames = 0;
+ pSmsHeader->iFrameBSize = 0;
+ pSmsHeader->iFormat = SMS_FORMAT_H;
+ pSmsHeader->iFrameRate = 0;
+ pSmsHeader->iStochasticType = SMS_STOC_APPROX;
+ pSmsHeader->nTracks = 0;
+ pSmsHeader->nStochasticCoeff = 0;
+ pSmsHeader->nEnvCoeff = 0;
+ pSmsHeader->iMaxFreq = 0;
+ pSmsHeader->fResidualPerc = 0;
+ pSmsHeader->nTextCharacters = 0;
+ pSmsHeader->pChTextCharacters = NULL;
+}
+
+/*! \brief fill an SMS header with necessary information for storage
+ *
+ * copies parameters from SMS_AnalParams, along with other values
+ * so an SMS file can be stored and correctly synthesized at a later
+ * time. This is somewhat of a convenience function.
+ *
+ * sms_initAnal() should be done first to properly set everything.
+ *
+ * \param pSmsHeader header for SMS file (to be stored)
+ * \param pAnalParams structure of analysis parameters
+ * \param pProgramString pointer to a string containing the name of the program that made the analysis data
+ */
+void sms_fillHeader (SMS_Header *pSmsHeader, SMS_AnalParams *pAnalParams,
+ char *pProgramString)
+{
+ sms_initHeader (pSmsHeader);
+ pSmsHeader->nFrames = pAnalParams->nFrames;
+ pSmsHeader->iFormat = pAnalParams->iFormat;
+ pSmsHeader->iFrameRate = pAnalParams->iFrameRate;
+ pSmsHeader->iStochasticType = pAnalParams->iStochasticType;
+ pSmsHeader->nTracks = pAnalParams->nTracks;
+ pSmsHeader->iSamplingRate = pAnalParams->iSamplingRate;
+ if(pAnalParams->iStochasticType == SMS_STOC_NONE)
+ pSmsHeader->nStochasticCoeff = 0;
+ else
+ pSmsHeader->nStochasticCoeff = pAnalParams->nStochasticCoeff;
+ pSmsHeader->iEnvType = pAnalParams->specEnvParams.iType;
+ pSmsHeader->nEnvCoeff = pAnalParams->specEnvParams.nCoeff;
+ pSmsHeader->iMaxFreq = (int) pAnalParams->fHighestFreq;
+ pSmsHeader->iFrameBSize = sms_frameSizeB(pSmsHeader);
+ sprintf (pChTextString,
+ "created by %s with parameters: format %d, soundType %d, "
+ "analysisDirection %d, windowSize %.2f,"
+ " windowType %d, frameRate %d, highestFreq %.2f, minPeakMag %.2f,"
+ " refHarmonic %d, minRefHarmMag %.2f, refHarmMagDiffFromMax %.2f,"
+ " defaultFund %.2f, lowestFund %.2f, highestFund %.2f, nGuides %d,"
+ " nTracks %d, freqDeviation %.2f, peakContToGuide %.2f,"
+ " fundContToGuide %.2f, cleanTracks %d, iMinTrackLength %d,"
+ "iMaxSleepingTime %d, stochasticType %d, nStocCoeff %d\n"
+ "iEnvType: %d, nEnvCoeff: %d",
+ pProgramString,
+ pAnalParams->iFormat, pAnalParams->iSoundType,
+ pAnalParams->iAnalysisDirection, pAnalParams->fSizeWindow,
+ pAnalParams->iWindowType, pAnalParams->iFrameRate,
+ pAnalParams->fHighestFreq, pAnalParams->fMinPeakMag,
+ pAnalParams->iRefHarmonic, pAnalParams->fMinRefHarmMag,
+ pAnalParams->fRefHarmMagDiffFromMax,
+ pAnalParams->fDefaultFundamental, pAnalParams->fLowestFundamental,
+ pAnalParams->fHighestFundamental, pAnalParams->nGuides,
+ pAnalParams->nTracks, pAnalParams->fFreqDeviation,
+ pAnalParams->fPeakContToGuide, pAnalParams->fFundContToGuide,
+ pAnalParams->iCleanTracks, pAnalParams->iMinTrackLength,
+ pAnalParams->iMaxSleepingTime, pAnalParams->iStochasticType,
+ pAnalParams->nStochasticCoeff, pSmsHeader->iEnvType, pSmsHeader->nEnvCoeff);
+
+ pSmsHeader->nTextCharacters = strlen (pChTextString) + 1;
+ pSmsHeader->pChTextCharacters = (char *) pChTextString;
+}
+
+/*! \brief write SMS header to file
+ *
+ * \param pChFileName file name for SMS file
+ * \param pSmsHeader header for SMS file
+ * \param ppSmsFile (double pointer to) file to be created
+ * \return error code \see SMS_WRERR in SMS_ERRORS
+ */
+int sms_writeHeader (char *pChFileName, SMS_Header *pSmsHeader,
+ FILE **ppSmsFile)
+{
+ int iVariableSize = 0;
+
+ if (pSmsHeader->iSmsMagic != SMS_MAGIC)
+ {
+ sms_error("not an SMS file");
+ return(-1);
+ }
+ if ((*ppSmsFile = fopen (pChFileName, "w+")) == NULL)
+ {
+ sms_error("cannot open file for writing");
+ return(-1);
+ }
+ /* check variable size of header */
+/* iVariableSize = sizeof (int) * pSmsHeader->nLoopRecords + */
+/* sizeof (sfloat) * pSmsHeader->nSpecEnvelopePoints + */
+/* sizeof(char) * pSmsHeader->nTextCharacters; */
+ iVariableSize = sizeof(char) * pSmsHeader->nTextCharacters;
+
+ pSmsHeader->iHeadBSize = sizeof(SMS_Header) + iVariableSize;
+
+ /* write header */
+ if (fwrite((void *)pSmsHeader, (size_t)1, (size_t)sizeof(SMS_Header),
+ *ppSmsFile) < (size_t)sizeof(SMS_Header))
+ {
+ sms_error("cannot write output file");
+ return(-1);
+ }
+ /* write variable part of header */
+ if (pSmsHeader->nTextCharacters > 0)
+ {
+ char *pChStart = (char *) pSmsHeader->pChTextCharacters;
+ int iSize = sizeof(char) * pSmsHeader->nTextCharacters;
+
+ if (fwrite ((void *)pChStart, (size_t)1, (size_t)iSize, *ppSmsFile) <
+ (size_t)iSize)
+ {
+ sms_error("cannot write output file (nTextCharacters)");
+ return(-1);
+ }
+ }
+ return (0);
+}
+
+/*! \brief rewrite SMS header and close file
+ *
+ * \param pSmsFile pointer to SMS file
+ * \param pSmsHeader pointer to header for SMS file
+ * \return error code \see SMS_WRERR in SMS_ERRORS
+ */
+int sms_writeFile (FILE *pSmsFile, SMS_Header *pSmsHeader)
+{
+ int iVariableSize;
+
+ rewind(pSmsFile);
+
+ /* check variable size of header */
+ iVariableSize = sizeof(char) * pSmsHeader->nTextCharacters;
+
+ pSmsHeader->iHeadBSize = sizeof(SMS_Header) + iVariableSize;
+
+ /* write header */
+ if (fwrite((void *)pSmsHeader, (size_t)1, (size_t)sizeof(SMS_Header),
+ pSmsFile) < (size_t)sizeof(SMS_Header))
+ {
+ sms_error("cannot write output file (header)");
+ return(-1);
+ }
+
+ if (pSmsHeader->nTextCharacters > 0)
+ {
+ char *pChStart = (char *) pSmsHeader->pChTextCharacters;
+ int iSize = sizeof(char) * pSmsHeader->nTextCharacters;
+
+ if (fwrite ((void *)pChStart, (size_t)1, (size_t)iSize, pSmsFile) <
+ (size_t)iSize)
+ {
+ sms_error("cannot write output file (nTextCharacters)");
+ return(-1);
+ }
+ }
+
+ fclose(pSmsFile);
+ return (0);
+}
+
+/*! \brief write SMS frame
+ *
+ * \param pSmsFile pointer to SMS file
+ * \param pSmsHeader pointer to SMS header
+ * \param pSmsFrame pointer to SMS data frame
+ * \return 0 on success, -1 on failure
+ */
+int sms_writeFrame (FILE *pSmsFile, SMS_Header *pSmsHeader,
+ SMS_Data *pSmsFrame)
+{
+ if (fwrite ((void *)pSmsFrame->pSmsData, 1, pSmsHeader->iFrameBSize,
+ pSmsFile) < (unsigned int) pSmsHeader->iFrameBSize)
+ {
+ sms_error("cannot write frame to output file");
+ return(-1);
+ }
+ else return (0);
+}
+
+
+/*! \brief get the size in bytes of the frame in a SMS file
+ *
+ * \param pSmsHeader pointer to SMS header
+ * \return the size in bytes of the frame
+ */
+int sms_frameSizeB (SMS_Header *pSmsHeader)
+{
+ int iSize, nDet;
+
+ if (pSmsHeader->iFormat == SMS_FORMAT_H ||
+ pSmsHeader->iFormat == SMS_FORMAT_IH)
+ nDet = 2;/* freq, mag */
+ else nDet = 3; /* freq, mag, phase */
+
+ iSize = sizeof (sfloat) * (nDet * pSmsHeader->nTracks);
+
+ if(pSmsHeader->iStochasticType == SMS_STOC_APPROX)
+ { /* stocCoeff + 1 (gain) */
+ iSize += sizeof(sfloat) * (pSmsHeader->nStochasticCoeff + 1);
+ }
+ else if(pSmsHeader->iStochasticType == SMS_STOC_IFFT)
+ {
+ /* sizeFFT*2 + 1 (gain) */
+ iSize += sizeof(sfloat) * (pSmsHeader->nStochasticCoeff * 2 + 1);
+ }
+ iSize += sizeof(sfloat) * pSmsHeader->nEnvCoeff;
+ return(iSize);
+}
+
+
+/*! \brief function to read SMS header
+ *
+ * \param pChFileName file name for SMS file
+ * \param ppSmsHeader (double pointer to) SMS header
+ * \param ppSmsFile (double pointer to) inputfile
+ * \return error code \see SMS_ERRORS
+ */
+int sms_getHeader (char *pChFileName, SMS_Header **ppSmsHeader,
+ FILE **ppSmsFile)
+{
+ int iHeadBSize, iFrameBSize, nFrames;
+ int iMagicNumber;
+
+ /* open file for reading */
+ if ((*ppSmsFile = fopen (pChFileName, "r")) == NULL)
+ {
+ sms_error("could not open SMS header");
+ return (-1);
+ }
+ /* read magic number */
+ if (fread ((void *) &iMagicNumber, (size_t) sizeof(int), (size_t)1,
+ *ppSmsFile) < (size_t)1)
+ {
+ sms_error("could not read SMS header");
+ return (-1);
+ }
+
+ if (iMagicNumber != SMS_MAGIC)
+ {
+ sms_error("not an SMS file");
+ return (-1);
+ }
+
+ /* read size of of header */
+ if (fread ((void *) &iHeadBSize, (size_t) sizeof(int), (size_t)1,
+ *ppSmsFile) < (size_t)1)
+ {
+ sms_error("could not read SMS header (iHeadBSize)");
+ return (-1);
+ }
+
+ if (iHeadBSize <= 0)
+ {
+ sms_error("bad SMS header size");
+ return (-1);
+ }
+
+ /* read number of data Frames */
+ if (fread ((void *) &nFrames, (size_t) sizeof(int), (size_t)1,
+ *ppSmsFile) < (size_t)1)
+ {
+ sms_error("could not read SMS number of frames");
+ return (-1);
+ }
+
+ if (nFrames <= 0)
+ {
+ sms_error("number of frames <= 0");
+ return (-1);
+ }
+
+ /* read size of data Frames */
+ if (fread ((void *) &iFrameBSize, (size_t) sizeof(int), (size_t)1,
+ *ppSmsFile) < (size_t)1)
+ {
+ sms_error("could not read size of SMS data");
+ return (-1);
+ }
+
+ if (iFrameBSize <= 0)
+ {
+ sms_error("size bytes of frames <= 0");
+ return (-1);
+ }
+
+ /* allocate memory for header */
+ if (((*ppSmsHeader) = (SMS_Header *)malloc (iHeadBSize)) == NULL)
+ {
+ sms_error("cannot allocate memory for header");
+ return (-1);
+ }
+
+ /* read header */
+ rewind (*ppSmsFile);
+ if (fread ((void *) (*ppSmsHeader), 1, iHeadBSize, *ppSmsFile) < (unsigned int) iHeadBSize)
+ {
+ sms_error("cannot read header of SMS file");
+ return (-1);
+ }
+
+ /* set pointers to variable part of header */
+/* if ((*ppSmsHeader)->nLoopRecords > 0) */
+/* (*ppSmsHeader)->pILoopRecords = (int *) ((char *)(*ppSmsHeader) + */
+/* sizeof(SMS_Header)); */
+
+/* if ((*ppSmsHeader)->nSpecEnvelopePoints > 0) */
+/* (*ppSmsHeader)->pFSpectralEnvelope = */
+/* (sfloat *) ((char *)(*ppSmsHeader) + sizeof(SMS_Header) + */
+/* sizeof(int) * (*ppSmsHeader)->nLoopRecords); */
+
+/* if ((*ppSmsHeader)->nTextCharacters > 0) */
+/* (*ppSmsHeader)->pChTextCharacters = */
+/* (char *) ((char *)(*ppSmsHeader) + sizeof(SMS_Header) + */
+/* sizeof(int) * (*ppSmsHeader)->nLoopRecords + */
+/* sizeof(sfloat) * (*ppSmsHeader)->nSpecEnvelopePoints); */
+ if ((*ppSmsHeader)->nTextCharacters > 0)
+ (*ppSmsHeader)->pChTextCharacters = (char *)(*ppSmsHeader) + sizeof(SMS_Header);
+
+ return (0);
+}
+
+/*! \brief read an SMS data frame
+ *
+ * \param pSmsFile pointer to SMS file
+ * \param pSmsHeader pointer to SMS header
+ * \param iFrame frame number
+ * \param pSmsFrame pointer to SMS frame
+ * \return 0 on sucess, -1 on error
+ */
+int sms_getFrame (FILE *pSmsFile, SMS_Header *pSmsHeader, int iFrame,
+ SMS_Data *pSmsFrame)
+{
+ if (fseek (pSmsFile, pSmsHeader->iHeadBSize + iFrame *
+ pSmsHeader->iFrameBSize, SEEK_SET) < 0)
+ {
+ sms_error ("cannot seek to the SMS frame");
+ return (-1);
+ }
+ if ((pSmsHeader->iFrameBSize =
+ fread ((void *)pSmsFrame->pSmsData, (size_t)1,
+ (size_t)pSmsHeader->iFrameBSize, pSmsFile))
+ != pSmsHeader->iFrameBSize)
+ {
+ sms_error ("cannot read SMS frame");
+ return (-1);
+ }
+ return (0);
+}
+
+/*! \brief allocate memory for a frame of SMS data
+ *
+ * \param pSmsFrame pointer to a frame of SMS data
+ * \param nTracks number of sinusoidal tracks in frame
+ * \param nStochCoeff number of stochastic coefficients in frame
+ * \param iPhase whether phase information is in the frame
+ * \param stochType stochastic resynthesis type
+ * \param nStochCoeff number of envelope coefficients in frame
+ * \param nEnvCoeff number of envelope coefficients in frame
+ * \return 0 on success, -1 on error
+ */
+int sms_allocFrame (SMS_Data *pSmsFrame, int nTracks, int nStochCoeff, int iPhase,
+ int stochType, int nEnvCoeff)
+{
+ sfloat *dataPos; /* a marker to locate specific data witin smsData */
+ /* calculate size of frame */
+ int sizeData = 2 * nTracks * sizeof(sfloat);
+ sizeData += 1 * sizeof(sfloat); //adding one for nSamples
+ if (iPhase > 0) sizeData += nTracks * sizeof(sfloat);
+ if (stochType == SMS_STOC_APPROX)
+ sizeData += (nStochCoeff + 1) * sizeof(sfloat);
+ else if (stochType == SMS_STOC_IFFT)
+ sizeData += (2*nStochCoeff + 1) * sizeof(sfloat);
+ sizeData += nEnvCoeff * sizeof(sfloat); /* add in number of envelope coefficients (cep or fbins) if any */
+ /* allocate memory for data */
+ if ((pSmsFrame->pSmsData = (sfloat *) malloc (sizeData)) == NULL)
+ {
+ sms_error("cannot allocate memory for SMS frame data");
+ return (-1);
+ }
+
+ /* set the variables in the structure */
+ /* \todo why not set these in init functions, then allocate with them?? */
+ pSmsFrame->sizeData = sizeData;
+ pSmsFrame->nTracks = nTracks;
+ pSmsFrame->nCoeff = nStochCoeff;
+ pSmsFrame->nEnvCoeff = nEnvCoeff;
+ /* set pointers to data types within smsData array */
+ pSmsFrame->pFSinFreq = pSmsFrame->pSmsData;
+ dataPos = (sfloat *)(pSmsFrame->pFSinFreq + nTracks);
+ memset(pSmsFrame->pFSinFreq, 0, sizeof(sfloat) * nTracks);
+
+ pSmsFrame->pFSinAmp = dataPos;
+ dataPos = (sfloat *)(pSmsFrame->pFSinAmp + nTracks);
+ memset(pSmsFrame->pFSinAmp, 0, sizeof(sfloat) * nTracks);
+
+ if (iPhase > 0)
+ {
+ pSmsFrame->pFSinPha = dataPos;
+ dataPos = (sfloat *) (pSmsFrame->pFSinPha + nTracks);
+ memset(pSmsFrame->pFSinPha, 0, sizeof(sfloat) * nTracks);
+ }
+ else pSmsFrame->pFSinPha = NULL;
+
+ if (stochType == SMS_STOC_APPROX)
+ {
+ pSmsFrame->pFStocCoeff = dataPos;
+ dataPos = (sfloat *) (pSmsFrame->pFStocCoeff + nStochCoeff);
+ memset(pSmsFrame->pFStocCoeff, 0, sizeof(sfloat) * nStochCoeff);
+
+ pSmsFrame->pFStocGain = dataPos;
+ dataPos = (sfloat *) (pSmsFrame->pFStocGain + 1);
+ }
+ else if (stochType == SMS_STOC_IFFT)
+ {
+ pSmsFrame->pFStocCoeff = dataPos;
+ dataPos = (sfloat *) (pSmsFrame->pFStocCoeff + nStochCoeff);
+ pSmsFrame->pResPhase = dataPos;
+ dataPos = (sfloat *) (pSmsFrame->pResPhase + nStochCoeff);
+ pSmsFrame->pFStocGain = dataPos;
+ dataPos = (sfloat *) (pSmsFrame->pFStocGain + 1);
+ }
+ else
+ {
+ pSmsFrame->pFStocCoeff = NULL;
+ pSmsFrame->pResPhase = NULL;
+ pSmsFrame->pFStocGain = NULL;
+ }
+ if (nEnvCoeff > 0)
+ pSmsFrame->pSpecEnv = dataPos;
+ else
+ pSmsFrame->pSpecEnv = NULL;
+ return (0);
+}
+
+/*! \brief function to allocate an SMS data frame using an SMS_Header
+ *
+ * this one is used when you have only read the header, such as after
+ * opening a file.
+ *
+ * \param pSmsHeader pointer to SMS header
+ * \param pSmsFrame pointer to SMS frame
+ * \return 0 on success, -1 on error
+ */
+int sms_allocFrameH (SMS_Header *pSmsHeader, SMS_Data *pSmsFrame)
+{
+ int iPhase = (pSmsHeader->iFormat == SMS_FORMAT_HP ||
+ pSmsHeader->iFormat == SMS_FORMAT_IHP) ? 1 : 0;
+ return (sms_allocFrame (pSmsFrame, pSmsHeader->nTracks,
+ pSmsHeader->nStochasticCoeff, iPhase,
+ pSmsHeader->iStochasticType, pSmsHeader->nEnvCoeff));
+}
+
+
+/*! \brief free the SMS data structure
+ *
+ * \param pSmsFrame pointer to frame of SMS data
+ */
+void sms_freeFrame (SMS_Data *pSmsFrame)
+{
+ free(pSmsFrame->pSmsData);
+ pSmsFrame->nTracks = 0;
+ pSmsFrame->nCoeff = 0;
+ pSmsFrame->sizeData = 0;
+ pSmsFrame->pFSinFreq = NULL;
+ pSmsFrame->pFSinAmp = NULL;
+ pSmsFrame->pFStocCoeff = NULL;
+ pSmsFrame->pResPhase = NULL;
+ pSmsFrame->pFStocGain = NULL;
+}
+
+/*! \brief clear the SMS data structure
+ *
+ * \param pSmsFrame pointer to frame of SMS data
+ */
+void sms_clearFrame (SMS_Data *pSmsFrame)
+{
+ memset ((char *) pSmsFrame->pSmsData, 0, pSmsFrame->sizeData);
+}
+
+/*! \brief copy a frame of SMS_Data
+ *
+ * \param pCopySmsData copy of frame
+ * \param pOriginalSmsData original frame
+ *
+ */
+void sms_copyFrame (SMS_Data *pCopySmsData, SMS_Data *pOriginalSmsData)
+{
+ /* if the two frames are the same size just copy data */
+ if (pCopySmsData->sizeData == pOriginalSmsData->sizeData &&
+ pCopySmsData->nTracks == pOriginalSmsData->nTracks)
+ {
+ memcpy ((char *)pCopySmsData->pSmsData,
+ (char *)pOriginalSmsData->pSmsData,
+ pCopySmsData->sizeData);
+ }
+ /* if frames is different size copy the smallest */
+ else
+ {
+ int nTracks = MIN (pCopySmsData->nTracks, pOriginalSmsData->nTracks);
+ int nCoeff = MIN (pCopySmsData->nCoeff, pOriginalSmsData->nCoeff);
+
+ pCopySmsData->nTracks = nTracks;
+ pCopySmsData->nCoeff = nCoeff;
+ memcpy ((char *)pCopySmsData->pFSinFreq,
+ (char *)pOriginalSmsData->pFSinFreq,
+ sizeof(sfloat) * nTracks);
+ memcpy ((char *)pCopySmsData->pFSinAmp,
+ (char *)pOriginalSmsData->pFSinAmp,
+ sizeof(sfloat) * nTracks);
+ if (pOriginalSmsData->pFSinPha != NULL &&
+ pCopySmsData->pFSinPha != NULL)
+ memcpy ((char *)pCopySmsData->pFSinPha,
+ (char *)pOriginalSmsData->pFSinPha,
+ sizeof(sfloat) * nTracks);
+ if (pOriginalSmsData->pFStocCoeff != NULL &&
+ pCopySmsData->pFStocCoeff != NULL)
+ {
+ if (pOriginalSmsData->pResPhase != NULL &&
+ pCopySmsData->pResPhase != NULL)
+ memcpy ((char *)pCopySmsData->pResPhase,
+ (char *)pOriginalSmsData->pResPhase,
+ sizeof(sfloat) * nCoeff);
+ }
+ if (pOriginalSmsData->pFStocGain != NULL &&
+ pCopySmsData->pFStocGain != NULL)
+ memcpy ((char *)pCopySmsData->pFStocGain,
+ (char *)pOriginalSmsData->pFStocGain,
+ sizeof(sfloat));
+ }
+}
+
+/*! \brief function to interpolate two SMS frames
+ *
+ * this assumes that the two frames are of the same size
+ *
+ * \param pSmsFrame1 sms frame 1
+ * \param pSmsFrame2 sms frame 2
+ * \param pSmsFrameOut sms output frame
+ * \param fInterpFactor interpolation factor
+ */
+void sms_interpolateFrames (SMS_Data *pSmsFrame1, SMS_Data *pSmsFrame2,
+ SMS_Data *pSmsFrameOut, sfloat fInterpFactor)
+{
+ int i;
+ sfloat fFreq1, fFreq2;
+
+ /* interpolate the deterministic part */
+ for (i = 0; i < pSmsFrame1->nTracks; i++)
+ {
+ fFreq1 = pSmsFrame1->pFSinFreq[i];
+ fFreq2 = pSmsFrame2->pFSinFreq[i];
+ if (fFreq1 == 0) fFreq1 = fFreq2;
+ if (fFreq2 == 0) fFreq2 = fFreq1;
+ pSmsFrameOut->pFSinFreq[i] =
+ fFreq1 + fInterpFactor * (fFreq2 - fFreq1);
+ pSmsFrameOut->pFSinAmp[i] =
+ pSmsFrame1->pFSinAmp[i] + fInterpFactor *
+ (pSmsFrame2->pFSinAmp[i] - pSmsFrame1->pFSinAmp[i]);
+ }
+
+ /* interpolate the stochastic part. The pointer is non-null when the frame contains
+ stochastic coefficients */
+ if (pSmsFrameOut->pFStocGain)
+ {
+ *(pSmsFrameOut->pFStocGain) =
+ *(pSmsFrame1->pFStocGain) + fInterpFactor *
+ (*(pSmsFrame2->pFStocGain) - *(pSmsFrame1->pFStocGain));
+ }
+ /*! \todo how to interpolate residual phase spectrum */
+ for (i = 0; i < pSmsFrame1->nCoeff; i++)
+ pSmsFrameOut->pFStocCoeff[i] =
+ pSmsFrame1->pFStocCoeff[i] + fInterpFactor *
+ (pSmsFrame2->pFStocCoeff[i] - pSmsFrame1->pFStocCoeff[i]);
+
+ /* DO NEXT: interpolate spec env here if fbins */
+ for (i = 0; i < pSmsFrame1->nEnvCoeff; i++)
+ pSmsFrameOut->pSpecEnv[i] =
+ pSmsFrame1->pSpecEnv[i] + fInterpFactor *
+ (pSmsFrame2->pSpecEnv[i] - pSmsFrame1->pSpecEnv[i]);
+
+
+}