mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1210 lines
35 KiB
1210 lines
35 KiB
/******************************************************************************
|
|
* PitchProsody.cpp *
|
|
*--------------------*
|
|
*
|
|
* This is an implementation of the PitchProsody class.
|
|
*------------------------------------------------------------------------------
|
|
* Copyright (C) 1999 Microsoft Corporation Date: 04/28/99
|
|
* All Rights Reserved
|
|
*
|
|
*********************************************************************** MC ****/
|
|
|
|
//--- Additional includes
|
|
#include "stdafx.h"
|
|
|
|
#ifndef SPDebug_h
|
|
#include <spdebug.h>
|
|
#endif
|
|
#ifndef AlloOps_H
|
|
#include "AlloOps.h"
|
|
#endif
|
|
#ifndef Frontend_H
|
|
#include "Frontend.h"
|
|
#endif
|
|
|
|
|
|
//-----------------------------
|
|
// Data.cpp
|
|
//-----------------------------
|
|
extern const float g_PitchScale[];
|
|
|
|
|
|
|
|
//--------------------------------
|
|
// Interpolation direction
|
|
//--------------------------------
|
|
enum INTERP_DIR
|
|
{
|
|
GOING_UP,
|
|
GOING_DOWN,
|
|
};
|
|
|
|
|
|
#define M_PI 3.14159265358979323846
|
|
#define MAX_ORDER 4096
|
|
|
|
|
|
class CFIRFilter
|
|
{
|
|
public:
|
|
//-------------------------------------------------
|
|
// Methods
|
|
//-------------------------------------------------
|
|
void Make_Filter( float freq1, long order, float srate );
|
|
void DoFilter (float *src, float *dst, long len);
|
|
|
|
void Design_Lowpass (float freq);
|
|
//-------------------------------------------------
|
|
// Member Variables
|
|
//-------------------------------------------------
|
|
float m_Coeff[MAX_ORDER];
|
|
float m_SRate;
|
|
long m_Order;
|
|
long m_HalfOrder;
|
|
};
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Design a low pass filter
|
|
//-----------------------------------------------
|
|
void CFIRFilter::Design_Lowpass (float freq)
|
|
{
|
|
float half_Filt[MAX_ORDER];
|
|
double w;
|
|
long i;
|
|
|
|
|
|
//----------------------------------------------------------
|
|
// generate one half of coefficients from sinc() function
|
|
//----------------------------------------------------------
|
|
w = 2.0 * M_PI * freq;
|
|
half_Filt[0] = (float)(w / M_PI);
|
|
for (i = 1; i <= m_HalfOrder; i++)
|
|
{
|
|
half_Filt[i] = (float)(sin(i * w) / (i * M_PI));
|
|
}
|
|
|
|
//----------------------------------------------------------
|
|
// window with (half-)hamming window
|
|
//----------------------------------------------------------
|
|
for (i = 0; i <= m_HalfOrder; i++)
|
|
{
|
|
half_Filt[i] *= (float)(0.54 + 0.46 * cos(i * M_PI / m_HalfOrder));
|
|
}
|
|
|
|
//----------------------------------------------------------
|
|
// copy as symmetric filter
|
|
//----------------------------------------------------------
|
|
for (i = 0; i < m_Order; i++)
|
|
{
|
|
m_Coeff[i] = half_Filt[abs(m_HalfOrder - i)];
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Do the filtering
|
|
//-----------------------------------------------
|
|
void CFIRFilter::DoFilter (float *src, float *dst, long len)
|
|
{
|
|
#define kWakeSamples 1000;
|
|
long i,j;
|
|
float *sp;
|
|
float sum;
|
|
long lenAdj;
|
|
|
|
lenAdj = len - m_Order;
|
|
if (lenAdj >= 0)
|
|
{
|
|
for (i = 0; i < m_HalfOrder; i++)
|
|
{
|
|
*dst++ = src[0];
|
|
}
|
|
|
|
for (i = 0; i < lenAdj; i++)
|
|
{
|
|
sum = (float)0.0;
|
|
sp = src + i;
|
|
for (j = 0; j < m_Order; j++)
|
|
{
|
|
sum += ((float)*sp++) * m_Coeff[j];
|
|
}
|
|
*dst++ = sum;
|
|
|
|
}
|
|
|
|
for (i = 0; i < m_HalfOrder; i++)
|
|
{
|
|
*dst++ = src[len-1];
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void CFIRFilter::Make_Filter( float freq1, long order, float srate )
|
|
{
|
|
m_SRate = srate;
|
|
m_Order = order;
|
|
m_HalfOrder = m_Order / 2;
|
|
|
|
Design_Lowpass (freq1 / m_SRate);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
* HzToOct *
|
|
*---------*
|
|
* Description:
|
|
* Convert liner freq ro exp pitch
|
|
* 0.69314718 is log of 2
|
|
* 1.021975 is offset for middle C
|
|
*
|
|
********************************************************************** MC ***/
|
|
float HzToOct( float cps)
|
|
{
|
|
SPDBG_FUNC( "HzToOct" );
|
|
|
|
return (float)(log(cps / 1.021975) / 0.69314718);
|
|
|
|
} /* HzToOct */
|
|
|
|
/*****************************************************************************
|
|
* OctToHz *
|
|
*---------*
|
|
* Description:
|
|
* Convert from exp pitch to linear freq
|
|
********************************************************************** MC ***/
|
|
float OctToHz( float oct)
|
|
{
|
|
SPDBG_FUNC( "OctToHz" );
|
|
|
|
return (float)(pow(2, oct) * 1.021975);
|
|
} /* OctToHz */
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
* CPitchProsody::DoPitchControl *
|
|
*-------------------------------*
|
|
* Description:
|
|
* Scale speech pitch to user control
|
|
*
|
|
********************************************************************** MC ***/
|
|
float CPitchProsody::DoPitchControl( long pitchControl, float basePitch )
|
|
{
|
|
SPDBG_FUNC( "CPitchProsody::DoPitchControl" );
|
|
float newPitch;
|
|
|
|
if( pitchControl < 0 )
|
|
{
|
|
//--------------------------------
|
|
// DECREASE the pitch
|
|
//--------------------------------
|
|
if( pitchControl < MIN_USER_PITCH )
|
|
{
|
|
pitchControl = MIN_USER_PITCH; // clip to min
|
|
}
|
|
newPitch = (float)basePitch / g_PitchScale[0 - pitchControl];
|
|
}
|
|
else
|
|
{
|
|
//--------------------------------
|
|
// INCREASE the pitch
|
|
//--------------------------------
|
|
if( pitchControl > MAX_USER_PITCH )
|
|
{
|
|
pitchControl = MAX_USER_PITCH; // clip to max
|
|
}
|
|
newPitch = (float)basePitch * g_PitchScale[pitchControl];
|
|
}
|
|
return newPitch;
|
|
} /* CPitchProsody::DoPitchControl */
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
* CPitchProsody::SetDefaultPitch *
|
|
*--------------------------------*
|
|
* Description:
|
|
* Init pitch knots to monotone in case there's a failure in this object.
|
|
*
|
|
********************************************************************** MC ***/
|
|
void CPitchProsody::SetDefaultPitch()
|
|
{
|
|
SPDBG_FUNC( "CPitchProsody::SetDefaultPitch" );
|
|
CAlloCell *pCurCell;
|
|
|
|
pCurCell = m_pAllos->GetHeadCell();
|
|
while( pCurCell )
|
|
{
|
|
float relTime, timeK;
|
|
float normalPitch;
|
|
long knot;
|
|
|
|
normalPitch = pCurCell->m_Pitch_LO + ((pCurCell->m_Pitch_HI - pCurCell->m_Pitch_LO) / 2);
|
|
timeK = pCurCell->m_ftDuration / KNOTS_PER_PHON;
|
|
relTime = 0;
|
|
for( knot = 0; knot < KNOTS_PER_PHON; knot++ )
|
|
{
|
|
pCurCell->m_ftPitch[knot] = normalPitch;
|
|
pCurCell->m_ftTime[knot] = relTime;
|
|
relTime += timeK;
|
|
}
|
|
pCurCell = m_pAllos->GetNextCell();
|
|
}
|
|
} /* CPitchProsody::SetDefaultPitch */
|
|
|
|
|
|
/*****************************************************************************
|
|
* CPitchProsody::AlloPitch *
|
|
*--------------------------*
|
|
* Description:
|
|
* Tag pitch highlights
|
|
*
|
|
********************************************************************** MC ***/
|
|
void CPitchProsody::AlloPitch( CAlloList *pAllos, float baseLine, float pitchRange )
|
|
{
|
|
SPDBG_FUNC( "CAlloOps::AlloPitch" );
|
|
CAlloCell *pCurCell;
|
|
bool skipInitialSil;
|
|
long quantTotal, index;
|
|
|
|
m_pAllos = pAllos;
|
|
m_numOfCells = m_pAllos->GetCount();
|
|
m_Tune_Style = DESCEND_TUNE; // NOTE: maybe set from rules
|
|
m_TotalDur = 0;
|
|
quantTotal = 0;
|
|
m_OffsTime = 0;
|
|
skipInitialSil = true;
|
|
|
|
|
|
//------------------------------
|
|
// Calculate total duration
|
|
// (exclude surrounding silence)
|
|
//------------------------------
|
|
index = 0;
|
|
pCurCell = m_pAllos->GetHeadCell();
|
|
while( pCurCell )
|
|
{
|
|
if( (skipInitialSil) && (pCurCell->m_allo == _SIL_) )
|
|
{
|
|
//---------------------------------
|
|
// Skip leading silence
|
|
//---------------------------------
|
|
m_OffsTime += pCurCell->m_ftDuration;
|
|
}
|
|
else if( (index == (m_numOfCells -1)) && (pCurCell->m_allo == _SIL_) )
|
|
{
|
|
//---------------------------------
|
|
// Skip term silence
|
|
//---------------------------------
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
pCurCell->m_PitchBufStart = quantTotal;
|
|
m_TotalDur += pCurCell->m_ftDuration;
|
|
quantTotal = (long)(m_TotalDur / PITCH_BUF_RES);
|
|
pCurCell->m_PitchBufEnd = quantTotal;
|
|
skipInitialSil = false;
|
|
}
|
|
index++;
|
|
pCurCell = pAllos->GetNextCell();
|
|
}
|
|
|
|
//------------------------------
|
|
// Init pitch range
|
|
//------------------------------
|
|
pCurCell = m_pAllos->GetHeadCell();
|
|
while( pCurCell )
|
|
{
|
|
float hzVal, pitchK, rangeTemp;
|
|
|
|
//---------------------------------------
|
|
// Scale to possible pitch control
|
|
//---------------------------------------
|
|
rangeTemp = pitchRange * pCurCell->m_PitchRangeScale;
|
|
|
|
hzVal = DoPitchControl( pCurCell->m_user_Pitch, baseLine );
|
|
pitchK = HzToOct( hzVal ) + pCurCell->m_PitchBaseOffs;
|
|
pCurCell->m_Pitch_HI = OctToHz( pitchK + rangeTemp );
|
|
pCurCell->m_Pitch_LO = OctToHz( pitchK - rangeTemp );
|
|
|
|
pCurCell = pAllos->GetNextCell();
|
|
}
|
|
|
|
//--------------------------------------------
|
|
// In case we fail somewhere, set values to
|
|
// a known valid state (monotone).
|
|
//--------------------------------------------
|
|
SetDefaultPitch();
|
|
|
|
if( m_TotalDur > 0 )
|
|
{
|
|
//--------------------------------------------
|
|
// Generate pitch targets
|
|
//--------------------------------------------
|
|
PitchTrack();
|
|
}
|
|
|
|
} /* CPitchProsody::AlloPitch */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
* LineInterpContour *
|
|
*--------------------*
|
|
* Description:
|
|
* Does linear interpolation over the pitch contour
|
|
*
|
|
********************************************************************** MC ***/
|
|
void LineInterpContour( long cNumOfPoints, float *pPoints )
|
|
{
|
|
SPDBG_FUNC( "LineInterpContour" );
|
|
long endAnch,startAnch, i;
|
|
float bPoint1, ePoint1;
|
|
|
|
|
|
//----------------------------------------------------
|
|
// Scan forward from beginning to find 1st non-zero enrty
|
|
// Use it as the START point.
|
|
//----------------------------------------------------
|
|
for( startAnch = 0; startAnch < cNumOfPoints; startAnch++ )
|
|
{
|
|
if( pPoints[startAnch] != 0 )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
bPoint1 = pPoints[startAnch];
|
|
|
|
|
|
//----------------------------------------------------
|
|
// Scan back from end to find 1st non-zero enrty
|
|
// Use it as the END point.
|
|
//----------------------------------------------------
|
|
for( endAnch = cNumOfPoints-1; endAnch >= 0; endAnch-- )
|
|
{
|
|
if( pPoints[endAnch] != 0 )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
ePoint1 = pPoints[endAnch];
|
|
|
|
|
|
long firstp = 0;
|
|
long lastp = 0;
|
|
|
|
while( firstp < cNumOfPoints )
|
|
{
|
|
//-------------------------------------------
|
|
// Find beginning and end of current section
|
|
//-------------------------------------------
|
|
while( pPoints[firstp] != 0 )
|
|
{
|
|
if( ++firstp >= cNumOfPoints-1 )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
if( firstp >= cNumOfPoints-1 )
|
|
{
|
|
//--------------------------------------
|
|
// There's nothing to interpolate!
|
|
//--------------------------------------
|
|
break;
|
|
}
|
|
|
|
|
|
lastp = firstp+1;
|
|
while( pPoints[lastp] == 0 )
|
|
{
|
|
if( ++lastp >= cNumOfPoints )
|
|
{
|
|
lastp = cNumOfPoints;
|
|
break;
|
|
}
|
|
}
|
|
lastp--;
|
|
|
|
if( lastp >= firstp )
|
|
{
|
|
if( (lastp >= cNumOfPoints) || (firstp >= cNumOfPoints) )
|
|
{
|
|
break;
|
|
}
|
|
//-------------------------------------------
|
|
// Do the interpolate
|
|
//-------------------------------------------
|
|
float bPoint,ePoint;
|
|
if( firstp == 0 )
|
|
{
|
|
bPoint = bPoint1;
|
|
}
|
|
else
|
|
{
|
|
bPoint = pPoints[firstp - 1];
|
|
}
|
|
|
|
if( lastp == cNumOfPoints-1 )
|
|
{
|
|
ePoint = ePoint1;
|
|
}
|
|
else
|
|
{
|
|
ePoint = pPoints[lastp + 1];
|
|
}
|
|
|
|
float pointSpread = ePoint - bPoint;
|
|
float timeSpread = (float) ((lastp - firstp)+2);
|
|
float inc = pointSpread / timeSpread;
|
|
float theBase = bPoint;
|
|
for( i = firstp; i <= lastp; i++ )
|
|
{
|
|
theBase += inc;
|
|
pPoints[i] = theBase;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pPoints[firstp] = pPoints[lastp+1];
|
|
}
|
|
firstp = lastp+1;
|
|
}
|
|
} /* LineInterpContour */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
* Interpolate2 *
|
|
*---------------*
|
|
* Description:
|
|
* Do a 2nd order interpolation, a little nicer than just linear
|
|
*
|
|
********************************************************************** MC ***/
|
|
void Interpolate2( INTERP_DIR direction, float *m_theFitPoints, long theStart, long len, float theAmp, float theBase)
|
|
{
|
|
SPDBG_FUNC( "Interpolate2" );
|
|
long midPoint = len / 2;
|
|
long i;
|
|
|
|
theAmp -= theBase;
|
|
|
|
for( i = theStart; i < theStart + len;i++ )
|
|
{
|
|
if (direction == GOING_UP)
|
|
{
|
|
if( i < theStart + midPoint )
|
|
{
|
|
m_theFitPoints[i] = theBase +
|
|
(2 * theAmp) * ((((float)i - (float)theStart) / (float)len) *
|
|
(((float)i - (float)theStart) / (float)len));
|
|
}
|
|
else
|
|
{
|
|
m_theFitPoints[i] = (theBase + theAmp) -
|
|
((2 * theAmp) * ((1 - ((float)i - (float)theStart) / (float)len) *
|
|
(1 - ((float)i - (float)theStart) / (float)len)));
|
|
}
|
|
}
|
|
else if( direction == GOING_DOWN )
|
|
{
|
|
|
|
if( i < theStart + midPoint )
|
|
{
|
|
m_theFitPoints[i] = theBase +
|
|
theAmp - (2 * theAmp) * ((((float)i - (float)theStart) / (float)len) *
|
|
(((float)i - (float)theStart) / (float)len));
|
|
}
|
|
else
|
|
{
|
|
m_theFitPoints[i] = theBase +
|
|
(2 * theAmp) * ((1 - ((float)i - (float)theStart) / (float)len) *
|
|
(1 - ((float)i - (float)theStart) / (float)len));
|
|
}
|
|
}
|
|
}
|
|
} /* Interpolate2 */
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
* SecondOrderInterp *
|
|
*--------------------*
|
|
* Description:
|
|
* Does 2nd order interpolation over the pitch contour
|
|
*
|
|
********************************************************************** MC ***/
|
|
void SecondOrderInterp( long cNumOfPoints, float *pPoints )
|
|
{
|
|
SPDBG_FUNC( "SecondOrderInterp" );
|
|
long endAnch,startAnch;
|
|
float bPoint1, ePoint1;
|
|
|
|
|
|
//----------------------------------------------------
|
|
// Scan forward from beginning to find 1st non-zero enrty
|
|
// Use it as the START point.
|
|
//----------------------------------------------------
|
|
for( startAnch = 0; startAnch < cNumOfPoints; startAnch++ )
|
|
{
|
|
if( pPoints[startAnch] != 0 )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
bPoint1 = pPoints[startAnch];
|
|
|
|
|
|
//----------------------------------------------------
|
|
// Scan back from end to find 1st non-zero enrty
|
|
// Use it as the END point.
|
|
//----------------------------------------------------
|
|
for( endAnch = cNumOfPoints-1; endAnch >= 0; endAnch-- )
|
|
{
|
|
if( pPoints[endAnch] != 0 )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
ePoint1 = pPoints[endAnch];
|
|
|
|
|
|
long firstp = 0;
|
|
long lastp = 0;
|
|
|
|
while( firstp < cNumOfPoints-1 )
|
|
{
|
|
|
|
//------------------------------------------------
|
|
// Find beginning and end of current section
|
|
//------------------------------------------------
|
|
while( pPoints[firstp] != 0 )
|
|
{
|
|
if( ++firstp >= cNumOfPoints-1 )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
if( firstp >= cNumOfPoints-1 )
|
|
{
|
|
//--------------------------------------
|
|
// There's nothing to interpolate!
|
|
//--------------------------------------
|
|
break;
|
|
}
|
|
|
|
lastp = firstp + 1;
|
|
while( pPoints[lastp] == 0 )
|
|
{
|
|
if( ++lastp >= cNumOfPoints )
|
|
{
|
|
lastp = cNumOfPoints;
|
|
break;
|
|
}
|
|
}
|
|
lastp--;
|
|
|
|
if( lastp >= firstp )
|
|
{
|
|
if( (lastp >= cNumOfPoints) || (firstp >= cNumOfPoints) )
|
|
{
|
|
break;
|
|
}
|
|
|
|
//--------------------------------
|
|
// Do the interpolate
|
|
//--------------------------------
|
|
float bPoint, ePoint;
|
|
|
|
if( firstp == 0 )
|
|
{
|
|
bPoint = bPoint1;
|
|
}
|
|
else
|
|
{
|
|
bPoint = pPoints[firstp - 1];
|
|
}
|
|
|
|
long theIndex = lastp + 1;
|
|
|
|
if( lastp == cNumOfPoints-1 )
|
|
{
|
|
ePoint = ePoint1;
|
|
}
|
|
else
|
|
{
|
|
ePoint = pPoints[theIndex];
|
|
}
|
|
|
|
//--------------------------------
|
|
// call the 2nd order routine
|
|
//--------------------------------
|
|
if( ePoint - bPoint > 0 )
|
|
{
|
|
Interpolate2( GOING_UP, pPoints, firstp, (lastp - firstp) + 1, ePoint, bPoint );
|
|
}
|
|
else
|
|
{
|
|
Interpolate2( GOING_DOWN, pPoints, firstp, (lastp - firstp) + 1, bPoint, ePoint );
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
pPoints[firstp] = pPoints[lastp+1];
|
|
}
|
|
|
|
firstp = lastp+1;
|
|
}
|
|
|
|
//---------------------------------
|
|
// FIR Filter
|
|
//---------------------------------
|
|
/*CFIRFilter filterObj;
|
|
float *pOrig;
|
|
|
|
|
|
pOrig = new float[cNumOfPoints];
|
|
memcpy( pOrig, pPoints, cNumOfPoints * sizeof(float));
|
|
if( pOrig )
|
|
{
|
|
filterObj.Make_Filter( 5, // Freq
|
|
10, // order
|
|
100 // SR
|
|
);
|
|
filterObj.DoFilter( pOrig, pPoints, cNumOfPoints);
|
|
delete pOrig;
|
|
}*/
|
|
|
|
//---------------------------------
|
|
// IIR Filter
|
|
//---------------------------------
|
|
#define kPointDelay 1
|
|
|
|
float filter_Out1, filter_In_Gain, filter_FB_Gain;
|
|
float lastPoint;
|
|
long i;
|
|
|
|
//--------------------------------------------------
|
|
// Skip filter if audio len less than delay
|
|
//--------------------------------------------------
|
|
if( cNumOfPoints > kPointDelay )
|
|
{
|
|
filter_In_Gain = 0.10f;
|
|
filter_FB_Gain = 1.0f - filter_In_Gain;
|
|
filter_Out1 = pPoints[0];
|
|
for( i = 0; i < cNumOfPoints; i++ )
|
|
{
|
|
filter_Out1 = (filter_In_Gain * pPoints[i]) + (filter_FB_Gain * filter_Out1);
|
|
pPoints[i] = filter_Out1;
|
|
}
|
|
for( i = kPointDelay; i < cNumOfPoints; i++ )
|
|
{
|
|
pPoints[i-kPointDelay] = pPoints[i];
|
|
}
|
|
i = (cNumOfPoints - kPointDelay) -1;
|
|
lastPoint = pPoints[i++];
|
|
for( ; i < cNumOfPoints; i++ )
|
|
{
|
|
pPoints[i] = lastPoint;
|
|
}
|
|
}
|
|
} /* SecondOrderInterp */
|
|
|
|
/*****************************************************************************
|
|
* CPitchProsody::NewTarget *
|
|
*---------------------------*
|
|
* Description:
|
|
* Insert pitch target into 'm_pContBuf'
|
|
*
|
|
********************************************************************** MC ***/
|
|
void CPitchProsody::NewTarget( long index, float value )
|
|
{
|
|
SPDBG_FUNC( "CPitchProsody::NewTarget" );
|
|
|
|
m_pContBuf[index] = value;
|
|
|
|
//--- Debug Macro - add pitch to target list for later debugging output
|
|
TTSDBG_ADDPITCHTARGET( m_OffsTime + (PITCH_BUF_RES * index), value, m_CurAccent );
|
|
|
|
} /* CPitchProsody::NewTarget */
|
|
|
|
|
|
/*****************************************************************************
|
|
* CPitchProsody::GetKnots *
|
|
*-------------------------*
|
|
* Description:
|
|
* Assign pitch knots based on entries in a contour buffer.
|
|
*
|
|
********************************************************************** MC ***/
|
|
void CPitchProsody::GetKnots ()
|
|
{
|
|
SPDBG_FUNC( "CPitchProsody::GetKnots" );
|
|
CAlloCell *pCurCell;
|
|
float distK, scale;
|
|
float pitchRange;
|
|
long knot, loc, index;
|
|
bool skipInitialSil;
|
|
|
|
skipInitialSil = true;
|
|
pCurCell = m_pAllos->GetHeadCell();
|
|
index = 0;
|
|
while( pCurCell )
|
|
{
|
|
if( index >= m_numOfCells-1 )
|
|
{
|
|
//-----------------------
|
|
// Skip last allo
|
|
//-----------------------
|
|
break;
|
|
}
|
|
if( (!skipInitialSil) || (pCurCell->m_allo != _SIL_) )
|
|
{
|
|
pitchRange = pCurCell->m_Pitch_HI - pCurCell->m_Pitch_LO;
|
|
distK = 1.0f / KNOTS_PER_PHON;
|
|
scale = 0;
|
|
for( knot = 0; knot < KNOTS_PER_PHON; knot++ )
|
|
{
|
|
loc = pCurCell->m_PitchBufStart + (long)((pCurCell->m_PitchBufEnd - pCurCell->m_PitchBufStart) * scale);
|
|
pCurCell->m_ftPitch[knot] = pCurCell->m_Pitch_LO + (m_pContBuf[loc] * pitchRange);
|
|
pCurCell->m_ftTime[knot] = scale * pCurCell->m_ftDuration;
|
|
scale += distK;
|
|
}
|
|
skipInitialSil = false;
|
|
}
|
|
pCurCell = m_pAllos->GetNextCell();
|
|
index++;
|
|
}
|
|
} /* CPitchProsody::GetKnots */
|
|
|
|
|
|
/*****************************************************************************
|
|
* CPitchProsody::PitchTrack *
|
|
*----------------------------*
|
|
* Description:
|
|
* Tag pitch highlights
|
|
*
|
|
********************************************************************** MC ***/
|
|
void CPitchProsody::PitchTrack()
|
|
{
|
|
SPDBG_FUNC( "CPitchProsody::PitchTrack" );
|
|
long i;
|
|
CAlloCell *pCurCell, *pNextCell;
|
|
bool initialWord; // 1st word in phrase
|
|
long wordCntDwn;
|
|
float curProm; // Current accent prominence
|
|
long cNumOfPoints;
|
|
float *pRefBuf, *pCeilBuf, *pFloorBuf;
|
|
float lastProm;
|
|
long loc;
|
|
float value;
|
|
|
|
pRefBuf = pCeilBuf = pFloorBuf = m_pContBuf = NULL;
|
|
cNumOfPoints = (long)(m_TotalDur / PITCH_BUF_RES);
|
|
pRefBuf = new float[cNumOfPoints];
|
|
pCeilBuf = new float[cNumOfPoints];
|
|
pFloorBuf = new float[cNumOfPoints];
|
|
m_pContBuf = new float[cNumOfPoints];
|
|
|
|
if( pRefBuf && pCeilBuf && pFloorBuf && m_pContBuf)
|
|
{
|
|
//--------------------------------------------
|
|
// Initialize buffers to zero
|
|
//--------------------------------------------
|
|
for (i = 0; i < cNumOfPoints; i++)
|
|
{
|
|
pCeilBuf[i] = 0;
|
|
pFloorBuf[i] = 0.00001f;
|
|
pRefBuf[i] = 0;
|
|
m_pContBuf[i] = 0;
|
|
}
|
|
|
|
//--------------------------------------------
|
|
// Linear CEILING slope
|
|
//--------------------------------------------
|
|
if( m_Tune_Style == DESCEND_TUNE )
|
|
{
|
|
pCeilBuf[0] = 1.0;
|
|
pCeilBuf[cNumOfPoints-1] = 0.70f;
|
|
::LineInterpContour( cNumOfPoints, pCeilBuf );
|
|
}
|
|
else if (m_Tune_Style == ASCEND_TUNE)
|
|
{
|
|
pCeilBuf[0] = 0.9f;
|
|
pCeilBuf[cNumOfPoints-1] = 1.0f;
|
|
::LineInterpContour( cNumOfPoints, pCeilBuf );
|
|
}
|
|
else if (m_Tune_Style == FLAT_TUNE)
|
|
{
|
|
pCeilBuf[0] = 1.0f;
|
|
pCeilBuf[cNumOfPoints-1] = 1.0f;
|
|
::LineInterpContour( cNumOfPoints, pCeilBuf );
|
|
}
|
|
|
|
//--------------------------------------------
|
|
// Linear REFERENCE slope
|
|
//--------------------------------------------
|
|
pRefBuf[0] = (float) (pFloorBuf[0] + (pCeilBuf[0] - pFloorBuf[0]) * 0.33f);
|
|
pRefBuf[cNumOfPoints-1] = (float) (pFloorBuf[0] + (pCeilBuf[cNumOfPoints-1] - pFloorBuf[cNumOfPoints-1]) * 0.33f);
|
|
::LineInterpContour( cNumOfPoints,pRefBuf );
|
|
|
|
//--------------------------------------------
|
|
// Final contour buffer
|
|
//--------------------------------------------
|
|
m_pContBuf[0] = pRefBuf[0];
|
|
m_pContBuf[cNumOfPoints-1] = 0.0001f; // Something very small
|
|
|
|
|
|
long iPrevBegin, iPrevEnd, iCurBegin;
|
|
long iCurEnd, iNextBegin, iNextEnd;
|
|
float cCurLen;
|
|
long iCellindex;
|
|
|
|
initialWord = true;
|
|
iCellindex = 0;
|
|
pCurCell = m_pAllos->GetHeadCell();
|
|
while( pCurCell->m_allo == _SIL_ )
|
|
{
|
|
//---------------------------------
|
|
// Skip leading silence
|
|
//---------------------------------
|
|
pCurCell = m_pAllos->GetNextCell();
|
|
iCellindex++;
|
|
}
|
|
wordCntDwn = 1; // Skip 1st word
|
|
lastProm = 0;
|
|
iPrevBegin = iPrevEnd = 0;
|
|
|
|
pNextCell = m_pAllos->GetNextCell();
|
|
while( pCurCell )
|
|
{
|
|
if( iCellindex >= m_numOfCells-1 )
|
|
{
|
|
//-----------------------
|
|
// Skip last allo
|
|
//-----------------------
|
|
break;
|
|
}
|
|
//-----------------------------------
|
|
// Get CURRENT allo
|
|
//-----------------------------------
|
|
iCurBegin = pCurCell->m_PitchBufStart;
|
|
iCurEnd = pCurCell->m_PitchBufEnd;
|
|
cCurLen = (float)(iCurEnd - iCurBegin);
|
|
curProm = pCurCell->m_Accent_Prom * (float)0.1;
|
|
|
|
//-----------------------------------
|
|
// Get NEXT allo
|
|
//-----------------------------------
|
|
iNextBegin = pNextCell->m_PitchBufStart;
|
|
iNextEnd = pNextCell->m_PitchBufEnd;
|
|
|
|
m_CurAccent = pCurCell->m_ToBI_Accent;
|
|
//---------------------
|
|
// Diagnostic
|
|
//---------------------
|
|
m_CurAccentSource = pCurCell->m_AccentSource;
|
|
m_CurBoundarySource = pCurCell->m_BoundarySource;
|
|
m_pCurTextStr = pCurCell->m_pTextStr;
|
|
|
|
switch( pCurCell->m_ToBI_Accent )
|
|
{
|
|
case K_RSTAR:
|
|
break;
|
|
|
|
case K_HSTAR:
|
|
{
|
|
if( !initialWord ) // We never add a 'leg' to a phrase-initial word
|
|
{
|
|
//----------------------------------------------
|
|
// Add a L leg to start to previous allo
|
|
//----------------------------------------------
|
|
if( iPrevBegin )
|
|
{
|
|
loc = (long) ((iCurBegin + (cCurLen * 0.1f)));
|
|
value = ((pCeilBuf[loc] - pRefBuf[loc]) * curProm); // H*
|
|
value = pRefBuf[loc] + (value * 0.25f); // L+H*
|
|
NewTarget( iPrevBegin, value );
|
|
//NewTarget( loc, value );
|
|
}
|
|
}
|
|
//----------------------------------------------
|
|
// Now plug in the H target
|
|
//
|
|
// If we're at a boundary, insert H at
|
|
// allo mid-point else insert at allo start
|
|
//----------------------------------------------
|
|
if( pCurCell->m_ToBI_Boundary != K_NOBND )
|
|
{
|
|
//---------------------------
|
|
// Insert H* at allo start
|
|
//---------------------------
|
|
loc = (long) iCurBegin;
|
|
}
|
|
else
|
|
{
|
|
//---------------------------
|
|
// Insert H* at allo mid-point
|
|
//---------------------------
|
|
loc = (long) (iCurBegin + (cCurLen * K_HSTAR_OFFSET));
|
|
}
|
|
value = pRefBuf[loc] + ((pCeilBuf[loc] - pRefBuf[loc]) * curProm); // H*
|
|
NewTarget( loc, value );
|
|
}
|
|
break;
|
|
|
|
case K_LSTAR:
|
|
{
|
|
//------------------------------------
|
|
// Insert L* at mid-point
|
|
//------------------------------------
|
|
loc = (long) (iCurBegin + (cCurLen * 0.3f));
|
|
value = pRefBuf[loc] - ((pRefBuf[loc] - pFloorBuf[loc]) * curProm); // L*
|
|
NewTarget( loc, value );
|
|
}
|
|
break;
|
|
|
|
case K_LSTARH:
|
|
{
|
|
//----------------------------------------------
|
|
// Insert L* at current start
|
|
//----------------------------------------------
|
|
value = pRefBuf[iCurBegin] - ((pRefBuf[iCurBegin] - pFloorBuf[iCurBegin]) * curProm); // L*+H
|
|
NewTarget( iCurBegin, value );
|
|
if( iNextBegin )
|
|
{
|
|
//----------------------------------------------
|
|
// Insert H at next end
|
|
// set prom gain?
|
|
//----------------------------------------------
|
|
value = pRefBuf[iNextEnd] - ((pRefBuf[iNextEnd] - pFloorBuf[iNextEnd]) * (curProm /* * .3 */ ));
|
|
NewTarget( iNextEnd, value );
|
|
}
|
|
lastProm = 0;
|
|
}
|
|
break;
|
|
|
|
case K_LHSTAR:
|
|
{
|
|
loc = (long) (iCurBegin + (cCurLen * 0.3f));
|
|
if( iPrevBegin )
|
|
{
|
|
//----------------------------------------------
|
|
// Insert L at previous start
|
|
//----------------------------------------------
|
|
value = (pRefBuf[iPrevBegin] - ((pRefBuf[iPrevBegin] - pFloorBuf[iPrevBegin]) * (curProm * 0.3f))); // L+H*
|
|
NewTarget( iPrevBegin, value );
|
|
}
|
|
//----------------------------------------------
|
|
// Insert H* at current mid-point
|
|
//----------------------------------------------
|
|
value = pRefBuf[loc] + ((pCeilBuf[loc] - pRefBuf[loc]) * curProm); // H*
|
|
NewTarget( loc, value );
|
|
lastProm = curProm;
|
|
}
|
|
break;
|
|
|
|
case K_HSTARLSTAR:
|
|
{
|
|
//value = pRefBuf[iCurBegin] + ((pCeilBuf[iCurBegin] - pRefBuf[iCurBegin]) * curProm); // H*
|
|
value = pRefBuf[0] + ((pCeilBuf[0] - pRefBuf[0]) * curProm); // H*
|
|
NewTarget( iCurBegin, value );
|
|
|
|
loc = (long) (iCurBegin + (cCurLen * 0.75f));
|
|
value = pRefBuf[loc] - ((pRefBuf[loc] - pFloorBuf[loc]) * curProm); // L*
|
|
NewTarget( loc, value );
|
|
lastProm = curProm;
|
|
}
|
|
break;
|
|
case K_DHSTAR:
|
|
{
|
|
loc = (long) ( iCurBegin + (cCurLen * 0.0f) );
|
|
if( lastProm )
|
|
{
|
|
lastProm *= K_HDOWNSTEP_COEFF;
|
|
value = pRefBuf[loc] + ((pCeilBuf[loc] - pRefBuf[loc]) * lastProm); // !H*
|
|
NewTarget( loc, value );
|
|
}
|
|
//-----------------------------------------
|
|
// no previous H*, treat !H* like an H*
|
|
//-----------------------------------------
|
|
else
|
|
{
|
|
value = pRefBuf[loc] + ((pCeilBuf[loc] - pRefBuf[loc]) * curProm); // H*
|
|
NewTarget( loc, value );
|
|
lastProm = curProm;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default: // Unknown accent specfied
|
|
break;
|
|
}
|
|
|
|
//-------------------------------------------------------------
|
|
// if there's a boundary, fill in pitch value(s)
|
|
// assume the boundary is set to correct (voiced) final phone
|
|
//-------------------------------------------------------------
|
|
curProm = pCurCell->m_Boundary_Prom * (float)0.1;
|
|
m_CurAccent =(TOBI_ACCENT) pCurCell->m_ToBI_Boundary;
|
|
//---------------------
|
|
// Diagnostic
|
|
//---------------------
|
|
m_CurAccentSource = pCurCell->m_AccentSource;
|
|
m_CurBoundarySource = pCurCell->m_BoundarySource;
|
|
m_pCurTextStr = pCurCell->m_pTextStr;
|
|
switch( pCurCell->m_ToBI_Boundary )
|
|
{
|
|
case K_LMINUS:
|
|
{
|
|
value = pRefBuf[iCurEnd] - ((pRefBuf[iCurEnd] - pFloorBuf[iCurEnd]) * curProm); // L-
|
|
NewTarget( iCurEnd, value );
|
|
}
|
|
break;
|
|
|
|
case K_HMINUS:
|
|
{
|
|
value = pRefBuf[iCurEnd] + ((pCeilBuf[iCurEnd] - pRefBuf[iCurEnd]) * curProm); // H-
|
|
NewTarget( iCurEnd, value );
|
|
}
|
|
break;
|
|
|
|
//case K_LPERC:
|
|
//case K_HPERC:
|
|
|
|
case K_LMINUSLPERC:
|
|
{
|
|
value = pFloorBuf[iCurEnd];
|
|
//NewTarget( iCurEnd, value );
|
|
NewTarget( iCurBegin, value );
|
|
}
|
|
break;
|
|
|
|
case K_HMINUSHPERC:
|
|
{
|
|
value = pCeilBuf[iCurEnd];
|
|
NewTarget( iCurEnd, value );
|
|
}
|
|
break;
|
|
|
|
case K_LMINUSHPERC: // L-H%
|
|
{
|
|
//---------------------------------------
|
|
// comma continuation rise
|
|
//---------------------------------------
|
|
//+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
// L starts at middle of previous phon
|
|
//+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
loc = iPrevBegin + (iPrevEnd - iPrevBegin) / 2;
|
|
value = pRefBuf[loc] - ((pRefBuf[loc] - pFloorBuf[loc]) * curProm); // L-
|
|
NewTarget( loc, value );
|
|
//+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
// H at end of current phon
|
|
//+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
value = pRefBuf[iCurEnd] + ((pCeilBuf[iCurEnd] - pRefBuf[iCurEnd]) * curProm); // H%
|
|
NewTarget( iCurEnd, value );
|
|
}
|
|
break;
|
|
|
|
case K_HMINUSLPERC:
|
|
{
|
|
//---------------------------------------
|
|
// accent extension followed by sharp drop
|
|
//---------------------------------------
|
|
value = pRefBuf[iCurBegin] + ((pCeilBuf[iCurBegin] - pRefBuf[iCurBegin]) * curProm); // H-
|
|
NewTarget( iCurBegin, value );
|
|
value = pFloorBuf[iCurEnd]; // L%
|
|
//loc = iCurBegin + ((iCurEnd - iCurBegin) * 0.1f);
|
|
NewTarget( iCurEnd, value );
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
//----------------------------
|
|
// Unflag initial word
|
|
//----------------------------
|
|
if( (initialWord) && (pCurCell->m_ctrlFlags & WORD_START) )
|
|
{
|
|
wordCntDwn--;
|
|
if( wordCntDwn < 0 )
|
|
{
|
|
initialWord = false;
|
|
}
|
|
}
|
|
|
|
//----------------------------
|
|
// Setup for next allo
|
|
//----------------------------
|
|
iPrevBegin = iCurBegin;
|
|
iPrevEnd = iCurEnd;
|
|
|
|
pCurCell = pNextCell;
|
|
pNextCell = m_pAllos->GetNextCell();
|
|
iCellindex++;
|
|
}
|
|
|
|
//--- Debug Macro - Log pitch data to stream
|
|
TTSDBG_LOGTOBI;
|
|
|
|
::SecondOrderInterp( cNumOfPoints, m_pContBuf );
|
|
GetKnots();
|
|
}
|
|
|
|
if( pRefBuf )
|
|
{
|
|
delete pRefBuf;
|
|
}
|
|
if( pCeilBuf )
|
|
{
|
|
delete pCeilBuf;
|
|
}
|
|
if( pFloorBuf )
|
|
{
|
|
delete pFloorBuf;
|
|
}
|
|
if( m_pContBuf )
|
|
{
|
|
delete m_pContBuf;
|
|
}
|
|
} /* CPitchProsody::PitchTrack */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|