From 9c8d9e54fe96f97bdda2be26bab571e4cbf0c597 Mon Sep 17 00:00:00 2001 From: ouwou <26526779+ouwou@users.noreply.github.com> Date: Fri, 2 Sep 2022 00:38:59 -0400 Subject: handle multiple speakers properly --- src/audio/manager.cpp | 49 +++++++++++++++++++++++++------------------------ src/audio/manager.hpp | 12 ++++-------- 2 files changed, 29 insertions(+), 32 deletions(-) (limited to 'src/audio') diff --git a/src/audio/manager.cpp b/src/audio/manager.cpp index e686a82..9ba8c51 100644 --- a/src/audio/manager.cpp +++ b/src/audio/manager.cpp @@ -9,8 +9,6 @@ #include #include -#define BUFFER_SAMPLES 4800 - const uint8_t *StripRTPExtensionHeader(const uint8_t *buf, int num_bytes, size_t &outlen) { if (buf[0] == 0xbe && buf[1] == 0xde && num_bytes > 4) { uint64_t offset = 4 + 4 * ((buf[2] << 8) | buf[3]); @@ -25,19 +23,24 @@ const uint8_t *StripRTPExtensionHeader(const uint8_t *buf, int num_bytes, size_t void data_callback(ma_device *pDevice, void *pOutput, const void *pInput, ma_uint32 frameCount) { AudioManager *mgr = reinterpret_cast(pDevice->pUserData); if (mgr == nullptr) return; - std::lock_guard _(mgr->m_dumb_mutex); + std::lock_guard _(mgr->m_mutex); - const auto buffered_frames = std::min(static_cast(mgr->m_dumb.size() / 2), frameCount); - auto *pOutputCast = static_cast(pOutput); - std::copy(mgr->m_dumb.begin(), mgr->m_dumb.begin() + buffered_frames * 2, pOutputCast); - mgr->m_dumb.erase(mgr->m_dumb.begin(), mgr->m_dumb.begin() + buffered_frames * 2); + auto *pOutputF32 = static_cast(pOutput); + for (auto &[ssrc, pair] : mgr->m_sources) { + auto &buf = pair.first; + const size_t n = std::min(buf.size(), frameCount * 2ULL); + for (size_t i = 0; i < n; i++) { + pOutputF32[i] += buf[i] / 32768.F; + } + buf.erase(buf.begin(), buf.begin() + n); + } } AudioManager::AudioManager() { m_ok = true; m_device_config = ma_device_config_init(ma_device_type_playback); - m_device_config.playback.format = ma_format_s16; + m_device_config.playback.format = ma_format_f32; m_device_config.playback.channels = 2; m_device_config.sampleRate = 48000; m_device_config.dataCallback = data_callback; @@ -55,33 +58,31 @@ AudioManager::AudioManager() { m_ok = false; return; } - - int err; - m_opus_decoder = opus_decoder_create(48000, 2, &err); - - m_active = true; } AudioManager::~AudioManager() { - m_active = false; ma_device_uninit(&m_device); + for (auto &[ssrc, pair] : m_sources) { + opus_decoder_destroy(pair.second); + } } -void AudioManager::FeedMeOpus(const std::vector &data) { +void AudioManager::FeedMeOpus(uint32_t ssrc, const std::vector &data) { size_t payload_size = 0; const auto *opus_encoded = StripRTPExtensionHeader(data.data(), static_cast(data.size()), payload_size); static std::array pcm; - int decoded = opus_decode(m_opus_decoder, opus_encoded, static_cast(payload_size), pcm.data(), 120 * 48, 0); + if (m_sources.find(ssrc) == m_sources.end()) { + int err; + auto *decoder = opus_decoder_create(48000, 2, &err); + m_sources.insert(std::make_pair(ssrc, std::make_pair(std::deque {}, decoder))); + } + int decoded = opus_decode(m_sources.at(ssrc).second, opus_encoded, static_cast(payload_size), pcm.data(), 120 * 48, 0); if (decoded <= 0) { - printf("failed decode: %d\n", decoded); } else { - m_buffer.insert(m_buffer.end(), pcm.begin(), pcm.begin() + decoded * 2); - if (m_buffer.size() >= BUFFER_SAMPLES * 2) { - m_dumb_mutex.lock(); - m_dumb.insert(m_dumb.end(), m_buffer.begin(), m_buffer.begin() + BUFFER_SAMPLES * 2); - m_dumb_mutex.unlock(); - m_buffer.erase(m_buffer.begin(), m_buffer.begin() + BUFFER_SAMPLES * 2); - } + m_mutex.lock(); + auto &buf = m_sources.at(ssrc).first; + buf.insert(buf.end(), pcm.begin(), pcm.begin() + decoded * 2); + m_mutex.unlock(); } } diff --git a/src/audio/manager.hpp b/src/audio/manager.hpp index 27e973b..55ecdc3 100644 --- a/src/audio/manager.hpp +++ b/src/audio/manager.hpp @@ -4,6 +4,7 @@ #include #include #include +#include #include #include #include @@ -13,25 +14,20 @@ public: AudioManager(); ~AudioManager(); - void FeedMeOpus(const std::vector &data); + void FeedMeOpus(uint32_t ssrc, const std::vector &data); [[nodiscard]] bool OK() const; private: friend void data_callback(ma_device *, void *, const void *, ma_uint32); - std::atomic m_active; std::thread m_thread; bool m_ok; - ma_engine m_engine; ma_device m_device; ma_device_config m_device_config; - std::mutex m_dumb_mutex; - std::deque m_dumb; - std::deque m_buffer; - - OpusDecoder *m_opus_decoder; + std::mutex m_mutex; + std::unordered_map, OpusDecoder *>> m_sources; }; -- cgit v1.2.3