blob: 8684e3cd5899c4eb1175aa8ec2900785f6e9d45d [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 "base/logging.h"
#include "base/memory/ptr_util.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_media_client.h"
#include "mojo/public/cpp/bindings/strong_binding.h"
#include "services/service_manager/public/interfaces/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/audio_renderer_sink.h"
#include "media/base/renderer_factory.h"
#include "media/base/video_renderer_sink.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)
namespace media {
namespace {
constexpr base::TimeDelta kServiceContextRefReleaseDelay =
base::TimeDelta::FromSeconds(5);
void DeleteServiceContextRef(service_manager::ServiceContextRef* ref) {
delete ref;
}
} // namespace
// Helper class to help delay the release of service_manager::ServiceContextRef
// by |kServiceContextRefReleaseDelay|, which will ultimately delay MediaService
// destruction by the same delay as well. This helps reduce service connection
// time in cases like navigation.
class DelayedReleaseServiceContextRef {
public:
explicit DelayedReleaseServiceContextRef(
std::unique_ptr<service_manager::ServiceContextRef> ref)
: ref_(std::move(ref)),
task_runner_(base::ThreadTaskRunnerHandle::Get()) {}
~DelayedReleaseServiceContextRef() {
service_manager::ServiceContextRef* ref_ptr = ref_.release();
if (!task_runner_->PostNonNestableDelayedTask(
FROM_HERE, base::BindOnce(&DeleteServiceContextRef, ref_ptr),
kServiceContextRefReleaseDelay)) {
DeleteServiceContextRef(ref_ptr);
}
}
private:
std::unique_ptr<service_manager::ServiceContextRef> ref_;
scoped_refptr<base::SingleThreadTaskRunner> task_runner_;
DISALLOW_COPY_AND_ASSIGN(DelayedReleaseServiceContextRef);
};
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::make_unique<DelayedReleaseServiceContextRef>(
std::move(connection_ref))),
mojo_media_client_(mojo_media_client) {
DVLOG(1) << __func__;
DCHECK(mojo_media_client_);
}
InterfaceFactoryImpl::~InterfaceFactoryImpl() {
DVLOG(1) << __func__;
}
// mojom::InterfaceFactory implementation.
void InterfaceFactoryImpl::CreateAudioDecoder(
mojo::InterfaceRequest<mojom::AudioDecoder> request) {
#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) {
LOG(ERROR) << "AudioDecoder creation failed.";
return;
}
audio_decoder_bindings_.AddBinding(
base::MakeUnique<MojoAudioDecoderService>(&cdm_service_context_,
std::move(audio_decoder)),
std::move(request));
#endif // BUILDFLAG(ENABLE_MOJO_AUDIO_DECODER)
}
void InterfaceFactoryImpl::CreateVideoDecoder(
mojom::VideoDecoderRequest request) {
#if BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
video_decoder_bindings_.AddBinding(
base::MakeUnique<MojoVideoDecoderService>(mojo_media_client_,
&cdm_service_context_),
std::move(request));
#endif // BUILDFLAG(ENABLE_MOJO_VIDEO_DECODER)
}
void InterfaceFactoryImpl::CreateRenderer(
const std::string& audio_device_id,
mojo::InterfaceRequest<mojom::Renderer> request) {
#if BUILDFLAG(ENABLE_MOJO_RENDERER)
RendererFactory* renderer_factory = GetRendererFactory();
if (!renderer_factory)
return;
scoped_refptr<base::SingleThreadTaskRunner> task_runner(
base::ThreadTaskRunnerHandle::Get());
auto audio_sink =
mojo_media_client_->CreateAudioRendererSink(audio_device_id);
auto video_sink = mojo_media_client_->CreateVideoRendererSink(task_runner);
// TODO(hubbe): Find out if gfx::ColorSpace() is correct for the
// target_color_space.
auto renderer = renderer_factory->CreateRenderer(
task_runner, task_runner, audio_sink.get(), video_sink.get(),
RequestOverlayInfoCB(), gfx::ColorSpace());
if (!renderer) {
LOG(ERROR) << "Renderer creation failed.";
return;
}
std::unique_ptr<MojoRendererService> mojo_renderer_service =
base::MakeUnique<MojoRendererService>(
&cdm_service_context_, std::move(audio_sink), std::move(video_sink),
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) {
#if BUILDFLAG(ENABLE_MOJO_CDM)
CdmFactory* cdm_factory = GetCdmFactory();
if (!cdm_factory)
return;
cdm_bindings_.AddBinding(
base::MakeUnique<MojoCdmService>(&cdm_service_context_, cdm_factory),
std::move(request));
#endif // BUILDFLAG(ENABLE_MOJO_CDM)
}
#if BUILDFLAG(ENABLE_MOJO_RENDERER)
RendererFactory* InterfaceFactoryImpl::GetRendererFactory() {
if (!renderer_factory_) {
renderer_factory_ = mojo_media_client_->CreateRendererFactory(media_log_);
LOG_IF(ERROR, !renderer_factory_) << "RendererFactory not available.";
}
return renderer_factory_.get();
}
#endif // BUILDFLAG(ENABLE_MOJO_RENDERER)
#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