blob: 0eb1732d914c7aeb0a4d4c9778f83f9e13b4f9a4 [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/mojo_video_decoder_service.h"
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/logging.h"
#include "base/optional.h"
#include "base/threading/thread_task_runner_handle.h"
#include "media/base/decoder_buffer.h"
#include "media/base/video_decoder.h"
#include "media/base/video_decoder_config.h"
#include "media/base/video_frame.h"
#include "media/mojo/common/media_type_converters.h"
#include "media/mojo/common/mojo_decoder_buffer_converter.h"
#include "media/mojo/services/mojo_media_client.h"
#include "media/mojo/services/mojo_media_log.h"
#include "mojo/public/c/system/types.h"
#include "mojo/public/cpp/system/buffer.h"
#include "mojo/public/cpp/system/handle.h"
namespace media {
MojoVideoDecoderService::MojoVideoDecoderService(
MojoMediaClient* mojo_media_client)
: mojo_media_client_(mojo_media_client), weak_factory_(this) {
weak_this_ = weak_factory_.GetWeakPtr();
}
MojoVideoDecoderService::~MojoVideoDecoderService() {}
void MojoVideoDecoderService::Construct(
mojom::VideoDecoderClientAssociatedPtrInfo client,
mojom::MediaLogAssociatedPtrInfo media_log,
mojo::ScopedDataPipeConsumerHandle decoder_buffer_pipe,
mojom::CommandBufferIdPtr command_buffer_id) {
DVLOG(1) << __func__;
if (decoder_) {
// TODO(sandersd): Close the channel.
return;
}
client_.Bind(std::move(client));
mojom::MediaLogAssociatedPtr media_log_ptr;
media_log_ptr.Bind(std::move(media_log));
media_log_ = base::MakeUnique<MojoMediaLog>(std::move(media_log_ptr));
mojo_decoder_buffer_reader_.reset(
new MojoDecoderBufferReader(std::move(decoder_buffer_pipe)));
decoder_ = mojo_media_client_->CreateVideoDecoder(
base::ThreadTaskRunnerHandle::Get(), media_log_.get(),
std::move(command_buffer_id),
base::Bind(&MojoVideoDecoderService::OnDecoderOutputWithReleaseCB,
weak_this_),
base::Bind(&MojoVideoDecoderService::OnDecoderRequestedOverlayInfo,
weak_this_));
}
void MojoVideoDecoderService::Initialize(const VideoDecoderConfig& config,
bool low_delay,
InitializeCallback callback) {
DVLOG(1) << __func__;
if (!decoder_) {
std::move(callback).Run(false, false, 1);
return;
}
decoder_->Initialize(
config, low_delay, nullptr,
base::Bind(&MojoVideoDecoderService::OnDecoderInitialized, weak_this_,
base::Passed(&callback)),
base::Bind(&MojoVideoDecoderService::OnDecoderOutput, weak_this_,
base::nullopt));
}
void MojoVideoDecoderService::Decode(mojom::DecoderBufferPtr buffer,
DecodeCallback callback) {
DVLOG(2) << __func__ << " pts=" << buffer->timestamp.InMilliseconds();
if (!decoder_) {
std::move(callback).Run(DecodeStatus::DECODE_ERROR);
return;
}
mojo_decoder_buffer_reader_->ReadDecoderBuffer(
std::move(buffer), base::BindOnce(&MojoVideoDecoderService::OnDecoderRead,
weak_this_, std::move(callback)));
}
void MojoVideoDecoderService::Reset(ResetCallback callback) {
DVLOG(1) << __func__;
if (!decoder_) {
std::move(callback).Run();
return;
}
decoder_->Reset(base::Bind(&MojoVideoDecoderService::OnDecoderReset,
weak_this_, base::Passed(&callback)));
}
void MojoVideoDecoderService::OnDecoderInitialized(InitializeCallback callback,
bool success) {
DVLOG(1) << __func__;
DCHECK(decoder_);
std::move(callback).Run(success, decoder_->NeedsBitstreamConversion(),
decoder_->GetMaxDecodeRequests());
}
void MojoVideoDecoderService::OnDecoderRead(
DecodeCallback callback,
scoped_refptr<DecoderBuffer> buffer) {
if (!buffer) {
// TODO(sandersd): Close the channel.
std::move(callback).Run(DecodeStatus::DECODE_ERROR);
return;
}
decoder_->Decode(
buffer, base::Bind(&MojoVideoDecoderService::OnDecoderDecoded, weak_this_,
base::Passed(&callback)));
}
void MojoVideoDecoderService::OnDecoderDecoded(DecodeCallback callback,
DecodeStatus status) {
DVLOG(2) << __func__;
std::move(callback).Run(status);
}
void MojoVideoDecoderService::OnDecoderReset(ResetCallback callback) {
DVLOG(1) << __func__;
std::move(callback).Run();
}
void MojoVideoDecoderService::OnDecoderOutputWithReleaseCB(
MojoMediaClient::ReleaseMailboxCB release_cb,
const scoped_refptr<VideoFrame>& frame) {
DVLOG(2) << __func__ << " pts=" << frame->timestamp().InMilliseconds();
DCHECK(client_);
DCHECK(decoder_);
base::Optional<base::UnguessableToken> release_token;
if (release_cb) {
release_token = base::UnguessableToken::Create();
release_mailbox_cbs_[*release_token] = std::move(release_cb);
}
OnDecoderOutput(std::move(release_token), frame);
}
void MojoVideoDecoderService::OnDecoderOutput(
base::Optional<base::UnguessableToken> release_token,
const scoped_refptr<VideoFrame>& frame) {
DCHECK(client_);
DCHECK(decoder_);
client_->OnVideoFrameDecoded(frame, decoder_->CanReadWithoutStalling(),
std::move(release_token));
}
void MojoVideoDecoderService::OnReleaseMailbox(
const base::UnguessableToken& release_token,
const gpu::SyncToken& release_sync_token) {
DVLOG(2) << __func__;
// TODO(sandersd): Is it a serious error for the client to call
// OnReleaseMailbox() with an invalid |release_token|?
auto it = release_mailbox_cbs_.find(release_token);
if (it == release_mailbox_cbs_.end())
return;
MojoMediaClient::ReleaseMailboxCB cb = std::move(it->second);
release_mailbox_cbs_.erase(it);
std::move(cb).Run(release_sync_token);
}
void MojoVideoDecoderService::OnOverlayInfoChanged(
const OverlayInfo& overlay_info) {
DVLOG(2) << __func__;
DCHECK(client_);
DCHECK(decoder_);
DCHECK(provide_overlay_info_cb_);
provide_overlay_info_cb_.Run(overlay_info);
}
void MojoVideoDecoderService::OnDecoderRequestedOverlayInfo(
bool restart_for_transitions,
const ProvideOverlayInfoCB& provide_overlay_info_cb) {
DVLOG(2) << __func__;
DCHECK(client_);
DCHECK(decoder_);
DCHECK(!provide_overlay_info_cb_);
provide_overlay_info_cb_ = std::move(provide_overlay_info_cb);
client_->RequestOverlayInfo(restart_for_transitions);
}
} // namespace media