blob: 15fc9783591670362c7aa41b1bfca674ea3605ec [file] [log] [blame]
// Copyright 2022 The Chromium 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 "components/cast_streaming/browser/demuxer_stream_data_provider.h"
#include <utility>
#include "base/test/bind.h"
#include "base/test/task_environment.h"
#include "media/base/audio_codecs.h"
#include "media/base/audio_decoder_config.h"
#include "media/base/channel_layout.h"
#include "media/base/decoder_buffer.h"
#include "media/base/media_util.h"
#include "media/base/sample_format.h"
#include "media/mojo/common/media_type_converters.h"
#include "media/mojo/common/mojo_data_pipe_read_write.h"
#include "mojo/public/cpp/bindings/remote.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace cast_streaming {
class DemuxerStreamDataProviderTest : public testing::Test {
public:
DemuxerStreamDataProviderTest()
: first_config_(media::AudioCodec::kMP3,
media::SampleFormat::kSampleFormatS16,
media::ChannelLayout::CHANNEL_LAYOUT_MONO,
24000 /* samples_per_second */,
media::EmptyExtraData(),
media::EncryptionScheme::kUnencrypted),
second_config_(media::AudioCodec::kOpus,
media::SampleFormat::kSampleFormatF32,
media::ChannelLayout::CHANNEL_LAYOUT_STEREO,
48000 /* samples_per_second */,
media::EmptyExtraData(),
media::EncryptionScheme::kUnencrypted) {
data_provider_ = std::make_unique<AudioDemuxerStreamDataProvider>(
remote_.BindNewPipeAndPassReceiver(),
base::BindRepeating(
&DemuxerStreamDataProviderTest::Callbacks::RequestBuffer,
base::Unretained(&callbacks_)),
base::BindOnce(
&DemuxerStreamDataProviderTest::Callbacks::OnMojoDisconnect,
base::Unretained(&callbacks_)),
second_config_);
data_provider_->SetOnNoBuffersAvailableCallback(base::BindRepeating(
&DemuxerStreamDataProviderTest::Callbacks::OnNoBuffers,
base::Unretained(&callbacks_)));
data_provider_->SetOnErrorCallback(
base::BindRepeating(&DemuxerStreamDataProviderTest::Callbacks::OnError,
base::Unretained(&callbacks_)));
std::vector<uint8_t> data = {1, 2, 3};
first_buffer_ = media::DecoderBuffer::CopyFrom(data.data(), 3);
first_buffer_->set_duration(base::Seconds(1));
first_buffer_->set_timestamp(base::Seconds(2));
data = {42, 43, 44};
second_buffer_ = media::DecoderBuffer::CopyFrom(data.data(), 3);
second_buffer_->set_duration(base::Seconds(32));
second_buffer_->set_timestamp(base::Seconds(42));
data = {7, 8, 9};
third_buffer_ = media::DecoderBuffer::CopyFrom(data.data(), 3);
third_buffer_->set_duration(base::Seconds(10));
third_buffer_->set_timestamp(base::Seconds(11));
task_environment_.RunUntilIdle();
}
~DemuxerStreamDataProviderTest() override = default;
protected:
class Callbacks {
public:
MOCK_METHOD0(OnNoBuffers, void());
MOCK_METHOD0(OnError, void());
MOCK_METHOD1(RequestBuffer, void(base::OnceClosure));
MOCK_METHOD0(OnMojoDisconnect, void());
MOCK_METHOD0(OnGetBufferDoneCalled, void());
void OnGetBufferDone(absl::optional<media::AudioDecoderConfig> config,
scoped_refptr<media::DecoderBuffer> buffer_expected,
mojom::AudioStreamInfoPtr data_stream_info,
media::mojom::DecoderBufferPtr buffer) {
ASSERT_TRUE(!!buffer);
scoped_refptr<media::DecoderBuffer> media_buffer(
buffer.To<scoped_refptr<media::DecoderBuffer>>());
EXPECT_TRUE(buffer_expected->MatchesMetadataForTesting(*media_buffer));
ASSERT_EQ(!!config, !!data_stream_info);
if (config) {
EXPECT_TRUE(config->Matches(data_stream_info->decoder_config));
}
OnGetBufferDoneCalled();
}
};
using MojoPipePair = std::pair<mojo::ScopedDataPipeProducerHandle,
mojo::ScopedDataPipeConsumerHandle>;
MojoPipePair GetMojoPipePair() {
mojo::ScopedDataPipeProducerHandle producer_handle;
mojo::ScopedDataPipeConsumerHandle consumer_handle;
mojo::CreateDataPipe(512 /* this constant is irrelevant for these tests */,
producer_handle, consumer_handle);
return MojoPipePair(std::move(producer_handle), std::move(consumer_handle));
}
testing::StrictMock<Callbacks> callbacks_;
base::test::SingleThreadTaskEnvironment task_environment_{
base::test::TaskEnvironment::TimeSource::MOCK_TIME};
scoped_refptr<media::DecoderBuffer> first_buffer_;
scoped_refptr<media::DecoderBuffer> second_buffer_;
scoped_refptr<media::DecoderBuffer> third_buffer_;
media::AudioDecoderConfig first_config_;
media::AudioDecoderConfig second_config_;
std::unique_ptr<AudioDemuxerStreamDataProvider> data_provider_;
mojo::Remote<mojom::AudioBufferRequester> remote_;
};
TEST_F(DemuxerStreamDataProviderTest, DataSentInOrderExpected) {
// Test first call, providing a config and a buffer.
EXPECT_CALL(callbacks_, RequestBuffer(testing::_));
remote_->GetBuffer(base::BindOnce(
&DemuxerStreamDataProviderTest::Callbacks::OnGetBufferDone,
base::Unretained(&callbacks_), first_config_, first_buffer_));
task_environment_.RunUntilIdle();
EXPECT_CALL(callbacks_, OnGetBufferDoneCalled());
MojoPipePair pipes = GetMojoPipePair();
data_provider_->OnNewStreamInfo(first_config_, std::move(pipes.second));
EXPECT_TRUE(first_config_.Matches(data_provider_->config()));
task_environment_.RunUntilIdle();
data_provider_->ProvideBuffer(
media::mojom::DecoderBuffer::From(*first_buffer_));
task_environment_.RunUntilIdle();
EXPECT_TRUE(first_config_.Matches(data_provider_->config()));
// Test second call, providing NO config but do provide a buffer.
EXPECT_CALL(callbacks_, RequestBuffer(testing::_));
remote_->GetBuffer(base::BindOnce(
&DemuxerStreamDataProviderTest::Callbacks::OnGetBufferDone,
base::Unretained(&callbacks_), absl::nullopt, second_buffer_));
task_environment_.RunUntilIdle();
EXPECT_CALL(callbacks_, OnGetBufferDoneCalled());
task_environment_.RunUntilIdle();
data_provider_->ProvideBuffer(
media::mojom::DecoderBuffer::From(*second_buffer_));
EXPECT_TRUE(first_config_.Matches(data_provider_->config()));
task_environment_.RunUntilIdle();
// Test third call, providing a different config and a buffer.
EXPECT_CALL(callbacks_, RequestBuffer(testing::_));
remote_->GetBuffer(base::BindOnce(
&DemuxerStreamDataProviderTest::Callbacks::OnGetBufferDone,
base::Unretained(&callbacks_), second_config_, third_buffer_));
task_environment_.RunUntilIdle();
EXPECT_CALL(callbacks_, OnGetBufferDoneCalled());
task_environment_.RunUntilIdle();
pipes = GetMojoPipePair();
EXPECT_TRUE(first_config_.Matches(data_provider_->config()));
data_provider_->OnNewStreamInfo(second_config_, std::move(pipes.second));
EXPECT_TRUE(second_config_.Matches(data_provider_->config()));
data_provider_->ProvideBuffer(
media::mojom::DecoderBuffer::From(*third_buffer_));
task_environment_.RunUntilIdle();
}
TEST_F(DemuxerStreamDataProviderTest, NoBuffersCallsWithCallback) {
EXPECT_CALL(callbacks_, RequestBuffer(testing::_))
.WillOnce([](base::OnceClosure no_buffers_cb) {
std::move(no_buffers_cb).Run();
});
EXPECT_CALL(callbacks_, OnNoBuffers());
remote_->GetBuffer(base::BindOnce(
&DemuxerStreamDataProviderTest::Callbacks::OnGetBufferDone,
base::Unretained(&callbacks_), first_config_, first_buffer_));
task_environment_.RunUntilIdle();
}
TEST_F(DemuxerStreamDataProviderTest, NoBuffersCallsNoCallback) {
data_provider_->SetOnNoBuffersAvailableCallback(base::RepeatingClosure());
EXPECT_CALL(callbacks_, RequestBuffer(testing::_))
.WillOnce(
[](base::OnceClosure no_buffers_cb) { ASSERT_FALSE(no_buffers_cb); });
remote_->GetBuffer(base::BindOnce(
&DemuxerStreamDataProviderTest::Callbacks::OnGetBufferDone,
base::Unretained(&callbacks_), first_config_, first_buffer_));
task_environment_.RunUntilIdle();
}
} // namespace cast_streaming