| // 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 |