blob: 8f0963ff71fd0f021745cd9888d6570a36ebec24 [file] [log] [blame]
// Copyright 2016 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CHROMECAST_MEDIA_SERVICE_CAST_RENDERER_H_
#define CHROMECAST_MEDIA_SERVICE_CAST_RENDERER_H_
#include <memory>
#include "base/memory/weak_ptr.h"
#include "base/no_destructor.h"
#include "base/unguessable_token.h"
#include "chromecast/common/mojom/multiroom.mojom.h"
#include "chromecast/common/mojom/service_connector.mojom.h"
#include "chromecast/external_mojo/external_service_support/external_connector.h"
#include "chromecast/media/api/cma_backend_factory.h"
#include "chromecast/media/base/video_resolution_policy.h"
#include "chromecast/media/service/mojom/video_geometry_setter.mojom.h"
#include "media/base/renderer.h"
#include "media/base/waiting.h"
#include "media/mojo/mojom/cast_application_media_info_manager.mojom.h"
#include "media/mojo/mojom/frame_interface_factory.mojom.h"
#include "mojo/public/cpp/bindings/receiver.h"
#include "mojo/public/cpp/bindings/remote.h"
#include "third_party/abseil-cpp/absl/types/optional.h"
#include "ui/gfx/geometry/rect_f.h"
#include "ui/gfx/overlay_transform.h"
namespace base {
class SingleThreadTaskRunner;
} // namespace base
namespace chromecast {
class TaskRunnerImpl;
namespace media {
class BalancedMediaTaskRunnerFactory;
class CastCdmContext;
class MediaPipelineImpl;
class VideoGeometrySetterService;
class VideoModeSwitcher;
class CastRenderer final : public ::media::Renderer,
public VideoResolutionPolicy::Observer,
public mojom::VideoGeometryChangeClient {
public:
// |frame_interfaces| provides interfaces tied to RenderFrameHost.
CastRenderer(CmaBackendFactory* backend_factory,
const scoped_refptr<base::SingleThreadTaskRunner>& task_runner,
VideoModeSwitcher* video_mode_switcher,
VideoResolutionPolicy* video_resolution_policy,
const base::UnguessableToken& overlay_plane_id,
::media::mojom::FrameInterfaceFactory* frame_interfaces,
external_service_support::ExternalConnector* connector,
bool is_buffering_enabled);
CastRenderer(const CastRenderer&) = delete;
CastRenderer& operator=(const CastRenderer&) = delete;
~CastRenderer() override;
// For CmaBackend implementation, CastRenderer must be connected to
// VideoGeometrySetterService.
void SetVideoGeometrySetterService(
VideoGeometrySetterService* video_geometry_setter_service);
// ::media::Renderer implementation.
void Initialize(::media::MediaResource* media_resource,
::media::RendererClient* client,
::media::PipelineStatusCallback init_cb) override;
void SetCdm(::media::CdmContext* cdm_context,
CdmAttachedCB cdm_attached_cb) override;
void SetLatencyHint(absl::optional<base::TimeDelta> latency_hint) override;
void Flush(base::OnceClosure flush_cb) override;
void StartPlayingFrom(base::TimeDelta time) override;
void SetPlaybackRate(double playback_rate) override;
void SetVolume(float volume) override;
base::TimeDelta GetMediaTime() override;
::media::RendererType GetRendererType() override;
// VideoResolutionPolicy::Observer implementation.
void OnVideoResolutionPolicyChanged() override;
// mojom::VideoGeometryChangeClient implementation.
void OnVideoGeometryChange(const gfx::RectF& rect_f,
gfx::OverlayTransform transform) override;
// TODO(guohuideng): For now we use a global callback to gain access to
// VideoPlaneController so CastRenderer can set the video geometry. We
// should separate the SetGeometry from VideoPlaneController and get rid
// of this callback. see b/79266094.
using OverlayCompositedCallback =
base::RepeatingCallback<void(const gfx::RectF&, gfx::OverlayTransform)>;
static void SetOverlayCompositedCallback(const OverlayCompositedCallback& cb);
private:
enum Stream { STREAM_AUDIO, STREAM_VIDEO };
void OnSubscribeToVideoGeometryChange(::media::MediaResource* media_resource,
::media::RendererClient* client);
void OnApplicationMediaInfoReceived(
::media::MediaResource* media_resource,
::media::RendererClient* client,
::media::mojom::CastApplicationMediaInfoPtr application_media_info);
void OnGetMultiroomInfo(
::media::MediaResource* media_resource,
::media::RendererClient* client,
::media::mojom::CastApplicationMediaInfoPtr application_media_info,
chromecast::mojom::MultiroomInfoPtr multiroom_info);
void OnError(::media::PipelineStatus status);
void OnEnded(Stream stream);
void OnStatisticsUpdate(const ::media::PipelineStatistics& stats);
void OnBufferingStateChange(::media::BufferingState state,
::media::BufferingStateChangeReason reason);
void OnWaiting(::media::WaitingReason reason);
void OnVideoNaturalSizeChange(const gfx::Size& size);
void OnVideoOpacityChange(bool opaque);
void CheckVideoResolutionPolicy();
void RunInitCallback(::media::PipelineStatus status);
void OnVideoInitializationFinished(::media::PipelineStatus status);
CmaBackendFactory* const backend_factory_;
scoped_refptr<base::SingleThreadTaskRunner> task_runner_;
VideoModeSwitcher* video_mode_switcher_;
VideoResolutionPolicy* video_resolution_policy_;
base::UnguessableToken overlay_plane_id_;
mojo::Remote<chromecast::mojom::ServiceConnector> service_connector_;
::media::mojom::FrameInterfaceFactory* frame_interfaces_;
external_service_support::ExternalConnector* const connector_;
::media::RendererClient* client_;
CastCdmContext* cast_cdm_context_;
scoped_refptr<BalancedMediaTaskRunnerFactory> media_task_runner_factory_;
std::unique_ptr<TaskRunnerImpl> backend_task_runner_;
std::unique_ptr<MediaPipelineImpl> pipeline_;
bool eos_[2];
gfx::Size video_res_;
mojo::Remote<::media::mojom::CastApplicationMediaInfoManager>
application_media_info_manager_remote_;
VideoGeometrySetterService* video_geometry_setter_service_;
mojo::Remote<mojom::VideoGeometryChangeSubscriber>
video_geometry_change_subcriber_remote_;
mojo::Remote<chromecast::mojom::MultiroomManager> multiroom_manager_;
::media::PipelineStatusCallback init_cb_;
mojo::Receiver<mojom::VideoGeometryChangeClient>
video_geometry_change_client_receiver_{this};
static OverlayCompositedCallback& GetOverlayCompositedCallback() {
static base::NoDestructor<OverlayCompositedCallback>
g_overlay_composited_callback;
return *g_overlay_composited_callback;
}
absl::optional<float> pending_volume_;
const bool is_buffering_enabled_;
base::WeakPtrFactory<CastRenderer> weak_factory_;
};
} // namespace media
} // namespace chromecast
#endif // CHROMECAST_MEDIA_SERVICE_CAST_RENDERER_H_