From e0bbc1b814b7838cc682f0cbdcbfc9efa0d8d6c5 Mon Sep 17 00:00:00 2001 From: gigaherz Date: Sun, 14 Aug 2011 22:06:18 +0000 Subject: [PATCH] SPU2-X: Reworked the circular buffer functions for audio output, with a thread-safe version. This hopefully avoids the issues appearing after long periods of continuous playing. git-svn-id: http://pcsx2.googlecode.com/svn/trunk@4871 96395faa-99c1-11dd-bbfe-3dabce05a288 --- plugins/spu2-x/src/SndOut.cpp | 157 +++++++++++++++++---------- plugins/spu2-x/src/SndOut.h | 20 +++- plugins/spu2-x/src/Timestretcher.cpp | 8 +- plugins/spu2-x/src/spu2sys.cpp | 2 +- 4 files changed, 118 insertions(+), 69 deletions(-) diff --git a/plugins/spu2-x/src/SndOut.cpp b/plugins/spu2-x/src/SndOut.cpp index 095025ef5d..e348b500dc 100644 --- a/plugins/spu2-x/src/SndOut.cpp +++ b/plugins/spu2-x/src/SndOut.cpp @@ -112,9 +112,8 @@ int FindOutputModuleById( const wchar_t* omodid ) StereoOut32 *SndBuffer::m_buffer; s32 SndBuffer::m_size; -s32 SndBuffer::m_rpos; -s32 SndBuffer::m_wpos; -s32 SndBuffer::m_data; +__aligned(4) volatile s32 SndBuffer::m_rpos; +__aligned(4) volatile s32 SndBuffer::m_wpos; bool SndBuffer::m_underrun_freeze; StereoOut32* SndBuffer::sndTempBuffer = NULL; @@ -131,6 +130,8 @@ int GetAlignedBufferSize( int comp ) bool SndBuffer::CheckUnderrunStatus( int& nSamples, int& quietSampleCount ) { quietSampleCount = 0; + + int data = _GetApproximateDataInBuffer(); if( m_underrun_freeze ) { int toFill = m_size / ( (SynchMode == 2) ? 32 : 400); // TimeStretch and Async off? @@ -138,7 +139,7 @@ bool SndBuffer::CheckUnderrunStatus( int& nSamples, int& quietSampleCount ) // toFill is now aligned to a SndOutPacket - if( m_data < toFill ) + if( data < toFill ) { quietSampleCount = nSamples; return false; @@ -149,10 +150,10 @@ bool SndBuffer::CheckUnderrunStatus( int& nSamples, int& quietSampleCount ) ConLog(" * SPU2 > Underrun compensation (%d packets buffered)\n", toFill / SndOutPacketSize ); lastPct = 0.0; // normalize timestretcher } - else if( m_data < nSamples ) + else if( data < nSamples ) { - nSamples = m_data; - quietSampleCount = SndOutPacketSize - m_data; + nSamples = data; + quietSampleCount = SndOutPacketSize - data; m_underrun_freeze = true; if( SynchMode == 0 ) // TimeStrech on @@ -172,6 +173,68 @@ void SndBuffer::_InitFail() mods[OutputModule]->Init(); } +int SndBuffer::_GetApproximateDataInBuffer() +{ + // WARNING: not necessarily 100% up to date by the time it's used, but it will have to do. + return (m_wpos + m_size - m_rpos) % m_size; +} + +void SndBuffer::_WriteSamples_Internal(StereoOut32 *bData, int nSamples) +{ + // WARNING: This assumes the write will NOT wrap around, + // and also assumes there's enough free space in the buffer. + + memcpy(m_buffer + m_wpos, bData, nSamples * sizeof(StereoOut32)); + m_wpos = (m_wpos + nSamples) % m_size; +} + +void SndBuffer::_DropSamples_Internal(int nSamples) +{ + m_rpos = (m_rpos + nSamples) % m_size; +} + +void SndBuffer::_ReadSamples_Internal(StereoOut32 *bData, int nSamples) +{ + // WARNING: This assumes the read will NOT wrap around, + // and also assumes there's enough data in the buffer. + memcpy(bData, m_buffer + m_rpos, nSamples * sizeof(StereoOut32)); + _DropSamples_Internal(nSamples); +} + +void SndBuffer::_WriteSamples_Safe(StereoOut32 *bData, int nSamples) +{ + // WARNING: This code assumes there's only ONE writing process. + if( (m_size - m_wpos) < nSamples) + { + int b1 = m_size - m_wpos; + int b2 = nSamples - b1; + + _WriteSamples_Internal(bData, b1); + _WriteSamples_Internal(bData+b1, b2); + } + else + { + _WriteSamples_Internal(bData, nSamples); + } +} + +void SndBuffer::_ReadSamples_Safe(StereoOut32* bData, int nSamples) +{ + // WARNING: This code assumes there's only ONE reading process. + if( (m_size - m_rpos) < nSamples) + { + int b1 = m_size - m_rpos; + int b2 = nSamples - b1; + + _ReadSamples_Internal(bData, b1); + _ReadSamples_Internal(bData+b1, b2); + } + else + { + _ReadSamples_Internal(bData, nSamples); + } +} + // Note: When using with 32 bit output buffers, the user of this function is responsible // for shifting the values to where they need to be manually. The fixed point depth of // the sample output is determined by the SndOutVolumeShift, which is the number of bits @@ -197,27 +260,23 @@ template void SndBuffer::ReadSamples(T* bData) if( CheckUnderrunStatus( nSamples, quietSamples ) ) { jASSUME( nSamples <= SndOutPacketSize ); + + // WARNING: This code assumes there's only ONE reading process. + int b1 = m_size - m_rpos; - // [Air] [TODO]: This loop is probably a candidate for SSE2 optimization. + if(b1 > nSamples) + b1 = nSamples; - const int endPos = m_rpos + nSamples; - const int secondCopyLen = endPos - m_size; - const StereoOut32* rposbuffer = &m_buffer[m_rpos]; + // First part + for( int i=0; i 0 ) - { - nSamples -= secondCopyLen; - for( int i=0; i (m_size-SndOutPacketSize) ) comp = m_size-SndOutPacketSize; } - m_data -= comp; - m_rpos = (m_rpos+comp) % m_size; + _DropSamples_Internal(comp); + if( MsgOverruns() ) ConLog(" * SPU2 > Overrun Compensation (%d packets tossed)\n", comp / SndOutPacketSize ); lastPct = 0.0; // normalize the timestretcher +#else + if( MsgOverruns() ) + ConLog(" * SPU2 > Overrun! 1 packet tossed)\n"); + lastPct = 0.0; // normalize the timestretcher + return; +#endif } - // copy in two phases, since there's a chance the packet - // wraps around the buffer (it'd be nice to deal in packets only, but - // the timestretcher and DSP options require flexibility). - - const int endPos = m_wpos + nSamples; - const int secondCopyLen = endPos - m_size; - StereoOut32* wposbuffer = &m_buffer[m_wpos]; - - m_data += nSamples; - if( secondCopyLen > 0 ) - { - nSamples -= secondCopyLen; - memcpy( m_buffer, &bData[nSamples], secondCopyLen * sizeof( *bData ) ); - m_wpos = secondCopyLen; - } - else - m_wpos += nSamples; - - memcpy( wposbuffer, bData, nSamples * sizeof( *bData ) ); - - // Use to monitor buffer levels in real time - /*int drvempty = mods[OutputModule]->GetEmptySampleCount(); - float result = (float)(m_data + m_predictData - drvempty) - (m_size/16); - result /= (m_size/16); - if (result > 0.6 || result < -0.5) - printf("buffer: %f\n",result); - }*/ + _WriteSamples_Safe(bData, nSamples); } void SndBuffer::Init() @@ -331,11 +370,9 @@ void SndBuffer::Init() // initialize sound buffer // Buffer actually attempts to run ~50%, so allocate near double what // the requested latency is: - - + m_rpos = 0; m_wpos = 0; - m_data = 0; try { diff --git a/plugins/spu2-x/src/SndOut.h b/plugins/spu2-x/src/SndOut.h index 15fa465f7e..cc0cb6da8e 100644 --- a/plugins/spu2-x/src/SndOut.h +++ b/plugins/spu2-x/src/SndOut.h @@ -385,7 +385,7 @@ private: static StereoOut32* sndTempBuffer; static StereoOut16* sndTempBuffer16; - + static int sndTempProgress; static int m_dsp_progress; @@ -394,17 +394,16 @@ private: static StereoOut32 *m_buffer; static s32 m_size; - static s32 m_rpos; - static s32 m_wpos; - static s32 m_data; + static __aligned(4) volatile s32 m_rpos; + static __aligned(4) volatile s32 m_wpos; + static float lastEmergencyAdj; static float cTempo; static float eTempo; static int ssFreeze; static void _InitFail(); - static void _WriteSamples(StereoOut32* bData, int nSamples); static bool CheckUnderrunStatus( int& nSamples, int& quietSampleCount ); static void soundtouchInit(); @@ -419,6 +418,17 @@ private: static void UpdateTempoChangeSoundTouch(); static void UpdateTempoChangeSoundTouch2(); + static void _WriteSamples(StereoOut32* bData, int nSamples); + + static void _WriteSamples_Safe(StereoOut32* bData, int nSamples); + static void _ReadSamples_Safe(StereoOut32* bData, int nSamples); + + static void _WriteSamples_Internal(StereoOut32 *bData, int nSamples); + static void _DropSamples_Internal(int nSamples); + static void _ReadSamples_Internal(StereoOut32 *bData, int nSamples); + + static int _GetApproximateDataInBuffer(); + public: static void UpdateTempoChangeAsyncMixing(); static void Init(); diff --git a/plugins/spu2-x/src/Timestretcher.cpp b/plugins/spu2-x/src/Timestretcher.cpp index 62ea69f132..58744fb959 100644 --- a/plugins/spu2-x/src/Timestretcher.cpp +++ b/plugins/spu2-x/src/Timestretcher.cpp @@ -54,7 +54,8 @@ float SndBuffer::GetStatusPct() //ConLog( "Data %d >>> driver: %d predict: %d\n", m_data, drvempty, m_predictData ); - float result = (float)(m_data + m_predictData - drvempty) - (m_size/16); + int data = _GetApproximateDataInBuffer(); + float result = (float)( data + m_predictData - drvempty) - (m_size/16); result /= (m_size/16); return result; } @@ -136,7 +137,8 @@ void SndBuffer::UpdateTempoChangeSoundTouch2() static int hys_ok_count=0; static float dynamicTargetFullness=baseTargetFullness; - float bufferFullness=(float)m_data;///(float)m_size; + int data = _GetApproximateDataInBuffer(); + float bufferFullness=(float)data;///(float)m_size; #ifdef NEWSTRETCHER_USE_DYNAMIC_TUNING {//test current iterations/sec every 0.5s, and change algo params accordingly if different than previous IPS more than 30% @@ -207,7 +209,7 @@ void SndBuffer::UpdateTempoChangeSoundTouch2() if(delta.GetMilliseconds()>1000){//report buffers state and tempo adjust every second ConLog("buffers: %4d ms (%3.0f%%), tempo: %f, comp: %2.3f, iters: %d, (N-IPS:%d -> avg:%d, minokc:%d, div:%d)\n", - (int)(m_data/48), (double)(100.0*bufferFullness/baseTargetFullness), (double)tempoAdjust, (double)(dynamicTargetFullness/baseTargetFullness), iters, (int)targetIPS + (int)(data/48), (double)(100.0*bufferFullness/baseTargetFullness), (double)tempoAdjust, (double)(dynamicTargetFullness/baseTargetFullness), iters, (int)targetIPS , STRETCH_AVERAGE_LEN, hys_min_ok_count, compensationDivider ); last=unow; diff --git a/plugins/spu2-x/src/spu2sys.cpp b/plugins/spu2-x/src/spu2sys.cpp index ab283307c3..ce9b10bd76 100644 --- a/plugins/spu2-x/src/spu2sys.cpp +++ b/plugins/spu2-x/src/spu2sys.cpp @@ -183,7 +183,7 @@ void V_Core::Init( int index ) Regs.STATX = 0x80; Regs.ENDX = 0xffffff; // PS2 confirmed - RevBuffers.NeedsUpdated = 1; + RevBuffers.NeedsUpdated = true; UpdateEffectsBufferSize(); }