Commit 2a0eb868 authored by Aman Gupta's avatar Aman Gupta

avcodec/mediacodecdec: add delay_flush option

The default behavior of the mediacodec decoder before this commit
was to delay flushes until all pending hardware frames were
returned to the decoder. This was useful for certain types of
applications, but was unexpected behavior for others.

The new default behavior with this commit is now to execute
flushes immediately to invalidate all pending frames. The old
behavior can be enabled by setting delay_flush=1.

With the new behavior, video players implementing seek can simply
call flush on the decoder without having to worry about whether
they have one or more mediacodec frames still buffered in their
rendering pipeline. Previously, all these frames had to be
explictly freed (or rendered) before the seek/flush would execute.

The new behavior matches the behavior of all other lavc decoders,
reducing the amount of special casing required when using the
mediacodec decoder.
Signed-off-by: 's avatarAman Gupta <aman@tmm1.net>
Signed-off-by: 's avatarMatthieu Bouron <matthieu.bouron@gmail.com>
parent 840f6eb7
...@@ -15,6 +15,11 @@ libavutil: 2017-10-21 ...@@ -15,6 +15,11 @@ libavutil: 2017-10-21
API changes, most recent first: API changes, most recent first:
2018-03-xx - xxxxxxx - lavc 58.14.100 - mediacodec.h
Change the default behavior of avcodec_flush() on mediacodec
video decoders. To restore the previous behavior, use the new
delay_flush=1 option.
2018-03-xx - xxxxxxx - lavu 56.8.100 - frame.h 2018-03-xx - xxxxxxx - lavu 56.8.100 - frame.h
Add av_frame_new_side_data_from_buf(). Add av_frame_new_side_data_from_buf().
......
...@@ -91,7 +91,7 @@ int av_mediacodec_release_buffer(AVMediaCodecBuffer *buffer, int render) ...@@ -91,7 +91,7 @@ int av_mediacodec_release_buffer(AVMediaCodecBuffer *buffer, int render)
MediaCodecDecContext *ctx = buffer->ctx; MediaCodecDecContext *ctx = buffer->ctx;
int released = atomic_fetch_add(&buffer->released, 1); int released = atomic_fetch_add(&buffer->released, 1);
if (!released) { if (!released && (ctx->delay_flush || buffer->serial == atomic_load(&ctx->serial))) {
return ff_AMediaCodec_releaseOutputBuffer(ctx->codec, buffer->index, render); return ff_AMediaCodec_releaseOutputBuffer(ctx->codec, buffer->index, render);
} }
......
...@@ -41,10 +41,14 @@ ...@@ -41,10 +41,14 @@
typedef struct MediaCodecH264DecContext { typedef struct MediaCodecH264DecContext {
AVClass *avclass;
MediaCodecDecContext *ctx; MediaCodecDecContext *ctx;
AVPacket buffered_pkt; AVPacket buffered_pkt;
int delay_flush;
} MediaCodecH264DecContext; } MediaCodecH264DecContext;
static av_cold int mediacodec_decode_close(AVCodecContext *avctx) static av_cold int mediacodec_decode_close(AVCodecContext *avctx)
...@@ -366,6 +370,8 @@ static av_cold int mediacodec_decode_init(AVCodecContext *avctx) ...@@ -366,6 +370,8 @@ static av_cold int mediacodec_decode_init(AVCodecContext *avctx)
goto done; goto done;
} }
s->ctx->delay_flush = s->delay_flush;
if ((ret = ff_mediacodec_dec_init(avctx, s->ctx, codec_mime, format)) < 0) { if ((ret = ff_mediacodec_dec_init(avctx, s->ctx, codec_mime, format)) < 0) {
s->ctx = NULL; s->ctx = NULL;
goto done; goto done;
...@@ -485,12 +491,30 @@ static const AVCodecHWConfigInternal *mediacodec_hw_configs[] = { ...@@ -485,12 +491,30 @@ static const AVCodecHWConfigInternal *mediacodec_hw_configs[] = {
NULL NULL
}; };
#define OFFSET(x) offsetof(MediaCodecH264DecContext, x)
#define VD AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_DECODING_PARAM
static const AVOption ff_mediacodec_vdec_options[] = {
{ "delay_flush", "Delay flush until hw output buffers are returned to the decoder",
OFFSET(delay_flush), AV_OPT_TYPE_BOOL, {.i64 = 0}, 0, 1, VD },
{ NULL }
};
#define DECLARE_MEDIACODEC_VCLASS(short_name) \
static const AVClass ff_##short_name##_mediacodec_dec_class = { \
.class_name = #short_name "_mediacodec", \
.item_name = av_default_item_name, \
.option = ff_mediacodec_vdec_options, \
.version = LIBAVUTIL_VERSION_INT, \
};
#define DECLARE_MEDIACODEC_VDEC(short_name, full_name, codec_id, bsf) \ #define DECLARE_MEDIACODEC_VDEC(short_name, full_name, codec_id, bsf) \
DECLARE_MEDIACODEC_VCLASS(short_name) \
AVCodec ff_##short_name##_mediacodec_decoder = { \ AVCodec ff_##short_name##_mediacodec_decoder = { \
.name = #short_name "_mediacodec", \ .name = #short_name "_mediacodec", \
.long_name = NULL_IF_CONFIG_SMALL(full_name " Android MediaCodec decoder"), \ .long_name = NULL_IF_CONFIG_SMALL(full_name " Android MediaCodec decoder"), \
.type = AVMEDIA_TYPE_VIDEO, \ .type = AVMEDIA_TYPE_VIDEO, \
.id = codec_id, \ .id = codec_id, \
.priv_class = &ff_##short_name##_mediacodec_dec_class, \
.priv_data_size = sizeof(MediaCodecH264DecContext), \ .priv_data_size = sizeof(MediaCodecH264DecContext), \
.init = mediacodec_decode_init, \ .init = mediacodec_decode_init, \
.receive_frame = mediacodec_receive_frame, \ .receive_frame = mediacodec_receive_frame, \
......
...@@ -178,10 +178,11 @@ static void mediacodec_buffer_release(void *opaque, uint8_t *data) ...@@ -178,10 +178,11 @@ static void mediacodec_buffer_release(void *opaque, uint8_t *data)
MediaCodecDecContext *ctx = buffer->ctx; MediaCodecDecContext *ctx = buffer->ctx;
int released = atomic_load(&buffer->released); int released = atomic_load(&buffer->released);
if (!released) { if (!released && (ctx->delay_flush || buffer->serial == atomic_load(&ctx->serial))) {
ff_AMediaCodec_releaseOutputBuffer(ctx->codec, buffer->index, 0); ff_AMediaCodec_releaseOutputBuffer(ctx->codec, buffer->index, 0);
} }
if (ctx->delay_flush)
ff_mediacodec_dec_unref(ctx); ff_mediacodec_dec_unref(ctx);
av_freep(&buffer); av_freep(&buffer);
} }
...@@ -236,6 +237,8 @@ FF_ENABLE_DEPRECATION_WARNINGS ...@@ -236,6 +237,8 @@ FF_ENABLE_DEPRECATION_WARNINGS
} }
buffer->ctx = s; buffer->ctx = s;
buffer->serial = atomic_load(&s->serial);
if (s->delay_flush)
ff_mediacodec_dec_ref(s); ff_mediacodec_dec_ref(s);
buffer->index = index; buffer->index = index;
...@@ -425,6 +428,7 @@ static int mediacodec_dec_flush_codec(AVCodecContext *avctx, MediaCodecDecContex ...@@ -425,6 +428,7 @@ static int mediacodec_dec_flush_codec(AVCodecContext *avctx, MediaCodecDecContex
s->draining = 0; s->draining = 0;
s->flushing = 0; s->flushing = 0;
s->eos = 0; s->eos = 0;
atomic_fetch_add(&s->serial, 1);
status = ff_AMediaCodec_flush(codec); status = ff_AMediaCodec_flush(codec);
if (status < 0) { if (status < 0) {
...@@ -449,6 +453,7 @@ int ff_mediacodec_dec_init(AVCodecContext *avctx, MediaCodecDecContext *s, ...@@ -449,6 +453,7 @@ int ff_mediacodec_dec_init(AVCodecContext *avctx, MediaCodecDecContext *s,
}; };
atomic_init(&s->refcount, 1); atomic_init(&s->refcount, 1);
atomic_init(&s->serial, 1);
pix_fmt = ff_get_format(avctx, pix_fmts); pix_fmt = ff_get_format(avctx, pix_fmts);
if (pix_fmt == AV_PIX_FMT_MEDIACODEC) { if (pix_fmt == AV_PIX_FMT_MEDIACODEC) {
......
...@@ -62,6 +62,9 @@ typedef struct MediaCodecDecContext { ...@@ -62,6 +62,9 @@ typedef struct MediaCodecDecContext {
uint64_t output_buffer_count; uint64_t output_buffer_count;
bool delay_flush;
atomic_int serial;
} MediaCodecDecContext; } MediaCodecDecContext;
int ff_mediacodec_dec_init(AVCodecContext *avctx, int ff_mediacodec_dec_init(AVCodecContext *avctx,
...@@ -93,6 +96,7 @@ typedef struct MediaCodecBuffer { ...@@ -93,6 +96,7 @@ typedef struct MediaCodecBuffer {
ssize_t index; ssize_t index;
int64_t pts; int64_t pts;
atomic_int released; atomic_int released;
int serial;
} MediaCodecBuffer; } MediaCodecBuffer;
......
...@@ -28,8 +28,8 @@ ...@@ -28,8 +28,8 @@
#include "libavutil/version.h" #include "libavutil/version.h"
#define LIBAVCODEC_VERSION_MAJOR 58 #define LIBAVCODEC_VERSION_MAJOR 58
#define LIBAVCODEC_VERSION_MINOR 13 #define LIBAVCODEC_VERSION_MINOR 14
#define LIBAVCODEC_VERSION_MICRO 102 #define LIBAVCODEC_VERSION_MICRO 100
#define LIBAVCODEC_VERSION_INT AV_VERSION_INT(LIBAVCODEC_VERSION_MAJOR, \ #define LIBAVCODEC_VERSION_INT AV_VERSION_INT(LIBAVCODEC_VERSION_MAJOR, \
LIBAVCODEC_VERSION_MINOR, \ LIBAVCODEC_VERSION_MINOR, \
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment