blob: c9df3802c8a6729d91aa9abd942ddc02df6a4972 [file] [log] [blame]
// Copyright 2017 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/gpu/android/codec_wrapper.h"
#include <stddef.h>
#include <stdint.h>
#include <string>
#include <vector>
#include "base/memory/ptr_util.h"
#include "base/optional.h"
#include "base/stl_util.h"
#include "media/base/android/media_codec_util.h"
#include "media/base/bind_to_current_loop.h"
namespace media {
// CodecWrapperImpl is the implementation for CodecWrapper but is separate so
// we can keep its refcounting as an implementation detail. CodecWrapper and
// CodecOutputBuffer are the only two things that hold references to it.
class CodecWrapperImpl : public base::RefCountedThreadSafe<CodecWrapperImpl> {
public:
CodecWrapperImpl(CodecSurfacePair codec_surface_pair,
base::Closure output_buffer_release_cb);
using DequeueStatus = CodecWrapper::DequeueStatus;
using QueueStatus = CodecWrapper::QueueStatus;
CodecSurfacePair TakeCodecSurfacePair();
bool HasUnreleasedOutputBuffers() const;
void DiscardOutputBuffers();
bool IsFlushed() const;
bool IsDraining() const;
bool IsDrained() const;
bool SupportsFlush(DeviceInfo* device_info) const;
bool Flush();
bool SetSurface(scoped_refptr<AVDASurfaceBundle> surface_bundle);
scoped_refptr<AVDASurfaceBundle> SurfaceBundle();
QueueStatus QueueInputBuffer(const DecoderBuffer& buffer,
const EncryptionScheme& encryption_scheme);
DequeueStatus DequeueOutputBuffer(
base::TimeDelta* presentation_time,
bool* end_of_stream,
std::unique_ptr<CodecOutputBuffer>* codec_buffer);
// Releases the codec buffer and optionally renders it. This is a noop if
// the codec buffer is not valid. Can be called on any thread. Returns true if
// the buffer was released.
bool ReleaseCodecOutputBuffer(int64_t id, bool render);
private:
enum class State {
kError,
kFlushed,
kRunning,
kDraining,
kDrained,
};
friend base::RefCountedThreadSafe<CodecWrapperImpl>;
~CodecWrapperImpl();
void DiscardOutputBuffers_Locked();
// |lock_| protects access to all member variables.
mutable base::Lock lock_;
State state_;
std::unique_ptr<MediaCodecBridge> codec_;
// The currently configured surface.
scoped_refptr<AVDASurfaceBundle> surface_bundle_;
// Buffer ids are unique for a given CodecWrapper and map to MediaCodec buffer
// indices.
int64_t next_buffer_id_;
base::flat_map<int64_t, int> buffer_ids_;
// An input buffer that was dequeued but subsequently rejected from
// QueueInputBuffer() because the codec didn't have the crypto key. We
// maintain ownership of it and reuse it next time.
base::Optional<int> owned_input_buffer_;
// The current output size. Updated when DequeueOutputBuffer() reports
// OUTPUT_FORMAT_CHANGED.
gfx::Size size_;
// A callback that's called whenever an output buffer is released back to the
// codec.
base::Closure output_buffer_release_cb_;
DISALLOW_COPY_AND_ASSIGN(CodecWrapperImpl);
};
CodecOutputBuffer::CodecOutputBuffer(scoped_refptr<CodecWrapperImpl> codec,
int64_t id,
gfx::Size size)
: codec_(std::move(codec)), id_(id), size_(size) {}
CodecOutputBuffer::~CodecOutputBuffer() {
codec_->ReleaseCodecOutputBuffer(id_, false);
}
bool CodecOutputBuffer::ReleaseToSurface() {
return codec_->ReleaseCodecOutputBuffer(id_, true);
}
CodecWrapperImpl::CodecWrapperImpl(CodecSurfacePair codec_surface_pair,
base::Closure output_buffer_release_cb)
: state_(State::kFlushed),
codec_(std::move(codec_surface_pair.first)),
surface_bundle_(std::move(codec_surface_pair.second)),
next_buffer_id_(0),
output_buffer_release_cb_(std::move(output_buffer_release_cb)) {
DVLOG(2) << __func__;
}
CodecWrapperImpl::~CodecWrapperImpl() = default;
CodecSurfacePair CodecWrapperImpl::TakeCodecSurfacePair() {
DVLOG(2) << __func__;
base::AutoLock l(lock_);
if (!codec_)
return {nullptr, nullptr};
DiscardOutputBuffers_Locked();
return {std::move(codec_), std::move(surface_bundle_)};
}
bool CodecWrapperImpl::IsFlushed() const {
base::AutoLock l(lock_);
return state_ == State::kFlushed;
}
bool CodecWrapperImpl::IsDraining() const {
base::AutoLock l(lock_);
return state_ == State::kDraining;
}
bool CodecWrapperImpl::IsDrained() const {
base::AutoLock l(lock_);
return state_ == State::kDrained;
}
bool CodecWrapperImpl::HasUnreleasedOutputBuffers() const {
base::AutoLock l(lock_);
return !buffer_ids_.empty();
}
void CodecWrapperImpl::DiscardOutputBuffers() {
DVLOG(2) << __func__;
base::AutoLock l(lock_);
DiscardOutputBuffers_Locked();
}
void CodecWrapperImpl::DiscardOutputBuffers_Locked() {
DVLOG(2) << __func__;
lock_.AssertAcquired();
for (auto& kv : buffer_ids_)
codec_->ReleaseOutputBuffer(kv.second, false);
buffer_ids_.clear();
}
bool CodecWrapperImpl::SupportsFlush(DeviceInfo* device_info) const {
DVLOG(2) << __func__;
base::AutoLock l(lock_);
return !device_info->CodecNeedsFlushWorkaround(codec_.get());
}
bool CodecWrapperImpl::Flush() {
DVLOG(2) << __func__;
base::AutoLock l(lock_);
DCHECK(codec_ && state_ != State::kError);
// Dequeued buffers are invalidated by flushing.
buffer_ids_.clear();
owned_input_buffer_.reset();
auto status = codec_->Flush();
if (status == MEDIA_CODEC_ERROR) {
state_ = State::kError;
return false;
}
state_ = State::kFlushed;
return true;
}
CodecWrapperImpl::QueueStatus CodecWrapperImpl::QueueInputBuffer(
const DecoderBuffer& buffer,
const EncryptionScheme& encryption_scheme) {
DVLOG(4) << __func__;
base::AutoLock l(lock_);
DCHECK(codec_ && state_ != State::kError);
// Dequeue an input buffer if we don't already own one.
int input_buffer;
if (owned_input_buffer_) {
input_buffer = *owned_input_buffer_;
owned_input_buffer_.reset();
} else {
MediaCodecStatus status =
codec_->DequeueInputBuffer(base::TimeDelta(), &input_buffer);
switch (status) {
case MEDIA_CODEC_ERROR:
state_ = State::kError;
return QueueStatus::kError;
case MEDIA_CODEC_TRY_AGAIN_LATER:
return QueueStatus::kTryAgainLater;
case MEDIA_CODEC_OK:
break;
default:
NOTREACHED();
return QueueStatus::kError;
}
}
// Queue EOS if it's an EOS buffer.
if (buffer.end_of_stream()) {
// Some MediaCodecs consider it an error to get an EOS as the first buffer
// (http://crbug.com/672268).
DCHECK_NE(state_, State::kFlushed);
codec_->QueueEOS(input_buffer);
state_ = State::kDraining;
return QueueStatus::kOk;
}
// Queue a buffer.
const DecryptConfig* decrypt_config = buffer.decrypt_config();
bool encrypted = decrypt_config && decrypt_config->is_encrypted();
MediaCodecStatus status;
if (encrypted) {
status = codec_->QueueSecureInputBuffer(
input_buffer, buffer.data(), buffer.data_size(),
decrypt_config->key_id(), decrypt_config->iv(),
decrypt_config->subsamples(), encryption_scheme, buffer.timestamp());
} else {
status = codec_->QueueInputBuffer(input_buffer, buffer.data(),
buffer.data_size(), buffer.timestamp());
}
switch (status) {
case MEDIA_CODEC_OK:
state_ = State::kRunning;
return QueueStatus::kOk;
case MEDIA_CODEC_ERROR:
state_ = State::kError;
return QueueStatus::kError;
case MEDIA_CODEC_NO_KEY:
// The input buffer remains owned by us, so save it for reuse.
owned_input_buffer_ = input_buffer;
return QueueStatus::kNoKey;
default:
NOTREACHED();
return QueueStatus::kError;
}
}
CodecWrapperImpl::DequeueStatus CodecWrapperImpl::DequeueOutputBuffer(
base::TimeDelta* presentation_time,
bool* end_of_stream,
std::unique_ptr<CodecOutputBuffer>* codec_buffer) {
DVLOG(4) << __func__;
base::AutoLock l(lock_);
DCHECK(codec_ && state_ != State::kError);
// If |*codec_buffer| were not null, deleting it would deadlock when its
// destructor calls ReleaseCodecOutputBuffer().
DCHECK(!*codec_buffer);
// Dequeue in a loop so we can avoid propagating the uninteresting
// OUTPUT_FORMAT_CHANGED and OUTPUT_BUFFERS_CHANGED statuses to our caller.
for (int attempt = 0; attempt < 3; ++attempt) {
int index = -1;
size_t unused;
bool eos = false;
auto status =
codec_->DequeueOutputBuffer(base::TimeDelta(), &index, &unused, &unused,
presentation_time, &eos, nullptr);
switch (status) {
case MEDIA_CODEC_OK: {
if (eos) {
state_ = State::kDrained;
// We assume that the EOS flag is only ever attached to empty output
// buffers because we submit the EOS flag on empty input buffers. The
// MediaCodec docs leave open the possibility that the last non-empty
// output buffer has the EOS flag but we haven't seen that happen.
codec_->ReleaseOutputBuffer(index, false);
if (end_of_stream)
*end_of_stream = true;
return DequeueStatus::kOk;
}
int64_t buffer_id = next_buffer_id_++;
buffer_ids_[buffer_id] = index;
*codec_buffer =
base::WrapUnique(new CodecOutputBuffer(this, buffer_id, size_));
return DequeueStatus::kOk;
}
case MEDIA_CODEC_TRY_AGAIN_LATER: {
return DequeueStatus::kTryAgainLater;
}
case MEDIA_CODEC_ERROR: {
state_ = State::kError;
return DequeueStatus::kError;
}
case MEDIA_CODEC_OUTPUT_FORMAT_CHANGED: {
if (codec_->GetOutputSize(&size_) == MEDIA_CODEC_ERROR) {
state_ = State::kError;
return DequeueStatus::kError;
}
continue;
}
case MEDIA_CODEC_OUTPUT_BUFFERS_CHANGED: {
continue;
}
case MEDIA_CODEC_NO_KEY: {
NOTREACHED();
return DequeueStatus::kError;
}
}
}
state_ = State::kError;
return DequeueStatus::kError;
}
bool CodecWrapperImpl::SetSurface(
scoped_refptr<AVDASurfaceBundle> surface_bundle) {
DVLOG(2) << __func__;
base::AutoLock l(lock_);
DCHECK(surface_bundle);
DCHECK(codec_ && state_ != State::kError);
if (!codec_->SetSurface(surface_bundle->GetJavaSurface())) {
state_ = State::kError;
return false;
}
surface_bundle_ = std::move(surface_bundle);
return true;
}
scoped_refptr<AVDASurfaceBundle> CodecWrapperImpl::SurfaceBundle() {
base::AutoLock l(lock_);
return surface_bundle_;
}
bool CodecWrapperImpl::ReleaseCodecOutputBuffer(int64_t id, bool render) {
base::AutoLock l(lock_);
if (!codec_ || state_ == State::kError)
return false;
auto buffer_it = buffer_ids_.find(id);
bool valid = buffer_it != buffer_ids_.end();
DVLOG(2) << __func__ << " id=" << id << " render=" << render
<< " valid=" << valid;
if (!valid)
return false;
// Discard the buffers preceding the one we're releasing. The buffers are in
// presentation order because the ids are generated in presentation order.
for (auto it = buffer_ids_.begin(); it < buffer_it; ++it) {
int index = it->second;
codec_->ReleaseOutputBuffer(index, false);
DVLOG(2) << __func__ << " discarded " << index;
}
int index = buffer_it->second;
codec_->ReleaseOutputBuffer(index, render);
buffer_ids_.erase(buffer_ids_.begin(), buffer_it + 1);
if (output_buffer_release_cb_)
output_buffer_release_cb_.Run();
return true;
}
CodecWrapper::CodecWrapper(CodecSurfacePair codec_surface_pair,
base::Closure output_buffer_release_cb)
: impl_(new CodecWrapperImpl(std::move(codec_surface_pair),
std::move(output_buffer_release_cb))) {}
CodecWrapper::~CodecWrapper() {
// The codec must have already been taken.
DCHECK(!impl_->TakeCodecSurfacePair().first);
}
CodecSurfacePair CodecWrapper::TakeCodecSurfacePair() {
return impl_->TakeCodecSurfacePair();
}
bool CodecWrapper::HasUnreleasedOutputBuffers() const {
return impl_->HasUnreleasedOutputBuffers();
}
void CodecWrapper::DiscardOutputBuffers() {
impl_->DiscardOutputBuffers();
}
bool CodecWrapper::SupportsFlush(DeviceInfo* device_info) const {
return impl_->SupportsFlush(device_info);
}
bool CodecWrapper::IsFlushed() const {
return impl_->IsFlushed();
}
bool CodecWrapper::IsDraining() const {
return impl_->IsDraining();
}
bool CodecWrapper::IsDrained() const {
return impl_->IsDrained();
}
bool CodecWrapper::Flush() {
return impl_->Flush();
}
CodecWrapper::QueueStatus CodecWrapper::QueueInputBuffer(
const DecoderBuffer& buffer,
const EncryptionScheme& encryption_scheme) {
return impl_->QueueInputBuffer(buffer, encryption_scheme);
}
CodecWrapper::DequeueStatus CodecWrapper::DequeueOutputBuffer(
base::TimeDelta* presentation_time,
bool* end_of_stream,
std::unique_ptr<CodecOutputBuffer>* codec_buffer) {
return impl_->DequeueOutputBuffer(presentation_time, end_of_stream,
codec_buffer);
}
bool CodecWrapper::SetSurface(scoped_refptr<AVDASurfaceBundle> surface_bundle) {
return impl_->SetSurface(std::move(surface_bundle));
}
scoped_refptr<AVDASurfaceBundle> CodecWrapper::SurfaceBundle() {
return impl_->SurfaceBundle();
}
} // namespace media