blob: 299f7cd759342573f1538774f60faaf38d6be6ce [file] [log] [blame]
// Copyright 2014 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 "chromecast/renderer/cast_content_renderer_client.h"
#include <utility>
#include "base/command_line.h"
#include "base/strings/string_number_conversions.h"
#include "chromecast/base/bitstream_audio_codecs.h"
#include "chromecast/base/chromecast_switches.h"
#include "chromecast/media/base/media_codec_support.h"
#include "chromecast/media/base/supported_codec_profile_levels_memo.h"
#include "chromecast/public/media/media_capabilities_shlib.h"
#include "chromecast/renderer/cast_media_playback_options.h"
#include "chromecast/renderer/media/key_systems_cast.h"
#include "chromecast/renderer/media/media_caps_observer_impl.h"
#include "components/network_hints/renderer/prescient_networking_dispatcher.h"
#include "content/public/common/content_switches.h"
#include "content/public/common/service_names.mojom.h"
#include "content/public/renderer/render_frame.h"
#include "content/public/renderer/render_thread.h"
#include "content/public/renderer/render_view.h"
#include "media/base/media.h"
#include "mojo/public/cpp/bindings/interface_request.h"
#include "services/service_manager/public/cpp/connector.h"
#include "services/service_manager/public/cpp/interface_provider.h"
#include "third_party/blink/public/platform/web_runtime_features.h"
#include "third_party/blink/public/web/web_frame_widget.h"
#include "third_party/blink/public/web/web_settings.h"
#include "third_party/blink/public/web/web_view.h"
#if defined(OS_ANDROID)
#include "media/base/android/media_codec_util.h"
#else
#include "chromecast/renderer/memory_pressure_observer_impl.h"
#endif // OS_ANDROID
#if !defined(OS_FUCHSIA)
#include "chromecast/crash/cast_crash_keys.h"
#endif // !defined(OS_FUCHSIA)
#if BUILDFLAG(ENABLE_CHROMECAST_EXTENSIONS)
#include "chromecast/common/cast_extensions_client.h"
#include "chromecast/renderer/cast_extensions_renderer_client.h"
#include "content/public/common/content_constants.h"
#include "extensions/common/common_manifest_handlers.h" // nogncheck
#include "extensions/common/extension_urls.h" // nogncheck
#include "extensions/renderer/dispatcher.h" // nogncheck
#include "extensions/renderer/extension_frame_helper.h" // nogncheck
#include "extensions/renderer/guest_view/extensions_guest_view_container.h" // nogncheck
#include "extensions/renderer/guest_view/extensions_guest_view_container_dispatcher.h" // nogncheck
#endif
namespace chromecast {
namespace shell {
CastContentRendererClient::CastContentRendererClient()
: supported_profiles_(new media::SupportedCodecProfileLevelsMemo()),
app_media_capabilities_observer_binding_(this),
supported_bitstream_audio_codecs_(kBitstreamAudioCodecNone) {
#if defined(OS_ANDROID)
DCHECK(::media::MediaCodecUtil::IsMediaCodecAvailable())
<< "MediaCodec is not available!";
// Platform decoder support must be enabled before we set the
// IsCodecSupportedCB because the latter instantiates the lazy MimeUtil
// instance, which caches the platform decoder supported state when it is
// constructed.
::media::EnablePlatformDecoderSupport();
#endif // OS_ANDROID
}
CastContentRendererClient::~CastContentRendererClient() = default;
void CastContentRendererClient::RenderThreadStarted() {
// Register as observer for media capabilities
content::RenderThread* thread = content::RenderThread::Get();
media::mojom::MediaCapsPtr media_caps;
thread->GetConnector()->BindInterface(content::mojom::kBrowserServiceName,
&media_caps);
media::mojom::MediaCapsObserverPtr proxy;
media_caps_observer_.reset(
new media::MediaCapsObserverImpl(&proxy, supported_profiles_.get()));
media_caps->AddObserver(std::move(proxy));
#if !defined(OS_ANDROID)
// Register to observe memory pressure changes
chromecast::mojom::MemoryPressureControllerPtr memory_pressure_controller;
thread->GetConnector()->BindInterface(content::mojom::kBrowserServiceName,
&memory_pressure_controller);
chromecast::mojom::MemoryPressureObserverPtr memory_pressure_proxy;
memory_pressure_observer_.reset(
new MemoryPressureObserverImpl(&memory_pressure_proxy));
memory_pressure_controller->AddObserver(std::move(memory_pressure_proxy));
#endif
prescient_networking_dispatcher_.reset(
new network_hints::PrescientNetworkingDispatcher());
#if !defined(OS_FUCHSIA)
// TODO(crbug.com/753619): Enable crash reporting on Fuchsia.
base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
std::string last_launched_app =
command_line->GetSwitchValueNative(switches::kLastLaunchedApp);
if (!last_launched_app.empty())
crash_keys::last_app.Set(last_launched_app);
std::string previous_app =
command_line->GetSwitchValueNative(switches::kPreviousApp);
if (!previous_app.empty())
crash_keys::previous_app.Set(previous_app);
#endif // !defined(OS_FUCHSIA)
#if BUILDFLAG(ENABLE_CHROMECAST_EXTENSIONS)
extensions_client_ = std::make_unique<extensions::CastExtensionsClient>();
extensions::ExtensionsClient::Set(extensions_client_.get());
extensions_renderer_client_ =
std::make_unique<extensions::CastExtensionsRendererClient>();
extensions::ExtensionsRendererClient::Set(extensions_renderer_client_.get());
thread->AddObserver(extensions_renderer_client_->GetDispatcher());
guest_view_container_dispatcher_ =
std::make_unique<extensions::ExtensionsGuestViewContainerDispatcher>();
thread->AddObserver(guest_view_container_dispatcher_.get());
#endif
}
void CastContentRendererClient::RenderViewCreated(
content::RenderView* render_view) {
blink::WebView* webview = render_view->GetWebView();
if (webview) {
if (auto* web_frame_widget = render_view->GetWebFrameWidget())
web_frame_widget->SetBaseBackgroundColor(chromecast::GetSwitchValueColor(
switches::kCastAppBackgroundColor, SK_ColorBLACK));
// Disable application cache as Chromecast doesn't support off-line
// application running.
webview->GetSettings()->SetOfflineWebApplicationCacheEnabled(false);
}
}
void CastContentRendererClient::RenderFrameCreated(
content::RenderFrame* render_frame) {
DCHECK(render_frame);
// Lifetime is tied to |render_frame| via content::RenderFrameObserver.
new CastMediaPlaybackOptions(render_frame);
if (!app_media_capabilities_observer_binding_.is_bound()) {
mojom::ApplicationMediaCapabilitiesObserverPtr observer;
app_media_capabilities_observer_binding_.Bind(mojo::MakeRequest(&observer));
mojom::ApplicationMediaCapabilitiesPtr app_media_capabilities;
render_frame->GetRemoteInterfaces()->GetInterface(
mojo::MakeRequest(&app_media_capabilities));
app_media_capabilities->AddObserver(std::move(observer));
}
#if BUILDFLAG(ENABLE_CHROMECAST_EXTENSIONS)
extensions::Dispatcher* dispatcher =
extensions_renderer_client_->GetDispatcher();
// ExtensionFrameHelper destroys itself when the RenderFrame is destroyed.
new extensions::ExtensionFrameHelper(render_frame, dispatcher);
dispatcher->OnRenderFrameCreated(render_frame);
#endif
}
content::BrowserPluginDelegate*
CastContentRendererClient::CreateBrowserPluginDelegate(
content::RenderFrame* render_frame,
const content::WebPluginInfo& info,
const std::string& mime_type,
const GURL& original_url) {
#if BUILDFLAG(ENABLE_CHROMECAST_EXTENSIONS)
if (mime_type == content::kBrowserPluginMimeType) {
return new extensions::ExtensionsGuestViewContainer(render_frame);
}
#endif
return nullptr;
}
void CastContentRendererClient::RunScriptsAtDocumentStart(
content::RenderFrame* render_frame) {
#if BUILDFLAG(ENABLE_CHROMECAST_EXTENSIONS)
extensions_renderer_client_->GetDispatcher()->RunScriptsAtDocumentStart(
render_frame);
#endif
}
void CastContentRendererClient::RunScriptsAtDocumentEnd(
content::RenderFrame* render_frame) {
#if BUILDFLAG(ENABLE_CHROMECAST_EXTENSIONS)
extensions_renderer_client_->GetDispatcher()->RunScriptsAtDocumentEnd(
render_frame);
#endif
}
void CastContentRendererClient::AddSupportedKeySystems(
std::vector<std::unique_ptr<::media::KeySystemProperties>>*
key_systems_properties) {
media::AddChromecastKeySystems(key_systems_properties,
false /* enable_persistent_license_support */,
false /* force_software_crypto */);
}
bool CastContentRendererClient::IsSupportedAudioConfig(
const ::media::AudioConfig& config) {
#if defined(OS_ANDROID)
// No ATV device we know of has (E)AC3 decoder, so it relies on the audio sink
// device.
if (config.codec == ::media::kCodecEAC3)
return kBitstreamAudioCodecEac3 & supported_bitstream_audio_codecs_;
if (config.codec == ::media::kCodecAC3)
return kBitstreamAudioCodecAc3 & supported_bitstream_audio_codecs_;
if (config.codec == ::media::kCodecMpegHAudio)
return kBitstreamAudioCodecMpegHAudio & supported_bitstream_audio_codecs_;
// TODO(sanfin): Implement this for Android.
return true;
#else
// If the HDMI sink supports bitstreaming the codec, then the vendor backend
// does not need to support it.
if (IsSupportedBitstreamAudioCodec(config.codec)) {
return true;
}
media::AudioCodec codec = media::ToCastAudioCodec(config.codec);
// Cast platform implements software decoding of Opus and FLAC, so only PCM
// support is necessary in order to support Opus and FLAC.
if (codec == media::kCodecOpus || codec == media::kCodecFLAC)
codec = media::kCodecPCM;
media::AudioConfig cast_audio_config;
cast_audio_config.codec = codec;
return media::MediaCapabilitiesShlib::IsSupportedAudioConfig(
cast_audio_config);
#endif
}
bool CastContentRendererClient::IsSupportedVideoConfig(
const ::media::VideoConfig& config) {
// TODO(servolk): make use of eotf.
#if defined(OS_ANDROID)
return supported_profiles_->IsSupportedVideoConfig(
media::ToCastVideoCodec(config.codec, config.profile),
media::ToCastVideoProfile(config.profile), config.level);
#else
return media::MediaCapabilitiesShlib::IsSupportedVideoConfig(
media::ToCastVideoCodec(config.codec, config.profile),
media::ToCastVideoProfile(config.profile), config.level);
#endif
}
bool CastContentRendererClient::IsSupportedBitstreamAudioCodec(
::media::AudioCodec codec) {
return (codec == ::media::kCodecAC3 &&
(kBitstreamAudioCodecAc3 & supported_bitstream_audio_codecs_)) ||
(codec == ::media::kCodecEAC3 &&
(kBitstreamAudioCodecEac3 & supported_bitstream_audio_codecs_)) ||
(codec == ::media::kCodecMpegHAudio &&
(kBitstreamAudioCodecMpegHAudio & supported_bitstream_audio_codecs_));
}
blink::WebPrescientNetworking*
CastContentRendererClient::GetPrescientNetworking() {
return prescient_networking_dispatcher_.get();
}
bool CastContentRendererClient::DeferMediaLoad(
content::RenderFrame* render_frame,
bool render_frame_has_played_media_before,
base::OnceClosure closure) {
return RunWhenInForeground(render_frame, std::move(closure));
}
bool CastContentRendererClient::RunWhenInForeground(
content::RenderFrame* render_frame,
base::OnceClosure closure) {
auto* playback_options = CastMediaPlaybackOptions::Get(render_frame);
DCHECK(playback_options);
return playback_options->RunWhenInForeground(std::move(closure));
}
bool CastContentRendererClient::IsIdleMediaSuspendEnabled() {
return false;
}
bool CastContentRendererClient::IsBackgroundMediaSuspendEnabled(
content::RenderFrame* render_frame) {
auto* playback_options = CastMediaPlaybackOptions::Get(render_frame);
DCHECK(playback_options);
return playback_options->IsBackgroundSuspendEnabled();
}
void CastContentRendererClient::
SetRuntimeFeaturesDefaultsBeforeBlinkInitialization() {
// Settings for ATV (Android defaults are not what we want).
blink::WebRuntimeFeatures::EnableMediaControlsOverlayPlayButton(false);
}
void CastContentRendererClient::OnSupportedBitstreamAudioCodecsChanged(
int codecs) {
supported_bitstream_audio_codecs_ = codecs;
}
} // namespace shell
} // namespace chromecast