blob: 529827234ad9110d8de6c18b8a7845b875b0c9c1 [file] [log] [blame]
// Copyright 2022 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <fuchsia/media/cpp/fidl.h>
#include <zircon/device/audio.h>
#include <cmath>
#include <optional>
#include <string>
#include <utility>
#include <vector>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include "src/media/audio/audio_core/device_id.h"
#include "src/media/audio/audio_core/testing/integration/hermetic_audio_realm.h"
#include "src/media/audio/audio_core/testing/integration/hermetic_audio_test.h"
#include "src/media/audio/audio_core/testing/integration/virtual_device.h"
#include "src/media/audio/lib/format/audio_buffer.h"
#include "src/media/audio/lib/format/format.h"
#include "src/media/audio/lib/format/traits.h"
#include "src/virtualization/tests/lib/enclosed_guest.h"
namespace {
using ::fuchsia::media::AudioSampleFormat;
using ::media::audio::AudioBuffer;
using ::media::audio::AudioBufferSlice;
using ::media::audio::Format;
using ::media::audio::SampleFormatTraits;
using ::media::audio::test::HermeticAudioRealm;
using ::media::audio::test::HermeticAudioTest;
using ::media::audio::test::VirtualOutput;
constexpr int32_t kOutputFrameRate = 48000;
constexpr int32_t kStereoChannelCount = 2;
constexpr AudioSampleFormat kSampleFormat = AudioSampleFormat::FLOAT;
constexpr audio_stream_unique_id_t kOutputId = AUDIO_STREAM_UNIQUE_ID_BUILTIN_SPEAKERS;
// TODO(https://fxbug.dev/42168790): Consider creating a `virtio_audio_test_util` that directly
// communicates with ALSA instead, to have better control over the output buffer.
constexpr char kAplayBinPath[] = "/tmp/vm_extras/aplay";
// TODO(https://fxbug.dev/42168790): Consider creating a `virtio_audio_test_util` that directly
// generates audio files on-the-fly.
constexpr char kTestFilePath[] = "/tmp/extras/stereo_ramp_48khz_16bit.wav";
constexpr int32_t kRampFrameCount = 65536;
constexpr int32_t kZeroPaddingFrameCount = 1024;
template <class GuestType>
class VirtioSoundGuestTest : public HermeticAudioTest {
protected:
void SetUp() override {
GuestLaunchInfo guest_launch_info;
enclosed_guest_.emplace(dispatcher(),
[this](fit::function<bool()> condition, zx::duration timeout) {
return RunLoopWithTimeoutOrUntil(std::move(condition), timeout);
});
HermeticAudioTest::SetTestSuiteRealmOptions([this, &guest_launch_info] {
return HermeticAudioRealm::Options{
.customize_realm = [this, &guest_launch_info](
::component_testing::RealmBuilder& realm_builder) -> zx_status_t {
auto status = enclosed_guest_->BuildLaunchInfo(&guest_launch_info);
if (status != ZX_OK) {
return status;
}
enclosed_guest_->InstallInRealm(realm_builder.root(), guest_launch_info);
using component_testing::ChildRef;
using component_testing::Protocol;
using component_testing::Route;
realm_builder.AddRoute(Route{
.capabilities = {Protocol{"fuchsia.media.Audio"}},
.source = ChildRef{HermeticAudioRealm::kAudioCore},
.targets = {ChildRef{"guest_manager"}},
});
return ZX_OK;
},
};
});
// Create the realm and start audio services.
HermeticAudioTest::SetUp();
// Now start the guest.
auto services =
std::make_unique<sys::ServiceDirectory>(realm().realm_root().component().CloneExposedDir());
ASSERT_EQ(enclosed_guest_->LaunchInRealm(std::move(services), guest_launch_info,
zx::time::infinite()),
ZX_OK)
<< "Failed to launch guest";
const auto format =
Format::Create<kSampleFormat>(kStereoChannelCount, kOutputFrameRate).take_value();
// Add some padding to ensure that there is enough headroom in the ring buffer.
output_ = CreateOutput(kOutputId, format,
kRampFrameCount + kZeroPaddingFrameCount + 10 * kOutputFrameRate);
}
void TearDown() override {
EXPECT_EQ(enclosed_guest_->Stop(zx::time::infinite()), ZX_OK);
enclosed_guest_.reset();
if constexpr (kEnableAllOverflowAndUnderflowChecksInRealtimeTests) {
ExpectNoOverflowsOrUnderflows();
}
HermeticAudioTest::TearDown();
}
zx_status_t Execute(const std::vector<std::string>& argv) {
return enclosed_guest_->Execute(argv, {}, zx::time::infinite(), nullptr, nullptr);
}
std::optional<int64_t> GetFirstNonSilentFrame(const AudioBuffer<kSampleFormat>& buffer) const {
for (int64_t frame = 0; frame < buffer.NumFrames(); ++frame) {
if (buffer.SampleAt(frame, 0) != SampleFormatTraits<kSampleFormat>::kSilentValue) {
return frame;
}
}
return std::nullopt;
}
AudioBuffer<kSampleFormat> GetOutputRingBuffer() { return output_->SnapshotRingBuffer(); }
bool OutputHasUnderflows() {
return DeviceHasUnderflows(media::audio::DeviceUniqueIdToString(kOutputId));
}
private:
std::optional<GuestType> enclosed_guest_;
VirtualOutput<kSampleFormat>* output_ = nullptr;
};
// We only support `TerminaEnclosedGuest` since the tests require `virtio-sound` and `alsa-lib`.
TYPED_TEST_SUITE(VirtioSoundGuestTest, ::testing::Types<TerminaEnclosedGuest>,
GuestTestNameGenerator);
TYPED_TEST(VirtioSoundGuestTest, OutputFidelity) {
// The input audio file consists of a stereo linear ramp that covers the entire 16-bit range of
// values with opposing direction in each channel. It is calculated for each frame as follows:
// `buffer[frame][0] = 0x7FFF - frame`
// `buffer[frame][1] = -0x8000 + frame`
// Note that the file begins with `kZeroPaddingFrameCount` frames of zeros, in order to compensate
// for the initial gain ramp. This is followed by `kRampFrameCount` ramp frames (described above).
ASSERT_EQ(this->Execute({kAplayBinPath, kTestFilePath}), ZX_OK);
const auto ring_buffer = this->GetOutputRingBuffer();
ASSERT_EQ(ring_buffer.format().channels(), kStereoChannelCount);
// TODO(https://fxbug.dev/42160300): Remove workarounds when underflow conditions are fixed.
if (this->OutputHasUnderflows()) {
GTEST_SKIP() << "Skipping fidelity checks due to underflows";
__builtin_unreachable();
}
const std::optional<int64_t> start_frame = this->GetFirstNonSilentFrame(ring_buffer);
ASSERT_TRUE(start_frame.has_value()) << "Could not find non-silent frame";
FX_LOGS(INFO) << "First non-silent frame: " << *start_frame;
const int64_t end_frame = *start_frame + kRampFrameCount;
ASSERT_LE(end_frame, ring_buffer.NumFrames()) << "Not enough frames in ring buffer";
const auto buffer_slice = AudioBufferSlice(&ring_buffer, *start_frame, end_frame);
// TODO(https://fxbug.dev/42177076): Temporarily limit the end frame to
// `24000 - kZeroPaddingFrameCount`, until the buffer repetition issue is resolved.
// When the issue is fixed, this would be replaced by `kRampFrameCount`.
for (int32_t frame = 0; frame < 24000 - kZeroPaddingFrameCount; ++frame) {
EXPECT_FLOAT_EQ(buffer_slice.SampleAt(frame, 0),
static_cast<float>(0x7FFF - frame) / static_cast<float>(kRampFrameCount / 2))
<< "at (" << frame << ", 0)";
EXPECT_FLOAT_EQ(buffer_slice.SampleAt(frame, 1),
static_cast<float>(-0x8000 + frame) / static_cast<float>(kRampFrameCount / 2))
<< "at (" << frame << ", 1)";
}
}
} // namespace