summaryrefslogtreecommitdiff
path: root/sms
diff options
context:
space:
mode:
authorJohn Glover <glover.john@gmail.com>2010-11-08 22:19:11 +0000
committerJohn Glover <glover.john@gmail.com>2010-11-08 22:19:11 +0000
commit8f932f334f9de58106814822d6a7371acbc9dee1 (patch)
treef454895658c4a9e7c6aeb01c61469511509686f1 /sms
parentc741f6ce7bb43b115d08e190b93e0ce090ae3475 (diff)
downloadsimpl-8f932f334f9de58106814822d6a7371acbc9dee1.tar.gz
simpl-8f932f334f9de58106814822d6a7371acbc9dee1.tar.bz2
simpl-8f932f334f9de58106814822d6a7371acbc9dee1.zip
SMS partial tracking bug fixes
Diffstat (limited to 'sms')
-rw-r--r--sms/peakContinuation.c830
1 files changed, 409 insertions, 421 deletions
diff --git a/sms/peakContinuation.c b/sms/peakContinuation.c
index 7af4388..571772f 100644
--- a/sms/peakContinuation.c
+++ b/sms/peakContinuation.c
@@ -30,104 +30,100 @@
#define GUIDE_ACTIVE 0
#define MAX_CONT_CANDIDATES 5 /*!< maximum number of peak continuation
- candidates */
+ candidates */
/*! \brief function to get the next closest peak from a guide
*
- * \param fGuideFreq guide's frequency
- * \param pFFreqDistance distance of last best peak from guide
- * \param pSpectralPeaks array of peaks
- * \param pAnalParams analysis parameters
- * \param fFreqDev maximum deviation from guide
+ * \param fGuideFreq guide's frequency
+ * \param pFFreqDistance distance of last best peak from guide
+ * \param pSpectralPeaks array of peaks
+ * \param pAnalParams analysis parameters
+ * \param fFreqDev maximum deviation from guide
* \return peak number or -1 if nothing is good
*/
-int GetNextClosestPeak (sfloat fGuideFreq, sfloat *pFFreqDistance,
- SMS_Peak *pSpectralPeaks, SMS_AnalParams *pAnalParams,
- sfloat fFreqDev)
+int GetNextClosestPeak(sfloat fGuideFreq, sfloat *pFFreqDistance,
+ SMS_Peak *pSpectralPeaks, SMS_AnalParams *pAnalParams,
+ sfloat fFreqDev)
{
- int iInitialPeak =
- SMS_MAX_NPEAKS * fGuideFreq / (pAnalParams->iSamplingRate * .5),
- iLowPeak, iHighPeak, iChosenPeak = -1;
- sfloat fLowDistance, fHighDistance, fFreq;
-
- if (pSpectralPeaks[iInitialPeak].fFreq <= 0)
- iInitialPeak = 0;
-
- /* find a low peak to start */
-// if(fGuideFreq > 825 && fGuideFreq < 827)
-// {
-// printf("%d\t%f\t%f\n", SMS_MAX_NPEAKS, *pFFreqDistance,fGuideFreq - pSpectralPeaks[iInitialPeak].fFreq);
-// }
- fLowDistance = fGuideFreq - pSpectralPeaks[iInitialPeak].fFreq;
- if (floor(fLowDistance) < floor(*pFFreqDistance))
- {
- while (floor(fLowDistance) <= floor(*pFFreqDistance) &&
- iInitialPeak > 0)
- {
- iInitialPeak--;
- fLowDistance = fGuideFreq - pSpectralPeaks[iInitialPeak].fFreq;
- }
- }
- else
- {
- while (floor(fLowDistance) >= floor(*pFFreqDistance) &&
- iInitialPeak < SMS_MAX_NPEAKS)
- {
- iInitialPeak++;
-// if(fGuideFreq > 825 && fGuideFreq < 827)
-// {
-// printf("%d, %f\n", iInitialPeak, pSpectralPeaks[iInitialPeak].fFreq);
-// }
- if ((fFreq = pSpectralPeaks[iInitialPeak].fFreq) == 0)
- return -1;
-
- fLowDistance = fGuideFreq - fFreq;
- }
- iInitialPeak--;
- fLowDistance = fGuideFreq - pSpectralPeaks[iInitialPeak].fFreq;
- }
-
- if (floor(fLowDistance) <= floor(*pFFreqDistance) ||
- fLowDistance > fFreqDev)
- iLowPeak = -1;
- else
- iLowPeak = iInitialPeak;
-
- /* find a high peak to finish */
- iHighPeak = iInitialPeak;
- fHighDistance = fGuideFreq - pSpectralPeaks[iHighPeak].fFreq;
- while (floor(fHighDistance) >= floor(-*pFFreqDistance) &&
- iHighPeak < SMS_MAX_NPEAKS)
- {
- iHighPeak++;
- if ((fFreq = pSpectralPeaks[iHighPeak].fFreq) == 0)
- {
- iHighPeak = -1;
- break;
- }
- fHighDistance = fGuideFreq - fFreq;
- }
- if (fHighDistance > 0 || fabs(fHighDistance) > fFreqDev ||
- floor(fabs(fHighDistance)) <= floor(*pFFreqDistance))
- iHighPeak = -1;
-
- /* chose between the two extrema */
- if (iHighPeak >= 0 && iLowPeak >= 0)
- {
- if (fabs(fHighDistance) > fLowDistance)
- iChosenPeak = iLowPeak;
- else
- iChosenPeak = iHighPeak;
- }
- else if (iHighPeak < 0 && iLowPeak >= 0)
- iChosenPeak = iLowPeak;
- else if (iHighPeak >= 0 && iLowPeak < 0)
- iChosenPeak = iHighPeak;
- else
- return (-1);
-
- *pFFreqDistance = fabs (fGuideFreq - pSpectralPeaks[iChosenPeak].fFreq);
- return (iChosenPeak);
-}
+ int iInitialPeak = SMS_MAX_NPEAKS * fGuideFreq / (pAnalParams->iSamplingRate * .5);
+ int iLowPeak, iHighPeak, iChosenPeak = -1;
+ sfloat fLowDistance, fHighDistance, fFreq;
+
+ if(pSpectralPeaks[iInitialPeak].fFreq <= 0)
+ iInitialPeak = 0;
+
+ if(iInitialPeak >= pAnalParams->maxPeaks)
+ iInitialPeak = 0;
+
+ fLowDistance = fGuideFreq - pSpectralPeaks[iInitialPeak].fFreq;
+ if(floor(fLowDistance) < floor(*pFFreqDistance))
+ {
+ while(floor(fLowDistance) <= floor(*pFFreqDistance) &&
+ iInitialPeak > 0)
+ {
+ iInitialPeak--;
+ fLowDistance = fGuideFreq - pSpectralPeaks[iInitialPeak].fFreq;
+ }
+ }
+ else
+ {
+ while(floor(fLowDistance) >= floor(*pFFreqDistance) &&
+ iInitialPeak < pAnalParams->maxPeaks)
+ {
+ iInitialPeak++;
+ /* TODO: is this really the correct behaviour? Will ignore
+ * iInitialPeak values of maxPeaks */
+ if((fFreq = pSpectralPeaks[iInitialPeak].fFreq) == 0 ||
+ (iInitialPeak == pAnalParams->maxPeaks))
+ return -1;
+
+ fLowDistance = fGuideFreq - fFreq;
+ }
+ iInitialPeak--;
+ fLowDistance = fGuideFreq - pSpectralPeaks[iInitialPeak].fFreq;
+ }
+
+ if(floor(fLowDistance) <= floor(*pFFreqDistance) ||
+ fLowDistance > fFreqDev)
+ iLowPeak = -1;
+ else
+ iLowPeak = iInitialPeak;
+
+ /* find a high peak to finish */
+ iHighPeak = iInitialPeak;
+ fHighDistance = fGuideFreq - pSpectralPeaks[iHighPeak].fFreq;
+ while(floor(fHighDistance) >= floor(-*pFFreqDistance) &&
+ iHighPeak < pAnalParams->maxPeaks)
+ {
+ iHighPeak++;
+ if((fFreq = pSpectralPeaks[iHighPeak].fFreq) == 0)
+ {
+ iHighPeak = -1;
+ break;
+ }
+ fHighDistance = fGuideFreq - fFreq;
+ }
+ if(fHighDistance > 0 || fabs(fHighDistance) > fFreqDev ||
+ floor(fabs(fHighDistance)) <= floor(*pFFreqDistance))
+ iHighPeak = -1;
+
+ /* chose between the two extrema */
+ if (iHighPeak >= 0 && iLowPeak >= 0)
+ {
+ if (fabs(fHighDistance) > fLowDistance)
+ iChosenPeak = iLowPeak;
+ else
+ iChosenPeak = iHighPeak;
+ }
+ else if (iHighPeak < 0 && iLowPeak >= 0)
+ iChosenPeak = iLowPeak;
+ else if (iHighPeak >= 0 && iLowPeak < 0)
+ iChosenPeak = iHighPeak;
+ else
+ return -1;
+
+ *pFFreqDistance = fabs (fGuideFreq - pSpectralPeaks[iChosenPeak].fFreq);
+ return iChosenPeak;
+}
/*! \brief choose the best candidate out of all
*
@@ -136,378 +132,370 @@ int GetNextClosestPeak (sfloat fGuideFreq, sfloat *pFFreqDistance,
* \param fFreqDev maximum frequency deviation allowed
* \return the peak number of the best candidate
*/
-static int ChooseBestCand (SMS_ContCandidate *pCandidate, int nCandidates,
- sfloat fFreqDev)
+static int ChooseBestCand(SMS_ContCandidate *pCandidate, int nCandidates, sfloat fFreqDev)
{
- int i, iHighestCand, iClosestCand, iBestCand = 0;
- sfloat fMaxMag, fClosestFreq;
-
- /* intial guess */
- iClosestCand = 0;
- fClosestFreq = pCandidate[iClosestCand].fFreqDev;
- iHighestCand = 0;
- fMaxMag = pCandidate[iHighestCand].fMagDev;
-
- /* get the best candidate */
- for (i = 1; i < nCandidates; i++)
- {
- /* look for the one with highest magnitude */
- if (pCandidate[i].fMagDev > fMaxMag)
- {
- fMaxMag = pCandidate[i].fMagDev;
- iHighestCand = i;
- }
- /* look for the closest one to the guide */
- if (pCandidate[i].fFreqDev < fClosestFreq)
- {
- fClosestFreq = pCandidate[i].fFreqDev;
- iClosestCand = i;
- }
- }
- iBestCand = iHighestCand;
-
- /* reconcile the two results */
- if (iBestCand != iClosestCand &&
- fabs(pCandidate[iHighestCand].fFreqDev - fClosestFreq) > fFreqDev / 2)
- iBestCand = iClosestCand;
-
- return(pCandidate[iBestCand].iPeak);
+ int i, iHighestCand, iClosestCand, iBestCand = 0;
+ sfloat fMaxMag, fClosestFreq;
+
+ /* intial guess */
+ iClosestCand = 0;
+ fClosestFreq = pCandidate[iClosestCand].fFreqDev;
+ iHighestCand = 0;
+ fMaxMag = pCandidate[iHighestCand].fMagDev;
+
+ /* get the best candidate */
+ for (i = 1; i < nCandidates; i++)
+ {
+ /* look for the one with highest magnitude */
+ if (pCandidate[i].fMagDev > fMaxMag)
+ {
+ fMaxMag = pCandidate[i].fMagDev;
+ iHighestCand = i;
+ }
+ /* look for the closest one to the guide */
+ if (pCandidate[i].fFreqDev < fClosestFreq)
+ {
+ fClosestFreq = pCandidate[i].fFreqDev;
+ iClosestCand = i;
+ }
+ }
+ iBestCand = iHighestCand;
+
+ /* reconcile the two results */
+ if(iBestCand != iClosestCand &&
+ fabs(pCandidate[iHighestCand].fFreqDev - fClosestFreq) > fFreqDev / 2)
+ iBestCand = iClosestCand;
+
+ return pCandidate[iBestCand].iPeak;
}
/*! \brief check for one guide that has choosen iBestPeak
*
- * \param iBestPeak choosen peak for a guide
- * \param pGuides array of guides
- * \param nGuides total number of guides
+ * \param iBestPeak choosen peak for a guide
+ * \param pGuides array of guides
+ * \param nGuides total number of guides
* \return number of guide that chose the peak, or -1 if none
*/
-static int CheckForConflict (int iBestPeak, SMS_Guide *pGuides, int nGuides)
+static int CheckForConflict(int iBestPeak, SMS_Guide *pGuides, int nGuides)
{
- int iGuide;
-
- for (iGuide = 0; iGuide < nGuides; iGuide++)
- if (pGuides[iGuide].iPeakChosen == iBestPeak)
- return iGuide;
-
- return -1;
+ int iGuide;
+
+ for (iGuide = 0; iGuide < nGuides; iGuide++)
+ if (pGuides[iGuide].iPeakChosen == iBestPeak)
+ return iGuide;
+
+ return -1;
}
/*! \brief chose the best of the two guides for the conflicting peak
*
- * \param iConflictingGuide conflicting guide number
- * \param iGuide guide number
- * \param pGuides array of guides
- * \param pSpectralPeaks array of peaks
+ * \param iConflictingGuide conflicting guide number
+ * \param iGuide guide number
+ * \param pGuides array of guides
+ * \param pSpectralPeaks array of peaks
* \return number of guide
*/
-static int BestGuide (int iConflictingGuide, int iGuide, SMS_Guide *pGuides,
- SMS_Peak *pSpectralPeaks)
+static int BestGuide(int iConflictingGuide, int iGuide, SMS_Guide *pGuides,
+ SMS_Peak *pSpectralPeaks)
{
- int iConflictingPeak = pGuides[iConflictingGuide].iPeakChosen;
- sfloat fGuideDistance = fabs (pSpectralPeaks[iConflictingPeak].fFreq -
- pGuides[iGuide].fFreq);
- sfloat fConfGuideDistance = fabs (pSpectralPeaks[iConflictingPeak].fFreq -
- pGuides[iConflictingGuide].fFreq);
-
- if (fGuideDistance > fConfGuideDistance)
- return (iConflictingGuide);
- else
- return (iGuide);
+ int iConflictingPeak = pGuides[iConflictingGuide].iPeakChosen;
+ sfloat fGuideDistance = fabs(pSpectralPeaks[iConflictingPeak].fFreq -
+ pGuides[iGuide].fFreq);
+ sfloat fConfGuideDistance = fabs(pSpectralPeaks[iConflictingPeak].fFreq -
+ pGuides[iConflictingGuide].fFreq);
+
+ if(fGuideDistance > fConfGuideDistance)
+ return iConflictingGuide;
+ else
+ return iGuide;
}
/*! \brief function to find the best continuation peak for a given guide
- * \param pGuides guide attributes
- * \param iGuide number of guide
- * \param pSpectralPeaks peak values at the current frame
- * \param pAnalParams analysis parameters
+ * \param pGuides guide attributes
+ * \param iGuide number of guide
+ * \param pSpectralPeaks peak values at the current frame
+ * \param pAnalParams analysis parameters
* \param fFreqDev frequency deviation allowed
* \return the peak number
*/
-int GetBestPeak (SMS_Guide *pGuides, int iGuide, SMS_Peak *pSpectralPeaks,
- SMS_AnalParams *pAnalParams, sfloat fFreqDev)
+int GetBestPeak(SMS_Guide *pGuides, int iGuide, SMS_Peak *pSpectralPeaks,
+ SMS_AnalParams *pAnalParams, sfloat fFreqDev)
{
- int iCand = 0, iPeak, iBestPeak, iConflictingGuide, iWinnerGuide;
- sfloat fGuideFreq = pGuides[iGuide].fFreq,
- fGuideMag = pGuides[iGuide].fMag,
- fMagDistance = 0;
- sfloat fFreqDistance = -1;
- SMS_ContCandidate pCandidate[MAX_CONT_CANDIDATES];
-
- /* find all possible candidates */
- while (iCand < MAX_CONT_CANDIDATES)
- {
- /* find the next best peak */
- if ((iPeak = GetNextClosestPeak (fGuideFreq, &fFreqDistance,
- pSpectralPeaks, pAnalParams,
- fFreqDev)) < 0)
- break;
-
- /* if the peak's magnitude is not too small accept it as */
- /* possible candidate */
- if ((fMagDistance = pSpectralPeaks[iPeak].fMag - fGuideMag) > -20.0)
- {
- pCandidate[iCand].fFreqDev = fabs(fFreqDistance);
- pCandidate[iCand].fMagDev = fMagDistance;
- pCandidate[iCand].iPeak = iPeak;
-
- if(pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
- pAnalParams->iDebugMode == SMS_DBG_ALL)
- fprintf (stdout, "candidate %d: freq %f mag %f\n",
- iCand, pSpectralPeaks[iPeak].fFreq,
- pSpectralPeaks[iPeak].fMag);
- iCand++;
- }
- }
- /* get best candidate */
- if (iCand < 1)
- return (0);
- else if (iCand == 1)
- iBestPeak = pCandidate[0].iPeak;
- else
- iBestPeak = ChooseBestCand (pCandidate, iCand,
- pAnalParams->fFreqDeviation);
-
- if(pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
- pAnalParams->iDebugMode == SMS_DBG_ALL)
- fprintf (stdout, "BestCandidate: freq %f\n",
- pSpectralPeaks[iBestPeak].fFreq);
-
- /* if peak taken by another guide resolve conflict */
- if ((iConflictingGuide = CheckForConflict (iBestPeak, pGuides,
- pAnalParams->nGuides)) >= 0)
- {
- iWinnerGuide = BestGuide (iConflictingGuide, iGuide, pGuides,
- pSpectralPeaks);
- if(pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
- pAnalParams->iDebugMode == SMS_DBG_ALL)
- fprintf (stdout,
- "Conflict: guide: %d (%f), and guide: %d (%f). best: %d\n",
- iGuide, pGuides[iGuide].fFreq,
- iConflictingGuide, pGuides[iConflictingGuide].fFreq,
- iWinnerGuide);
-
- if (iGuide == iWinnerGuide)
- {
- pGuides[iGuide].iPeakChosen = iBestPeak;
- pGuides[iConflictingGuide].iPeakChosen = -1;
- }
- }
- else
- pGuides[iGuide].iPeakChosen = iBestPeak;
-
- return (iBestPeak);
+ int iCand = 0, iPeak, iBestPeak, iConflictingGuide, iWinnerGuide;
+ sfloat fGuideFreq = pGuides[iGuide].fFreq,
+ fGuideMag = pGuides[iGuide].fMag,
+ fMagDistance = 0;
+ sfloat fFreqDistance = -1;
+ SMS_ContCandidate pCandidate[MAX_CONT_CANDIDATES];
+
+ /* find all possible candidates */
+ while (iCand < MAX_CONT_CANDIDATES)
+ {
+ /* find the next best peak */
+ if((iPeak = GetNextClosestPeak(fGuideFreq, &fFreqDistance,
+ pSpectralPeaks, pAnalParams, fFreqDev)) < 0)
+ break;
+
+ /* if the peak's magnitude is not too small accept it as */
+ /* possible candidate */
+ if ((fMagDistance = pSpectralPeaks[iPeak].fMag - fGuideMag) > -20.0)
+ {
+ pCandidate[iCand].fFreqDev = fabs(fFreqDistance);
+ pCandidate[iCand].fMagDev = fMagDistance;
+ pCandidate[iCand].iPeak = iPeak;
+
+ if(pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
+ pAnalParams->iDebugMode == SMS_DBG_ALL)
+ fprintf(stdout, "candidate %d: freq %f mag %f\n",
+ iCand, pSpectralPeaks[iPeak].fFreq,
+ pSpectralPeaks[iPeak].fMag);
+ iCand++;
+ }
+ }
+ /* get best candidate */
+ if(iCand < 1)
+ return 0;
+ else if (iCand == 1)
+ iBestPeak = pCandidate[0].iPeak;
+ else
+ iBestPeak = ChooseBestCand (pCandidate, iCand,
+ pAnalParams->fFreqDeviation);
+
+ if(pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
+ pAnalParams->iDebugMode == SMS_DBG_ALL)
+ fprintf (stdout, "BestCandidate: freq %f\n",
+ pSpectralPeaks[iBestPeak].fFreq);
+
+ /* if peak taken by another guide resolve conflict */
+ if ((iConflictingGuide = CheckForConflict (iBestPeak, pGuides,
+ pAnalParams->nGuides)) >= 0)
+ {
+ iWinnerGuide = BestGuide (iConflictingGuide, iGuide, pGuides,
+ pSpectralPeaks);
+ if(pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
+ pAnalParams->iDebugMode == SMS_DBG_ALL)
+ fprintf (stdout,
+ "Conflict: guide: %d (%f), and guide: %d (%f). best: %d\n",
+ iGuide, pGuides[iGuide].fFreq,
+ iConflictingGuide, pGuides[iConflictingGuide].fFreq,
+ iWinnerGuide);
+
+ if (iGuide == iWinnerGuide)
+ {
+ pGuides[iGuide].iPeakChosen = iBestPeak;
+ pGuides[iConflictingGuide].iPeakChosen = -1;
+ }
+ }
+ else
+ pGuides[iGuide].iPeakChosen = iBestPeak;
+
+ return iBestPeak;
}
/*! \brief function to get the next maximum (magnitude) peak
- * \param pSpectralPeaks array of peaks
- * \param pFCurrentMax last peak maximum
+ * \param pSpectralPeaks array of peaks
+ * \param pFCurrentMax last peak maximum
* \return the number of the maximum peak
*/
static int GetNextMax (SMS_Peak *pSpectralPeaks, sfloat *pFCurrentMax)
{
- sfloat fPeakMag;
- sfloat fMaxMag = 0.;
- int iPeak, iMaxPeak = -1;
-
- for (iPeak = 0; iPeak < SMS_MAX_NPEAKS; iPeak++)
- {
- fPeakMag = pSpectralPeaks[iPeak].fMag;
-
- if (fPeakMag == 0)
- break;
-
- if (fPeakMag > fMaxMag && fPeakMag < *pFCurrentMax)
- {
- iMaxPeak = iPeak;
- fMaxMag = fPeakMag;
- }
- }
- *pFCurrentMax = fMaxMag;
- return (iMaxPeak);
+ sfloat fPeakMag;
+ sfloat fMaxMag = 0.;
+ int iPeak, iMaxPeak = -1;
+
+ for (iPeak = 0; iPeak < SMS_MAX_NPEAKS; iPeak++)
+ {
+ fPeakMag = pSpectralPeaks[iPeak].fMag;
+
+ if (fPeakMag == 0)
+ break;
+
+ if (fPeakMag > fMaxMag && fPeakMag < *pFCurrentMax)
+ {
+ iMaxPeak = iPeak;
+ fMaxMag = fPeakMag;
+ }
+ }
+ *pFCurrentMax = fMaxMag;
+ return (iMaxPeak);
}
/*! \brief function to get a good starting peak for a track
*
- * \param iGuide current guide
- * \param pGuides array of guides
- * \param nGuides total number of guides
- * \param pSpectralPeaks array of peaks
- * \param pFCurrentMax current peak maximum
+ * \param iGuide current guide
+ * \param pGuides array of guides
+ * \param nGuides total number of guides
+ * \param pSpectralPeaks array of peaks
+ * \param pFCurrentMax current peak maximum
* \return \todo should this return something?
*/
-static int GetStartingPeak (int iGuide, SMS_Guide *pGuides, int nGuides,
- SMS_Peak *pSpectralPeaks, sfloat *pFCurrentMax)
+static int GetStartingPeak(int iGuide, SMS_Guide *pGuides, int nGuides,
+ SMS_Peak *pSpectralPeaks, sfloat *pFCurrentMax)
{
- int iPeak = -1;
- short peakNotFound = 1;
-
- while (peakNotFound == 1 && *pFCurrentMax > 0)
- {
- /* \todo I don't think this ever returns -1, but check */
- if ((iPeak = GetNextMax (pSpectralPeaks, pFCurrentMax)) < 0)
- return (-1);
-
- if (CheckForConflict (iPeak, pGuides, nGuides) < 0)
- {
- pGuides[iGuide].iPeakChosen = iPeak;
- pGuides[iGuide].iStatus = GUIDE_BEG;
- pGuides[iGuide].fFreq = pSpectralPeaks[iPeak].fFreq;
- peakNotFound = 0;
- }
- }
- return (1);
+ int iPeak = -1;
+ short peakNotFound = 1;
+
+ while (peakNotFound == 1 && *pFCurrentMax > 0)
+ {
+ /* \todo I don't think this ever returns -1, but check */
+ if ((iPeak = GetNextMax(pSpectralPeaks, pFCurrentMax)) < 0)
+ return (-1);
+
+ if (CheckForConflict (iPeak, pGuides, nGuides) < 0)
+ {
+ pGuides[iGuide].iPeakChosen = iPeak;
+ pGuides[iGuide].iStatus = GUIDE_BEG;
+ pGuides[iGuide].fFreq = pSpectralPeaks[iPeak].fFreq;
+ peakNotFound = 0;
+ }
+ }
+ return (1);
}
/*! \brief function to advance the guides through the next frame
*
* the output is the frequency, magnitude, and phase tracks
*
- * \param iFrame current frame number
+ * \param iFrame current frame number
* \param pAnalParams analysis parameters
* \return error code \see SMS_ERRORS
*/
-int sms_peakContinuation (int iFrame, SMS_AnalParams *pAnalParams)
+int sms_peakContinuation(int iFrame, SMS_AnalParams *pAnalParams)
{
- int iGuide, iCurrentPeak = -1, iGoodPeak = -1;
- sfloat fFund = pAnalParams->ppFrames[iFrame]->fFundamental,
- fFreqDev = fFund * pAnalParams->fFreqDeviation, fCurrentMax = 1000;
- static SMS_Guide *pGuides = NULL;
- static int nGuides = 0;
-
- if (pGuides == NULL || nGuides != pAnalParams->nGuides || pAnalParams->resetGuides == 1)
- {
- if ((pGuides = (SMS_Guide *) calloc(pAnalParams->nGuides, sizeof(SMS_Guide)))
- == NULL)
- return (SMS_MALLOC);
- if (pAnalParams->iFormat == SMS_FORMAT_H ||
- pAnalParams->iFormat == SMS_FORMAT_HP)
- for (iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
- pGuides[iGuide].fFreq = pAnalParams->fDefaultFundamental
- * (iGuide + 1);
- nGuides = pAnalParams->nGuides;
- pAnalParams->resetGuides = 0;
- }
-
- /* update guides with fundamental contribution */
- if (fFund > 0 &&
- (pAnalParams->iFormat == SMS_FORMAT_H ||
- pAnalParams->iFormat == SMS_FORMAT_HP))
- for(iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
- pGuides[iGuide].fFreq =
- (1 - pAnalParams->fFundContToGuide) * pGuides[iGuide].fFreq +
- pAnalParams->fFundContToGuide * fFund * (iGuide + 1);
-
- if (pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
- pAnalParams->iDebugMode == SMS_DBG_ALL)
- fprintf (stdout,
- "Frame %d Peak Continuation: \n",
- pAnalParams->ppFrames[iFrame]->iFrameNum);
-
- /* continue all guides */
- for (iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
- {
- sfloat fPreviousFreq =
- pAnalParams->ppFrames[iFrame-1]->deterministic.pFSinFreq[iGuide];
-
- /* get the guide value by upgrading the previous guide */
- if (fPreviousFreq > 0)
- pGuides[iGuide].fFreq =
- (1 - pAnalParams->fPeakContToGuide) * pGuides[iGuide].fFreq +
- pAnalParams->fPeakContToGuide * fPreviousFreq;
-
- if (pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
- pAnalParams->iDebugMode == SMS_DBG_ALL)
- fprintf (stdout, "Guide %d: freq %f, mag %f\n",
- iGuide, pGuides[iGuide].fFreq, pGuides[iGuide].fMag);
-
- if (pGuides[iGuide].fFreq <= 0.0 ||
- pGuides[iGuide].fFreq > pAnalParams->fHighestFreq)
- {
- pGuides[iGuide].iStatus = GUIDE_DEAD;
- pGuides[iGuide].fFreq = 0;
- continue;
- }
-
- pGuides[iGuide].iPeakChosen = -1;
-
- if (pAnalParams->iFormat == SMS_FORMAT_IH ||
- pAnalParams->iFormat == SMS_FORMAT_IHP)
- fFreqDev = pGuides[iGuide].fFreq * pAnalParams->fFreqDeviation;
-
- /* get the best peak for the guide */
- iGoodPeak =
- GetBestPeak(pGuides, iGuide, pAnalParams->ppFrames[iFrame]->pSpectralPeaks,
- pAnalParams, fFreqDev);
-
-// printf("%f\t->\t%f (", fPreviousFreq, pAnalParams->ppFrames[iFrame]->pSpectralPeaks[iGoodPeak].fFreq);
-// int i;
-// for(i = 0; i < pAnalParams->ppFrames[iFrame]->nPeaks; i++)
-// {
-// printf("%f ", pAnalParams->ppFrames[iFrame]->pSpectralPeaks[i].fFreq);
-// }
-// printf(")\n");
- }
-
- /* try to find good peaks for the GUIDE_DEAD guides */
- if (pAnalParams->iFormat == SMS_FORMAT_IH ||
- pAnalParams->iFormat == SMS_FORMAT_IHP)
- for(iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
- {
- if (pGuides[iGuide].iStatus != GUIDE_DEAD)
- continue;
-
- if (GetStartingPeak (iGuide, pGuides, pAnalParams->nGuides,
- pAnalParams->ppFrames[iFrame]->pSpectralPeaks,
- &fCurrentMax) == -1)
- break;
- }
-
- /* save all the continuation values,
- * assume output tracks are already clear */
- for (iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
- {
- if (pGuides[iGuide].iStatus == GUIDE_DEAD)
- continue;
-
- if (pAnalParams->iFormat == SMS_FORMAT_IH ||
- pAnalParams->iFormat == SMS_FORMAT_IHP)
- {
- if (pGuides[iGuide].iStatus > 0 &&
- pGuides[iGuide].iPeakChosen == -1)
- {
- if(pGuides[iGuide].iStatus++ > pAnalParams->iMaxSleepingTime)
- {
- pGuides[iGuide].iStatus = GUIDE_DEAD;
- pGuides[iGuide].fFreq = 0;
- pGuides[iGuide].fMag = 0;
- pGuides[iGuide].iPeakChosen = -1;
- }
- else
- pGuides[iGuide].iStatus++;
- continue;
- }
-
- if (pGuides[iGuide].iStatus == GUIDE_ACTIVE &&
- pGuides[iGuide].iPeakChosen == -1)
- {
- pGuides[iGuide].iStatus = 1;
- continue;
- }
- }
-
- /* if good continuation peak found, save it */
- if ((iCurrentPeak = pGuides[iGuide].iPeakChosen) >= 0)
- {
- pAnalParams->ppFrames[iFrame]->deterministic.pFSinFreq[iGuide] =
- pAnalParams->ppFrames[iFrame]->pSpectralPeaks[iCurrentPeak].fFreq;
- pAnalParams->ppFrames[iFrame]->deterministic.pFSinAmp[iGuide] =
- pAnalParams->ppFrames[iFrame]->pSpectralPeaks[iCurrentPeak].fMag;
- pAnalParams->ppFrames[iFrame]->deterministic.pFSinPha[iGuide] =
- pAnalParams->ppFrames[iFrame]->pSpectralPeaks[iCurrentPeak].fPhase;
-
- pGuides[iGuide].iStatus = GUIDE_ACTIVE;
- pGuides[iGuide].iPeakChosen = -1;
- }
- }
- return(SMS_OK);
+ int iGuide, iCurrentPeak = -1, iGoodPeak = -1;
+ sfloat fFund = pAnalParams->ppFrames[iFrame]->fFundamental;
+ sfloat fFreqDev = fFund * pAnalParams->fFreqDeviation, fCurrentMax = 1000;
+ static SMS_Guide *pGuides = NULL;
+ static int nGuides = 0;
+
+ if(pGuides == NULL || nGuides != pAnalParams->nGuides || pAnalParams->resetGuides == 1)
+ {
+ if((pGuides = (SMS_Guide *) calloc(pAnalParams->nGuides, sizeof(SMS_Guide)))
+ == NULL)
+ return SMS_MALLOC;
+ if (pAnalParams->iFormat == SMS_FORMAT_H ||
+ pAnalParams->iFormat == SMS_FORMAT_HP)
+ for (iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
+ pGuides[iGuide].fFreq = pAnalParams->fDefaultFundamental * (iGuide + 1);
+ nGuides = pAnalParams->nGuides;
+ pAnalParams->resetGuides = 0;
+ }
+
+ /* update guides with fundamental contribution */
+ if(fFund > 0 &&
+ (pAnalParams->iFormat == SMS_FORMAT_H ||
+ pAnalParams->iFormat == SMS_FORMAT_HP))
+ for(iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
+ pGuides[iGuide].fFreq =
+ (1 - pAnalParams->fFundContToGuide) * pGuides[iGuide].fFreq +
+ pAnalParams->fFundContToGuide * fFund * (iGuide + 1);
+
+ if (pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
+ pAnalParams->iDebugMode == SMS_DBG_ALL)
+ fprintf(stdout,
+ "Frame %d Peak Continuation: \n",
+ pAnalParams->ppFrames[iFrame]->iFrameNum);
+
+ /* continue all guides */
+ for(iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
+ {
+ sfloat fPreviousFreq =
+ pAnalParams->ppFrames[iFrame-1]->deterministic.pFSinFreq[iGuide];
+
+ /* get the guide value by upgrading the previous guide */
+ if(fPreviousFreq > 0)
+ pGuides[iGuide].fFreq =
+ (1 - pAnalParams->fPeakContToGuide) * pGuides[iGuide].fFreq +
+ pAnalParams->fPeakContToGuide * fPreviousFreq;
+
+ if(pAnalParams->iDebugMode == SMS_DBG_PEAK_CONT ||
+ pAnalParams->iDebugMode == SMS_DBG_ALL)
+ fprintf(stdout, "Guide %d: freq %f, mag %f\n",
+ iGuide, pGuides[iGuide].fFreq, pGuides[iGuide].fMag);
+
+ if(pGuides[iGuide].fFreq <= 0.0 ||
+ pGuides[iGuide].fFreq > pAnalParams->fHighestFreq)
+ {
+ pGuides[iGuide].iStatus = GUIDE_DEAD;
+ pGuides[iGuide].fFreq = 0;
+ continue;
+ }
+
+ pGuides[iGuide].iPeakChosen = -1;
+
+ if(pAnalParams->iFormat == SMS_FORMAT_IH ||
+ pAnalParams->iFormat == SMS_FORMAT_IHP)
+ fFreqDev = pGuides[iGuide].fFreq * pAnalParams->fFreqDeviation;
+
+ /* get the best peak for the guide */
+ GetBestPeak(pGuides, iGuide, pAnalParams->ppFrames[iFrame]->pSpectralPeaks,
+ pAnalParams, fFreqDev);
+ }
+
+ /* try to find good peaks for the GUIDE_DEAD guides */
+ if(pAnalParams->iFormat == SMS_FORMAT_IH ||
+ pAnalParams->iFormat == SMS_FORMAT_IHP)
+ {
+ for(iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
+ {
+ if(pGuides[iGuide].iStatus != GUIDE_DEAD)
+ continue;
+
+ /* TODO: make this function use pAnalParams->maxPeaks instead
+ * of SMS_MAX_NPEAKS */
+ if(GetStartingPeak (iGuide, pGuides, pAnalParams->nGuides,
+ pAnalParams->ppFrames[iFrame]->pSpectralPeaks,
+ &fCurrentMax) == -1)
+ break;
+ }
+ }
+
+ /* save all the continuation values,
+ * assume output tracks are already clear */
+ for(iGuide = 0; iGuide < pAnalParams->nGuides; iGuide++)
+ {
+ if(pGuides[iGuide].iStatus == GUIDE_DEAD)
+ continue;
+
+ if(pAnalParams->iFormat == SMS_FORMAT_IH ||
+ pAnalParams->iFormat == SMS_FORMAT_IHP)
+ {
+ if(pGuides[iGuide].iStatus > 0 &&
+ pGuides[iGuide].iPeakChosen == -1)
+ {
+ if(pGuides[iGuide].iStatus++ > pAnalParams->iMaxSleepingTime)
+ {
+ pGuides[iGuide].iStatus = GUIDE_DEAD;
+ pGuides[iGuide].fFreq = 0;
+ pGuides[iGuide].fMag = 0;
+ pGuides[iGuide].iPeakChosen = -1;
+ }
+ else
+ pGuides[iGuide].iStatus++;
+ continue;
+ }
+
+ if(pGuides[iGuide].iStatus == GUIDE_ACTIVE &&
+ pGuides[iGuide].iPeakChosen == -1)
+ {
+ pGuides[iGuide].iStatus = 1;
+ continue;
+ }
+ }
+
+ /* if good continuation peak found, save it */
+ if((iCurrentPeak = pGuides[iGuide].iPeakChosen) >= 0)
+ {
+ pAnalParams->ppFrames[iFrame]->deterministic.pFSinFreq[iGuide] =
+ pAnalParams->ppFrames[iFrame]->pSpectralPeaks[iCurrentPeak].fFreq;
+ pAnalParams->ppFrames[iFrame]->deterministic.pFSinAmp[iGuide] =
+ pAnalParams->ppFrames[iFrame]->pSpectralPeaks[iCurrentPeak].fMag;
+ pAnalParams->ppFrames[iFrame]->deterministic.pFSinPha[iGuide] =
+ pAnalParams->ppFrames[iFrame]->pSpectralPeaks[iCurrentPeak].fPhase;
+
+ pGuides[iGuide].iStatus = GUIDE_ACTIVE;
+ pGuides[iGuide].iPeakChosen = -1;
+ }
+ }
+ return SMS_OK;
}