| // 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 <memory> |
| |
| #include "base/bind.h" |
| #include "base/files/file_path.h" |
| #include "base/run_loop.h" |
| #include "base/test/scoped_task_environment.h" |
| #include "build/build_config.h" |
| #include "media/cdm/default_cdm_factory.h" |
| #include "media/media_buildflags.h" |
| #include "media/mojo/mojom/constants.mojom.h" |
| #include "media/mojo/services/cdm_service.h" |
| #include "media/mojo/services/media_interface_provider.h" |
| #include "services/service_manager/public/cpp/test/test_connector_factory.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "url/gurl.h" |
| #include "url/origin.h" |
| |
| namespace media { |
| |
| namespace { |
| |
| using testing::_; |
| using testing::Invoke; |
| using testing::InvokeWithoutArgs; |
| |
| MATCHER_P(MatchesResult, success, "") { |
| return arg->success == success; |
| } |
| |
| const char kClearKeyKeySystem[] = "org.w3.clearkey"; |
| const char kInvalidKeySystem[] = "invalid.key.system"; |
| const char kSecurityOrigin[] = "https://foo.com"; |
| |
| class MockCdmServiceClient : public media::CdmService::Client { |
| public: |
| MockCdmServiceClient() = default; |
| ~MockCdmServiceClient() override = default; |
| |
| // media::CdmService::Client implementation. |
| MOCK_METHOD0(EnsureSandboxed, void()); |
| |
| std::unique_ptr<media::CdmFactory> CreateCdmFactory( |
| service_manager::mojom::InterfaceProvider* host_interfaces) override { |
| return std::make_unique<media::DefaultCdmFactory>(); |
| } |
| |
| #if BUILDFLAG(ENABLE_CDM_HOST_VERIFICATION) |
| void AddCdmHostFilePaths(std::vector<media::CdmHostFilePath>*) override {} |
| #endif // BUILDFLAG(ENABLE_CDM_HOST_VERIFICATION) |
| }; |
| |
| class CdmServiceTest : public testing::Test { |
| public: |
| CdmServiceTest() : connector_(test_connector_factory_.CreateConnector()) {} |
| ~CdmServiceTest() override = default; |
| |
| service_manager::Connector* connector() const { return connector_.get(); } |
| |
| MOCK_METHOD0(CdmServiceConnectionClosed, void()); |
| MOCK_METHOD0(CdmFactoryConnectionClosed, void()); |
| MOCK_METHOD0(CdmConnectionClosed, void()); |
| |
| void Initialize() { |
| auto mock_cdm_service_client = std::make_unique<MockCdmServiceClient>(); |
| mock_cdm_service_client_ = mock_cdm_service_client.get(); |
| |
| service_ = |
| std::make_unique<CdmService>(std::move(mock_cdm_service_client), |
| test_connector_factory_.RegisterInstance( |
| media::mojom::kCdmServiceName)); |
| service_->SetServiceReleaseDelayForTesting(service_release_delay_); |
| service_->set_termination_closure(base::BindOnce( |
| &CdmServiceTest::DestroyService, base::Unretained(this))); |
| |
| connector()->BindInterface(media::mojom::kCdmServiceName, |
| &cdm_service_ptr_); |
| cdm_service_ptr_.set_connection_error_handler(base::BindOnce( |
| &CdmServiceTest::CdmServiceConnectionClosed, base::Unretained(this))); |
| |
| service_manager::mojom::InterfaceProviderPtr interfaces; |
| auto provider = std::make_unique<MediaInterfaceProvider>( |
| mojo::MakeRequest(&interfaces)); |
| |
| ASSERT_FALSE(cdm_factory_ptr_); |
| cdm_service_ptr_->CreateCdmFactory(mojo::MakeRequest(&cdm_factory_ptr_), |
| std::move(interfaces)); |
| cdm_service_ptr_.FlushForTesting(); |
| ASSERT_TRUE(cdm_factory_ptr_); |
| cdm_factory_ptr_.set_connection_error_handler(base::BindOnce( |
| &CdmServiceTest::CdmFactoryConnectionClosed, base::Unretained(this))); |
| } |
| |
| void InitializeWithServiceReleaseDelay(base::TimeDelta delay) { |
| service_release_delay_ = delay; |
| Initialize(); |
| } |
| |
| MOCK_METHOD3(OnCdmInitialized, |
| void(mojom::CdmPromiseResultPtr result, |
| int cdm_id, |
| mojom::DecryptorPtr decryptor)); |
| |
| void InitializeCdm(const std::string& key_system, bool expected_result) { |
| base::RunLoop run_loop; |
| cdm_factory_ptr_->CreateCdm(key_system, mojo::MakeRequest(&cdm_ptr_)); |
| cdm_ptr_.set_connection_error_handler(base::BindOnce( |
| &CdmServiceTest::CdmConnectionClosed, base::Unretained(this))); |
| EXPECT_CALL(*this, OnCdmInitialized(MatchesResult(expected_result), _, _)) |
| .WillOnce(InvokeWithoutArgs(&run_loop, &base::RunLoop::Quit)); |
| cdm_ptr_->Initialize(key_system, url::Origin::Create(GURL(kSecurityOrigin)), |
| CdmConfig(), |
| base::BindOnce(&CdmServiceTest::OnCdmInitialized, |
| base::Unretained(this))); |
| run_loop.Run(); |
| } |
| |
| void DestroyService() { service_.reset(); } |
| |
| MockCdmServiceClient* mock_cdm_service_client() { |
| return mock_cdm_service_client_; |
| } |
| |
| CdmService* cdm_service() { return service_.get(); } |
| |
| base::test::ScopedTaskEnvironment task_environment_; |
| mojom::CdmServicePtr cdm_service_ptr_; |
| mojom::CdmFactoryPtr cdm_factory_ptr_; |
| mojom::ContentDecryptionModulePtr cdm_ptr_; |
| |
| private: |
| service_manager::TestConnectorFactory test_connector_factory_; |
| std::unique_ptr<service_manager::Connector> connector_; |
| |
| // Delayed service release involves a posted delayed task which will not |
| // block *.RunUntilIdle() and hence cause a memory leak in the test. So by |
| // default use a zero value delay to disable the delay. |
| base::TimeDelta service_release_delay_; |
| |
| std::unique_ptr<CdmService> service_; |
| MockCdmServiceClient* mock_cdm_service_client_ = nullptr; |
| |
| DISALLOW_COPY_AND_ASSIGN(CdmServiceTest); |
| }; |
| |
| } // namespace |
| |
| TEST_F(CdmServiceTest, LoadCdm) { |
| Initialize(); |
| |
| // Even with a dummy path where the CDM cannot be loaded, EnsureSandboxed() |
| // should still be called to ensure the process is sandboxed. |
| EXPECT_CALL(*mock_cdm_service_client(), EnsureSandboxed()); |
| |
| base::FilePath cdm_path(FILE_PATH_LITERAL("dummy path")); |
| #if defined(OS_MACOSX) |
| // Token provider will not be used since the path is a dummy path. |
| cdm_service_ptr_->LoadCdm(cdm_path, nullptr); |
| #else |
| cdm_service_ptr_->LoadCdm(cdm_path); |
| #endif |
| |
| cdm_service_ptr_.FlushForTesting(); |
| } |
| |
| TEST_F(CdmServiceTest, InitializeCdm_Success) { |
| Initialize(); |
| InitializeCdm(kClearKeyKeySystem, true); |
| } |
| |
| TEST_F(CdmServiceTest, InitializeCdm_InvalidKeySystem) { |
| Initialize(); |
| InitializeCdm(kInvalidKeySystem, false); |
| } |
| |
| TEST_F(CdmServiceTest, DestroyAndRecreateCdm) { |
| Initialize(); |
| InitializeCdm(kClearKeyKeySystem, true); |
| cdm_ptr_.reset(); |
| InitializeCdm(kClearKeyKeySystem, true); |
| } |
| |
| // CdmFactory connection error will NOT destroy CDMs. Instead, it will only be |
| // destroyed after |cdm_ptr_| is reset. |
| TEST_F(CdmServiceTest, DestroyCdmFactory) { |
| Initialize(); |
| auto* service = cdm_service(); |
| |
| InitializeCdm(kClearKeyKeySystem, true); |
| EXPECT_EQ(service->BoundCdmFactorySizeForTesting(), 1u); |
| EXPECT_EQ(service->UnboundCdmFactorySizeForTesting(), 0u); |
| |
| cdm_factory_ptr_.reset(); |
| base::RunLoop().RunUntilIdle(); |
| EXPECT_EQ(service->BoundCdmFactorySizeForTesting(), 0u); |
| EXPECT_EQ(service->UnboundCdmFactorySizeForTesting(), 1u); |
| |
| cdm_ptr_.reset(); |
| base::RunLoop().RunUntilIdle(); |
| |
| // The service should be destroyed by the time we get here. |
| EXPECT_FALSE(cdm_service()); |
| } |
| |
| // Same as DestroyCdmFactory test, but do not disable delayed service release. |
| // TODO(xhwang): Use ScopedTaskEnvironment::TimeSource::MOCK_TIME and |
| // ScopedTaskEnvironment::FastForwardBy() so we don't have to really wait for |
| // the delay in the test. But currently FastForwardBy() doesn't support delayed |
| // task yet. |
| TEST_F(CdmServiceTest, DestroyCdmFactory_DelayedServiceRelease) { |
| constexpr base::TimeDelta kKeepaliveIdleTimeout = |
| base::TimeDelta::FromSeconds(1); |
| InitializeWithServiceReleaseDelay(kKeepaliveIdleTimeout); |
| |
| InitializeCdm(kClearKeyKeySystem, true); |
| cdm_factory_ptr_.reset(); |
| base::RunLoop().RunUntilIdle(); |
| |
| base::RunLoop run_loop; |
| auto start_time = base::Time::Now(); |
| cdm_ptr_.reset(); |
| EXPECT_CALL(*this, CdmServiceConnectionClosed()) |
| .WillOnce(Invoke(&run_loop, &base::RunLoop::Quit)); |
| run_loop.Run(); |
| auto time_passed = base::Time::Now() - start_time; |
| EXPECT_GE(time_passed, kKeepaliveIdleTimeout); |
| } |
| |
| // Destroy service will destroy the CdmFactory and all CDMs. |
| TEST_F(CdmServiceTest, DestroyCdmService) { |
| Initialize(); |
| InitializeCdm(kClearKeyKeySystem, true); |
| |
| base::RunLoop run_loop; |
| // Ideally we should not care about order, and should only quit the loop when |
| // both connections are closed. |
| EXPECT_CALL(*this, CdmServiceConnectionClosed()); |
| EXPECT_CALL(*this, CdmFactoryConnectionClosed()); |
| EXPECT_CALL(*this, CdmConnectionClosed()) |
| .WillOnce(Invoke(&run_loop, &base::RunLoop::Quit)); |
| DestroyService(); |
| run_loop.Run(); |
| } |
| |
| } // namespace media |