| // Copyright 2015 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 "content/utility/utility_service_factory.h" |
| |
| #include <utility> |
| #include <vector> |
| |
| #include "base/bind.h" |
| #include "base/command_line.h" |
| #include "base/trace_event/trace_log.h" |
| #include "build/build_config.h" |
| #include "content/child/child_process.h" |
| #include "content/public/common/content_client.h" |
| #include "content/public/common/content_features.h" |
| #include "content/public/common/content_switches.h" |
| #include "content/public/common/service_names.mojom.h" |
| #include "content/public/utility/content_utility_client.h" |
| #include "content/public/utility/utility_thread.h" |
| #include "content/utility/utility_thread_impl.h" |
| #include "media/media_buildflags.h" |
| #include "services/audio/public/mojom/constants.mojom.h" |
| #include "services/audio/service_factory.h" |
| #include "services/data_decoder/data_decoder_service.h" |
| #include "services/data_decoder/public/mojom/constants.mojom.h" |
| #include "services/network/network_service.h" |
| #include "services/network/public/cpp/features.h" |
| #include "services/service_manager/public/mojom/service.mojom.h" |
| #include "services/video_capture/public/mojom/constants.mojom.h" |
| #include "services/video_capture/service_impl.h" |
| #include "services/viz/public/interfaces/constants.mojom.h" |
| #include "services/viz/service.h" |
| |
| #if BUILDFLAG(ENABLE_LIBRARY_CDMS) |
| #include "media/cdm/cdm_adapter_factory.h" // nogncheck |
| #include "media/mojo/interfaces/constants.mojom.h" // nogncheck |
| #include "media/mojo/services/cdm_service.h" // nogncheck |
| #include "media/mojo/services/mojo_cdm_helper.h" // nogncheck |
| #include "media/mojo/services/mojo_media_client.h" // nogncheck |
| #if BUILDFLAG(ENABLE_CDM_HOST_VERIFICATION) |
| #include "media/cdm/cdm_host_file.h" |
| #endif // BUILDFLAG(ENABLE_CDM_HOST_VERIFICATION) |
| #endif |
| |
| #if defined(OS_MACOSX) |
| #include "sandbox/mac/system_services.h" |
| #include "services/service_manager/sandbox/features.h" |
| #endif |
| |
| #if defined(OS_WIN) |
| #include "sandbox/win/src/sandbox.h" |
| |
| extern sandbox::TargetServices* g_utility_target_services; |
| #endif |
| |
| #if defined(OS_CHROMEOS) |
| #include "chromeos/assistant/buildflags.h" // nogncheck |
| #if BUILDFLAG(ENABLE_CROS_LIBASSISTANT) |
| #include "chromeos/services/assistant/audio_decoder/assistant_audio_decoder_service.h" // nogncheck |
| #include "chromeos/services/assistant/public/mojom/constants.mojom.h" // nogncheck |
| #endif // BUILDFLAG(ENABLE_CROS_LIBASSISTANT) |
| #endif |
| |
| namespace content { |
| |
| namespace { |
| |
| #if BUILDFLAG(ENABLE_LIBRARY_CDMS) |
| |
| std::unique_ptr<media::CdmAuxiliaryHelper> CreateCdmHelper( |
| service_manager::mojom::InterfaceProvider* interface_provider) { |
| return std::make_unique<media::MojoCdmHelper>(interface_provider); |
| } |
| |
| class ContentCdmServiceClient final : public media::CdmService::Client { |
| public: |
| ContentCdmServiceClient() {} |
| ~ContentCdmServiceClient() override {} |
| |
| void EnsureSandboxed() override { |
| #if defined(OS_WIN) |
| // |g_utility_target_services| can be null if --no-sandbox is specified. |
| if (g_utility_target_services) |
| g_utility_target_services->LowerToken(); |
| #endif |
| } |
| |
| std::unique_ptr<media::CdmFactory> CreateCdmFactory( |
| service_manager::mojom::InterfaceProvider* host_interfaces) override { |
| return std::make_unique<media::CdmAdapterFactory>( |
| base::Bind(&CreateCdmHelper, host_interfaces)); |
| } |
| |
| #if BUILDFLAG(ENABLE_CDM_HOST_VERIFICATION) |
| void AddCdmHostFilePaths( |
| std::vector<media::CdmHostFilePath>* cdm_host_file_paths) override { |
| GetContentClient()->AddContentDecryptionModules(nullptr, |
| cdm_host_file_paths); |
| } |
| #endif // BUILDFLAG(ENABLE_CDM_HOST_VERIFICATION) |
| }; |
| |
| std::unique_ptr<service_manager::Service> CreateCdmService() { |
| return std::unique_ptr<service_manager::Service>( |
| new ::media::CdmService(std::make_unique<ContentCdmServiceClient>())); |
| } |
| #endif // BUILDFLAG(ENABLE_LIBRARY_CDMS) |
| |
| std::unique_ptr<service_manager::Service> CreateVizService() { |
| return std::make_unique<viz::Service>(); |
| } |
| |
| } // namespace |
| |
| UtilityServiceFactory::UtilityServiceFactory() |
| : network_registry_(std::make_unique<service_manager::BinderRegistry>()), |
| audio_registry_(std::make_unique<service_manager::BinderRegistry>()) {} |
| |
| UtilityServiceFactory::~UtilityServiceFactory() {} |
| |
| void UtilityServiceFactory::CreateService( |
| service_manager::mojom::ServiceRequest request, |
| const std::string& name, |
| service_manager::mojom::PIDReceiverPtr pid_receiver) { |
| auto* trace_log = base::trace_event::TraceLog::GetInstance(); |
| if (trace_log->IsProcessNameEmpty()) |
| trace_log->set_process_name("Service: " + name); |
| ServiceFactory::CreateService(std::move(request), name, |
| std::move(pid_receiver)); |
| } |
| |
| void UtilityServiceFactory::RegisterServices(ServiceMap* services) { |
| GetContentClient()->utility()->RegisterServices(services); |
| |
| GetContentClient()->utility()->RegisterAudioBinders(audio_registry_.get()); |
| service_manager::EmbeddedServiceInfo audio_info; |
| audio_info.factory = base::BindRepeating( |
| &UtilityServiceFactory::CreateAudioService, base::Unretained(this)); |
| services->insert(std::make_pair(audio::mojom::kServiceName, audio_info)); |
| |
| #if BUILDFLAG(ENABLE_LIBRARY_CDMS) |
| service_manager::EmbeddedServiceInfo info; |
| info.factory = base::Bind(&CreateCdmService); |
| services->emplace(media::mojom::kCdmServiceName, info); |
| #endif |
| |
| #if defined(OS_CHROMEOS) |
| #if BUILDFLAG(ENABLE_CROS_LIBASSISTANT) |
| { |
| service_manager::EmbeddedServiceInfo assistant_audio_decoder_info; |
| assistant_audio_decoder_info.factory = base::BindRepeating( |
| &chromeos::assistant::AssistantAudioDecoderService::CreateService); |
| services->emplace(chromeos::assistant::mojom::kAudioDecoderServiceName, |
| assistant_audio_decoder_info); |
| } |
| #endif // BUILDFLAG(ENABLE_CROS_LIBASSISTANT) |
| #endif |
| |
| service_manager::EmbeddedServiceInfo viz_info; |
| viz_info.factory = base::Bind(&CreateVizService); |
| services->insert(std::make_pair(viz::mojom::kVizServiceName, viz_info)); |
| |
| if (base::FeatureList::IsEnabled(network::features::kNetworkService)) { |
| GetContentClient()->utility()->RegisterNetworkBinders( |
| network_registry_.get()); |
| service_manager::EmbeddedServiceInfo network_info; |
| network_info.factory = base::Bind( |
| &UtilityServiceFactory::CreateNetworkService, base::Unretained(this)); |
| network_info.task_runner = ChildProcess::current()->io_task_runner(); |
| services->insert( |
| std::make_pair(content::mojom::kNetworkServiceName, network_info)); |
| } |
| } |
| |
| bool UtilityServiceFactory::HandleServiceRequest( |
| const std::string& name, |
| service_manager::mojom::ServiceRequest request) { |
| if (name == data_decoder::mojom::kServiceName) { |
| content::UtilityThread::Get()->EnsureBlinkInitialized(); |
| running_service_ = |
| std::make_unique<data_decoder::DataDecoderService>(std::move(request)); |
| } else if (name == video_capture::mojom::kServiceName) { |
| running_service_ = |
| std::make_unique<video_capture::ServiceImpl>(std::move(request)); |
| } |
| |
| if (running_service_) { |
| // If we actually started a service for this request, make sure its |
| // self-termination results in full process termination. |
| running_service_->set_termination_closure(base::BindOnce( |
| &UtilityServiceFactory::OnServiceQuit, base::Unretained(this))); |
| return true; |
| } |
| |
| return false; |
| } |
| |
| void UtilityServiceFactory::OnServiceQuit() { |
| UtilityThread::Get()->ReleaseProcess(); |
| } |
| |
| void UtilityServiceFactory::OnLoadFailed() { |
| UtilityThreadImpl* utility_thread = |
| static_cast<UtilityThreadImpl*>(UtilityThread::Get()); |
| utility_thread->Shutdown(); |
| utility_thread->ReleaseProcess(); |
| } |
| |
| std::unique_ptr<service_manager::Service> |
| UtilityServiceFactory::CreateNetworkService() { |
| return std::make_unique<network::NetworkService>( |
| std::move(network_registry_)); |
| } |
| |
| std::unique_ptr<service_manager::Service> |
| UtilityServiceFactory::CreateAudioService() { |
| #if defined(OS_MACOSX) |
| // Don't connect to launch services when running sandboxed |
| // (https://crbug.com/874785). |
| if (base::FeatureList::IsEnabled( |
| service_manager::features::kAudioServiceSandbox)) { |
| sandbox::DisableLaunchServices(); |
| } |
| #endif |
| |
| return audio::CreateStandaloneService(std::move(audio_registry_)); |
| } |
| |
| } // namespace content |