/* * Copyright (c) 2021, kleines Filmröllchen * Copyright (c) 2021, JJ Roberts-White * * SPDX-License-Identifier: BSD-2-Clause */ #include "AudioPlayerLoop.h" #include "Music.h" #include "TrackManager.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include struct AudioLoopDeferredInvoker final : public IPC::DeferredInvoker { static constexpr size_t INLINE_FUNCTIONS = 4; virtual ~AudioLoopDeferredInvoker() = default; virtual void schedule(Function function) override { deferred_functions.append(move(function)); } void run_functions() { if (deferred_functions.size() > INLINE_FUNCTIONS) dbgln("Warning: Audio loop has more than {} deferred functions, audio might glitch!", INLINE_FUNCTIONS); while (!deferred_functions.is_empty()) { auto function = deferred_functions.take_last(); function(); } } Vector, INLINE_FUNCTIONS> deferred_functions; }; AudioPlayerLoop::AudioPlayerLoop(TrackManager& track_manager, Atomic& need_to_write_wav, Atomic& wav_percent_written, Threading::MutexProtected& wav_writer) : m_track_manager(track_manager) , m_buffer(FixedArray::must_create_but_fixme_should_propagate_errors(sample_count)) , m_pipeline_thread(Threading::Thread::construct([this]() { return this->pipeline_thread_main(); }, "Audio pipeline"sv)) , m_need_to_write_wav(need_to_write_wav) , m_wav_percent_written(wav_percent_written) , m_wav_writer(wav_writer) { m_audio_client = Audio::ConnectionToServer::try_create().release_value_but_fixme_should_propagate_errors(); auto target_sample_rate = m_audio_client->get_sample_rate(); if (target_sample_rate == 0) target_sample_rate = Music::sample_rate; m_resampler = Audio::ResampleHelper(Music::sample_rate, target_sample_rate); MUST(m_pipeline_thread->set_priority(sched_get_priority_max(0))); m_pipeline_thread->start(); } AudioPlayerLoop::~AudioPlayerLoop() { // Tell the pipeline to exit and wait for the last audio cycle to finish. m_exit_requested.store(true); auto result = m_pipeline_thread->join(); // FIXME: Get rid of the EINVAL/ESRCH check once we allow to join dead threads. VERIFY(!result.is_error() || result.error() == EINVAL || result.error() == ESRCH); m_audio_client->shutdown(); } intptr_t AudioPlayerLoop::pipeline_thread_main() { m_audio_client->set_deferred_invoker(make()); auto& deferred_invoker = static_cast(m_audio_client->deferred_invoker()); m_audio_client->async_start_playback(); while (!m_exit_requested.load()) { deferred_invoker.run_functions(); // The track manager guards against allocations itself. m_track_manager.fill_buffer(m_buffer); auto result = send_audio_to_server(); // Tolerate errors in the audio pipeline; we don't want this thread to crash the program. This might likely happen with OOM. if (result.is_error()) [[unlikely]] { dbgln("Error in audio pipeline: {}", result.error()); m_track_manager.reset(); } write_wav_if_needed(); } m_audio_client->async_pause_playback(); return static_cast(0); } ErrorOr AudioPlayerLoop::send_audio_to_server() { TRY(m_resampler->try_resample_into_end(m_remaining_samples, m_buffer)); auto sample_rate = static_cast(m_resampler->target()); auto buffer_play_time_ns = 1'000'000'000.0 / (sample_rate / static_cast(Audio::AUDIO_BUFFER_SIZE)); auto good_sleep_time = Time::from_nanoseconds(static_cast(buffer_play_time_ns)).to_timespec(); size_t start_of_chunk_to_write = 0; while (start_of_chunk_to_write + Audio::AUDIO_BUFFER_SIZE <= m_remaining_samples.size()) { auto const exact_chunk = m_remaining_samples.span().slice(start_of_chunk_to_write, Audio::AUDIO_BUFFER_SIZE); auto exact_chunk_array = Array::from_span(exact_chunk); TRY(m_audio_client->blocking_realtime_enqueue(exact_chunk_array, [&]() { nanosleep(&good_sleep_time, nullptr); })); start_of_chunk_to_write += Audio::AUDIO_BUFFER_SIZE; } m_remaining_samples.remove(0, start_of_chunk_to_write); VERIFY(m_remaining_samples.size() < Audio::AUDIO_BUFFER_SIZE); return {}; } void AudioPlayerLoop::write_wav_if_needed() { bool _true = true; if (m_need_to_write_wav.compare_exchange_strong(_true, false)) { m_audio_client->async_pause_playback(); m_wav_writer.with_locked([this](auto& wav_writer) { m_track_manager.reset(); m_track_manager.set_should_loop(false); do { // FIXME: This progress detection is crude, but it works for now. m_wav_percent_written.store(static_cast(static_cast(m_track_manager.transport()->time()) / roll_length * 100.0f)); m_track_manager.fill_buffer(m_buffer); wav_writer.write_samples(m_buffer.span()); } while (m_track_manager.transport()->time()); // FIXME: Make sure that the new TrackManager APIs aren't as bad. m_wav_percent_written.store(100); m_track_manager.reset(); m_track_manager.set_should_loop(true); wav_writer.finalize(); }); m_audio_client->async_start_playback(); } } void AudioPlayerLoop::toggle_paused() { m_should_play_audio = !m_should_play_audio; if (m_should_play_audio) m_audio_client->async_start_playback(); else m_audio_client->async_pause_playback(); }