#ifndef MEDIA_GPU_CHROMEOS_OOP_VIDEO_DECODER_H_
#define MEDIA_GPU_CHROMEOS_OOP_VIDEO_DECODER_H_
#include "base/containers/lru_cache.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/scoped_refptr.h"
#include "base/memory/weak_ptr.h"
#include "base/sequence_checker.h"
#include "base/task/sequenced_task_runner.h"
#include "base/time/time.h"
#include "media/base/media_log.h"
#include "media/gpu/chromeos/video_decoder_pipeline.h"
#include "media/gpu/media_gpu_export.h"
#include "media/mojo/mojom/media_log.mojom.h"
#include "media/mojo/mojom/video_decoder.mojom.h"
#include "mojo/public/cpp/bindings/associated_receiver.h"
#include "mojo/public/cpp/bindings/receiver.h"
#include "mojo/public/cpp/bindings/remote.h"
#if BUILDFLAG(IS_CHROMEOS)
namespace chromeos {
class CdmContextForOOPVDImpl;
}
#endif
namespace media {
class MediaLog;
class MojoDecoderBufferWriter;
class OOPVideoDecoder : public VideoDecoderMixin,
public mojom::VideoDecoderClient,
public mojom::MediaLog {
public:
OOPVideoDecoder(const OOPVideoDecoder&) = delete;
OOPVideoDecoder& operator=(const OOPVideoDecoder&) = delete;
static std::unique_ptr<VideoDecoderMixin> Create(
mojo::PendingRemote<mojom::VideoDecoder> pending_remote_decoder,
std::unique_ptr<media::MediaLog> media_log,
scoped_refptr<base::SequencedTaskRunner> decoder_task_runner,
base::WeakPtr<VideoDecoderMixin::Client> client);
static void NotifySupportKnown(
mojo::PendingRemote<mojom::VideoDecoder> oop_video_decoder,
base::OnceCallback<void(mojo::PendingRemote<mojom::VideoDecoder>)> cb);
static std::optional<SupportedVideoDecoderConfigs> GetSupportedConfigs();
static void ResetGlobalStateForTesting();
void Initialize(const VideoDecoderConfig& config,
bool low_delay,
CdmContext* cdm_context,
InitCB init_cb,
const PipelineOutputCB& output_cb,
const WaitingCB& waiting_cb) override;
void Decode(scoped_refptr<DecoderBuffer> buffer, DecodeCB decode_cb) override;
void Reset(base::OnceClosure reset_cb) override;
bool NeedsBitstreamConversion() const override;
bool CanReadWithoutStalling() const override;
int GetMaxDecodeRequests() const override;
VideoDecoderType GetDecoderType() const override;
bool IsPlatformDecoder() const override;
void ApplyResolutionChange() override;
bool NeedsTranscryption() override;
void OnVideoFrameDecoded(
const scoped_refptr<VideoFrame>& frame,
bool can_read_without_stalling,
const std::optional<base::UnguessableToken>& release_token) final;
void OnWaiting(WaitingReason reason) final;
void RequestOverlayInfo() final;
void AddLogRecord(const MediaLogRecord& event) final;
FrameResource* GetOriginalFrame(const base::UnguessableToken& tracking_token);
private:
OOPVideoDecoder(
std::unique_ptr<media::MediaLog> media_log,
scoped_refptr<base::SequencedTaskRunner> decoder_task_runner,
base::WeakPtr<VideoDecoderMixin::Client> client,
mojo::PendingRemote<mojom::VideoDecoder> pending_remote_decoder);
~OOPVideoDecoder() override;
void OnInitializeDone(const DecoderStatus& status,
bool needs_bitstream_conversion,
int32_t max_decode_requests,
VideoDecoderType decoder_type,
bool needs_transcryption);
void OnDecodeDone(uint64_t decode_id,
bool is_flush_cb,
const DecoderStatus& status);
void DeferDecodeCallback(DecodeCB decode_cb, const DecoderStatus& status);
void CallDeferredDecodeCallback(DecodeCB decode_cb,
const DecoderStatus& status);
bool HasPendingDecodeCallbacks() const;
void OnResetDone();
void CallResetCallback();
void Stop();
void ReleaseVideoFrame(const base::UnguessableToken& release_token);
InitCB init_cb_ GUARDED_BY_CONTEXT(sequence_checker_);
PipelineOutputCB output_cb_ GUARDED_BY_CONTEXT(sequence_checker_);
WaitingCB waiting_cb_ GUARDED_BY_CONTEXT(sequence_checker_);
uint64_t decode_counter_ GUARDED_BY_CONTEXT(sequence_checker_) = 0;
std::map<uint64_t, DecodeCB> pending_decodes_
GUARDED_BY_CONTEXT(sequence_checker_);
uint64_t num_deferred_decode_cbs_ GUARDED_BY_CONTEXT(sequence_checker_) = 0;
bool is_flushing_ GUARDED_BY_CONTEXT(sequence_checker_) = false;
base::TimeDelta current_fake_timestamp_
GUARDED_BY_CONTEXT(sequence_checker_) = base::Microseconds(0u);
base::LRUCache<base::TimeDelta, base::TimeDelta>
fake_timestamp_to_real_timestamp_cache_
GUARDED_BY_CONTEXT(sequence_checker_);
base::OnceClosure reset_cb_ GUARDED_BY_CONTEXT(sequence_checker_);
mojo::AssociatedReceiver<mojom::VideoDecoderClient> client_receiver_
GUARDED_BY_CONTEXT(sequence_checker_){this};
mojo::Receiver<mojom::MediaLog> media_log_receiver_
GUARDED_BY_CONTEXT(sequence_checker_){this};
#if BUILDFLAG(IS_CHROMEOS)
std::unique_ptr<chromeos::CdmContextForOOPVDImpl> cdm_context_for_oopvd_
GUARDED_BY_CONTEXT(sequence_checker_);
std::unique_ptr<mojo::Receiver<mojom::CdmContextForOOPVD>>
cdm_context_for_oopvd_receiver_ GUARDED_BY_CONTEXT(sequence_checker_);
#endif
bool initialized_for_protected_content_
GUARDED_BY_CONTEXT(sequence_checker_) = false;
bool needs_bitstream_conversion_ GUARDED_BY_CONTEXT(sequence_checker_) =
false;
int32_t max_decode_requests_ GUARDED_BY_CONTEXT(sequence_checker_) = 8u;
VideoDecoderType remote_decoder_type_ GUARDED_BY_CONTEXT(sequence_checker_) =
VideoDecoderType::kUnknown;
mojo::Remote<mojom::VideoDecoder> remote_decoder_
GUARDED_BY_CONTEXT(sequence_checker_);
bool has_error_ GUARDED_BY_CONTEXT(sequence_checker_) = false;
mojo::Remote<mojom::VideoFrameHandleReleaser>
video_frame_handle_releaser_remote_ GUARDED_BY_CONTEXT(sequence_checker_);
std::unique_ptr<MojoDecoderBufferWriter> mojo_decoder_buffer_writer_
GUARDED_BY_CONTEXT(sequence_checker_);
bool can_read_without_stalling_ GUARDED_BY_CONTEXT(sequence_checker_) = false;
bool needs_transcryption_ GUARDED_BY_CONTEXT(sequence_checker_) = false;
base::flat_map<base::UnguessableToken, scoped_refptr<FrameResource>>
received_token_to_decoded_frame_map_
GUARDED_BY_CONTEXT(sequence_checker_);
base::flat_map<base::UnguessableToken,
raw_ptr<FrameResource, CtnExperimental>>
generated_token_to_decoded_frame_map_
GUARDED_BY_CONTEXT(sequence_checker_);
SEQUENCE_CHECKER(sequence_checker_);
base::WeakPtrFactory<OOPVideoDecoder> weak_this_factory_
GUARDED_BY_CONTEXT(sequence_checker_);
};
}
#endif