123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769 |
- #ifndef R8B_CDSPRESAMPLER_INCLUDED
- #define R8B_CDSPRESAMPLER_INCLUDED
- #include "CDSPHBDownsampler.h"
- #include "CDSPHBUpsampler.h"
- #include "CDSPBlockConvolver.h"
- #include "CDSPFracInterpolator.h"
- namespace r8b {
- class CDSPResampler : public CDSPProcessor
- {
- public:
-
- CDSPResampler( const double SrcSampleRate, const double DstSampleRate,
- const int aMaxInLen, const double ReqTransBand = 2.0,
- const double ReqAtten = 206.91,
- const EDSPFilterPhaseResponse ReqPhase = fprLinearPhase )
- : StepCapacity( 0 )
- , StepCount( 0 )
- , MaxInLen( aMaxInLen )
- , CurMaxOutLen( aMaxInLen )
- , LatencyFrac( 0.0 )
- {
- R8BASSERT( SrcSampleRate > 0.0 );
- R8BASSERT( DstSampleRate > 0.0 );
- R8BASSERT( MaxInLen > 0 );
- R8BCONSOLE( "* CDSPResampler: src=%.1f dst=%.1f len=%i tb=%.1f "
- "att=%.2f ph=%i\n", SrcSampleRate, DstSampleRate, aMaxInLen,
- ReqTransBand, ReqAtten, (int) ReqPhase );
- if( SrcSampleRate == DstSampleRate )
- {
- return;
- }
- TmpBufCapacities[ 0 ] = 0;
- TmpBufCapacities[ 1 ] = 0;
- CurTmpBuf = 0;
-
- const int CommonRatioCount = 5;
- const int CommonRatios[ CommonRatioCount ][ 2 ] = {
- { 1, 2 },
- { 1, 3 },
- { 2, 3 },
- { 3, 2 },
- { 3, 4 }
- };
- int i;
- for( i = 0; i < CommonRatioCount; i++ )
- {
- const int num = CommonRatios[ i ][ 0 ];
- const int den = CommonRatios[ i ][ 1 ];
- if( SrcSampleRate * num == DstSampleRate * den )
- {
- addProcessor( new CDSPBlockConvolver(
- CDSPFIRFilterCache :: getLPFilter(
- 1.0 / ( num > den ? num : den ), ReqTransBand,
- ReqAtten, ReqPhase, num ), num, den, LatencyFrac ));
- createTmpBuffers();
- return;
- }
- }
-
- for( i = 2; i <= 3; i++ )
- {
- bool WasFound = false;
- int c = 0;
- while( true )
- {
- const double NewSR = SrcSampleRate * ( i << c );
- if( NewSR == DstSampleRate )
- {
- WasFound = true;
- break;
- }
- if( NewSR > DstSampleRate )
- {
- break;
- }
- c++;
- }
- if( WasFound )
- {
- addProcessor( new CDSPBlockConvolver(
- CDSPFIRFilterCache :: getLPFilter( 1.0 / i, ReqTransBand,
- ReqAtten, ReqPhase, i ), i, 1, LatencyFrac ));
- const bool IsThird = ( i == 3 );
- for( i = 0; i < c; i++ )
- {
- addProcessor( new CDSPHBUpsampler( ReqAtten, i, IsThird,
- LatencyFrac ));
- }
- createTmpBuffers();
- return;
- }
- }
- if( DstSampleRate * 2.0 > SrcSampleRate )
- {
-
- const double NormFreq = ( DstSampleRate > SrcSampleRate ? 0.5 :
- 0.5 * DstSampleRate / SrcSampleRate );
- addProcessor( new CDSPBlockConvolver(
- CDSPFIRFilterCache :: getLPFilter( NormFreq, ReqTransBand,
- ReqAtten, ReqPhase, 2.0 ), 2, 1, LatencyFrac ));
-
-
- const double tbw = 0.0175;
-
- const double ThreshSampleRate = SrcSampleRate /
- ( 1.0 - tbw * ReqTransBand );
-
-
- int c = 0;
- int div = 1;
- while( true )
- {
- const int ndiv = div * 2;
- if( DstSampleRate < ThreshSampleRate * ndiv )
- {
- break;
- }
- div = ndiv;
- c++;
- }
- int c2 = 0;
- int div2 = 1;
- while( true )
- {
- const int ndiv = div * ( c2 == 0 ? 3 : 2 );
- if( DstSampleRate < ThreshSampleRate * ndiv )
- {
- break;
- }
- div2 = ndiv;
- c2++;
- }
- const double SrcSampleRate2 = SrcSampleRate * 2.0;
- int tmp1;
- int tmp2;
- if( c == 1 && getWholeStepping( SrcSampleRate2, DstSampleRate,
- tmp1, tmp2 ))
- {
-
-
- c = 0;
- }
- if( c > 0 )
- {
-
- int num;
- if( c2 > 0 && div2 > div )
- {
- div = div2;
- c = c2;
- num = 3;
- }
- else
- {
- num = 2;
- }
- addProcessor( new CDSPFracInterpolator( SrcSampleRate2 * div,
- DstSampleRate, ReqAtten, false, LatencyFrac ));
- double tb = ( 1.0 - SrcSampleRate * div / DstSampleRate ) /
- tbw;
-
- if( tb > CDSPFIRFilter :: getLPMaxTransBand() )
- {
- tb = CDSPFIRFilter :: getLPMaxTransBand();
- }
- addProcessor( new CDSPBlockConvolver(
- CDSPFIRFilterCache :: getLPFilter( 1.0 / num, tb,
- ReqAtten, ReqPhase, num ), num, 1, LatencyFrac ));
- const bool IsThird = ( num == 3 );
- for( i = 1; i < c; i++ )
- {
- addProcessor( new CDSPHBUpsampler( ReqAtten, i - 1,
- IsThird, LatencyFrac ));
- }
- }
- else
- {
- addProcessor( new CDSPFracInterpolator( SrcSampleRate2,
- DstSampleRate, ReqAtten, false, LatencyFrac ));
- }
- createTmpBuffers();
- return;
- }
-
- double CheckSR = DstSampleRate * 4.0;
- int c = 0;
- double FinGain = 1.0;
- while( CheckSR <= SrcSampleRate )
- {
- c++;
- CheckSR *= 2.0;
- FinGain *= 0.5;
- }
- const int SrcSRDiv = ( 1 << c );
- int downf;
- double NormFreq = 0.5;
- bool UseInterp = true;
- bool IsThird = false;
- for( downf = 2; downf <= 3; downf++ )
- {
- if( DstSampleRate * SrcSRDiv * downf == SrcSampleRate )
- {
- NormFreq = 1.0 / downf;
- UseInterp = false;
- IsThird = ( downf == 3 );
- break;
- }
- }
- if( UseInterp )
- {
- downf = 1;
- NormFreq = DstSampleRate * SrcSRDiv / SrcSampleRate;
- IsThird = ( NormFreq * 3.0 <= 1.0 );
- }
- for( i = 0; i < c; i++ )
- {
-
-
-
-
- addProcessor( new CDSPHBDownsampler( ReqAtten, c - 1 - i, IsThird,
- LatencyFrac ));
- }
- addProcessor( new CDSPBlockConvolver(
- CDSPFIRFilterCache :: getLPFilter( NormFreq, ReqTransBand,
- ReqAtten, ReqPhase, FinGain ), 1, downf, LatencyFrac ));
- if( UseInterp )
- {
- addProcessor( new CDSPFracInterpolator( SrcSampleRate,
- DstSampleRate * SrcSRDiv, ReqAtten, IsThird, LatencyFrac ));
- }
- createTmpBuffers();
- }
- virtual ~CDSPResampler()
- {
- int i;
- for( i = 0; i < StepCount; i++ )
- {
- delete Steps[ i ];
- }
- }
- virtual int getLatency() const
- {
- return( 0 );
- }
- virtual double getLatencyFrac() const
- {
- return( LatencyFrac );
- }
-
- virtual int getMaxOutLen( const int ) const
- {
- return( CurMaxOutLen );
- }
-
- virtual void clear()
- {
- int i;
- for( i = 0; i < StepCount; i++ )
- {
- Steps[ i ] -> clear();
- }
- }
-
- virtual int process( double* ip0, int l, double*& op0 )
- {
- R8BASSERT( l >= 0 );
- double* ip = ip0;
- int i;
- for( i = 0; i < StepCount; i++ )
- {
- double* op = TmpBufs[ i & 1 ];
- l = Steps[ i ] -> process( ip, l, op );
- ip = op;
- }
- op0 = ip;
- return( l );
- }
-
- template< typename Tin, typename Tout >
- void oneshot( const Tin* ip, int iplen, Tout* op, int oplen )
- {
- CFixedBuffer< double > Buf( MaxInLen );
- bool IsZero = false;
- while( oplen > 0 )
- {
- int rc;
- double* p;
- int i;
- if( iplen == 0 )
- {
- rc = MaxInLen;
- p = &Buf[ 0 ];
- if( !IsZero )
- {
- IsZero = true;
- memset( p, 0, MaxInLen * sizeof( p[ 0 ]));
- }
- }
- else
- {
- rc = min( iplen, MaxInLen );
- if( sizeof( Tin ) == sizeof( double ))
- {
- p = (double*) ip;
- }
- else
- {
- p = &Buf[ 0 ];
- for( i = 0; i < rc; i++ )
- {
- p[ i ] = ip[ i ];
- }
- }
- ip += rc;
- iplen -= rc;
- }
- double* op0;
- int wc = process( p, rc, op0 );
- wc = min( oplen, wc );
- for( i = 0; i < wc; i++ )
- {
- op[ i ] = (Tout) op0[ i ];
- }
- op += wc;
- oplen -= wc;
- }
- clear();
- }
-
- int getInLenBeforeOutStart()
- {
- int inc = 0;
- while( true )
- {
- double ins = 0.0;
- double* op;
- if( process( &ins, 1, op ) > 0 )
- {
- clear();
- return( inc );
- }
- inc++;
- }
- }
- private:
- CFixedBuffer< CDSPProcessor* > Steps;
-
- int StepCapacity;
-
- int StepCount;
-
- int MaxInLen;
-
- CFixedBuffer< double > TmpBufAll;
-
-
- double* TmpBufs[ 2 ];
-
- int TmpBufCapacities[ 2 ];
-
-
- int CurTmpBuf;
-
- int CurMaxOutLen;
-
- double LatencyFrac;
-
-
-
-
- void addProcessor( CDSPProcessor* const Proc )
- {
- if( StepCount == StepCapacity )
- {
-
- const int NewCapacity = StepCapacity + 8;
- Steps.realloc( StepCapacity, NewCapacity );
- StepCapacity = NewCapacity;
- }
- LatencyFrac = Proc -> getLatencyFrac();
- CurMaxOutLen = Proc -> getMaxOutLen( CurMaxOutLen );
- if( CurMaxOutLen > TmpBufCapacities[ CurTmpBuf ])
- {
- TmpBufCapacities[ CurTmpBuf ] = CurMaxOutLen;
- }
- CurTmpBuf ^= 1;
- Steps[ StepCount ] = Proc;
- StepCount++;
- }
-
- void createTmpBuffers()
- {
- const int ol = TmpBufCapacities[ 0 ] + TmpBufCapacities[ 1 ];
- if( ol > 0 )
- {
- TmpBufAll.alloc( ol );
- TmpBufs[ 0 ] = &TmpBufAll[ 0 ];
- TmpBufs[ 1 ] = &TmpBufAll[ TmpBufCapacities[ 0 ]];
- }
- R8BCONSOLE( "* CDSPResampler: init done\n" );
- }
- };
- class CDSPResampler16 : public CDSPResampler
- {
- public:
-
- CDSPResampler16( const double SrcSampleRate, const double DstSampleRate,
- const int aMaxInLen, const double ReqTransBand = 2.0 )
- : CDSPResampler( SrcSampleRate, DstSampleRate, aMaxInLen, ReqTransBand,
- 136.45, fprLinearPhase )
- {
- }
- };
- class CDSPResampler16IR : public CDSPResampler
- {
- public:
-
- CDSPResampler16IR( const double SrcSampleRate, const double DstSampleRate,
- const int aMaxInLen, const double ReqTransBand = 2.0 )
- : CDSPResampler( SrcSampleRate, DstSampleRate, aMaxInLen, ReqTransBand,
- 109.56, fprLinearPhase )
- {
- }
- };
- class CDSPResampler24 : public CDSPResampler
- {
- public:
-
- CDSPResampler24( const double SrcSampleRate, const double DstSampleRate,
- const int aMaxInLen, const double ReqTransBand = 2.0 )
- : CDSPResampler( SrcSampleRate, DstSampleRate, aMaxInLen, ReqTransBand,
- 180.15, fprLinearPhase )
- {
- }
- };
- }
- #endif
|