| // Copyright 2016 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "content/browser/renderer_host/media/media_devices_manager.h" |
| |
| #include <memory> |
| #include <string> |
| |
| #include "base/containers/contains.h" |
| #include "base/functional/bind.h" |
| #include "base/functional/callback_helpers.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/memory/ref_counted.h" |
| #include "base/run_loop.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "base/task/single_thread_task_runner.h" |
| #include "base/test/metrics/histogram_tester.h" |
| #include "build/build_config.h" |
| #include "content/browser/media/media_devices_permission_checker.h" |
| #include "content/browser/renderer_host/media/in_process_video_capture_provider.h" |
| #include "content/browser/renderer_host/media/mock_video_capture_provider.h" |
| #include "content/browser/renderer_host/media/video_capture_manager.h" |
| #include "content/public/test/browser_task_environment.h" |
| #include "media/audio/audio_device_name.h" |
| #include "media/audio/audio_system_impl.h" |
| #include "media/audio/fake_audio_log_factory.h" |
| #include "media/audio/fake_audio_manager.h" |
| #include "media/audio/test_audio_thread.h" |
| #include "media/capture/video/fake_video_capture_device_factory.h" |
| #include "media/capture/video/video_capture_system_impl.h" |
| #include "mojo/public/cpp/bindings/pending_remote.h" |
| #include "mojo/public/cpp/bindings/receiver_set.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "url/gurl.h" |
| #include "url/origin.h" |
| |
| using base::HistogramTester; |
| using blink::mojom::MediaDeviceType; |
| using media::mojom::DeviceEnumerationResult; |
| using testing::_; |
| using testing::Invoke; |
| using testing::SaveArg; |
| |
| namespace content { |
| |
| namespace { |
| |
| const int kRenderProcessId = 1; |
| const int kRenderFrameId = 3; |
| |
| // Number of client enumerations to simulate on each test run. |
| // This allows testing that a single call to low-level enumeration functions |
| // is performed when cache is enabled, regardless of the number of client calls. |
| const int kNumCalls = 3; |
| |
| const size_t kNumAudioInputDevices = 2; |
| |
| const auto kIgnoreLogMessageCB = base::DoNothing(); |
| |
| std::string salt = "fake_media_device_salt"; |
| MediaDeviceSaltAndOrigin GetSaltAndOrigin(int /* process_id */, |
| int /* frame_id */) { |
| return MediaDeviceSaltAndOrigin(salt, "fake_group_id_salt", |
| url::Origin::Create(GURL("https://test.com")), |
| /*has_focus=*/true, /*is_background=*/false); |
| } |
| |
| // This class mocks the audio manager and overrides some methods to ensure that |
| // we can run simulate device changes. |
| class MockAudioManager : public media::FakeAudioManager { |
| public: |
| MockAudioManager() |
| : FakeAudioManager(std::make_unique<media::TestAudioThread>(), |
| &fake_audio_log_factory_), |
| num_output_devices_(2), |
| num_input_devices_(kNumAudioInputDevices) {} |
| |
| MockAudioManager(const MockAudioManager&) = delete; |
| MockAudioManager& operator=(const MockAudioManager&) = delete; |
| |
| ~MockAudioManager() override {} |
| |
| MOCK_METHOD1(MockGetAudioInputDeviceNames, void(media::AudioDeviceNames*)); |
| MOCK_METHOD1(MockGetAudioOutputDeviceNames, void(media::AudioDeviceNames*)); |
| |
| std::string GetDefaultInputDeviceID() override { return default_device_id_; } |
| |
| std::string GetCommunicationsInputDeviceID() override { |
| return communications_device_id_; |
| } |
| |
| void GetAudioInputDeviceNames( |
| media::AudioDeviceNames* device_names) override { |
| DCHECK(device_names->empty()); |
| if (default_device_id_ != std::string()) { |
| device_names->push_back(media::AudioDeviceName( |
| media::AudioDeviceDescription::kDefaultDeviceId, |
| media::AudioDeviceDescription::kDefaultDeviceId)); |
| } |
| if (communications_device_id_ != std::string()) { |
| device_names->push_back(media::AudioDeviceName( |
| std::string(), |
| media::AudioDeviceDescription::kCommunicationsDeviceId)); |
| } |
| |
| size_t num_devices_to_create = num_input_devices_; |
| size_t start_id_trailer = 0; |
| while (num_devices_to_create > 0) { |
| size_t trailer = ++start_id_trailer; |
| std::string id("fake_device_id_" + base::NumberToString(trailer)); |
| if (base::Contains(removed_input_audio_device_ids_, id)) |
| continue; |
| |
| device_names->push_back(media::AudioDeviceName( |
| std::string("fake_device_name_") + base::NumberToString(trailer), |
| id)); |
| --num_devices_to_create; |
| } |
| MockGetAudioInputDeviceNames(device_names); |
| } |
| |
| void GetAudioOutputDeviceNames( |
| media::AudioDeviceNames* device_names) override { |
| DCHECK(device_names->empty()); |
| for (size_t i = 0; i < num_output_devices_; i++) { |
| device_names->push_back(media::AudioDeviceName( |
| std::string("fake_device_name_") + base::NumberToString(i), |
| std::string("fake_device_id_") + base::NumberToString(i))); |
| } |
| MockGetAudioOutputDeviceNames(device_names); |
| } |
| |
| media::AudioParameters GetDefaultOutputStreamParameters() override { |
| return media::AudioParameters(media::AudioParameters::AUDIO_PCM_LOW_LATENCY, |
| media::ChannelLayoutConfig::Stereo(), 48000, |
| 128); |
| } |
| |
| media::AudioParameters GetOutputStreamParameters( |
| const std::string& device_id) override { |
| return media::AudioParameters(media::AudioParameters::AUDIO_PCM_LOW_LATENCY, |
| media::ChannelLayoutConfig::Stereo(), 48000, |
| 128); |
| } |
| |
| void RemoveInputAudioDeviceById(const std::string& device_id) { |
| --num_input_devices_; |
| removed_input_audio_device_ids_.insert(device_id); |
| if (device_id == default_device_id_) |
| default_device_id_ = std::string(); |
| if (device_id == communications_device_id_) |
| communications_device_id_ = std::string(); |
| } |
| |
| void SetNumAudioOutputDevices(size_t num_devices) { |
| num_output_devices_ = num_devices; |
| } |
| |
| void SetNumAudioInputDevices(size_t num_devices) { |
| num_input_devices_ = num_devices; |
| } |
| |
| void SetDefaultDeviceToId(const std::string& device_id) { |
| default_device_id_ = device_id; |
| } |
| |
| void SetCommunicationsDeviceToId(const std::string& device_id) { |
| communications_device_id_ = device_id; |
| } |
| |
| private: |
| media::FakeAudioLogFactory fake_audio_log_factory_; |
| size_t num_output_devices_; |
| size_t num_input_devices_; |
| std::string default_device_id_; |
| std::string communications_device_id_; |
| std::set<std::string> removed_input_audio_device_ids_; |
| }; |
| |
| // This class mocks the video capture device factory and overrides some methods |
| // to ensure that we can simulate device changes. |
| class MockVideoCaptureDeviceFactory |
| : public media::FakeVideoCaptureDeviceFactory { |
| public: |
| MockVideoCaptureDeviceFactory() {} |
| ~MockVideoCaptureDeviceFactory() override {} |
| |
| MOCK_METHOD0(MockGetDevicesInfo, void()); |
| void GetDevicesInfo(GetDevicesInfoCallback callback) override { |
| media::FakeVideoCaptureDeviceFactory::GetDevicesInfo(std::move(callback)); |
| MockGetDevicesInfo(); |
| } |
| }; |
| |
| class MockMediaDevicesListener : public blink::mojom::MediaDevicesListener { |
| public: |
| MockMediaDevicesListener() {} |
| |
| MOCK_METHOD2(OnDevicesChanged, |
| void(MediaDeviceType, const blink::WebMediaDeviceInfoArray&)); |
| |
| mojo::PendingRemote<blink::mojom::MediaDevicesListener> |
| CreateInterfacePtrAndBind() { |
| mojo::PendingRemote<blink::mojom::MediaDevicesListener> listener; |
| receivers_.Add(this, listener.InitWithNewPipeAndPassReceiver()); |
| return listener; |
| } |
| |
| private: |
| mojo::ReceiverSet<blink::mojom::MediaDevicesListener> receivers_; |
| }; |
| |
| class MockMediaDevicesManagerClient { |
| public: |
| MOCK_METHOD2(StopRemovedInputDevice, |
| void(MediaDeviceType type, |
| const blink::WebMediaDeviceInfo& media_device_info)); |
| MOCK_METHOD2(InputDevicesChangedUI, |
| void(MediaDeviceType stream_type, |
| const blink::WebMediaDeviceInfoArray& devices)); |
| }; |
| |
| void VerifyDeviceAndGroupID( |
| const std::vector<blink::WebMediaDeviceInfoArray>& array) { |
| for (const auto& device_infos : array) { |
| for (const auto& device_info : device_infos) { |
| EXPECT_FALSE(device_info.device_id.empty()); |
| EXPECT_FALSE(device_info.group_id.empty()); |
| } |
| } |
| } |
| |
| } // namespace |
| |
| class MediaDevicesManagerTest : public ::testing::Test { |
| public: |
| MediaDevicesManagerTest() |
| : task_environment_(content::BrowserTaskEnvironment::IO_MAINLOOP), |
| video_capture_device_factory_(nullptr) {} |
| |
| MediaDevicesManagerTest(const MediaDevicesManagerTest&) = delete; |
| MediaDevicesManagerTest& operator=(const MediaDevicesManagerTest&) = delete; |
| |
| ~MediaDevicesManagerTest() override { audio_manager_->Shutdown(); } |
| |
| MOCK_METHOD1(MockCallback, void(const MediaDeviceEnumeration&)); |
| |
| void TrackRemovedDevice(MediaDeviceType type, |
| const blink::WebMediaDeviceInfo& media_device_info) { |
| removed_device_ids_.insert(media_device_info.device_id); |
| } |
| |
| void RunEnumerateDevices() { |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| } |
| |
| void EnumerateCallback(base::RunLoop* run_loop, |
| const MediaDeviceEnumeration& result) { |
| for (int i = 0; |
| i < static_cast<int>(MediaDeviceType::NUM_MEDIA_DEVICE_TYPES); ++i) { |
| for (const auto& device_info : result[i]) { |
| EXPECT_FALSE(device_info.device_id.empty()); |
| EXPECT_FALSE(device_info.group_id.empty()); |
| } |
| } |
| MockCallback(result); |
| run_loop->Quit(); |
| } |
| |
| void EnumerateWithCapabilitiesCallback( |
| const std::vector<media::FakeVideoCaptureDeviceSettings>& |
| expected_video_capture_device_settings, |
| base::RunLoop* run_loop, |
| const std::vector<blink::WebMediaDeviceInfoArray>& devices, |
| std::vector<VideoInputDeviceCapabilitiesPtr> video_capabilities, |
| std::vector<AudioInputDeviceCapabilitiesPtr> audio_capabilities) { |
| EXPECT_EQ(video_capabilities.size(), |
| expected_video_capture_device_settings.size()); |
| for (size_t i = 0; i < video_capabilities.size(); ++i) { |
| EXPECT_EQ( |
| video_capabilities[i]->formats.size(), |
| expected_video_capture_device_settings[i].supported_formats.size()); |
| for (size_t j = 0; j < video_capabilities[i]->formats.size(); ++j) { |
| EXPECT_EQ( |
| video_capabilities[i]->formats[j], |
| expected_video_capture_device_settings[i].supported_formats[j]); |
| } |
| } |
| EXPECT_EQ(audio_capabilities.size(), kNumAudioInputDevices); |
| for (size_t i = 0; i < audio_capabilities.size(); ++i) { |
| EXPECT_TRUE(audio_capabilities[i]->parameters.IsValid()); |
| EXPECT_EQ(audio_capabilities[i]->parameters.channel_layout(), |
| media::CHANNEL_LAYOUT_STEREO); |
| EXPECT_EQ(audio_capabilities[i]->parameters.sample_rate(), |
| media::AudioParameters::kAudioCDSampleRate); |
| EXPECT_EQ(audio_capabilities[i]->parameters.frames_per_buffer(), |
| media::AudioParameters::kAudioCDSampleRate / 100); |
| } |
| run_loop->Quit(); |
| } |
| |
| protected: |
| void SetUp() override { |
| audio_manager_ = std::make_unique<MockAudioManager>(); |
| audio_system_ = |
| std::make_unique<media::AudioSystemImpl>(audio_manager_.get()); |
| auto video_capture_device_factory = |
| std::make_unique<MockVideoCaptureDeviceFactory>(); |
| video_capture_device_factory_ = video_capture_device_factory.get(); |
| auto video_capture_system = std::make_unique<media::VideoCaptureSystemImpl>( |
| std::move(video_capture_device_factory)); |
| in_process_video_capture_provider_ = |
| std::make_unique<InProcessVideoCaptureProvider>( |
| std::move(video_capture_system), |
| base::SingleThreadTaskRunner::GetCurrentDefault(), |
| kIgnoreLogMessageCB); |
| |
| auto mock_video_capture_provider = |
| std::make_unique<MockVideoCaptureProvider>(); |
| mock_video_capture_provider_ = mock_video_capture_provider.get(); |
| // By default, forward calls to the real InProcessVideoCaptureProvider. |
| ON_CALL(*mock_video_capture_provider_, GetDeviceInfosAsync(_)) |
| .WillByDefault(Invoke( |
| [&](VideoCaptureProvider::GetDeviceInfosCallback result_callback) { |
| in_process_video_capture_provider_->GetDeviceInfosAsync( |
| std::move(result_callback)); |
| })); |
| |
| video_capture_manager_ = new VideoCaptureManager( |
| std::move(mock_video_capture_provider), kIgnoreLogMessageCB); |
| media_devices_manager_ = std::make_unique<MediaDevicesManager>( |
| audio_system_.get(), video_capture_manager_, |
| base::BindRepeating( |
| &MockMediaDevicesManagerClient::StopRemovedInputDevice, |
| base::Unretained(&media_devices_manager_client_)), |
| base::BindRepeating( |
| &MockMediaDevicesManagerClient::InputDevicesChangedUI, |
| base::Unretained(&media_devices_manager_client_))); |
| media_devices_manager_->set_salt_and_origin_callback_for_testing( |
| base::BindRepeating(&GetSaltAndOrigin)); |
| media_devices_manager_->SetPermissionChecker( |
| std::make_unique<MediaDevicesPermissionChecker>(true)); |
| } |
| |
| void EnableCache(MediaDeviceType type) { |
| media_devices_manager_->SetCachePolicy( |
| type, MediaDevicesManager::CachePolicy::SYSTEM_MONITOR); |
| } |
| |
| void ExpectVideoEnumerationHistogramReport(int success_count, |
| int error_count = 0) { |
| histogram_tester_.ExpectTotalCount( |
| "Media.MediaDevicesManager.VideoDeviceEnumeration.Start", |
| success_count + error_count); |
| histogram_tester_.ExpectBucketCount( |
| "Media.MediaDevicesManager.VideoDeviceEnumeration.Result", |
| DeviceEnumerationResult::kSuccess, success_count); |
| histogram_tester_.ExpectBucketCount( |
| "Media.MediaDevicesManager.VideoDeviceEnumeration.Result", |
| DeviceEnumerationResult::kUnknownError, error_count); |
| } |
| |
| // Must outlive MediaDevicesManager as ~MediaDevicesManager() verifies it's |
| // running on the IO thread. |
| BrowserTaskEnvironment task_environment_; |
| |
| std::unique_ptr<MediaDevicesManager> media_devices_manager_; |
| scoped_refptr<VideoCaptureManager> video_capture_manager_; |
| raw_ptr<MockVideoCaptureDeviceFactory> video_capture_device_factory_; |
| std::unique_ptr<MockAudioManager> audio_manager_; |
| std::unique_ptr<media::AudioSystem> audio_system_; |
| testing::StrictMock<MockMediaDevicesManagerClient> |
| media_devices_manager_client_; |
| std::set<std::string> removed_device_ids_; |
| raw_ptr<MockVideoCaptureProvider> mock_video_capture_provider_; |
| std::unique_ptr<InProcessVideoCaptureProvider> |
| in_process_video_capture_provider_; |
| HistogramTester histogram_tester_; |
| }; |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateNoCacheAudioInput) { |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)) |
| .Times(kNumCalls); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(0); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*this, MockCallback(_)).Times(kNumCalls); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| } |
| ExpectVideoEnumerationHistogramReport(0); |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateNoCacheVideoInput) { |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()) |
| .Times(kNumCalls); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*this, MockCallback(_)).Times(kNumCalls); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| } |
| ExpectVideoEnumerationHistogramReport(kNumCalls); |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateNoCacheAudioOutput) { |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(0); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)) |
| .Times(kNumCalls); |
| EXPECT_CALL(*this, MockCallback(_)).Times(kNumCalls); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_OUTPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| } |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateNoCacheAudio) { |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)) |
| .Times(kNumCalls); |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)) |
| .Times(kNumCalls); |
| EXPECT_CALL(*this, MockCallback(_)).Times(kNumCalls); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_OUTPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| } |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateCacheAudio) { |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(1); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(0); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(1); |
| EXPECT_CALL(*this, MockCallback(_)).Times(kNumCalls); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| EnableCache(MediaDeviceType::MEDIA_AUDIO_INPUT); |
| EnableCache(MediaDeviceType::MEDIA_AUDIO_OUTPUT); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_OUTPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| } |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateCacheVideo) { |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(1); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*this, MockCallback(_)).Times(kNumCalls); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| EnableCache(MediaDeviceType::MEDIA_VIDEO_INPUT); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| } |
| ExpectVideoEnumerationHistogramReport(1); |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateCacheAudioWithDeviceChanges) { |
| MediaDeviceEnumeration enumeration; |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(3); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(0); |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(3); |
| EXPECT_CALL(*this, MockCallback(_)) |
| .Times(3 * kNumCalls) |
| .WillRepeatedly(SaveArg<0>(&enumeration)); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| size_t num_audio_input_devices = 5; |
| size_t num_audio_output_devices = 3; |
| audio_manager_->SetNumAudioInputDevices(num_audio_input_devices); |
| audio_manager_->SetNumAudioOutputDevices(num_audio_output_devices); |
| EnableCache(MediaDeviceType::MEDIA_AUDIO_INPUT); |
| EnableCache(MediaDeviceType::MEDIA_AUDIO_OUTPUT); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_OUTPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| EXPECT_EQ( |
| num_audio_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_INPUT)] |
| .size()); |
| EXPECT_EQ( |
| num_audio_output_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_OUTPUT)] |
| .size()); |
| } |
| |
| // Simulate removal of devices. |
| size_t old_num_audio_input_devices = num_audio_input_devices; |
| num_audio_input_devices = 3; |
| num_audio_output_devices = 2; |
| ASSERT_LT(num_audio_input_devices, old_num_audio_input_devices); |
| size_t num_removed_audio_input_devices = |
| old_num_audio_input_devices - num_audio_input_devices; |
| EXPECT_CALL(media_devices_manager_client_, StopRemovedInputDevice(_, _)) |
| .Times(num_removed_audio_input_devices); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| |
| audio_manager_->SetNumAudioInputDevices(num_audio_input_devices); |
| audio_manager_->SetNumAudioOutputDevices(num_audio_output_devices); |
| media_devices_manager_->OnDevicesChanged(base::SystemMonitor::DEVTYPE_AUDIO); |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| EXPECT_EQ( |
| num_audio_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_INPUT)] |
| .size()); |
| EXPECT_EQ( |
| num_audio_output_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_OUTPUT)] |
| .size()); |
| } |
| |
| // Simulate addition of devices. |
| old_num_audio_input_devices = num_audio_input_devices; |
| num_audio_input_devices = 4; |
| num_audio_output_devices = 3; |
| ASSERT_GT(num_audio_input_devices, old_num_audio_input_devices); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| |
| audio_manager_->SetNumAudioInputDevices(num_audio_input_devices); |
| audio_manager_->SetNumAudioOutputDevices(num_audio_output_devices); |
| media_devices_manager_->OnDevicesChanged(base::SystemMonitor::DEVTYPE_AUDIO); |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| EXPECT_EQ( |
| num_audio_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_INPUT)] |
| .size()); |
| EXPECT_EQ( |
| num_audio_output_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_OUTPUT)] |
| .size()); |
| } |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateCacheVideoWithDeviceChanges) { |
| MediaDeviceEnumeration enumeration; |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(3); |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*this, MockCallback(_)) |
| .Times(3 * kNumCalls) |
| .WillRepeatedly(SaveArg<0>(&enumeration)); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| |
| // First enumeration. |
| size_t num_video_input_devices = 5; |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| EnableCache(MediaDeviceType::MEDIA_VIDEO_INPUT); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| EXPECT_EQ( |
| num_video_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_VIDEO_INPUT)] |
| .size()); |
| } |
| |
| // Simulate addition of devices. |
| size_t old_num_video_input_devices = num_video_input_devices; |
| num_video_input_devices = 9; |
| ASSERT_GT(num_video_input_devices, old_num_video_input_devices); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| media_devices_manager_->OnDevicesChanged( |
| base::SystemMonitor::DEVTYPE_VIDEO_CAPTURE); |
| |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| EXPECT_EQ( |
| num_video_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_VIDEO_INPUT)] |
| .size()); |
| } |
| |
| // Simulate removal of devices. |
| old_num_video_input_devices = num_video_input_devices; |
| num_video_input_devices = 7; |
| ASSERT_LT(num_video_input_devices, old_num_video_input_devices); |
| size_t num_removed_video_input_devices = |
| old_num_video_input_devices - num_video_input_devices; |
| EXPECT_CALL(media_devices_manager_client_, StopRemovedInputDevice(_, _)) |
| .Times(num_removed_video_input_devices); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| media_devices_manager_->OnDevicesChanged( |
| base::SystemMonitor::DEVTYPE_VIDEO_CAPTURE); |
| |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| EXPECT_EQ( |
| num_video_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_VIDEO_INPUT)] |
| .size()); |
| } |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateCacheAllWithDeviceChanges) { |
| MediaDeviceEnumeration enumeration; |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(2); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(2); |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(2); |
| EXPECT_CALL(*this, MockCallback(_)) |
| .Times(2 * kNumCalls) |
| .WillRepeatedly(SaveArg<0>(&enumeration)); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(2); |
| |
| size_t num_audio_input_devices = 5; |
| size_t num_video_input_devices = 4; |
| size_t num_audio_output_devices = 3; |
| audio_manager_->SetNumAudioInputDevices(num_audio_input_devices); |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| audio_manager_->SetNumAudioOutputDevices(num_audio_output_devices); |
| EnableCache(MediaDeviceType::MEDIA_AUDIO_INPUT); |
| EnableCache(MediaDeviceType::MEDIA_AUDIO_OUTPUT); |
| EnableCache(MediaDeviceType::MEDIA_VIDEO_INPUT); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_OUTPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| EXPECT_EQ( |
| num_audio_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_INPUT)] |
| .size()); |
| EXPECT_EQ( |
| num_video_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_VIDEO_INPUT)] |
| .size()); |
| EXPECT_EQ( |
| num_audio_output_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_OUTPUT)] |
| .size()); |
| } |
| |
| // Simulate device changes |
| size_t old_num_audio_input_devices = num_audio_input_devices; |
| size_t old_num_video_input_devices = num_video_input_devices; |
| num_audio_input_devices = 3; |
| num_video_input_devices = 2; |
| num_audio_output_devices = 4; |
| ASSERT_LT(num_audio_input_devices, old_num_audio_input_devices); |
| ASSERT_LT(num_video_input_devices, old_num_video_input_devices); |
| size_t num_removed_input_devices = |
| old_num_audio_input_devices - num_audio_input_devices + |
| old_num_video_input_devices - num_video_input_devices; |
| EXPECT_CALL(media_devices_manager_client_, StopRemovedInputDevice(_, _)) |
| .Times(num_removed_input_devices); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(2); |
| audio_manager_->SetNumAudioInputDevices(num_audio_input_devices); |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| audio_manager_->SetNumAudioOutputDevices(num_audio_output_devices); |
| media_devices_manager_->OnDevicesChanged(base::SystemMonitor::DEVTYPE_AUDIO); |
| media_devices_manager_->OnDevicesChanged( |
| base::SystemMonitor::DEVTYPE_VIDEO_CAPTURE); |
| |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| EXPECT_EQ( |
| num_audio_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_INPUT)] |
| .size()); |
| EXPECT_EQ( |
| num_video_input_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_VIDEO_INPUT)] |
| .size()); |
| EXPECT_EQ( |
| num_audio_output_devices, |
| enumeration[static_cast<size_t>(MediaDeviceType::MEDIA_AUDIO_OUTPUT)] |
| .size()); |
| } |
| } |
| |
| TEST_F(MediaDevicesManagerTest, SubscribeDeviceChanges) { |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(3); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(3); |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(3); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(2); |
| |
| size_t num_audio_input_devices = 5; |
| size_t num_video_input_devices = 4; |
| size_t num_audio_output_devices = 3; |
| audio_manager_->SetNumAudioInputDevices(num_audio_input_devices); |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| audio_manager_->SetNumAudioOutputDevices(num_audio_output_devices); |
| |
| // Run an enumeration to make sure |media_devices_manager_| has the new |
| // configuration. |
| EXPECT_CALL(*this, MockCallback(_)); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_OUTPUT)] = true; |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| |
| // Add device-change event listeners. |
| MockMediaDevicesListener listener_audio_input; |
| MediaDevicesManager::BoolDeviceTypes audio_input_devices_to_subscribe; |
| audio_input_devices_to_subscribe[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| uint32_t audio_input_subscription_id = |
| media_devices_manager_->SubscribeDeviceChangeNotifications( |
| kRenderProcessId, kRenderFrameId, audio_input_devices_to_subscribe, |
| listener_audio_input.CreateInterfacePtrAndBind()); |
| |
| MockMediaDevicesListener listener_video_input; |
| MediaDevicesManager::BoolDeviceTypes video_input_devices_to_subscribe; |
| video_input_devices_to_subscribe[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| uint32_t video_input_subscription_id = |
| media_devices_manager_->SubscribeDeviceChangeNotifications( |
| kRenderProcessId, kRenderFrameId, video_input_devices_to_subscribe, |
| listener_video_input.CreateInterfacePtrAndBind()); |
| |
| MockMediaDevicesListener listener_audio_output; |
| MediaDevicesManager::BoolDeviceTypes audio_output_devices_to_subscribe; |
| audio_output_devices_to_subscribe[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_OUTPUT)] = true; |
| uint32_t audio_output_subscription_id = |
| media_devices_manager_->SubscribeDeviceChangeNotifications( |
| kRenderProcessId, kRenderFrameId, audio_output_devices_to_subscribe, |
| listener_audio_output.CreateInterfacePtrAndBind()); |
| |
| MockMediaDevicesListener listener_all; |
| MediaDevicesManager::BoolDeviceTypes all_devices_to_subscribe; |
| all_devices_to_subscribe[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| all_devices_to_subscribe[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| all_devices_to_subscribe[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_OUTPUT)] = true; |
| media_devices_manager_->SubscribeDeviceChangeNotifications( |
| kRenderProcessId, kRenderFrameId, all_devices_to_subscribe, |
| listener_all.CreateInterfacePtrAndBind()); |
| |
| blink::WebMediaDeviceInfoArray notification_audio_input; |
| blink::WebMediaDeviceInfoArray notification_video_input; |
| blink::WebMediaDeviceInfoArray notification_audio_output; |
| blink::WebMediaDeviceInfoArray notification_all_audio_input; |
| blink::WebMediaDeviceInfoArray notification_all_video_input; |
| blink::WebMediaDeviceInfoArray notification_all_audio_output; |
| EXPECT_CALL(listener_audio_input, |
| OnDevicesChanged(MediaDeviceType::MEDIA_AUDIO_INPUT, _)) |
| .Times(1) |
| .WillOnce(SaveArg<1>(¬ification_audio_input)); |
| EXPECT_CALL(listener_video_input, |
| OnDevicesChanged(MediaDeviceType::MEDIA_VIDEO_INPUT, _)) |
| .Times(1) |
| .WillOnce(SaveArg<1>(¬ification_video_input)); |
| EXPECT_CALL(listener_audio_output, |
| OnDevicesChanged(MediaDeviceType::MEDIA_AUDIO_OUTPUT, _)) |
| .Times(1) |
| .WillOnce(SaveArg<1>(¬ification_audio_output)); |
| EXPECT_CALL(listener_all, |
| OnDevicesChanged(MediaDeviceType::MEDIA_AUDIO_INPUT, _)) |
| .Times(2) |
| .WillRepeatedly(SaveArg<1>(¬ification_all_audio_input)); |
| EXPECT_CALL(listener_all, |
| OnDevicesChanged(MediaDeviceType::MEDIA_VIDEO_INPUT, _)) |
| .Times(2) |
| .WillRepeatedly(SaveArg<1>(¬ification_all_video_input)); |
| EXPECT_CALL(listener_all, |
| OnDevicesChanged(MediaDeviceType::MEDIA_AUDIO_OUTPUT, _)) |
| .Times(2) |
| .WillRepeatedly(SaveArg<1>(¬ification_all_audio_output)); |
| |
| // Simulate device changes. |
| size_t old_num_audio_input_devices = num_audio_input_devices; |
| size_t old_num_video_input_devices = num_video_input_devices; |
| num_audio_input_devices = 3; |
| num_video_input_devices = 2; |
| num_audio_output_devices = 4; |
| ASSERT_LT(num_audio_input_devices, old_num_audio_input_devices); |
| ASSERT_LT(num_video_input_devices, old_num_video_input_devices); |
| size_t num_removed_input_devices = |
| old_num_audio_input_devices - num_audio_input_devices + |
| old_num_video_input_devices - num_video_input_devices; |
| EXPECT_CALL(media_devices_manager_client_, StopRemovedInputDevice(_, _)) |
| .Times(num_removed_input_devices); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(2); |
| audio_manager_->SetNumAudioInputDevices(num_audio_input_devices); |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| audio_manager_->SetNumAudioOutputDevices(num_audio_output_devices); |
| media_devices_manager_->OnDevicesChanged(base::SystemMonitor::DEVTYPE_AUDIO); |
| media_devices_manager_->OnDevicesChanged( |
| base::SystemMonitor::DEVTYPE_VIDEO_CAPTURE); |
| base::RunLoop().RunUntilIdle(); |
| EXPECT_EQ(num_audio_input_devices, notification_audio_input.size()); |
| EXPECT_EQ(num_video_input_devices, notification_video_input.size()); |
| EXPECT_EQ(num_audio_output_devices, notification_audio_output.size()); |
| EXPECT_EQ(num_audio_input_devices, notification_all_audio_input.size()); |
| EXPECT_EQ(num_video_input_devices, notification_all_video_input.size()); |
| EXPECT_EQ(num_audio_output_devices, notification_all_audio_output.size()); |
| VerifyDeviceAndGroupID( |
| {notification_audio_input, notification_video_input, |
| notification_audio_output, notification_all_audio_input, |
| notification_all_video_input, notification_all_audio_output}); |
| |
| media_devices_manager_->UnsubscribeDeviceChangeNotifications( |
| audio_input_subscription_id); |
| media_devices_manager_->UnsubscribeDeviceChangeNotifications( |
| video_input_subscription_id); |
| media_devices_manager_->UnsubscribeDeviceChangeNotifications( |
| audio_output_subscription_id); |
| |
| // Simulate further device changes. Only the objects still subscribed to the |
| // device-change events will receive notifications. |
| old_num_audio_input_devices = num_audio_input_devices; |
| old_num_video_input_devices = num_video_input_devices; |
| num_audio_input_devices = 2; |
| num_video_input_devices = 1; |
| num_audio_output_devices = 3; |
| ASSERT_LT(num_audio_input_devices, old_num_audio_input_devices); |
| ASSERT_LT(num_video_input_devices, old_num_video_input_devices); |
| num_removed_input_devices = |
| old_num_audio_input_devices - num_audio_input_devices + |
| old_num_video_input_devices - num_video_input_devices; |
| EXPECT_CALL(media_devices_manager_client_, StopRemovedInputDevice(_, _)) |
| .Times(num_removed_input_devices); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(2); |
| audio_manager_->SetNumAudioInputDevices(num_audio_input_devices); |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| audio_manager_->SetNumAudioOutputDevices(num_audio_output_devices); |
| media_devices_manager_->OnDevicesChanged(base::SystemMonitor::DEVTYPE_AUDIO); |
| media_devices_manager_->OnDevicesChanged( |
| base::SystemMonitor::DEVTYPE_VIDEO_CAPTURE); |
| base::RunLoop().RunUntilIdle(); |
| EXPECT_EQ(num_audio_input_devices, notification_all_audio_input.size()); |
| EXPECT_EQ(num_video_input_devices, notification_all_video_input.size()); |
| EXPECT_EQ(num_audio_output_devices, notification_all_audio_output.size()); |
| VerifyDeviceAndGroupID({notification_all_audio_input, |
| notification_all_video_input, |
| notification_all_audio_output}); |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateDevicesWithCapabilities) { |
| // Audio is enumerated due to heuristics to compute video group IDs. |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)); |
| EXPECT_CALL(media_devices_manager_client_, |
| InputDevicesChangedUI(MediaDeviceType::MEDIA_AUDIO_INPUT, _)); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()); |
| EXPECT_CALL(media_devices_manager_client_, |
| InputDevicesChangedUI(MediaDeviceType::MEDIA_VIDEO_INPUT, _)); |
| // Configure fake devices with video formats different from the fallback |
| // formats to make sure that expected capabilities are what devices actually |
| // report. |
| media::FakeVideoCaptureDeviceSettings fake_device1; |
| fake_device1.device_id = "fake_id_1"; |
| fake_device1.delivery_mode = |
| media::FakeVideoCaptureDevice::DeliveryMode::USE_DEVICE_INTERNAL_BUFFERS; |
| fake_device1.supported_formats = { |
| {{1000, 1000}, 60.0, media::PIXEL_FORMAT_I420}, |
| {{2000, 2000}, 120.0, media::PIXEL_FORMAT_I420}}; |
| |
| media::FakeVideoCaptureDeviceSettings fake_device2; |
| fake_device2.device_id = "fake_id_2"; |
| fake_device2.delivery_mode = |
| media::FakeVideoCaptureDevice::DeliveryMode::USE_DEVICE_INTERNAL_BUFFERS; |
| fake_device2.supported_formats = { |
| {{100, 100}, 6.0, media::PIXEL_FORMAT_I420}, |
| {{200, 200}, 12.0, media::PIXEL_FORMAT_I420}}; |
| |
| std::vector<media::FakeVideoCaptureDeviceSettings> |
| fake_capture_device_settings = {fake_device1, fake_device2}; |
| video_capture_device_factory_->SetToCustomDevicesConfig( |
| fake_capture_device_settings); |
| |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_AUDIO_INPUT)] = true; |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| -1, -1, devices_to_enumerate, true, true, |
| base::BindOnce( |
| &MediaDevicesManagerTest::EnumerateWithCapabilitiesCallback, |
| base::Unretained(this), fake_capture_device_settings, &run_loop)); |
| run_loop.Run(); |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateDevicesUnplugDefaultDevice) { |
| // This tests does not apply to CrOS, which is to seamlessly switch device. |
| #if !BUILDFLAG(IS_CHROMEOS) |
| std::string default_device_id("fake_device_id_1"); |
| std::string new_default_device_id("fake_device_id_2"); |
| |
| EXPECT_EQ(removed_device_ids_.size(), 0u); |
| |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(2); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(0); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*this, MockCallback(_)).Times(2); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(2); |
| // Since we will be removing the default device, we expect that we will remove |
| // both the actual device as well as the old instance of the device with |
| // 'default' ID. |
| blink::WebMediaDeviceInfo removed_devices; |
| EXPECT_CALL(media_devices_manager_client_, StopRemovedInputDevice(_, _)) |
| .Times(2) |
| .WillRepeatedly( |
| Invoke(this, &MediaDevicesManagerTest::TrackRemovedDevice)); |
| |
| // Setup the configuration and run the devices. |
| audio_manager_->SetDefaultDeviceToId(default_device_id); |
| RunEnumerateDevices(); |
| |
| // Unplug the default device and switch default to a different device. |
| audio_manager_->RemoveInputAudioDeviceById(default_device_id); |
| audio_manager_->SetDefaultDeviceToId(new_default_device_id); |
| RunEnumerateDevices(); |
| |
| EXPECT_EQ(removed_device_ids_.size(), 2u); |
| EXPECT_TRUE(base::Contains(removed_device_ids_, default_device_id)); |
| EXPECT_TRUE(base::Contains(removed_device_ids_, |
| media::AudioDeviceDescription::kDefaultDeviceId)); |
| #endif // !BUILDFLAG(IS_CHROMEOS) |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateDevicesUnplugCommunicationsDevice) { |
| // This test has only significance on Windows devices, since communication |
| // devices can only be found on windows. |
| #if BUILDFLAG(IS_WIN) |
| std::string communications_device_id("fake_device_id_1"); |
| std::string new_communications_device_id("fake_device_id_2"); |
| |
| EXPECT_EQ(removed_device_ids_.size(), 0u); |
| |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(2); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(0); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*this, MockCallback(_)).Times(2); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(2); |
| // Since we will be removing the communications device, we expect that we will |
| // remove both the actual device as well as the old instance of the device |
| // with 'communications' ID. |
| blink::WebMediaDeviceInfo removed_devices; |
| EXPECT_CALL(media_devices_manager_client_, StopRemovedInputDevice(_, _)) |
| .Times(2) |
| .WillRepeatedly( |
| Invoke(this, &MediaDevicesManagerTest::TrackRemovedDevice)); |
| |
| // Setup the configuration and run the devices. |
| audio_manager_->SetCommunicationsDeviceToId(communications_device_id); |
| RunEnumerateDevices(); |
| |
| // Unplug the default device and switch default to a different device. |
| audio_manager_->RemoveInputAudioDeviceById(communications_device_id); |
| audio_manager_->SetCommunicationsDeviceToId(new_communications_device_id); |
| RunEnumerateDevices(); |
| |
| EXPECT_EQ(removed_device_ids_.size(), 2u); |
| EXPECT_TRUE(base::Contains(removed_device_ids_, communications_device_id)); |
| EXPECT_TRUE( |
| base::Contains(removed_device_ids_, |
| media::AudioDeviceDescription::kCommunicationsDeviceId)); |
| #endif // BUILDFLAG(IS_WIN) |
| } |
| |
| TEST_F(MediaDevicesManagerTest, |
| EnumerateDevicesUnplugDefaultAndCommunicationsDevice) { |
| // This test has only significance on Windows devices, since communication |
| // devices can only be found on windows. |
| #if BUILDFLAG(IS_WIN) |
| // The two device IDs that will be used as 'default' and 'communications' |
| // devices. |
| std::string target_device_id("fake_device_id_1"); |
| std::string new_target_device_id("fake_device_id_2"); |
| |
| EXPECT_EQ(removed_device_ids_.size(), 0u); |
| |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(2); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(0); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*this, MockCallback(_)).Times(2); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(2); |
| // Since we will be removing the device identified as 'default' and |
| // 'communications, we expect that we will |
| // remove the actual device as well as the old instances of the device |
| // with 'communications' and 'default' IDs. |
| blink::WebMediaDeviceInfo removed_devices; |
| EXPECT_CALL(media_devices_manager_client_, StopRemovedInputDevice(_, _)) |
| .Times(3) |
| .WillRepeatedly( |
| Invoke(this, &MediaDevicesManagerTest::TrackRemovedDevice)); |
| |
| // Setup the configuration and run the devices. |
| audio_manager_->SetDefaultDeviceToId(target_device_id); |
| audio_manager_->SetCommunicationsDeviceToId(target_device_id); |
| RunEnumerateDevices(); |
| |
| // Unplug the default device and switch default to a different device. |
| audio_manager_->RemoveInputAudioDeviceById(target_device_id); |
| audio_manager_->SetDefaultDeviceToId(new_target_device_id); |
| audio_manager_->SetCommunicationsDeviceToId(new_target_device_id); |
| RunEnumerateDevices(); |
| |
| EXPECT_EQ(removed_device_ids_.size(), 3u); |
| EXPECT_TRUE(base::Contains(removed_device_ids_, target_device_id)); |
| EXPECT_TRUE(base::Contains(removed_device_ids_, |
| media::AudioDeviceDescription::kDefaultDeviceId)); |
| EXPECT_TRUE( |
| base::Contains(removed_device_ids_, |
| media::AudioDeviceDescription::kCommunicationsDeviceId)); |
| #endif // BUILDFLAG(IS_WIN) |
| } |
| |
| TEST_F(MediaDevicesManagerTest, GuessVideoGroupID) { |
| blink::WebMediaDeviceInfoArray audio_devices = { |
| {media::AudioDeviceDescription::kDefaultDeviceId, |
| "Default - Unique USB Mic (1234:5678)", "group_1"}, |
| {media::AudioDeviceDescription::kCommunicationsDeviceId, |
| "communications - Unique USB Mic (1234:5678)", "group_1"}, |
| {"device_id_1", "Microphone (Logitech Webcam C930e)", "group_1"}, |
| {"device_id_2", "HD Pro Webcam C920", "group_2"}, |
| {"device_id_3", "Microsoft® LifeCam Cinema(TM)", "group_3"}, |
| {"device_id_4", "Repeated webcam", "group_4"}, |
| {"device_id_5", "Repeated webcam", "group_5"}, |
| {"device_id_6", "Dual-mic webcam device", "group_6"}, |
| {"device_id_7", "Dual-mic webcam device", "group_6"}, |
| {"device_id_8", "Repeated dual-mic webcam device", "group_7"}, |
| {"device_id_9", "Repeated dual-mic webcam device", "group_7"}, |
| {"device_id_10", "Repeated dual-mic webcam device", "group_8"}, |
| {"device_id_11", "Repeated dual-mic webcam device", "group_8"}, |
| {"device_id_12", "Unique USB Mic (1234:5678)", "group_1"}, |
| {"device_id_13", "Another Unique USB Mic (5678:9abc)", "group_9"}, |
| {"device_id_14", "Repeated USB Mic (8765:4321)", "group_10"}, |
| {"device_id_15", "Repeated USB Mic (8765:4321)", "group_11"}, |
| // Extra entry for communications device added here just to make sure |
| // that it is not incorrectly detected as an extra real device. Real |
| // enumerations contain only the first entry for the communications |
| // device. |
| {media::AudioDeviceDescription::kCommunicationsDeviceId, |
| "communications - Unique USB Mic (1234:5678)", "group_1"}, |
| }; |
| |
| blink::WebMediaDeviceInfo logitech_video( |
| "logitech_video", "Logitech Webcam C930e (046d:0843)", ""); |
| blink::WebMediaDeviceInfo hd_pro_video("hd_pro_video", |
| "HD Pro Webcam C920 (046d:082d)", ""); |
| blink::WebMediaDeviceInfo lifecam_video( |
| "lifecam_video", "Microsoft® LifeCam Cinema(TM) (045e:075d)", ""); |
| blink::WebMediaDeviceInfo repeated_webcam1_video("repeated_webcam_1", |
| "Repeated webcam", ""); |
| blink::WebMediaDeviceInfo repeated_webcam2_video("repeated_webcam_2", |
| "Repeated webcam", ""); |
| blink::WebMediaDeviceInfo dual_mic_video( |
| "dual_mic_video", "Dual-mic webcam device (1111:1111)", ""); |
| blink::WebMediaDeviceInfo webcam_only_video("webcam_only_video", |
| "Webcam-only device", ""); |
| blink::WebMediaDeviceInfo repeated_dual_mic1_video( |
| "repeated_dual_mic1_video", "Repeated dual-mic webcam device", ""); |
| blink::WebMediaDeviceInfo repeated_dual_mic2_video( |
| "repeated_dual_mic2_video", "Repeated dual-mic webcam device", ""); |
| blink::WebMediaDeviceInfo short_label_video("short_label_video", " ()", ""); |
| blink::WebMediaDeviceInfo unique_usb_video( |
| "unique_usb_video", "Unique USB webcam (1234:5678)", ""); |
| blink::WebMediaDeviceInfo another_unique_usb_video( |
| "another_unique_usb_video", "Another Unique USB webcam (5678:9abc)", ""); |
| blink::WebMediaDeviceInfo repeated_usb1_video( |
| "repeated_usb1_video", "Repeated USB webcam (8765:4321)", ""); |
| blink::WebMediaDeviceInfo repeated_usb2_video( |
| "repeated_usb2_video", "Repeated USB webcam (8765:4321)", ""); |
| |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, logitech_video), "group_1"); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, hd_pro_video), "group_2"); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, lifecam_video), "group_3"); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, repeated_webcam1_video), |
| repeated_webcam1_video.device_id); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, repeated_webcam2_video), |
| repeated_webcam2_video.device_id); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, dual_mic_video), "group_6"); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, webcam_only_video), |
| webcam_only_video.device_id); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, repeated_dual_mic1_video), |
| repeated_dual_mic1_video.device_id); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, repeated_dual_mic2_video), |
| repeated_dual_mic2_video.device_id); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, short_label_video), |
| short_label_video.device_id); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, unique_usb_video), "group_1"); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, another_unique_usb_video), |
| "group_9"); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, repeated_usb1_video), |
| repeated_usb1_video.device_id); |
| EXPECT_EQ(GuessVideoGroupID(audio_devices, repeated_usb2_video), |
| repeated_usb2_video.device_id); |
| } |
| |
| TEST_F(MediaDevicesManagerTest, DeviceIdSaltReset) { |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()).Times(2); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)) |
| .Times(1); |
| |
| size_t num_video_input_devices = 4; |
| video_capture_device_factory_->SetToDefaultDevicesConfig( |
| num_video_input_devices); |
| |
| // Run an enumeration to make sure |media_devices_manager_| has the new |
| // configuration. |
| EXPECT_CALL(*this, MockCallback(_)); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| |
| // Add device-change event listener. |
| MockMediaDevicesListener listener_video_input; |
| MediaDevicesManager::BoolDeviceTypes video_input_devices_to_subscribe; |
| video_input_devices_to_subscribe[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| media_devices_manager_->SubscribeDeviceChangeNotifications( |
| kRenderProcessId, kRenderFrameId, video_input_devices_to_subscribe, |
| listener_video_input.CreateInterfacePtrAndBind()); |
| |
| // Expect an OnDevicesChanged event. |
| blink::WebMediaDeviceInfoArray notification_video_input; |
| EXPECT_CALL(listener_video_input, |
| OnDevicesChanged(MediaDeviceType::MEDIA_VIDEO_INPUT, _)) |
| .Times(1) |
| .WillOnce(SaveArg<1>(¬ification_video_input)); |
| |
| base::RunLoop().RunUntilIdle(); |
| |
| // Set a new salt and notify MDM. |
| salt = "new-device-id-salt"; |
| media_devices_manager_->OnDevicesChanged( |
| base::SystemMonitor::DEVTYPE_VIDEO_CAPTURE); |
| |
| base::RunLoop().RunUntilIdle(); |
| EXPECT_EQ(num_video_input_devices, notification_video_input.size()); |
| } |
| |
| TEST_F(MediaDevicesManagerTest, EnumerateVideoInputFailsOnce) { |
| // Inject an UnknownError on the first call to GetDeviceInfosAsync, otherwise |
| // fall through to the in_process_video_capture_provider_. |
| EXPECT_CALL(*mock_video_capture_provider_, GetDeviceInfosAsync(_)) |
| .Times(kNumCalls) |
| .WillOnce(Invoke( |
| [&](VideoCaptureProvider::GetDeviceInfosCallback result_callback) { |
| std::move(result_callback) |
| .Run(DeviceEnumerationResult::kUnknownError, {}); |
| })) |
| .WillRepeatedly(Invoke( |
| [&](VideoCaptureProvider::GetDeviceInfosCallback result_callback) { |
| in_process_video_capture_provider_->GetDeviceInfosAsync( |
| std::move(result_callback)); |
| })); |
| EXPECT_CALL(*video_capture_device_factory_, MockGetDevicesInfo()) |
| .Times(kNumCalls - 1); |
| |
| EXPECT_CALL(*audio_manager_, MockGetAudioInputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*audio_manager_, MockGetAudioOutputDeviceNames(_)).Times(0); |
| EXPECT_CALL(*this, MockCallback(_)).Times(kNumCalls); |
| EXPECT_CALL(media_devices_manager_client_, InputDevicesChangedUI(_, _)); |
| MediaDevicesManager::BoolDeviceTypes devices_to_enumerate; |
| devices_to_enumerate[static_cast<size_t>( |
| MediaDeviceType::MEDIA_VIDEO_INPUT)] = true; |
| for (int i = 0; i < kNumCalls; i++) { |
| base::RunLoop run_loop; |
| media_devices_manager_->EnumerateDevices( |
| devices_to_enumerate, |
| base::BindOnce(&MediaDevicesManagerTest::EnumerateCallback, |
| base::Unretained(this), &run_loop)); |
| run_loop.Run(); |
| } |
| ExpectVideoEnumerationHistogramReport(/*success_count=*/kNumCalls - 1, |
| /*error_count=*/1); |
| } |
| |
| } // namespace content |