ladybird/Tests/LibAudio/TestPlaybackStream.cpp
Zaggy1024 bb156f8133 LibAudio: Add a Serenity implementation of PlaybackStream
This implementation is very naive compared to the PulseAudio one.

Instead of using a callback implemented by the audio server connection
to push audio to the buffer, we have to poll on a timer to check when
we need to push the audio buffers. Implementing cross-process condition
variables into the audio queue class could allow us to avoid polling,
which may prove beneficial to CPU usage.

Audio timestamps will be accurate to the number of samples available,
but will count in increments of about 100ms and run ahead of the actual
audio being pushed to the device by the server.

Buffer underruns are completely ignored for now as well, since the
`AudioServer` has no way to know how many samples are actually written
in a single audio buffer.
2023-08-29 11:31:10 +02:00

59 lines
1.8 KiB
C++

/*
* Copyright (c) 2023, Gregory Bertilson <zaggy1024@gmail.com>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <AK/Math.h>
#include <AK/MemoryStream.h>
#include <AK/WeakPtr.h>
#include <LibAudio/PlaybackStream.h>
#include <LibCore/EventLoop.h>
#include <LibTest/TestSuite.h>
#include <unistd.h>
#if defined(HAVE_PULSEAUDIO)
# include <LibAudio/PulseAudioWrappers.h>
#endif
// FIXME: CI doesn't run an AudioServer currently. Creating one in /etc/SystemServer.ini does not
// allow this test to pass since CI runs in a Shell that will setsid() if it finds that the
// current session ID is 0, and AudioServer's socket address depends on the current sid.
// If we can fix that, this test can run on CI.
// https://github.com/SerenityOS/serenity/issues/20538
#if defined(AK_OS_SERENITY)
# define STREAM_TEST BENCHMARK_CASE
#else
# define STREAM_TEST TEST_CASE
#endif
STREAM_TEST(create_and_destroy_playback_stream)
{
Core::EventLoop event_loop;
bool has_implementation = false;
#if defined(AK_OS_SERENITY) || defined(HAVE_PULSEAUDIO)
has_implementation = true;
#endif
{
auto stream_result = Audio::PlaybackStream::create(Audio::OutputState::Playing, 44100, 2, 100, [](Bytes buffer, Audio::PcmSampleFormat format, size_t sample_count) -> ReadonlyBytes {
VERIFY(format == Audio::PcmSampleFormat::Float32);
FixedMemoryStream writing_stream { buffer };
for (size_t i = 0; i < sample_count; i++) {
MUST(writing_stream.write_value(0.0f));
MUST(writing_stream.write_value(0.0f));
}
return buffer.trim(writing_stream.offset());
});
EXPECT_EQ(!stream_result.is_error(), has_implementation);
usleep(10000);
}
#if defined(HAVE_PULSEAUDIO)
VERIFY(!Audio::PulseAudioContext::weak_instance());
#endif
}