From dc42e4f38e731cda0651540b4629f3e196792a85 Mon Sep 17 00:00:00 2001 From: 2 * r + 2 * t <61896496+soramanew@users.noreply.github.com> Date: Mon, 8 Sep 2025 23:12:25 +1000 Subject: plugin/ac: use double buffer Double buffer instead of circular buffer --- plugin/src/Caelestia/audiocollector.cpp | 80 ++++++++++++--------------------- 1 file changed, 28 insertions(+), 52 deletions(-) (limited to 'plugin/src/Caelestia/audiocollector.cpp') diff --git a/plugin/src/Caelestia/audiocollector.cpp b/plugin/src/Caelestia/audiocollector.cpp index cf3a673..036a383 100644 --- a/plugin/src/Caelestia/audiocollector.cpp +++ b/plugin/src/Caelestia/audiocollector.cpp @@ -2,12 +2,10 @@ #include "service.hpp" #include -#include #include #include #include #include -#include #include #include #include @@ -157,13 +155,14 @@ unsigned int PipeWireWorker::nextPowerOf2(unsigned int n) { return n; } -AudioCollector::AudioCollector(uint32_t sampleRate, uint32_t chunkSize, uint32_t bufferSize, QObject* parent) +AudioCollector::AudioCollector(uint32_t sampleRate, uint32_t chunkSize, QObject* parent) : Service(parent) - , m_buffer(bufferSize, 0.0f) - , m_bufferIndex(bufferSize) + , m_buffer1(chunkSize) + , m_buffer2(chunkSize) + , m_readBuffer(&m_buffer1) + , m_writeBuffer(&m_buffer2) , m_sampleRate(sampleRate) - , m_chunkSize(chunkSize) - , m_bufferSize(bufferSize) {} + , m_chunkSize(chunkSize) {} AudioCollector::~AudioCollector() { stop(); @@ -185,72 +184,49 @@ uint32_t AudioCollector::chunkSize() const { return m_chunkSize; } -uint32_t AudioCollector::bufferSize() const { - return m_bufferSize; -} - void AudioCollector::clearBuffer() { - std::lock_guard lock(m_bufferMutex); - std::fill(m_buffer.begin(), m_buffer.end(), 0.0f); - m_bufferIndex = m_bufferSize; + auto* writeBuffer = m_writeBuffer.load(std::memory_order_relaxed); + std::fill(writeBuffer->begin(), writeBuffer->end(), 0.0f); + + auto* oldRead = m_readBuffer.exchange(writeBuffer, std::memory_order_acq_rel); + m_writeBuffer.store(oldRead, std::memory_order_release); } void AudioCollector::loadChunk(const int16_t* samples, uint32_t count) { - std::lock_guard lock(m_bufferMutex); - - while (count > 0) { - const auto spaceToEnd = m_bufferSize - m_bufferIndex; - const auto toCopy = (count < spaceToEnd) ? count : spaceToEnd; + if (count > m_chunkSize) { + count = m_chunkSize; + } - std::transform(samples, samples + toCopy, m_buffer.begin() + m_bufferIndex, [](int16_t sample) { - return sample / 32768.0f; - }); + auto* writeBuffer = m_writeBuffer.load(std::memory_order_relaxed); + std::transform(samples, samples + count, writeBuffer->begin(), [](int16_t sample) { + return sample / 32768.0f; + }); - m_bufferIndex = (m_bufferIndex + toCopy) % m_bufferSize; - samples += toCopy; - count -= toCopy; - } + auto* oldRead = m_readBuffer.exchange(writeBuffer, std::memory_order_acq_rel); + m_writeBuffer.store(oldRead, std::memory_order_release); } uint32_t AudioCollector::readChunk(float* out, uint32_t count) { - std::lock_guard lock(m_bufferMutex); - - if (count == 0 || count > m_bufferSize) { - count = m_bufferSize; + if (count == 0 || count > m_chunkSize) { + count = m_chunkSize; } - const auto start = (m_bufferIndex + m_bufferSize - count) % m_bufferSize; - const auto firstChunk = std::min(count, m_bufferSize - start); - - std::copy(m_buffer.begin() + start, m_buffer.begin() + start + firstChunk, out); - - if (firstChunk < count) { - std::copy(m_buffer.begin(), m_buffer.begin() + (count - firstChunk), out + firstChunk); - } + auto* readBuffer = m_readBuffer.load(std::memory_order_acquire); + std::memcpy(out, readBuffer->data(), count * sizeof(float)); return count; } uint32_t AudioCollector::readChunk(double* out, uint32_t count) { - std::lock_guard lock(m_bufferMutex); - - if (count == 0 || count > m_bufferSize) { - count = m_bufferSize; + if (count == 0 || count > m_chunkSize) { + count = m_chunkSize; } - const auto start = (m_bufferIndex + m_bufferSize - count) % m_bufferSize; - const auto firstChunk = std::min(count, m_bufferSize - start); - - std::transform(m_buffer.begin() + start, m_buffer.begin() + start + firstChunk, out, [](float sample) { + auto* readBuffer = m_readBuffer.load(std::memory_order_acquire); + std::transform(readBuffer->begin(), readBuffer->begin() + count, out, [](float sample) { return static_cast(sample); }); - if (firstChunk < count) { - std::transform(m_buffer.begin(), m_buffer.begin() + (count - firstChunk), out + firstChunk, [](float sample) { - return static_cast(sample); - }); - } - return count; } -- cgit v1.2.3-freya