blob: 1465a28b8934559de24e4b6a6f22d9a8a99fc357 [file] [log] [blame]
// 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 "media/mojo/services/interface_factory_impl.h"
#include <memory>
#include "base/guid.h"
#include "base/logging.h"
#include "base/single_thread_task_runner.h"
#include "base/threading/thread_task_runner_handle.h"
#include "media/base/media_log.h"
#include "media/mojo/services/mojo_decryptor_service.h"
#include "media/mojo/services/mojo_media_client.h"
#include "mojo/public/cpp/bindings/strong_binding.h"
#include "services/service_manager/public/mojom/interface_provider.mojom.h"
#if BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
#include "media/mojo/services/mojo_audio_decoder_service.h"
#endif // BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
#if BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
#include "media/mojo/services/mojo_video_decoder_service.h"
#endif // BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
#if BUILDFLAG(ENABLE_MOJO_RENDERER)
#include "base/bind_helpers.h"
#include "media/base/renderer.h"
#include "media/mojo/services/mojo_renderer_service.h"
#endif // BUILDFLAG(ENABLE_MOJO_RENDERER)
#if BUILDFLAG(ENABLE_MOJO_CDM)
#include "media/base/cdm_factory.h"
#include "media/mojo/services/mojo_cdm_service.h"
#endif // BUILDFLAG(ENABLE_MOJO_CDM)
#if BUILDFLAG(ENABLE_LIBRARY_CDMS)
#include "media/mojo/services/mojo_cdm_proxy_service.h"
#endif // BUILDFLAG(ENABLE_LIBRARY_CDMS)
namespace media {
InterfaceFactoryImpl::InterfaceFactoryImpl(
service_manager::mojom::InterfaceProviderPtr interfaces,
MediaLog* media_log,
std::unique_ptr<service_manager::ServiceContextRef> connection_ref,
MojoMediaClient* mojo_media_client)
:
#if BUILDFLAG(ENABLE_MOJO_RENDERER)
media_log_(media_log),
#endif
#if BUILDFLAG(ENABLE_MOJO_CDM)
interfaces_(std::move(interfaces)),
#endif
connection_ref_(std::move(connection_ref)),
mojo_media_client_(mojo_media_client) {
DVLOG(1) << __func__;
DCHECK(mojo_media_client_);
SetBindingConnectionErrorHandler();
}
InterfaceFactoryImpl::~InterfaceFactoryImpl() {
DVLOG(1) << __func__;
}
// mojom::InterfaceFactory implementation.
void InterfaceFactoryImpl::CreateAudioDecoder(
mojo::InterfaceRequest<mojom::AudioDecoder> request) {
DVLOG(2) << __func__;
#if BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
scoped_refptr<base::SingleThreadTaskRunner> task_runner(
base::ThreadTaskRunnerHandle::Get());
std::unique_ptr<AudioDecoder> audio_decoder =
mojo_media_client_->CreateAudioDecoder(task_runner);
if (!audio_decoder) {
DLOG(ERROR) << "AudioDecoder creation failed.";
return;
}
audio_decoder_bindings_.AddBinding(
std::make_unique<MojoAudioDecoderService>(&cdm_service_context_,
std::move(audio_decoder)),
std::move(request));
#endif // BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
}
void InterfaceFactoryImpl::CreateVideoDecoder(
mojom::VideoDecoderRequest request) {
DVLOG(2) << __func__;
#if BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
video_decoder_bindings_.AddBinding(
std::make_unique<MojoVideoDecoderService>(mojo_media_client_,
&cdm_service_context_),
std::move(request));
#endif // BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
}
void InterfaceFactoryImpl::CreateRenderer(
media::mojom::HostedRendererType type,
const std::string& type_specific_id,
mojo::InterfaceRequest<mojom::Renderer> request) {
DVLOG(2) << __func__;
#if BUILDFLAG(ENABLE_MOJO_RENDERER)
// Creation requests for non default renderers should have already been
// handled by now, in a different layer.
if (type != media::mojom::HostedRendererType::kDefault) {
DLOG(ERROR) << "Creation of specialized renderers is not supported.";
return;
}
// For HostedRendererType::kDefault type, |type_specific_id| represents an
// audio device ID. See interface_factory.mojom.
const std::string& audio_device_id = type_specific_id;
auto renderer = mojo_media_client_->CreateRenderer(
interfaces_.get(), base::ThreadTaskRunnerHandle::Get(), media_log_,
audio_device_id);
if (!renderer) {
DLOG(ERROR) << "Renderer creation failed.";
return;
}
std::unique_ptr<MojoRendererService> mojo_renderer_service =
std::make_unique<MojoRendererService>(
&cdm_service_context_, std::move(renderer),
MojoRendererService::InitiateSurfaceRequestCB());
MojoRendererService* mojo_renderer_service_ptr = mojo_renderer_service.get();
mojo::BindingId binding_id = renderer_bindings_.AddBinding(
std::move(mojo_renderer_service), std::move(request));
// base::Unretained() is safe because the callback will be fired by
// |mojo_renderer_service|, which is owned by |renderer_bindings_|.
mojo_renderer_service_ptr->set_bad_message_cb(
base::Bind(base::IgnoreResult(
&mojo::StrongBindingSet<mojom::Renderer>::RemoveBinding),
base::Unretained(&renderer_bindings_), binding_id));
#endif // BUILDFLAG(ENABLE_MOJO_RENDERER)
}
void InterfaceFactoryImpl::CreateCdm(
const std::string& /* key_system */,
mojo::InterfaceRequest<mojom::ContentDecryptionModule> request) {
DVLOG(2) << __func__;
#if BUILDFLAG(ENABLE_MOJO_CDM)
CdmFactory* cdm_factory = GetCdmFactory();
if (!cdm_factory)
return;
cdm_bindings_.AddBinding(
std::make_unique<MojoCdmService>(cdm_factory, &cdm_service_context_),
std::move(request));
#endif // BUILDFLAG(ENABLE_MOJO_CDM)
}
void InterfaceFactoryImpl::CreateDecryptor(int cdm_id,
mojom::DecryptorRequest request) {
DVLOG(2) << __func__;
auto mojo_decryptor_service =
MojoDecryptorService::Create(cdm_id, &cdm_service_context_);
if (!mojo_decryptor_service) {
DLOG(ERROR) << "MojoDecryptorService creation failed.";
return;
}
decryptor_bindings_.AddBinding(std::move(mojo_decryptor_service),
std::move(request));
}
void InterfaceFactoryImpl::CreateCdmProxy(const std::string& cdm_guid,
mojom::CdmProxyRequest request) {
DVLOG(2) << __func__;
#if BUILDFLAG(ENABLE_LIBRARY_CDMS)
if (!base::IsValidGUID(cdm_guid)) {
DLOG(ERROR) << "Invalid CDM GUID: " << cdm_guid;
return;
}
auto cdm_proxy = mojo_media_client_->CreateCdmProxy(cdm_guid);
if (!cdm_proxy) {
DLOG(ERROR) << "CdmProxy creation failed.";
return;
}
cdm_proxy_bindings_.AddBinding(
std::make_unique<MojoCdmProxyService>(std::move(cdm_proxy),
&cdm_service_context_),
std::move(request));
#endif // BUILDFLAG(ENABLE_LIBRARY_CDMS)
}
void InterfaceFactoryImpl::OnDestroyPending(base::OnceClosure destroy_cb) {
DVLOG(1) << __func__;
destroy_cb_ = std::move(destroy_cb);
if (IsEmpty())
std::move(destroy_cb_).Run();
// else the callback will be called when IsEmpty() becomes true.
}
bool InterfaceFactoryImpl::IsEmpty() {
#if BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
if (!audio_decoder_bindings_.empty())
return false;
#endif // BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
#if BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
if (!video_decoder_bindings_.empty())
return false;
#endif // BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
#if BUILDFLAG(ENABLE_MOJO_RENDERER)
if (!renderer_bindings_.empty())
return false;
#endif // BUILDFLAG(ENABLE_MOJO_RENDERER)
#if BUILDFLAG(ENABLE_MOJO_CDM)
if (!cdm_bindings_.empty())
return false;
#endif // BUILDFLAG(ENABLE_MOJO_CDM)
#if BUILDFLAG(ENABLE_LIBRARY_CDMS)
if (!cdm_proxy_bindings_.empty())
return false;
#endif // BUILDFLAG(ENABLE_LIBRARY_CDMS)
if (!decryptor_bindings_.empty())
return false;
return true;
}
void InterfaceFactoryImpl::SetBindingConnectionErrorHandler() {
// base::Unretained is safe because all bindings are owned by |this|. If
// |this| is destructed, the bindings will be destructed as well and the
// connection error handler should never be called.
auto connection_error_cb = base::BindRepeating(
&InterfaceFactoryImpl::OnBindingConnectionError, base::Unretained(this));
#if BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
audio_decoder_bindings_.set_connection_error_handler(connection_error_cb);
#endif // BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
#if BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
video_decoder_bindings_.set_connection_error_handler(connection_error_cb);
#endif // BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
#if BUILDFLAG(ENABLE_MOJO_RENDERER)
renderer_bindings_.set_connection_error_handler(connection_error_cb);
#endif // BUILDFLAG(ENABLE_MOJO_RENDERER)
#if BUILDFLAG(ENABLE_MOJO_CDM)
cdm_bindings_.set_connection_error_handler(connection_error_cb);
#endif // BUILDFLAG(ENABLE_MOJO_CDM)
#if BUILDFLAG(ENABLE_LIBRARY_CDMS)
cdm_proxy_bindings_.set_connection_error_handler(connection_error_cb);
#endif // BUILDFLAG(ENABLE_LIBRARY_CDMS)
decryptor_bindings_.set_connection_error_handler(connection_error_cb);
}
void InterfaceFactoryImpl::OnBindingConnectionError() {
DVLOG(2) << __func__;
if (destroy_cb_ && IsEmpty())
std::move(destroy_cb_).Run();
}
#if BUILDFLAG(ENABLE_MOJO_CDM)
CdmFactory* InterfaceFactoryImpl::GetCdmFactory() {
if (!cdm_factory_) {
cdm_factory_ = mojo_media_client_->CreateCdmFactory(interfaces_.get());
LOG_IF(ERROR, !cdm_factory_) << "CdmFactory not available.";
}
return cdm_factory_.get();
}
#endif // BUILDFLAG(ENABLE_MOJO_CDM)
} // namespace media