From 140f99c1631bc93bd280cdaf70428d63e1bf5d54 Mon Sep 17 00:00:00 2001 From: saker Date: Mon, 16 Dec 2024 23:50:34 -0500 Subject: [PATCH 01/22] Remove FIFO thread --- include/AudioDevice.h | 7 +-- include/AudioDummy.h | 10 ---- include/AudioEngine.h | 45 ++-------------- src/core/AudioEngine.cpp | 93 ++-------------------------------- src/core/ProjectRenderer.cpp | 6 +-- src/core/audio/AudioDevice.cpp | 19 ++----- 6 files changed, 15 insertions(+), 165 deletions(-) diff --git a/include/AudioDevice.h b/include/AudioDevice.h index 376aee26be4..942b4bd8098 100644 --- a/include/AudioDevice.h +++ b/include/AudioDevice.h @@ -83,11 +83,7 @@ class AudioDevice void processNextBuffer(); - virtual void startProcessing() - { - m_inProcess = true; - } - + virtual void startProcessing(); virtual void stopProcessing(); protected: @@ -130,7 +126,6 @@ class AudioDevice sample_rate_t m_sampleRate; ch_cnt_t m_channels; AudioEngine* m_audioEngine; - bool m_inProcess; QMutex m_devMutex; diff --git a/include/AudioDummy.h b/include/AudioDummy.h index 607ea40f2f7..17826f89ddb 100644 --- a/include/AudioDummy.h +++ b/include/AudioDummy.h @@ -94,16 +94,6 @@ class AudioDummy : public QThread, public AudioDevice while( true ) { timer.reset(); - const SampleFrame* b = audioEngine()->nextBuffer(); - if( !b ) - { - break; - } - if( audioEngine()->hasFifoWriter() ) - { - delete[] b; - } - const int microseconds = static_cast( audioEngine()->framesPerPeriod() * 1000000.0f / audioEngine()->outputSampleRate() - timer.elapsed() ); if( microseconds > 0 ) { diff --git a/include/AudioEngine.h b/include/AudioEngine.h index b22830221d9..dc8ead8d1fb 100644 --- a/include/AudioEngine.h +++ b/include/AudioEngine.h @@ -37,7 +37,6 @@ #include "lmms_basics.h" #include "SampleFrame.h" #include "LocklessList.h" -#include "FifoBuffer.h" #include "AudioEngineProfiler.h" #include "PlayHandle.h" @@ -160,10 +159,7 @@ class LMMS_EXPORT AudioEngine : public QObject //! Set new audio device. Old device will be deleted, //! unless it's stored using storeAudioDevice - void setAudioDevice( AudioDevice * _dev, - const struct qualitySettings & _qs, - bool _needs_fifo, - bool startNow ); + void setAudioDevice(AudioDevice* _dev, const struct qualitySettings& _qs, bool startNow); void storeAudioDevice(); void restoreAudioDevice(); inline AudioDevice * audioDev() @@ -278,11 +274,6 @@ class LMMS_EXPORT AudioEngine : public QObject bool criticalXRuns() const; - inline bool hasFifoWriter() const - { - return m_fifoWriter != nullptr; - } - void pushInputFrames( SampleFrame* _ab, const f_cnt_t _frames ); inline const SampleFrame* inputBuffer() @@ -295,10 +286,7 @@ class LMMS_EXPORT AudioEngine : public QObject return m_inputBufferFrames[ m_inputBufferRead ]; } - inline const SampleFrame* nextBuffer() - { - return hasFifoWriter() ? m_fifo->read() : renderNextBuffer(); - } + const SampleFrame* renderNextBuffer(); void changeQuality(const struct qualitySettings & qs); @@ -322,31 +310,10 @@ class LMMS_EXPORT AudioEngine : public QObject private: - using Fifo = FifoBuffer; - - class fifoWriter : public QThread - { - public: - fifoWriter( AudioEngine * audioEngine, Fifo * fifo ); - - void finish(); - - - private: - AudioEngine * m_audioEngine; - Fifo * m_fifo; - volatile bool m_writing; - - void run() override; - - void write(SampleFrame* buffer); - } ; - - AudioEngine( bool renderOnly ); ~AudioEngine() override; - void startProcessing(bool needsFifo = true); + void startProcessing(); void stopProcessing(); @@ -358,8 +325,6 @@ class LMMS_EXPORT AudioEngine : public QObject void renderStageEffects(); void renderStageMix(); - const SampleFrame* renderNextBuffer(); - void swapBuffers(); void clearInternal(); @@ -405,10 +370,6 @@ class LMMS_EXPORT AudioEngine : public QObject MidiClient * m_midiClient; QString m_midiClientName; - // FIFO stuff - Fifo * m_fifo; - fifoWriter * m_fifoWriter; - AudioEngineProfiler m_profiler; bool m_clearSignal; diff --git a/src/core/AudioEngine.cpp b/src/core/AudioEngine.cpp index 435fa38fa56..e73f67a3850 100644 --- a/src/core/AudioEngine.cpp +++ b/src/core/AudioEngine.cpp @@ -98,9 +98,6 @@ AudioEngine::AudioEngine( bool renderOnly ) : zeroSampleFrames(m_inputBuffer[i], m_inputBufferSize[i]); } - // determine FIFO size and number of frames per period - int fifoSize = 1; - // if not only rendering (that is, using the GUI), load the buffer // size from user configuration if( renderOnly == false ) @@ -124,14 +121,10 @@ AudioEngine::AudioEngine( bool renderOnly ) : // DEFAULT_BUFFER_SIZE, it's set to DEFAULT_BUFFER_SIZE and the rest is handled by an increased fifoSize. else if( m_framesPerPeriod > DEFAULT_BUFFER_SIZE ) { - fifoSize = m_framesPerPeriod / DEFAULT_BUFFER_SIZE; m_framesPerPeriod = DEFAULT_BUFFER_SIZE; } } - // allocte the FIFO from the determined size - m_fifo = new Fifo( fifoSize ); - // now that framesPerPeriod is fixed initialize global BufferManager BufferManager::init( m_framesPerPeriod ); @@ -167,12 +160,6 @@ AudioEngine::~AudioEngine() m_workers[w]->wait( 500 ); } - while( m_fifo->available() ) - { - delete[] m_fifo->read(); - } - delete m_fifo; - delete m_midiClient; delete m_audioDev; @@ -205,18 +192,8 @@ void AudioEngine::initDevices() -void AudioEngine::startProcessing(bool needsFifo) +void AudioEngine::startProcessing() { - if (needsFifo) - { - m_fifoWriter = new fifoWriter( this, m_fifo ); - m_fifoWriter->start( QThread::HighPriority ); - } - else - { - m_fifoWriter = nullptr; - } - m_audioDev->startProcessing(); } @@ -225,18 +202,7 @@ void AudioEngine::startProcessing(bool needsFifo) void AudioEngine::stopProcessing() { - if( m_fifoWriter != nullptr ) - { - m_fifoWriter->finish(); - m_fifoWriter->wait(); - m_audioDev->stopProcessing(); - delete m_fifoWriter; - m_fifoWriter = nullptr; - } - else - { - m_audioDev->stopProcessing(); - } + m_audioDev->stopProcessing(); } @@ -504,13 +470,7 @@ void AudioEngine::doSetAudioDevice( AudioDevice * _dev ) } } - - - -void AudioEngine::setAudioDevice(AudioDevice * _dev, - const struct qualitySettings & _qs, - bool _needs_fifo, - bool startNow) +void AudioEngine::setAudioDevice(AudioDevice* _dev, const struct qualitySettings& _qs, bool startNow) { stopProcessing(); @@ -521,7 +481,7 @@ void AudioEngine::setAudioDevice(AudioDevice * _dev, emit qualitySettingsChanged(); emit sampleRateChanged(); - if (startNow) {startProcessing( _needs_fifo );} + if (startNow) { startProcessing(); } } @@ -1073,49 +1033,4 @@ MidiClient * AudioEngine::tryMidiClients() return new MidiDummy; } - - - - - - - - -AudioEngine::fifoWriter::fifoWriter( AudioEngine* audioEngine, Fifo * fifo ) : - m_audioEngine( audioEngine ), - m_fifo( fifo ), - m_writing( true ) -{ - setObjectName("AudioEngine::fifoWriter"); -} - - - - -void AudioEngine::fifoWriter::finish() -{ - m_writing = false; -} - - - - -void AudioEngine::fifoWriter::run() -{ - disable_denormals(); - - const fpp_t frames = m_audioEngine->framesPerPeriod(); - while( m_writing ) - { - auto buffer = new SampleFrame[frames]; - const SampleFrame* b = m_audioEngine->renderNextBuffer(); - memcpy(buffer, b, frames * sizeof(SampleFrame)); - m_fifo->write(buffer); - } - - // Let audio backend stop processing - m_fifo->write(nullptr); - m_fifo->waitUntilRead(); -} - } // namespace lmms diff --git a/src/core/ProjectRenderer.cpp b/src/core/ProjectRenderer.cpp index c56c34068b4..f61e417c955 100644 --- a/src/core/ProjectRenderer.cpp +++ b/src/core/ProjectRenderer.cpp @@ -145,7 +145,7 @@ void ProjectRenderer::startProcessing() { // Have to do audio engine stuff with GUI-thread affinity in order to // make slots connected to sampleRateChanged()-signals being called immediately. - Engine::audioEngine()->setAudioDevice( m_fileDev, m_qualitySettings, false, false ); + Engine::audioEngine()->setAudioDevice(m_fileDev, m_qualitySettings, false); start( #ifndef LMMS_BUILD_WIN32 @@ -163,12 +163,12 @@ void ProjectRenderer::run() Engine::getSong()->startExport(); // Skip first empty buffer. - Engine::audioEngine()->nextBuffer(); + Engine::audioEngine()->renderNextBuffer(); m_progress = 0; // Now start processing - Engine::audioEngine()->startProcessing(false); + Engine::audioEngine()->startProcessing(); // Continually track and emit progress percentage to listeners. while (!Engine::getSong()->isExportDone() && !m_abort) diff --git a/src/core/audio/AudioDevice.cpp b/src/core/audio/AudioDevice.cpp index 2047fffe98b..095549f3ce8 100644 --- a/src/core/audio/AudioDevice.cpp +++ b/src/core/audio/AudioDevice.cpp @@ -58,37 +58,26 @@ void AudioDevice::processNextBuffer() { const fpp_t frames = getNextBuffer( m_buffer ); if (frames) { writeBuffer(m_buffer, frames); } - else - { - m_inProcess = false; - } } fpp_t AudioDevice::getNextBuffer(SampleFrame* _ab) { fpp_t frames = audioEngine()->framesPerPeriod(); - const SampleFrame* b = audioEngine()->nextBuffer(); + const SampleFrame* b = audioEngine()->renderNextBuffer(); if (!b) { return 0; } memcpy(_ab, b, frames * sizeof(SampleFrame)); - - if (audioEngine()->hasFifoWriter()) { delete[] b; } return frames; } - +void AudioDevice::startProcessing() +{ +} void AudioDevice::stopProcessing() { - if( audioEngine()->hasFifoWriter() ) - { - while( m_inProcess ) - { - processNextBuffer(); - } - } } From a85b1451cf8546d546dfe689c90cbe7be467bb11 Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 00:55:41 -0500 Subject: [PATCH 02/22] Add AudioEngine::renderNextBufferChunked and use it in SDL audio device --- include/AudioEngine.h | 3 +++ include/AudioSdl.h | 2 -- src/core/AudioEngine.cpp | 43 ++++++++++++++----------------------- src/core/audio/AudioSdl.cpp | 34 +++-------------------------- 4 files changed, 22 insertions(+), 60 deletions(-) diff --git a/include/AudioEngine.h b/include/AudioEngine.h index dc8ead8d1fb..1e0f01b5246 100644 --- a/include/AudioEngine.h +++ b/include/AudioEngine.h @@ -210,6 +210,7 @@ class LMMS_EXPORT AudioEngine : public QObject return m_framesPerPeriod; } + fpp_t userFramesPerPeriod() const { return m_userFramesPerPeriod; } AudioEngineProfiler& profiler() { @@ -287,6 +288,7 @@ class LMMS_EXPORT AudioEngine : public QObject } const SampleFrame* renderNextBuffer(); + void renderNextBufferChunked(SampleFrame* dst, std::size_t size); void changeQuality(const struct qualitySettings & qs); @@ -334,6 +336,7 @@ class LMMS_EXPORT AudioEngine : public QObject std::vector m_audioPorts; fpp_t m_framesPerPeriod; + fpp_t m_userFramesPerPeriod; SampleFrame* m_inputBuffer[2]; f_cnt_t m_inputBufferFrames[2]; diff --git a/include/AudioSdl.h b/include/AudioSdl.h index 651ed96be57..4c79c4b1a70 100644 --- a/include/AudioSdl.h +++ b/include/AudioSdl.h @@ -84,8 +84,6 @@ class AudioSdl : public AudioDevice SDL_AudioSpec m_audioHandle; - SampleFrame* m_outBuf; - size_t m_currentBufferFramePos; size_t m_currentBufferFramesCount; diff --git a/src/core/AudioEngine.cpp b/src/core/AudioEngine.cpp index e73f67a3850..13e753f9fbb 100644 --- a/src/core/AudioEngine.cpp +++ b/src/core/AudioEngine.cpp @@ -74,6 +74,7 @@ static thread_local bool s_renderingThread = false; AudioEngine::AudioEngine( bool renderOnly ) : m_renderOnly( renderOnly ), m_framesPerPeriod( DEFAULT_BUFFER_SIZE ), + m_userFramesPerPeriod(ConfigManager::inst()->value("audioengine", "framesperaudiobuffer").toInt()), m_baseSampleRate(std::max(ConfigManager::inst()->value("audioengine", "samplerate").toInt(), 44100)), m_inputBufferRead( 0 ), m_inputBufferWrite( 1 ), @@ -98,33 +99,6 @@ AudioEngine::AudioEngine( bool renderOnly ) : zeroSampleFrames(m_inputBuffer[i], m_inputBufferSize[i]); } - // if not only rendering (that is, using the GUI), load the buffer - // size from user configuration - if( renderOnly == false ) - { - m_framesPerPeriod = - ( fpp_t ) ConfigManager::inst()->value( "audioengine", "framesperaudiobuffer" ).toInt(); - - // if the value read from user configuration is not set or - // lower than the minimum allowed, use the default value and - // save it to the configuration - if( m_framesPerPeriod < MINIMUM_BUFFER_SIZE ) - { - ConfigManager::inst()->setValue( "audioengine", - "framesperaudiobuffer", - QString::number( DEFAULT_BUFFER_SIZE ) ); - - m_framesPerPeriod = DEFAULT_BUFFER_SIZE; - } - // lmms works with chunks of size DEFAULT_BUFFER_SIZE (256) and only the final mix will use the actual - // buffer size. Plugins don't see a larger buffer size than 256. If m_framesPerPeriod is larger than - // DEFAULT_BUFFER_SIZE, it's set to DEFAULT_BUFFER_SIZE and the rest is handled by an increased fifoSize. - else if( m_framesPerPeriod > DEFAULT_BUFFER_SIZE ) - { - m_framesPerPeriod = DEFAULT_BUFFER_SIZE; - } - } - // now that framesPerPeriod is fixed initialize global BufferManager BufferManager::init( m_framesPerPeriod ); @@ -382,7 +356,22 @@ const SampleFrame* AudioEngine::renderNextBuffer() return m_outputBufferRead.get(); } +void AudioEngine::renderNextBufferChunked(SampleFrame* dst, std::size_t size) +{ + auto buffer = static_cast(nullptr); + auto framesCopied = 0; + + while (framesCopied != size) + { + if (!buffer) { buffer = renderNextBuffer(); } + const auto numFramesToCopy = std::min(m_framesPerPeriod, size - framesCopied); + std::copy_n(buffer + (framesCopied % m_framesPerPeriod), numFramesToCopy, dst + framesCopied); + + framesCopied += numFramesToCopy; + if (framesCopied % m_framesPerPeriod == 0) { buffer = nullptr; } + } +} void AudioEngine::swapBuffers() diff --git a/src/core/audio/AudioSdl.cpp b/src/core/audio/AudioSdl.cpp index 8f533119c8a..f8a474cbcc3 100644 --- a/src/core/audio/AudioSdl.cpp +++ b/src/core/audio/AudioSdl.cpp @@ -43,8 +43,7 @@ constexpr auto PlaybackDeviceSDL = "device"; constexpr auto InputDeviceSDL = "inputdevice"; AudioSdl::AudioSdl( bool & _success_ful, AudioEngine* _audioEngine ) : - AudioDevice( DEFAULT_CHANNELS, _audioEngine ), - m_outBuf(new SampleFrame[audioEngine()->framesPerPeriod()]) + AudioDevice(DEFAULT_CHANNELS, _audioEngine) { _success_ful = false; @@ -63,7 +62,7 @@ AudioSdl::AudioSdl( bool & _success_ful, AudioEngine* _audioEngine ) : // to convert the buffers m_audioHandle.channels = channels(); - m_audioHandle.samples = std::max(f_cnt_t{1024}, audioEngine()->framesPerPeriod() * 2); + m_audioHandle.samples = _audioEngine->userFramesPerPeriod(); m_audioHandle.callback = sdlAudioCallback; m_audioHandle.userdata = this; @@ -130,8 +129,6 @@ AudioSdl::~AudioSdl() SDL_CloseAudioDevice(m_outputDevice); SDL_Quit(); - - delete[] m_outBuf; } @@ -183,32 +180,7 @@ void AudioSdl::sdlAudioCallback( Uint8 * _buf, int _len ) return; } - // SDL2: process float samples - while( _len ) - { - if( m_currentBufferFramePos == 0 ) - { - // frames depend on the sample rate - const fpp_t frames = getNextBuffer( m_outBuf ); - if( !frames ) - { - memset( _buf, 0, _len ); - return; - } - m_currentBufferFramesCount = frames; - - } - const uint min_frames_count = std::min(_len/sizeof(SampleFrame), - m_currentBufferFramesCount - - m_currentBufferFramePos); - - memcpy( _buf, m_outBuf + m_currentBufferFramePos, min_frames_count*sizeof(SampleFrame) ); - _buf += min_frames_count*sizeof(SampleFrame); - _len -= min_frames_count*sizeof(SampleFrame); - m_currentBufferFramePos += min_frames_count; - - m_currentBufferFramePos %= m_currentBufferFramesCount; - } + audioEngine()->renderNextBufferChunked(reinterpret_cast(_buf), _len / sizeof(SampleFrame)); } void AudioSdl::sdlInputAudioCallback(void *_udata, Uint8 *_buf, int _len) { From 4a3ca482ea942979520045983d17bda5eac935c4 Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 02:22:18 -0500 Subject: [PATCH 03/22] Use new chunking function in JACK audio device --- include/AudioJack.h | 5 ---- src/core/audio/AudioJack.cpp | 48 +++++------------------------------- 2 files changed, 6 insertions(+), 47 deletions(-) diff --git a/include/AudioJack.h b/include/AudioJack.h index 234f6ebf234..e131b6c1cd6 100644 --- a/include/AudioJack.h +++ b/include/AudioJack.h @@ -110,11 +110,6 @@ private slots: std::atomic m_midiClient; std::vector m_outputPorts; - jack_default_audio_sample_t** m_tempOutBufs; - SampleFrame* m_outBuf; - - f_cnt_t m_framesDoneInCurBuf; - f_cnt_t m_framesToDoInCurBuf; #ifdef AUDIO_PORT_SUPPORT struct StereoPort diff --git a/src/core/audio/AudioJack.cpp b/src/core/audio/AudioJack.cpp index bd5b8e514de..05ec6cc694f 100644 --- a/src/core/audio/AudioJack.cpp +++ b/src/core/audio/AudioJack.cpp @@ -55,10 +55,6 @@ AudioJack::AudioJack(bool& successful, AudioEngine* audioEngineParam) , m_client(nullptr) , m_active(false) , m_midiClient(nullptr) - , m_tempOutBufs(new jack_default_audio_sample_t*[channels()]) - , m_outBuf(new SampleFrame[audioEngine()->framesPerPeriod()]) - , m_framesDoneInCurBuf(0) - , m_framesToDoInCurBuf(0) { m_stopped = true; @@ -86,10 +82,6 @@ AudioJack::~AudioJack() if (m_active) { jack_deactivate(m_client); } jack_client_close(m_client); } - - delete[] m_tempOutBufs; - - delete[] m_outBuf; } @@ -299,11 +291,6 @@ int AudioJack::processCallback(jack_nframes_t nframes) m_midiClient.load()->JackMidiWrite(nframes); } - for (int c = 0; c < channels(); ++c) - { - m_tempOutBufs[c] = (jack_default_audio_sample_t*)jack_port_get_buffer(m_outputPorts[c], nframes); - } - #ifdef AUDIO_PORT_SUPPORT const int frames = std::min(nframes, audioEngine()->framesPerPeriod()); for (JackPortMap::iterator it = m_portMap.begin(); it != m_portMap.end(); ++it) @@ -321,38 +308,15 @@ int AudioJack::processCallback(jack_nframes_t nframes) } #endif - jack_nframes_t done = 0; - while (done < nframes && !m_stopped) - { - jack_nframes_t todo = std::min(nframes - done, m_framesToDoInCurBuf - m_framesDoneInCurBuf); - for (int c = 0; c < channels(); ++c) - { - jack_default_audio_sample_t* o = m_tempOutBufs[c]; - for (jack_nframes_t frame = 0; frame < todo; ++frame) - { - o[done + frame] = m_outBuf[m_framesDoneInCurBuf + frame][c]; - } - } - done += todo; - m_framesDoneInCurBuf += todo; - if (m_framesDoneInCurBuf == m_framesToDoInCurBuf) - { - m_framesToDoInCurBuf = getNextBuffer(m_outBuf); - m_framesDoneInCurBuf = 0; - if (!m_framesToDoInCurBuf) - { - m_stopped = true; - break; - } - } - } + static auto buf = std::vector(nframes); + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); - if (nframes != done) + for (int c = 0; c < channels(); ++c) { - for (int c = 0; c < channels(); ++c) + auto o = static_cast(jack_port_get_buffer(m_outputPorts[c], nframes)); + for (jack_nframes_t frame = 0; frame < nframes; ++frame) { - jack_default_audio_sample_t* b = m_tempOutBufs[c] + done; - memset(b, 0, sizeof(*b) * (nframes - done)); + o[frame] = buf[frame][c]; } } From e1938605116b3c5b99fd74c7c349b65da0a84c58 Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 02:28:52 -0500 Subject: [PATCH 04/22] Use new chunking function in OSS audio device --- src/core/audio/AudioOss.cpp | 23 ++++++----------------- 1 file changed, 6 insertions(+), 17 deletions(-) diff --git a/src/core/audio/AudioOss.cpp b/src/core/audio/AudioOss.cpp index bd427523520..d6382cb38dc 100644 --- a/src/core/audio/AudioOss.cpp +++ b/src/core/audio/AudioOss.cpp @@ -100,7 +100,7 @@ AudioOss::AudioOss( bool & _success_ful, AudioEngine* _audioEngine ) : int frag_spec; for (frag_spec = 0; - 1u << frag_spec < audioEngine()->framesPerPeriod() * channels() * BYTES_PER_INT_SAMPLE; + 1u << frag_spec < audioEngine()->userFramesPerPeriod() * channels() * BYTES_PER_INT_SAMPLE; ++frag_spec) { } @@ -255,26 +255,15 @@ void AudioOss::stopProcessing() void AudioOss::run() { - auto temp = new SampleFrame[audioEngine()->framesPerPeriod()]; - auto outbuf = new int_sample_t[audioEngine()->framesPerPeriod() * channels()]; + static auto buf = std::vector(audioEngine()->userFramesPerPeriod()); + static auto outbuf = std::vector(buf.size() * channels()); while( true ) { - const fpp_t frames = getNextBuffer( temp ); - if( !frames ) - { - break; - } - - int bytes = convertToS16(temp, frames, outbuf, m_convertEndian); - if( write( m_audioFD, outbuf, bytes ) != bytes ) - { - break; - } + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); + const auto bytes = convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); + if (write(m_audioFD, outbuf.data(), bytes) != bytes) { break; } } - - delete[] temp; - delete[] outbuf; } From 608a7d1c7920ea5ff8a72e05222203a698a12f93 Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 02:35:39 -0500 Subject: [PATCH 05/22] Use new chunking function in PortAudio device --- include/AudioPortAudio.h | 5 ---- src/core/audio/AudioPortAudio.cpp | 40 +++---------------------------- 2 files changed, 3 insertions(+), 42 deletions(-) diff --git a/include/AudioPortAudio.h b/include/AudioPortAudio.h index fbfa9b60dd6..554fbc769e2 100644 --- a/include/AudioPortAudio.h +++ b/include/AudioPortAudio.h @@ -145,11 +145,6 @@ class AudioPortAudio : public AudioDevice PaStreamParameters m_inputParameters; bool m_wasPAInitError; - - SampleFrame* m_outBuf; - std::size_t m_outBufPos; - fpp_t m_outBufSize; - bool m_stopped; } ; diff --git a/src/core/audio/AudioPortAudio.cpp b/src/core/audio/AudioPortAudio.cpp index eb5058bc6b5..29e459d6cb9 100644 --- a/src/core/audio/AudioPortAudio.cpp +++ b/src/core/audio/AudioPortAudio.cpp @@ -66,14 +66,10 @@ AudioPortAudio::AudioPortAudio( bool & _success_ful, AudioEngine * _audioEngine DEFAULT_CHANNELS, DEFAULT_CHANNELS), _audioEngine), m_paStream( nullptr ), - m_wasPAInitError( false ), - m_outBuf(new SampleFrame[audioEngine()->framesPerPeriod()]), - m_outBufPos( 0 ) + m_wasPAInitError(false) { _success_ful = false; - m_outBufSize = audioEngine()->framesPerPeriod(); - PaError err = Pa_Initialize(); if( err != paNoError ) { @@ -123,7 +119,7 @@ AudioPortAudio::AudioPortAudio( bool & _success_ful, AudioEngine * _audioEngine //inLatency = Pa_GetDeviceInfo( inDevIdx )->defaultLowInputLatency; //outLatency = Pa_GetDeviceInfo( outDevIdx )->defaultLowOutputLatency; - const int samples = audioEngine()->framesPerPeriod(); + const int samples = audioEngine()->userFramesPerPeriod(); // Configure output parameters. m_outputParameters.device = outDevIdx; @@ -193,7 +189,6 @@ AudioPortAudio::~AudioPortAudio() { Pa_Terminate(); } - delete[] m_outBuf; } @@ -243,36 +238,7 @@ int AudioPortAudio::process_callback(const float* _inputBuffer, float* _outputBu return paComplete; } - while( _framesPerBuffer ) - { - if( m_outBufPos == 0 ) - { - // frames depend on the sample rate - const fpp_t frames = getNextBuffer( m_outBuf ); - if( !frames ) - { - m_stopped = true; - memset( _outputBuffer, 0, _framesPerBuffer * - channels() * sizeof(float) ); - return paComplete; - } - m_outBufSize = frames; - } - const auto min_len = std::min(_framesPerBuffer, m_outBufSize - m_outBufPos); - - for( fpp_t frame = 0; frame < min_len; ++frame ) - { - for( ch_cnt_t chnl = 0; chnl < channels(); ++chnl ) - { - (_outputBuffer + frame * channels())[chnl] = AudioEngine::clip(m_outBuf[frame][chnl]); - } - } - - _outputBuffer += min_len * channels(); - _framesPerBuffer -= min_len; - m_outBufPos += min_len; - m_outBufPos %= m_outBufSize; - } + audioEngine()->renderNextBufferChunked(reinterpret_cast(_outputBuffer), _framesPerBuffer); return paContinue; } From 0a956c6788455046a8efbaacb8cd9643e4363078 Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 02:47:58 -0500 Subject: [PATCH 06/22] Use new chunking function in PulseAudio device --- src/core/audio/AudioPulseAudio.cpp | 32 +++++++----------------------- 1 file changed, 7 insertions(+), 25 deletions(-) diff --git a/src/core/audio/AudioPulseAudio.cpp b/src/core/audio/AudioPulseAudio.cpp index aa434487154..6241b0d3e7d 100644 --- a/src/core/audio/AudioPulseAudio.cpp +++ b/src/core/audio/AudioPulseAudio.cpp @@ -160,7 +160,7 @@ static void context_state_callback(pa_context *c, void *userdata) buffer_attr.minreq = (uint32_t)(-1); buffer_attr.fragsize = (uint32_t)(-1); - double latency = (double)( Engine::audioEngine()->framesPerPeriod() ) / (double)_this->sampleRate(); + double latency = static_cast(Engine::audioEngine()->userFramesPerPeriod()) / _this->sampleRate(); // ask PulseAudio for the desired latency (which might not be approved) buffer_attr.tlength = pa_usec_to_bytes( latency * PA_USEC_PER_MSEC, @@ -229,7 +229,7 @@ void AudioPulseAudio::run() } else { - const fpp_t fpp = audioEngine()->framesPerPeriod(); + const fpp_t fpp = audioEngine()->userFramesPerPeriod(); auto temp = new SampleFrame[fpp]; while( getNextBuffer( temp ) ) { @@ -248,30 +248,12 @@ void AudioPulseAudio::run() void AudioPulseAudio::streamWriteCallback( pa_stream *s, size_t length ) { - const fpp_t fpp = audioEngine()->framesPerPeriod(); - auto temp = new SampleFrame[fpp]; - auto pcmbuf = (int_sample_t*)pa_xmalloc(fpp * channels() * sizeof(int_sample_t)); + static auto buf = std::vector(audioEngine()->userFramesPerPeriod()); + static auto pcmbuf = std::vector(buf.size() * channels() * sizeof(int_sample_t)); - size_t fd = 0; - while( fd < length/4 && m_quit == false ) - { - const fpp_t frames = getNextBuffer( temp ); - if( !frames ) - { - m_quit = true; - break; - } - int bytes = convertToS16(temp, frames, pcmbuf, m_convertEndian); - if( bytes > 0 ) - { - pa_stream_write( m_s, pcmbuf, bytes, nullptr, 0, - PA_SEEK_RELATIVE ); - } - fd += frames; - } - - pa_xfree( pcmbuf ); - delete[] temp; + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); + const auto bytes = convertToS16(buf.data(), buf.size(), pcmbuf.data(), m_convertEndian); + if (bytes > 0) { pa_stream_write(m_s, pcmbuf.data(), bytes, nullptr, 0, PA_SEEK_RELATIVE); } } From 850a8427c43b757769d0edce0e930c58df78df34 Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 08:47:18 -0500 Subject: [PATCH 07/22] Use new chunking function in ALSA device --- src/core/audio/AudioAlsa.cpp | 73 ++++-------------------------------- 1 file changed, 8 insertions(+), 65 deletions(-) diff --git a/src/core/audio/AudioAlsa.cpp b/src/core/audio/AudioAlsa.cpp index c41703e2271..09319bde69f 100644 --- a/src/core/audio/AudioAlsa.cpp +++ b/src/core/audio/AudioAlsa.cpp @@ -242,74 +242,17 @@ void AudioAlsa::stopProcessing() void AudioAlsa::run() { - auto temp = new SampleFrame[audioEngine()->framesPerPeriod()]; - auto outbuf = new int_sample_t[audioEngine()->framesPerPeriod() * channels()]; - auto pcmbuf = new int_sample_t[m_periodSize * channels()]; + static auto buf = std::vector(m_periodSize); + static auto outbuf = std::vector(buf.size() * channels()); - int outbuf_size = audioEngine()->framesPerPeriod() * channels(); - int outbuf_pos = 0; - int pcmbuf_size = m_periodSize * channels(); - - bool quit = false; - while( quit == false ) + while (true) { - int_sample_t * ptr = pcmbuf; - int len = pcmbuf_size; - while( len ) - { - if( outbuf_pos == 0 ) - { - // frames depend on the sample rate - const fpp_t frames = getNextBuffer( temp ); - if( !frames ) - { - quit = true; - memset( ptr, 0, len - * sizeof( int_sample_t ) ); - break; - } - outbuf_size = frames * channels(); - - convertToS16(temp, frames, outbuf, m_convertEndian); - } - int min_len = std::min(len, outbuf_size - outbuf_pos); - memcpy( ptr, outbuf + outbuf_pos, - min_len * sizeof( int_sample_t ) ); - ptr += min_len; - len -= min_len; - outbuf_pos += min_len; - outbuf_pos %= outbuf_size; - } - - f_cnt_t frames = m_periodSize; - ptr = pcmbuf; + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); + convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); - while( frames ) - { - int err = snd_pcm_writei( m_handle, ptr, frames ); - - if( err == -EAGAIN ) - { - continue; - } - - if( err < 0 ) - { - if( handleError( err ) < 0 ) - { - printf( "Write error: %s\n", - snd_strerror( err ) ); - } - break; // skip this buffer - } - ptr += err * channels(); - frames -= err; - } + const auto err = snd_pcm_writei(m_handle, outbuf.data(), buf.size()); + if (handleError(err) < 0) { printf("Write error: %s\n", snd_strerror(err)); } } - - delete[] temp; - delete[] outbuf; - delete[] pcmbuf; } @@ -369,7 +312,7 @@ int AudioAlsa::setHWParams( const ch_cnt_t _channels, snd_pcm_access_t _access ) } } - m_periodSize = audioEngine()->framesPerPeriod(); + m_periodSize = audioEngine()->userFramesPerPeriod(); m_bufferSize = m_periodSize * 8; int dir; if (int err = snd_pcm_hw_params_set_period_size_near(m_handle, m_hwParams, &m_periodSize, &dir); err < 0) From 25802da3093770b9944ddd488ab536e8492c602d Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 08:52:25 -0500 Subject: [PATCH 08/22] Use new chunking function in sndio device --- src/core/audio/AudioSndio.cpp | 25 +++++++------------------ 1 file changed, 7 insertions(+), 18 deletions(-) diff --git a/src/core/audio/AudioSndio.cpp b/src/core/audio/AudioSndio.cpp index 95fab473a79..35586c306cb 100644 --- a/src/core/audio/AudioSndio.cpp +++ b/src/core/audio/AudioSndio.cpp @@ -74,7 +74,7 @@ AudioSndio::AudioSndio(bool & _success_ful, AudioEngine * _audioEngine) : m_par.bits = 16; m_par.le = SIO_LE_NATIVE; m_par.rate = sampleRate(); - m_par.round = audioEngine()->framesPerPeriod(); + m_par.round = audioEngine()->userFramesPerPeriod(); m_par.appbufsz = m_par.round * 2; if ( (isLittleEndian() && (m_par.le == 0)) || @@ -141,26 +141,15 @@ void AudioSndio::stopProcessing() void AudioSndio::run() { - SampleFrame* temp = new SampleFrame[audioEngine()->framesPerPeriod()]; - int_sample_t * outbuf = new int_sample_t[audioEngine()->framesPerPeriod() * channels()]; + static auto buf = std::vector(audioEngine()->userFramesPerPeriod()); + static auto outbuf = std::vector(buf.size() * channels()); - while( true ) + while (true) { - const fpp_t frames = getNextBuffer( temp ); - if( !frames ) - { - break; - } - - uint bytes = convertToS16(temp, frames, outbuf, m_convertEndian); - if( sio_write( m_hdl, outbuf, bytes ) != bytes ) - { - break; - } + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); + convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); + sio_write(m_hdl, outbuf.data(), outbuf.size()); } - - delete[] temp; - delete[] outbuf; } From 663ca7eb699933df1cd7c7938b6f089477d41d3e Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 10:20:56 -0500 Subject: [PATCH 09/22] Use new chunking function in soundio device --- include/AudioSoundIo.h | 5 ---- src/core/audio/AudioSoundIo.cpp | 52 +++++++++------------------------ 2 files changed, 13 insertions(+), 44 deletions(-) diff --git a/include/AudioSoundIo.h b/include/AudioSoundIo.h index 1260951bffc..4115ffd3362 100644 --- a/include/AudioSoundIo.h +++ b/include/AudioSoundIo.h @@ -110,11 +110,6 @@ class AudioSoundIo : public AudioDevice SoundIo *m_soundio; SoundIoOutStream *m_outstream; - SampleFrame* m_outBuf; - int m_outBufSize; - fpp_t m_outBufFramesTotal; - fpp_t m_outBufFrameIndex; - bool m_stopped; bool m_outstreamStarted; diff --git a/src/core/audio/AudioSoundIo.cpp b/src/core/audio/AudioSoundIo.cpp index c7fa380e402..4c589fd0d39 100644 --- a/src/core/audio/AudioSoundIo.cpp +++ b/src/core/audio/AudioSoundIo.cpp @@ -47,10 +47,7 @@ AudioSoundIo::AudioSoundIo( bool & outSuccessful, AudioEngine * _audioEngine ) : outSuccessful = false; m_soundio = nullptr; m_outstream = nullptr; - m_outBuf = nullptr; m_disconnectErr = 0; - m_outBufFrameIndex = 0; - m_outBufFramesTotal = 0; m_stopped = true; m_outstreamStarted = false; @@ -165,7 +162,7 @@ AudioSoundIo::AudioSoundIo( bool & outSuccessful, AudioEngine * _audioEngine ) : } m_outstream->name = "LMMS"; - m_outstream->software_latency = (double)audioEngine()->framesPerPeriod() / (double)currentSampleRate; + m_outstream->software_latency = static_cast(audioEngine()->userFramesPerPeriod()) / currentSampleRate; m_outstream->userdata = this; m_outstream->write_callback = staticWriteCallback; m_outstream->error_callback = staticErrorCallback; @@ -209,12 +206,6 @@ AudioSoundIo::~AudioSoundIo() void AudioSoundIo::startProcessing() { - m_outBufFrameIndex = 0; - m_outBufFramesTotal = 0; - m_outBufSize = audioEngine()->framesPerPeriod(); - - m_outBuf = new SampleFrame[m_outBufSize]; - if (! m_outstreamStarted) { if (int err = soundio_outstream_start(m_outstream)) @@ -250,12 +241,6 @@ void AudioSoundIo::stopProcessing() soundio_strerror(err)); } } - - if (m_outBuf) - { - delete[] m_outBuf; - m_outBuf = nullptr; - } } void AudioSoundIo::errorCallback(int err) @@ -270,11 +255,10 @@ void AudioSoundIo::underflowCallback() void AudioSoundIo::writeCallback(int frameCountMin, int frameCountMax) { - if (m_stopped) {return;} - const struct SoundIoChannelLayout *layout = &m_outstream->layout; - SoundIoChannelArea* areas; - int bytesPerSample = m_outstream->bytes_per_sample; - int framesLeft = frameCountMax; + const auto layout = &m_outstream->layout; + + auto areas = static_cast(nullptr); + auto framesLeft = frameCountMax; while (framesLeft > 0) { @@ -288,37 +272,27 @@ void AudioSoundIo::writeCallback(int frameCountMin, int frameCountMax) if (!frameCount) break; - if (m_stopped) { for (int channel = 0; channel < layout->channel_count; ++channel) { - memset(areas[channel].ptr, 0, bytesPerSample * frameCount); + memset(areas[channel].ptr, 0, m_outstream->bytes_per_sample * frameCount); areas[channel].ptr += areas[channel].step * frameCount; } continue; } - for (int frame = 0; frame < frameCount; frame += 1) - { - if (m_outBufFrameIndex >= m_outBufFramesTotal) - { - m_outBufFramesTotal = getNextBuffer(m_outBuf); - if (m_outBufFramesTotal == 0) - { - m_stopped = true; - break; - } - m_outBufFrameIndex = 0; - } + auto buf = std::vector(frameCount); + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); - for (int channel = 0; channel < layout->channel_count; channel += 1) + for (const auto& frame : buf) + { + for (auto channel = 0; channel < layout->channel_count; ++channel) { - float sample = m_outBuf[m_outBufFrameIndex][channel]; - memcpy(areas[channel].ptr, &sample, bytesPerSample); + const auto sample = frame[channel]; + memcpy(areas[channel].ptr, &sample, m_outstream->bytes_per_sample); areas[channel].ptr += areas[channel].step; } - m_outBufFrameIndex += 1; } if (int err = soundio_outstream_end_write(m_outstream)) From 17a1d3439d21adc18e35ac8cede1b7cf9d81b87a Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 10:35:15 -0500 Subject: [PATCH 10/22] Remove unused getNextBuffer function --- include/AudioDevice.h | 7 ------- src/core/audio/AudioDevice.cpp | 21 ++++----------------- src/core/audio/AudioPulseAudio.cpp | 9 --------- 3 files changed, 4 insertions(+), 33 deletions(-) diff --git a/include/AudioDevice.h b/include/AudioDevice.h index 942b4bd8098..b7b049f2120 100644 --- a/include/AudioDevice.h +++ b/include/AudioDevice.h @@ -91,9 +91,6 @@ class AudioDevice // processNextBuffer() virtual void writeBuffer(const SampleFrame* /* _buf*/, const fpp_t /*_frames*/) {} - // called by according driver for fetching new sound-data - fpp_t getNextBuffer(SampleFrame* _ab); - // convert a given audio-buffer to a buffer in signed 16-bit samples // returns num of bytes in outbuf int convertToS16(const SampleFrame* _ab, @@ -126,11 +123,7 @@ class AudioDevice sample_rate_t m_sampleRate; ch_cnt_t m_channels; AudioEngine* m_audioEngine; - QMutex m_devMutex; - - SampleFrame* m_buffer; - }; } // namespace lmms diff --git a/src/core/audio/AudioDevice.cpp b/src/core/audio/AudioDevice.cpp index 095549f3ce8..f6bfaa0ac1e 100644 --- a/src/core/audio/AudioDevice.cpp +++ b/src/core/audio/AudioDevice.cpp @@ -36,8 +36,7 @@ AudioDevice::AudioDevice( const ch_cnt_t _channels, AudioEngine* _audioEngine ) m_supportsCapture( false ), m_sampleRate( _audioEngine->outputSampleRate() ), m_channels( _channels ), - m_audioEngine( _audioEngine ), - m_buffer(new SampleFrame[audioEngine()->framesPerPeriod()]) + m_audioEngine(_audioEngine) { } @@ -46,7 +45,6 @@ AudioDevice::AudioDevice( const ch_cnt_t _channels, AudioEngine* _audioEngine ) AudioDevice::~AudioDevice() { - delete[] m_buffer; m_devMutex.tryLock(); unlock(); } @@ -56,22 +54,11 @@ AudioDevice::~AudioDevice() void AudioDevice::processNextBuffer() { - const fpp_t frames = getNextBuffer( m_buffer ); - if (frames) { writeBuffer(m_buffer, frames); } + // TODO: This function can probably be in AudioFileDevice instead + const auto buffer = m_audioEngine->renderNextBuffer(); + writeBuffer(buffer, m_audioEngine->framesPerPeriod()); } -fpp_t AudioDevice::getNextBuffer(SampleFrame* _ab) -{ - fpp_t frames = audioEngine()->framesPerPeriod(); - const SampleFrame* b = audioEngine()->renderNextBuffer(); - - if (!b) { return 0; } - - memcpy(_ab, b, frames * sizeof(SampleFrame)); - return frames; -} - - void AudioDevice::startProcessing() { } diff --git a/src/core/audio/AudioPulseAudio.cpp b/src/core/audio/AudioPulseAudio.cpp index 6241b0d3e7d..7fb8e28d5f5 100644 --- a/src/core/audio/AudioPulseAudio.cpp +++ b/src/core/audio/AudioPulseAudio.cpp @@ -227,15 +227,6 @@ void AudioPulseAudio::run() pa_stream_disconnect( m_s ); pa_stream_unref( m_s ); } - else - { - const fpp_t fpp = audioEngine()->userFramesPerPeriod(); - auto temp = new SampleFrame[fpp]; - while( getNextBuffer( temp ) ) - { - } - delete[] temp; - } pa_context_disconnect( context ); pa_context_unref( context ); From 24a2a28572a7f63471a51102ecaa89f7d434c53d Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 17 Dec 2024 10:40:47 -0500 Subject: [PATCH 11/22] Minor changes --- src/core/AudioEngine.cpp | 2 +- src/core/audio/AudioPulseAudio.cpp | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/core/AudioEngine.cpp b/src/core/AudioEngine.cpp index 13e753f9fbb..cd691130f8e 100644 --- a/src/core/AudioEngine.cpp +++ b/src/core/AudioEngine.cpp @@ -359,7 +359,7 @@ const SampleFrame* AudioEngine::renderNextBuffer() void AudioEngine::renderNextBufferChunked(SampleFrame* dst, std::size_t size) { auto buffer = static_cast(nullptr); - auto framesCopied = 0; + auto framesCopied = std::size_t{0}; while (framesCopied != size) { diff --git a/src/core/audio/AudioPulseAudio.cpp b/src/core/audio/AudioPulseAudio.cpp index 7fb8e28d5f5..a913376e6bd 100644 --- a/src/core/audio/AudioPulseAudio.cpp +++ b/src/core/audio/AudioPulseAudio.cpp @@ -240,11 +240,11 @@ void AudioPulseAudio::run() void AudioPulseAudio::streamWriteCallback( pa_stream *s, size_t length ) { static auto buf = std::vector(audioEngine()->userFramesPerPeriod()); - static auto pcmbuf = std::vector(buf.size() * channels() * sizeof(int_sample_t)); + static auto outbuf = std::vector(buf.size() * channels() * sizeof(int_sample_t)); audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); - const auto bytes = convertToS16(buf.data(), buf.size(), pcmbuf.data(), m_convertEndian); - if (bytes > 0) { pa_stream_write(m_s, pcmbuf.data(), bytes, nullptr, 0, PA_SEEK_RELATIVE); } + const auto bytes = convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); + if (bytes > 0) { pa_stream_write(m_s, outbuf.data(), bytes, nullptr, 0, PA_SEEK_RELATIVE); } } From 2dbe458fcd4452d08a14bfbf695b7f3cb22fec97 Mon Sep 17 00:00:00 2001 From: saker Date: Wed, 18 Dec 2024 07:20:38 -0500 Subject: [PATCH 12/22] Make renderNextBufferChunked persist buffers across calls to avoid dropped frames --- src/core/AudioEngine.cpp | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/src/core/AudioEngine.cpp b/src/core/AudioEngine.cpp index cd691130f8e..19d58afc38c 100644 --- a/src/core/AudioEngine.cpp +++ b/src/core/AudioEngine.cpp @@ -358,18 +358,19 @@ const SampleFrame* AudioEngine::renderNextBuffer() void AudioEngine::renderNextBufferChunked(SampleFrame* dst, std::size_t size) { - auto buffer = static_cast(nullptr); - auto framesCopied = std::size_t{0}; + static auto buffer = std::vector(m_framesPerPeriod); + static auto index = 0; + auto framesCopied = std::size_t{0}; while (framesCopied != size) { - if (!buffer) { buffer = renderNextBuffer(); } + if (index == 0) { std::copy_n(renderNextBuffer(), m_framesPerPeriod, buffer.begin()); } - const auto numFramesToCopy = std::min(m_framesPerPeriod, size - framesCopied); - std::copy_n(buffer + (framesCopied % m_framesPerPeriod), numFramesToCopy, dst + framesCopied); + const auto numFramesToCopy = std::min(m_framesPerPeriod - index, size - framesCopied); + std::copy_n(buffer.begin() + index, numFramesToCopy, dst + framesCopied); framesCopied += numFramesToCopy; - if (framesCopied % m_framesPerPeriod == 0) { buffer = nullptr; } + index = (index + numFramesToCopy) % m_framesPerPeriod; } } From 30db2ef3ca036ca485347212a543d4dc176592df Mon Sep 17 00:00:00 2001 From: saker Date: Wed, 18 Dec 2024 07:21:17 -0500 Subject: [PATCH 13/22] Make some style changes in AudioSoundIo --- src/core/audio/AudioSoundIo.cpp | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/core/audio/AudioSoundIo.cpp b/src/core/audio/AudioSoundIo.cpp index 4c589fd0d39..5e535765b9f 100644 --- a/src/core/audio/AudioSoundIo.cpp +++ b/src/core/audio/AudioSoundIo.cpp @@ -256,7 +256,6 @@ void AudioSoundIo::underflowCallback() void AudioSoundIo::writeCallback(int frameCountMin, int frameCountMax) { const auto layout = &m_outstream->layout; - auto areas = static_cast(nullptr); auto framesLeft = frameCountMax; @@ -269,8 +268,7 @@ void AudioSoundIo::writeCallback(int frameCountMin, int frameCountMax) return; } - if (!frameCount) - break; + if (!frameCount) { break; } if (m_stopped) { From 0a0122e6bc6aa1c52bf9999221a3cbbe4cda2cb6 Mon Sep 17 00:00:00 2001 From: saker Date: Wed, 18 Dec 2024 07:26:01 -0500 Subject: [PATCH 14/22] Remove redundancy in AudioOss --- src/core/audio/AudioOss.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/core/audio/AudioOss.cpp b/src/core/audio/AudioOss.cpp index d6382cb38dc..003139d79b4 100644 --- a/src/core/audio/AudioOss.cpp +++ b/src/core/audio/AudioOss.cpp @@ -258,11 +258,11 @@ void AudioOss::run() static auto buf = std::vector(audioEngine()->userFramesPerPeriod()); static auto outbuf = std::vector(buf.size() * channels()); - while( true ) + while (true) { audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); const auto bytes = convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); - if (write(m_audioFD, outbuf.data(), bytes) != bytes) { break; } + write(m_audioFD, outbuf.data(), bytes); } } From b2f3579a4d8c06dec3327c4ebf056ee3168095e2 Mon Sep 17 00:00:00 2001 From: saker Date: Wed, 18 Dec 2024 07:56:08 -0500 Subject: [PATCH 15/22] Check for result from write call again --- src/core/audio/AudioOss.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/core/audio/AudioOss.cpp b/src/core/audio/AudioOss.cpp index 003139d79b4..eb6145df058 100644 --- a/src/core/audio/AudioOss.cpp +++ b/src/core/audio/AudioOss.cpp @@ -262,7 +262,7 @@ void AudioOss::run() { audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); const auto bytes = convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); - write(m_audioFD, outbuf.data(), bytes); + if (write(m_audioFD, outbuf.data(), bytes) != bytes) { break; } } } From bf890325262a634493d0ca0fdf4c11971c6d58e1 Mon Sep 17 00:00:00 2001 From: saker Date: Wed, 18 Dec 2024 08:30:07 -0500 Subject: [PATCH 16/22] Consider if the audio device has stopped for JACK devices Removed this functionality by accident --- src/core/audio/AudioJack.cpp | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/src/core/audio/AudioJack.cpp b/src/core/audio/AudioJack.cpp index 05ec6cc694f..2b19d75f680 100644 --- a/src/core/audio/AudioJack.cpp +++ b/src/core/audio/AudioJack.cpp @@ -308,6 +308,17 @@ int AudioJack::processCallback(jack_nframes_t nframes) } #endif + if (m_stopped) + { + for (int c = 0; c < channels(); ++c) + { + auto o = static_cast(jack_port_get_buffer(m_outputPorts[c], nframes)); + std::fill_n(o, 0, nframes); + } + + return 0; + } + static auto buf = std::vector(nframes); audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); From e4fc133872f59750c3d6f76c5a3448959c436f92 Mon Sep 17 00:00:00 2001 From: saker Date: Wed, 18 Dec 2024 09:15:06 -0500 Subject: [PATCH 17/22] Consider if the audio device has stopped --- include/AudioAlsa.h | 1 + include/AudioOss.h | 1 + include/AudioPulseAudio.h | 5 +++-- include/AudioSndio.h | 1 + src/core/audio/AudioAlsa.cpp | 4 ++++ src/core/audio/AudioJack.cpp | 2 +- src/core/audio/AudioOss.cpp | 4 ++++ src/core/audio/AudioPulseAudio.cpp | 13 ++++++------- src/core/audio/AudioSndio.cpp | 8 ++++++-- 9 files changed, 27 insertions(+), 12 deletions(-) diff --git a/include/AudioAlsa.h b/include/AudioAlsa.h index 92c47a6aceb..c5ab1eb244f 100644 --- a/include/AudioAlsa.h +++ b/include/AudioAlsa.h @@ -100,6 +100,7 @@ class AudioAlsa : public QThread, public AudioDevice snd_pcm_sw_params_t * m_swParams; bool m_convertEndian; + std::atomic m_stopped; } ; diff --git a/include/AudioOss.h b/include/AudioOss.h index 91d45607344..8fe81bcc427 100644 --- a/include/AudioOss.h +++ b/include/AudioOss.h @@ -84,6 +84,7 @@ class setupWidget : public gui::AudioDeviceSetupWidget int m_audioFD; bool m_convertEndian; + std::atomic m_stopped; } ; diff --git a/include/AudioPulseAudio.h b/include/AudioPulseAudio.h index db3c566bf53..b080f556350 100644 --- a/include/AudioPulseAudio.h +++ b/include/AudioPulseAudio.h @@ -90,11 +90,12 @@ class AudioPulseAudio : public QThread, public AudioDevice void stopProcessing() override; void run() override; - volatile bool m_quit; - bool m_convertEndian; bool m_connected; + + std::atomic m_stopped; + QSemaphore m_connectedSemaphore; } ; diff --git a/include/AudioSndio.h b/include/AudioSndio.h index beb4913eb4b..6088c64bf17 100644 --- a/include/AudioSndio.h +++ b/include/AudioSndio.h @@ -81,6 +81,7 @@ class AudioSndio : public QThread, public AudioDevice struct sio_par m_par; bool m_convertEndian; + std::atomic m_stopped; } ; diff --git a/src/core/audio/AudioAlsa.cpp b/src/core/audio/AudioAlsa.cpp index 09319bde69f..9d807271870 100644 --- a/src/core/audio/AudioAlsa.cpp +++ b/src/core/audio/AudioAlsa.cpp @@ -228,6 +228,7 @@ void AudioAlsa::startProcessing() { if( !isRunning() ) { + m_stopped = false; start( QThread::HighPriority ); } } @@ -237,6 +238,7 @@ void AudioAlsa::startProcessing() void AudioAlsa::stopProcessing() { + m_stopped = true; stopProcessingThread( this ); } @@ -247,6 +249,8 @@ void AudioAlsa::run() while (true) { + if (m_stopped) { break; } + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); diff --git a/src/core/audio/AudioJack.cpp b/src/core/audio/AudioJack.cpp index 2b19d75f680..d002bc8a348 100644 --- a/src/core/audio/AudioJack.cpp +++ b/src/core/audio/AudioJack.cpp @@ -313,7 +313,7 @@ int AudioJack::processCallback(jack_nframes_t nframes) for (int c = 0; c < channels(); ++c) { auto o = static_cast(jack_port_get_buffer(m_outputPorts[c], nframes)); - std::fill_n(o, 0, nframes); + memset(o, 0, nframes); } return 0; diff --git a/src/core/audio/AudioOss.cpp b/src/core/audio/AudioOss.cpp index eb6145df058..b594186e924 100644 --- a/src/core/audio/AudioOss.cpp +++ b/src/core/audio/AudioOss.cpp @@ -241,6 +241,7 @@ void AudioOss::startProcessing() { if( !isRunning() ) { + m_stopped = false; start( QThread::HighPriority ); } } @@ -250,6 +251,7 @@ void AudioOss::startProcessing() void AudioOss::stopProcessing() { + m_stopped = true; stopProcessingThread( this ); } @@ -260,6 +262,8 @@ void AudioOss::run() while (true) { + if (m_stopped) { break; } + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); const auto bytes = convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); if (write(m_audioFD, outbuf.data(), bytes) != bytes) { break; } diff --git a/src/core/audio/AudioPulseAudio.cpp b/src/core/audio/AudioPulseAudio.cpp index a913376e6bd..d7407a23162 100644 --- a/src/core/audio/AudioPulseAudio.cpp +++ b/src/core/audio/AudioPulseAudio.cpp @@ -51,7 +51,6 @@ AudioPulseAudio::AudioPulseAudio( bool & _success_ful, AudioEngine* _audioEngin DEFAULT_CHANNELS, DEFAULT_CHANNELS), _audioEngine), m_s( nullptr ), - m_quit( false ), m_convertEndian( false ) { _success_ful = false; @@ -95,6 +94,7 @@ void AudioPulseAudio::startProcessing() { if( !isRunning() ) { + m_stopped = false; start( QThread::HighPriority ); } } @@ -104,7 +104,7 @@ void AudioPulseAudio::startProcessing() void AudioPulseAudio::stopProcessing() { - m_quit = true; + m_stopped = true; stopProcessingThread( this ); } @@ -218,11 +218,8 @@ void AudioPulseAudio::run() if( m_connected ) { int ret = 0; - m_quit = false; - while( m_quit == false - && pa_mainloop_iterate( mainLoop, 1, &ret ) >= 0 ) - { - } + m_stopped = false; + while (!m_stopped && pa_mainloop_iterate(mainLoop, 1, &ret) >= 0) {} pa_stream_disconnect( m_s ); pa_stream_unref( m_s ); @@ -242,6 +239,8 @@ void AudioPulseAudio::streamWriteCallback( pa_stream *s, size_t length ) static auto buf = std::vector(audioEngine()->userFramesPerPeriod()); static auto outbuf = std::vector(buf.size() * channels() * sizeof(int_sample_t)); + if (m_stopped) { return; } + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); const auto bytes = convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); if (bytes > 0) { pa_stream_write(m_s, outbuf.data(), bytes, nullptr, 0, PA_SEEK_RELATIVE); } diff --git a/src/core/audio/AudioSndio.cpp b/src/core/audio/AudioSndio.cpp index 35586c306cb..10350791225 100644 --- a/src/core/audio/AudioSndio.cpp +++ b/src/core/audio/AudioSndio.cpp @@ -129,6 +129,7 @@ void AudioSndio::startProcessing() { if( !isRunning() ) { + m_stopped = false; start( QThread::HighPriority ); } } @@ -136,6 +137,7 @@ void AudioSndio::startProcessing() void AudioSndio::stopProcessing() { + m_stopped = true; stopProcessingThread( this ); } @@ -146,9 +148,11 @@ void AudioSndio::run() while (true) { + if (m_stopped) { break; } + audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); - convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); - sio_write(m_hdl, outbuf.data(), outbuf.size()); + const auto bytes = convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); + if (sio_write(m_hdl, outbuf.data(), outbuf.size()) != bytes) { break; } } } From ecf952359f0b7b254dfc6f5d6f8c410085d2da1e Mon Sep 17 00:00:00 2001 From: saker Date: Wed, 18 Dec 2024 22:23:16 -0500 Subject: [PATCH 18/22] Cast bytes to std::size_t --- src/core/audio/AudioSndio.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/core/audio/AudioSndio.cpp b/src/core/audio/AudioSndio.cpp index 10350791225..583c4851507 100644 --- a/src/core/audio/AudioSndio.cpp +++ b/src/core/audio/AudioSndio.cpp @@ -152,7 +152,7 @@ void AudioSndio::run() audioEngine()->renderNextBufferChunked(buf.data(), buf.size()); const auto bytes = convertToS16(buf.data(), buf.size(), outbuf.data(), m_convertEndian); - if (sio_write(m_hdl, outbuf.data(), outbuf.size()) != bytes) { break; } + if (sio_write(m_hdl, outbuf.data(), outbuf.size()) != static_cast(bytes)) { break; } } } From a03bd179b014451922e8f18a652bd3430a70516a Mon Sep 17 00:00:00 2001 From: saker Date: Tue, 24 Dec 2024 11:05:14 -0500 Subject: [PATCH 19/22] Avoid copying of rendered buffer in renderNextBufferChunked --- src/core/AudioEngine.cpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/core/AudioEngine.cpp b/src/core/AudioEngine.cpp index 19d58afc38c..982f0bd5790 100644 --- a/src/core/AudioEngine.cpp +++ b/src/core/AudioEngine.cpp @@ -358,16 +358,16 @@ const SampleFrame* AudioEngine::renderNextBuffer() void AudioEngine::renderNextBufferChunked(SampleFrame* dst, std::size_t size) { - static auto buffer = std::vector(m_framesPerPeriod); + static auto buffer = static_cast(nullptr); static auto index = 0; auto framesCopied = std::size_t{0}; while (framesCopied != size) { - if (index == 0) { std::copy_n(renderNextBuffer(), m_framesPerPeriod, buffer.begin()); } + if (index == 0) { buffer = renderNextBuffer(); } const auto numFramesToCopy = std::min(m_framesPerPeriod - index, size - framesCopied); - std::copy_n(buffer.begin() + index, numFramesToCopy, dst + framesCopied); + std::copy_n(buffer + index, numFramesToCopy, dst + framesCopied); framesCopied += numFramesToCopy; index = (index + numFramesToCopy) % m_framesPerPeriod; From 885cb32ee9f2b2d4d147c78cbbfff321315706af Mon Sep 17 00:00:00 2001 From: saker Date: Wed, 25 Dec 2024 15:31:43 -0500 Subject: [PATCH 20/22] Remove processNextBuffer function and call writeBuffer directly --- include/AudioDevice.h | 6 ------ include/AudioFileDevice.h | 1 + include/AudioSampleRecorder.h | 2 +- src/core/ProjectRenderer.cpp | 5 ++++- src/core/audio/AudioDevice.cpp | 9 --------- 5 files changed, 6 insertions(+), 17 deletions(-) diff --git a/include/AudioDevice.h b/include/AudioDevice.h index b7b049f2120..91634ff181b 100644 --- a/include/AudioDevice.h +++ b/include/AudioDevice.h @@ -81,16 +81,10 @@ class AudioDevice return m_channels; } - void processNextBuffer(); - virtual void startProcessing(); virtual void stopProcessing(); protected: - // subclasses can re-implement this for being used in conjunction with - // processNextBuffer() - virtual void writeBuffer(const SampleFrame* /* _buf*/, const fpp_t /*_frames*/) {} - // convert a given audio-buffer to a buffer in signed 16-bit samples // returns num of bytes in outbuf int convertToS16(const SampleFrame* _ab, diff --git a/include/AudioFileDevice.h b/include/AudioFileDevice.h index dc9a786a4a8..41ef64ae920 100644 --- a/include/AudioFileDevice.h +++ b/include/AudioFileDevice.h @@ -49,6 +49,7 @@ class AudioFileDevice : public AudioDevice OutputSettings const & getOutputSettings() const { return m_outputSettings; } + virtual void writeBuffer(const SampleFrame* buf, const fpp_t frames) = 0; protected: int writeData( const void* data, int len ); diff --git a/include/AudioSampleRecorder.h b/include/AudioSampleRecorder.h index 691196be62c..402cbb8909a 100644 --- a/include/AudioSampleRecorder.h +++ b/include/AudioSampleRecorder.h @@ -48,7 +48,7 @@ class AudioSampleRecorder : public AudioDevice std::shared_ptr createSampleBuffer(); private: - void writeBuffer(const SampleFrame* _ab, const fpp_t _frames) override; + void writeBuffer(const SampleFrame* _ab, const fpp_t _frames); using BufferList = QList>; BufferList m_buffers; diff --git a/src/core/ProjectRenderer.cpp b/src/core/ProjectRenderer.cpp index f61e417c955..cdb3f1d3920 100644 --- a/src/core/ProjectRenderer.cpp +++ b/src/core/ProjectRenderer.cpp @@ -173,7 +173,10 @@ void ProjectRenderer::run() // Continually track and emit progress percentage to listeners. while (!Engine::getSong()->isExportDone() && !m_abort) { - m_fileDev->processNextBuffer(); + const auto buffer = Engine::audioEngine()->renderNextBuffer(); + const auto framesPerPeriod = Engine::audioEngine()->framesPerPeriod(); + m_fileDev->writeBuffer(buffer, framesPerPeriod); + const int nprog = Engine::getSong()->getExportProgress(); if (m_progress != nprog) { diff --git a/src/core/audio/AudioDevice.cpp b/src/core/audio/AudioDevice.cpp index f6bfaa0ac1e..2b05092ec00 100644 --- a/src/core/audio/AudioDevice.cpp +++ b/src/core/audio/AudioDevice.cpp @@ -50,15 +50,6 @@ AudioDevice::~AudioDevice() } - - -void AudioDevice::processNextBuffer() -{ - // TODO: This function can probably be in AudioFileDevice instead - const auto buffer = m_audioEngine->renderNextBuffer(); - writeBuffer(buffer, m_audioEngine->framesPerPeriod()); -} - void AudioDevice::startProcessing() { } From 3a72faab0684023d0446a8b791a3648c3829d5b7 Mon Sep 17 00:00:00 2001 From: saker Date: Fri, 27 Dec 2024 02:07:26 -0500 Subject: [PATCH 21/22] Restore functionality to render audio within AudioDummy --- include/AudioDummy.h | 1 + 1 file changed, 1 insertion(+) diff --git a/include/AudioDummy.h b/include/AudioDummy.h index 17826f89ddb..d922a1009d2 100644 --- a/include/AudioDummy.h +++ b/include/AudioDummy.h @@ -93,6 +93,7 @@ class AudioDummy : public QThread, public AudioDevice MicroTimer timer; while( true ) { + audioEngine()->renderNextBuffer(); timer.reset(); const int microseconds = static_cast( audioEngine()->framesPerPeriod() * 1000000.0f / audioEngine()->outputSampleRate() - timer.elapsed() ); if( microseconds > 0 ) From c5a16a7107edc0f9fe604c48c27e2bc418bcd2b5 Mon Sep 17 00:00:00 2001 From: saker Date: Sat, 28 Dec 2024 11:16:36 -0500 Subject: [PATCH 22/22] Add stopped variable in AudioDummy --- include/AudioDummy.h | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/include/AudioDummy.h b/include/AudioDummy.h index d922a1009d2..f0b08676314 100644 --- a/include/AudioDummy.h +++ b/include/AudioDummy.h @@ -80,18 +80,20 @@ class AudioDummy : public QThread, public AudioDevice private: void startProcessing() override { + m_stopped = false; start(); } void stopProcessing() override { + m_stopped = true; stopProcessingThread( this ); } void run() override { MicroTimer timer; - while( true ) + while (!m_stopped) { audioEngine()->renderNextBuffer(); timer.reset(); @@ -103,6 +105,7 @@ class AudioDummy : public QThread, public AudioDevice } } + std::atomic m_stopped; } ; } // namespace lmms