| // Copyright 2018 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 "services/audio/input_stream.h" |
| |
| #include <memory> |
| #include <string> |
| #include <utility> |
| |
| #include "base/bind.h" |
| #include "base/test/scoped_task_environment.h" |
| #include "media/audio/audio_io.h" |
| #include "media/audio/mock_audio_manager.h" |
| #include "media/audio/test_audio_thread.h" |
| #include "mojo/core/embedder/embedder.h" |
| #include "mojo/public/cpp/bindings/associated_binding.h" |
| #include "services/audio/stream_factory.h" |
| #include "services/audio/test/mock_log.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| using testing::NiceMock; |
| using testing::NotNull; |
| using testing::Return; |
| using testing::StrictMock; |
| using testing::_; |
| |
| namespace audio { |
| |
| namespace { |
| |
| const uint32_t kDefaultSharedMemoryCount = 10; |
| const bool kDoEnableAGC = true; |
| const bool kDoNotEnableAGC = false; |
| const bool kValidStream = true; |
| const bool kInvalidStream = false; |
| const bool kMuted = true; |
| const bool kNotMuted = false; |
| const char* kDefaultDeviceId = "default"; |
| |
| class MockStreamClient : public media::mojom::AudioInputStreamClient { |
| public: |
| MockStreamClient() : binding_(this) {} |
| |
| media::mojom::AudioInputStreamClientPtr MakePtr() { |
| DCHECK(!binding_.is_bound()); |
| media::mojom::AudioInputStreamClientPtr ptr; |
| binding_.Bind(mojo::MakeRequest(&ptr)); |
| binding_.set_connection_error_handler(base::BindOnce( |
| &MockStreamClient::BindingConnectionError, base::Unretained(this))); |
| return ptr; |
| } |
| |
| void CloseBinding() { binding_.Close(); } |
| |
| MOCK_METHOD0(OnError, void()); |
| MOCK_METHOD1(OnMutedStateChanged, void(bool)); |
| MOCK_METHOD0(BindingConnectionError, void()); |
| |
| private: |
| mojo::Binding<media::mojom::AudioInputStreamClient> binding_; |
| |
| DISALLOW_COPY_AND_ASSIGN(MockStreamClient); |
| }; |
| |
| class MockStreamObserver : public media::mojom::AudioInputStreamObserver { |
| public: |
| MockStreamObserver() : binding_(this) {} |
| |
| media::mojom::AudioInputStreamObserverPtr MakePtr() { |
| DCHECK(!binding_.is_bound()); |
| media::mojom::AudioInputStreamObserverPtr ptr; |
| binding_.Bind(mojo::MakeRequest(&ptr)); |
| binding_.set_connection_error_handler(base::BindOnce( |
| &MockStreamObserver::BindingConnectionError, base::Unretained(this))); |
| return ptr; |
| } |
| |
| void CloseBinding() { binding_.Close(); } |
| |
| MOCK_METHOD0(DidStartRecording, void()); |
| |
| MOCK_METHOD0(BindingConnectionError, void()); |
| |
| private: |
| mojo::Binding<media::mojom::AudioInputStreamObserver> binding_; |
| |
| DISALLOW_COPY_AND_ASSIGN(MockStreamObserver); |
| }; |
| |
| class MockStream : public media::AudioInputStream { |
| public: |
| MockStream() {} |
| |
| double GetMaxVolume() override { return 1; } |
| |
| MOCK_METHOD0(Open, bool()); |
| MOCK_METHOD1(Start, void(AudioInputCallback*)); |
| MOCK_METHOD0(Stop, void()); |
| MOCK_METHOD0(Close, void()); |
| MOCK_METHOD1(SetVolume, void(double)); |
| MOCK_METHOD0(GetVolume, double()); |
| MOCK_METHOD1(SetAutomaticGainControl, bool(bool)); |
| MOCK_METHOD0(GetAutomaticGainControl, bool()); |
| MOCK_METHOD0(IsMuted, bool()); |
| MOCK_METHOD1(SetOutputDeviceForAec, void(const std::string&)); |
| |
| private: |
| DISALLOW_COPY_AND_ASSIGN(MockStream); |
| }; |
| |
| } // namespace |
| |
| class AudioServiceInputStreamTest : public testing::Test { |
| public: |
| AudioServiceInputStreamTest() |
| : audio_manager_(std::make_unique<media::TestAudioThread>(false)), |
| stream_factory_(&audio_manager_), |
| stream_factory_binding_(&stream_factory_, |
| mojo::MakeRequest(&stream_factory_ptr_)) {} |
| |
| ~AudioServiceInputStreamTest() override { audio_manager_.Shutdown(); } |
| |
| void SetUp() override { |
| mojo::core::SetDefaultProcessErrorCallback( |
| base::BindRepeating(&AudioServiceInputStreamTest::BadMessageCallback, |
| base::Unretained(this))); |
| } |
| |
| void TearDown() override { |
| mojo::core::SetDefaultProcessErrorCallback( |
| mojo::core::ProcessErrorCallback()); |
| } |
| |
| media::mojom::AudioInputStreamPtr CreateStream(bool enable_agc) { |
| media::mojom::AudioInputStreamPtr stream_ptr; |
| stream_factory_ptr_->CreateInputStream( |
| mojo::MakeRequest(&stream_ptr), client_.MakePtr(), observer_.MakePtr(), |
| log_.MakePtr(), kDefaultDeviceId, |
| media::AudioParameters::UnavailableDeviceParams(), |
| kDefaultSharedMemoryCount, enable_agc, mojo::ScopedSharedBufferHandle(), |
| nullptr, |
| base::BindOnce(&AudioServiceInputStreamTest::OnCreated, |
| base::Unretained(this))); |
| return stream_ptr; |
| } |
| |
| media::mojom::AudioInputStreamPtr CreateStreamWithNullptrLog() { |
| media::mojom::AudioInputStreamPtr stream_ptr; |
| stream_factory_ptr_->CreateInputStream( |
| mojo::MakeRequest(&stream_ptr), client_.MakePtr(), observer_.MakePtr(), |
| nullptr, kDefaultDeviceId, |
| media::AudioParameters::UnavailableDeviceParams(), |
| kDefaultSharedMemoryCount, false, mojo::ScopedSharedBufferHandle(), |
| nullptr, |
| base::BindOnce(&AudioServiceInputStreamTest::OnCreated, |
| base::Unretained(this))); |
| return stream_ptr; |
| } |
| |
| media::mojom::AudioInputStreamPtr CreateStreamWithNullptrObserver() { |
| media::mojom::AudioInputStreamPtr stream_ptr; |
| stream_factory_ptr_->CreateInputStream( |
| mojo::MakeRequest(&stream_ptr), client_.MakePtr(), nullptr, |
| log_.MakePtr(), kDefaultDeviceId, |
| media::AudioParameters::UnavailableDeviceParams(), |
| kDefaultSharedMemoryCount, false, mojo::ScopedSharedBufferHandle(), |
| nullptr, |
| base::BindOnce(&AudioServiceInputStreamTest::OnCreated, |
| base::Unretained(this))); |
| return stream_ptr; |
| } |
| |
| media::MockAudioManager& audio_manager() { return audio_manager_; } |
| |
| MockStreamClient& client() { return client_; } |
| |
| MockStreamObserver& observer() { return observer_; } |
| |
| MockLog& log() { return log_; } |
| |
| void OnCreated(media::mojom::ReadOnlyAudioDataPipePtr ptr, |
| bool initially_muted, |
| const base::Optional<base::UnguessableToken>& stream_id) { |
| EXPECT_EQ(stream_id.has_value(), !!ptr); |
| CreatedCallback(!!ptr, initially_muted); |
| } |
| |
| MOCK_METHOD2(CreatedCallback, void(bool /*valid*/, bool /*initially_muted*/)); |
| MOCK_METHOD1(BadMessageCallback, void(const std::string&)); |
| |
| private: |
| base::test::ScopedTaskEnvironment scoped_task_env_; |
| media::MockAudioManager audio_manager_; |
| StreamFactory stream_factory_; |
| mojom::StreamFactoryPtr stream_factory_ptr_; |
| mojo::Binding<mojom::StreamFactory> stream_factory_binding_; |
| StrictMock<MockStreamClient> client_; |
| StrictMock<MockStreamObserver> observer_; |
| NiceMock<MockLog> log_; |
| |
| DISALLOW_COPY_AND_ASSIGN(AudioServiceInputStreamTest); |
| }; |
| |
| TEST_F(AudioServiceInputStreamTest, ConstructDestruct) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, ConstructDestructNullptrLog) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStreamWithNullptrLog(); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, ConstructDestructNullptrObserver) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = |
| CreateStreamWithNullptrObserver(); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| stream_ptr.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, |
| ConstructStreamAndCloseClientBinding_DestructsStream) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| client().CloseBinding(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, |
| ConstructStreamAndCloseObserverBinding_DestructsStream) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| observer().CloseBinding(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, |
| ConstructStreamAndResetStreamPtr_DestructsStream) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, Record) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Start(NotNull())); |
| EXPECT_CALL(log(), OnStarted()); |
| EXPECT_CALL(observer(), DidStartRecording()); |
| stream_ptr->Record(); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Stop()); |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, SetVolume) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| const double new_volume = 0.618; |
| EXPECT_CALL(mock_stream, SetVolume(new_volume)); |
| EXPECT_CALL(log(), OnSetVolume(new_volume)); |
| stream_ptr->SetVolume(new_volume); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, SetNegativeVolume_BadMessage) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(*this, BadMessageCallback(_)); |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr->SetVolume(-0.618); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, SetVolumeGreaterThanOne_BadMessage) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(*this, BadMessageCallback(_)); |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr->SetVolume(1.618); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, CreateStreamWithAGCEnable_PropagateAGC) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kNotMuted)); |
| EXPECT_CALL(mock_stream, SetAutomaticGainControl(kDoEnableAGC)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kNotMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, |
| CreateInitiallyMutedStream_PropagateInitiallyMuted) { |
| NiceMock<MockStream> mock_stream; |
| audio_manager().SetMakeInputStreamCB(base::BindRepeating( |
| [](media::AudioInputStream* stream, const media::AudioParameters& params, |
| const std::string& device_id) { return stream; }, |
| &mock_stream)); |
| |
| EXPECT_CALL(mock_stream, Open()).WillOnce(Return(true)); |
| EXPECT_CALL(mock_stream, IsMuted()).WillOnce(Return(kMuted)); |
| EXPECT_CALL(log(), OnCreated(_, _)); |
| EXPECT_CALL(*this, CreatedCallback(kValidStream, kMuted)); |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoEnableAGC); |
| base::RunLoop().RunUntilIdle(); |
| |
| EXPECT_CALL(log(), OnClosed()); |
| EXPECT_CALL(mock_stream, Close()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| stream_ptr.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(AudioServiceInputStreamTest, |
| ConstructWithStreamCreationFailure_SignalsError) { |
| // By default, MockAudioManager fails to create a stream. |
| |
| media::mojom::AudioInputStreamPtr stream_ptr = CreateStream(kDoNotEnableAGC); |
| |
| EXPECT_CALL(*this, CreatedCallback(kInvalidStream, kNotMuted)); |
| EXPECT_CALL(log(), OnError()); |
| EXPECT_CALL(client(), OnError()); |
| EXPECT_CALL(client(), BindingConnectionError()); |
| EXPECT_CALL(observer(), BindingConnectionError()); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| } // namespace audio |