blob: 51cce17cb097108f33927b4d4521091c11c41630 [file] [log] [blame]
// Copyright 2016 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/gpu_mojo_media_client.h"
#include <utility>
#include "base/bind.h"
#include "base/feature_list.h"
#include "base/memory/ptr_util.h"
#include "gpu/ipc/service/gpu_channel.h"
#include "media/base/audio_decoder.h"
#include "media/base/cdm_factory.h"
#include "media/base/fallback_video_decoder.h"
#include "media/base/media_switches.h"
#include "media/base/video_decoder.h"
#include "media/gpu/buildflags.h"
#include "media/gpu/gpu_video_accelerator_util.h"
#include "media/gpu/gpu_video_decode_accelerator_factory.h"
#include "media/gpu/gpu_video_decode_accelerator_helpers.h"
#include "media/gpu/ipc/service/media_gpu_channel_manager.h"
#include "media/gpu/ipc/service/vda_video_decoder.h"
#include "media/mojo/mojom/video_decoder.mojom.h"
#include "media/video/video_decode_accelerator.h"
#if defined(OS_ANDROID)
#include "base/memory/ptr_util.h"
#include "media/base/android/android_cdm_factory.h"
#include "media/filters/android/media_codec_audio_decoder.h"
#include "media/gpu/android/android_video_surface_chooser_impl.h"
#include "media/gpu/android/codec_allocator.h"
#include "media/gpu/android/direct_shared_image_video_provider.h"
#include "media/gpu/android/maybe_render_early_manager.h"
#include "media/gpu/android/media_codec_video_decoder.h"
#include "media/gpu/android/video_frame_factory_impl.h"
#include "media/mojo/mojom/media_drm_storage.mojom.h"
#include "media/mojo/mojom/provision_fetcher.mojom.h"
#include "media/mojo/services/mojo_media_drm_storage.h"
#include "media/mojo/services/mojo_provision_fetcher.h"
#include "services/service_manager/public/cpp/connect.h"
#endif // defined(OS_ANDROID)
#if defined(OS_WIN)
#include "media/gpu/windows/d3d11_video_decoder.h"
#include "ui/gl/gl_angle_util_win.h"
#endif // defined(OS_WIN)
#if defined(OS_CHROMEOS)
#include "media/gpu/chromeos/chromeos_video_decoder_factory.h"
#if BUILDFLAG(USE_V4L2_CODEC) || BUILDFLAG(USE_VAAPI)
#include "media/gpu/linux/mailbox_video_frame_converter.h"
#include "media/gpu/linux/platform_video_frame_pool.h"
#endif // BUILDFLAG(USE_V4L2_CODEC) || BUILDFLAG(USE_VAAPI)
#endif // defined(OS_CHROMEOS)
#if defined(OS_ANDROID)
#include "media/mojo/services/android_mojo_util.h"
using media::android_mojo_util::CreateProvisionFetcher;
using media::android_mojo_util::CreateMediaDrmStorage;
#endif // defined(OS_ANDROID)
namespace media {
namespace {
#if defined(OS_ANDROID) || defined(OS_CHROMEOS) || defined(OS_MACOSX) || \
defined(OS_WIN) || defined(OS_LINUX)
gpu::CommandBufferStub* GetCommandBufferStub(
scoped_refptr<base::SingleThreadTaskRunner> gpu_task_runner,
base::WeakPtr<MediaGpuChannelManager> media_gpu_channel_manager,
base::UnguessableToken channel_token,
int32_t route_id) {
DCHECK(gpu_task_runner->BelongsToCurrentThread());
if (!media_gpu_channel_manager)
return nullptr;
gpu::GpuChannel* channel =
media_gpu_channel_manager->LookupChannel(channel_token);
if (!channel)
return nullptr;
gpu::CommandBufferStub* stub = channel->LookupCommandBuffer(route_id);
if (!stub)
return nullptr;
// Only allow stubs that have a ContextGroup, that is, the GLES2 ones. Later
// code assumes the ContextGroup is valid.
if (!stub->decoder_context()->GetContextGroup())
return nullptr;
return stub;
}
#endif
#if defined(OS_WIN)
// Return a callback to get the D3D11 device for D3D11VideoDecoder. Since it
// only supports the ANGLE device right now, that's what we return.
D3D11VideoDecoder::GetD3D11DeviceCB GetD3D11DeviceCallback() {
return base::BindRepeating(
[]() { return gl::QueryD3D11DeviceObjectFromANGLE(); });
}
#endif
} // namespace
GpuMojoMediaClient::GpuMojoMediaClient(
const gpu::GpuPreferences& gpu_preferences,
const gpu::GpuDriverBugWorkarounds& gpu_workarounds,
const gpu::GpuFeatureInfo& gpu_feature_info,
scoped_refptr<base::SingleThreadTaskRunner> gpu_task_runner,
base::WeakPtr<MediaGpuChannelManager> media_gpu_channel_manager,
AndroidOverlayMojoFactoryCB android_overlay_factory_cb,
CdmProxyFactoryCB cdm_proxy_factory_cb)
: gpu_preferences_(gpu_preferences),
gpu_workarounds_(gpu_workarounds),
gpu_feature_info_(gpu_feature_info),
gpu_task_runner_(std::move(gpu_task_runner)),
media_gpu_channel_manager_(std::move(media_gpu_channel_manager)),
android_overlay_factory_cb_(std::move(android_overlay_factory_cb)),
cdm_proxy_factory_cb_(std::move(cdm_proxy_factory_cb)) {}
GpuMojoMediaClient::~GpuMojoMediaClient() = default;
void GpuMojoMediaClient::Initialize(service_manager::Connector* connector) {}
std::unique_ptr<AudioDecoder> GpuMojoMediaClient::CreateAudioDecoder(
scoped_refptr<base::SingleThreadTaskRunner> task_runner) {
#if defined(OS_ANDROID)
return std::make_unique<MediaCodecAudioDecoder>(task_runner);
#else
return nullptr;
#endif // defined(OS_ANDROID)
}
SupportedVideoDecoderConfigMap
GpuMojoMediaClient::GetSupportedVideoDecoderConfigs() {
#if defined(OS_ANDROID)
static SupportedVideoDecoderConfigMap supported_configs{
{VideoDecoderImplementation::kDefault,
MediaCodecVideoDecoder::GetSupportedConfigs()},
};
return supported_configs;
#else
SupportedVideoDecoderConfigMap supported_config_map;
#if defined(OS_WIN)
// Start with the configurations supported by D3D11VideoDecoder.
// VdaVideoDecoder is still used as a fallback.
if (!d3d11_supported_configs_) {
d3d11_supported_configs_ =
D3D11VideoDecoder::GetSupportedVideoDecoderConfigs(
gpu_preferences_, gpu_workarounds_, GetD3D11DeviceCallback());
}
supported_config_map[VideoDecoderImplementation::kAlternate] =
*d3d11_supported_configs_;
#elif defined(OS_CHROMEOS)
if (base::FeatureList::IsEnabled(kChromeosVideoDecoder)) {
if (!cros_supported_configs_) {
cros_supported_configs_ =
ChromeosVideoDecoderFactory::GetSupportedConfigs();
}
supported_config_map[VideoDecoderImplementation::kDefault] =
*cros_supported_configs_;
return supported_config_map;
}
#endif // defined(OS_WIN)
auto& default_configs =
supported_config_map[VideoDecoderImplementation::kDefault];
// VdaVideoDecoder will be used to wrap a VDA. Add the configs supported
// by the VDA implementation.
// TODO(sandersd): Move conversion code into VdaVideoDecoder.
VideoDecodeAccelerator::Capabilities capabilities =
GpuVideoAcceleratorUtil::ConvertGpuToMediaDecodeCapabilities(
GpuVideoDecodeAcceleratorFactory::GetDecoderCapabilities(
gpu_preferences_, gpu_workarounds_));
bool allow_encrypted =
capabilities.flags &
VideoDecodeAccelerator::Capabilities::SUPPORTS_ENCRYPTED_STREAMS;
SupportedVideoDecoderConfigs supported_configs = ConvertFromSupportedProfiles(
capabilities.supported_profiles, allow_encrypted);
default_configs.insert(default_configs.end(), supported_configs.begin(),
supported_configs.end());
return supported_config_map;
#endif // defined(OS_ANDROID)
}
std::unique_ptr<VideoDecoder> GpuMojoMediaClient::CreateVideoDecoder(
scoped_refptr<base::SingleThreadTaskRunner> task_runner,
MediaLog* media_log,
mojom::CommandBufferIdPtr command_buffer_id,
VideoDecoderImplementation implementation,
RequestOverlayInfoCB request_overlay_info_cb,
const gfx::ColorSpace& target_color_space) {
// All implementations require a command buffer.
if (!command_buffer_id)
return nullptr;
std::unique_ptr<VideoDecoder> video_decoder;
switch (implementation) {
case VideoDecoderImplementation::kDefault: {
#if defined(OS_ANDROID)
auto get_stub_cb = base::Bind(
&GetCommandBufferStub, gpu_task_runner_, media_gpu_channel_manager_,
command_buffer_id->channel_token, command_buffer_id->route_id);
auto image_provider = std::make_unique<DirectSharedImageVideoProvider>(
gpu_task_runner_, std::move(get_stub_cb));
video_decoder = std::make_unique<MediaCodecVideoDecoder>(
gpu_preferences_, gpu_feature_info_, DeviceInfo::GetInstance(),
CodecAllocator::GetInstance(gpu_task_runner_),
std::make_unique<AndroidVideoSurfaceChooserImpl>(
DeviceInfo::GetInstance()->IsSetOutputSurfaceSupported()),
android_overlay_factory_cb_, std::move(request_overlay_info_cb),
std::make_unique<VideoFrameFactoryImpl>(
gpu_task_runner_, gpu_preferences_, std::move(image_provider),
MaybeRenderEarlyManager::Create(gpu_task_runner_)));
#elif defined(OS_CHROMEOS)
std::unique_ptr<VideoDecoder> cros_video_decoder;
if (base::FeatureList::IsEnabled(kChromeosVideoDecoder)) {
#if BUILDFLAG(USE_V4L2_CODEC) || BUILDFLAG(USE_VAAPI)
auto frame_pool = std::make_unique<PlatformVideoFramePool>();
auto frame_converter = std::make_unique<MailboxVideoFrameConverter>(
base::BindRepeating(&DmabufVideoFramePool::UnwrapFrame,
base::Unretained(frame_pool.get())),
gpu_task_runner_,
base::BindOnce(&GetCommandBufferStub, gpu_task_runner_,
media_gpu_channel_manager_,
command_buffer_id->channel_token,
command_buffer_id->route_id));
cros_video_decoder = ChromeosVideoDecoderFactory::Create(
task_runner, std::move(frame_pool), std::move(frame_converter));
#endif // BUILDFLAG(USE_V4L2_CODEC) || BUILDFLAG(USE_VAAPI)
}
if (cros_video_decoder) {
video_decoder = std::move(cros_video_decoder);
} else {
video_decoder = VdaVideoDecoder::Create(
task_runner, gpu_task_runner_, media_log->Clone(),
target_color_space, gpu_preferences_, gpu_workarounds_,
base::BindRepeating(&GetCommandBufferStub, gpu_task_runner_,
media_gpu_channel_manager_,
command_buffer_id->channel_token,
command_buffer_id->route_id));
}
#elif defined(OS_MACOSX) || defined(OS_WIN) || defined(OS_LINUX)
video_decoder = VdaVideoDecoder::Create(
task_runner, gpu_task_runner_, media_log->Clone(), target_color_space,
gpu_preferences_, gpu_workarounds_,
base::BindRepeating(&GetCommandBufferStub, gpu_task_runner_,
media_gpu_channel_manager_,
command_buffer_id->channel_token,
command_buffer_id->route_id));
#endif // defined(OS_ANDROID)
} break;
case VideoDecoderImplementation::kAlternate:
#if defined(OS_WIN)
if (base::FeatureList::IsEnabled(kD3D11VideoDecoder)) {
// If nothing has cached the configs yet, then do so now.
if (!d3d11_supported_configs_)
GetSupportedVideoDecoderConfigs();
video_decoder = D3D11VideoDecoder::Create(
gpu_task_runner_, media_log->Clone(), gpu_preferences_,
gpu_workarounds_,
base::BindRepeating(&GetCommandBufferStub, gpu_task_runner_,
media_gpu_channel_manager_,
command_buffer_id->channel_token,
command_buffer_id->route_id),
GetD3D11DeviceCallback(), *d3d11_supported_configs_);
}
#endif // defined(OS_WIN)
break;
}; // switch
// |video_decoder| may be null if we don't support |implementation|.
return video_decoder;
}
std::unique_ptr<CdmFactory> GpuMojoMediaClient::CreateCdmFactory(
service_manager::mojom::InterfaceProvider* interface_provider) {
#if defined(OS_ANDROID)
return std::make_unique<AndroidCdmFactory>(
base::Bind(&CreateProvisionFetcher, interface_provider),
base::Bind(&CreateMediaDrmStorage, interface_provider));
#else
return nullptr;
#endif // defined(OS_ANDROID)
}
#if BUILDFLAG(ENABLE_LIBRARY_CDMS)
std::unique_ptr<CdmProxy> GpuMojoMediaClient::CreateCdmProxy(
const base::Token& cdm_guid) {
if (cdm_proxy_factory_cb_)
return cdm_proxy_factory_cb_.Run(cdm_guid);
return nullptr;
}
#endif // BUILDFLAG(ENABLE_LIBRARY_CDMS)
} // namespace media