ext/ffmpeg/gstffmpegdec.c: Change the pad_alloc calculations for weird clipped sizes, refactor the code a bit.

Original commit message from CVS:
* ext/ffmpeg/gstffmpegdec.c: (gst_ffmpegdec_setcaps),
(alloc_output_buffer), (gst_ffmpegdec_get_buffer),
(gst_ffmpegdec_release_buffer), (gst_ffmpegdec_negotiate),
(get_output_buffer):
Change the pad_alloc calculations for weird clipped sizes, refactor the
code a bit.
Add support for some different refcounting algorithm.
Direct rendering still disabled by default.
This commit is contained in:
Wim Taymans 2008-01-23 16:08:27 +00:00
parent 31131add1f
commit e57ac5b514
2 changed files with 133 additions and 64 deletions

View file

@ -1,3 +1,14 @@
2008-01-23 Wim Taymans <wim.taymans@collabora.co.uk>
* ext/ffmpeg/gstffmpegdec.c: (gst_ffmpegdec_setcaps),
(alloc_output_buffer), (gst_ffmpegdec_get_buffer),
(gst_ffmpegdec_release_buffer), (gst_ffmpegdec_negotiate),
(get_output_buffer):
Change the pad_alloc calculations for weird clipped sizes, refactor the
code a bit.
Add support for some different refcounting algorithm.
Direct rendering still disabled by default.
2008-01-22 Edward Hervey <edward.hervey@collabora.co.uk> 2008-01-22 Edward Hervey <edward.hervey@collabora.co.uk>
* ext/ffmpeg/gstffmpegdec.c: (gst_ffmpegdec_class_init): * ext/ffmpeg/gstffmpegdec.c: (gst_ffmpegdec_class_init):

View file

@ -35,6 +35,9 @@
#include "gstffmpeg.h" #include "gstffmpeg.h"
#include "gstffmpegcodecmap.h" #include "gstffmpegcodecmap.h"
/* define to enable alternative buffer refcounting algorithm */
#undef EXTRA_REF
typedef struct _GstFFMpegDec GstFFMpegDec; typedef struct _GstFFMpegDec GstFFMpegDec;
struct _GstFFMpegDec struct _GstFFMpegDec
@ -77,6 +80,7 @@ struct _GstFFMpegDec
GValue *par; /* pixel aspect ratio of incoming data */ GValue *par; /* pixel aspect ratio of incoming data */
gboolean current_dr; /* if direct rendering is enabled */ gboolean current_dr; /* if direct rendering is enabled */
gboolean extra_ref; /* keep extra ref around in get/release */
/* some properties */ /* some properties */
gint hurry_up; gint hurry_up;
@ -636,22 +640,27 @@ gst_ffmpegdec_setcaps (GstPad * pad, GstCaps * caps)
/* figure out if we can use direct rendering */ /* figure out if we can use direct rendering */
ffmpegdec->current_dr = FALSE; ffmpegdec->current_dr = FALSE;
ffmpegdec->extra_ref = FALSE;
if (ffmpegdec->direct_rendering) { if (ffmpegdec->direct_rendering) {
GST_DEBUG_OBJECT (ffmpegdec, "trying to enable direct rendering"); GST_DEBUG_OBJECT (ffmpegdec, "trying to enable direct rendering");
if (!oclass->in_plugin->capabilities & CODEC_CAP_DR1) { if (!oclass->in_plugin->capabilities & CODEC_CAP_DR1) {
GST_DEBUG_OBJECT (ffmpegdec, "direct rendering not supported"); GST_DEBUG_OBJECT (ffmpegdec, "direct rendering not supported");
} }
if (oclass->in_plugin->id == CODEC_ID_H264) { if (oclass->in_plugin->id == CODEC_ID_H264) {
GST_DEBUG_OBJECT (ffmpegdec, "direct rendering disabled for H264"); GST_DEBUG_OBJECT (ffmpegdec, "direct rendering setup for H264");
ffmpegdec->current_dr = TRUE;
ffmpegdec->extra_ref = TRUE;
} }
else { else {
GST_DEBUG_OBJECT (ffmpegdec, "enabled direct rendering"); GST_DEBUG_OBJECT (ffmpegdec, "enabled direct rendering");
/* do *not* draw edges when in direct rendering, for some reason it draws /* do *not* draw edges when in direct rendering, for some reason it draws
* outside of the memory. */ * outside of the memory. */
ffmpegdec->current_dr = TRUE; ffmpegdec->current_dr = TRUE;
ffmpegdec->context->flags |= CODEC_FLAG_EMU_EDGE;
} }
} }
if (ffmpegdec->current_dr) {
ffmpegdec->context->flags |= CODEC_FLAG_EMU_EDGE;
}
/* workaround encoder bugs */ /* workaround encoder bugs */
ffmpegdec->context->workaround_bugs |= FF_BUG_AUTODETECT; ffmpegdec->context->workaround_bugs |= FF_BUG_AUTODETECT;
@ -694,20 +703,66 @@ open_failed:
} }
} }
static GstFlowReturn
alloc_output_buffer (GstFFMpegDec * ffmpegdec, GstBuffer ** outbuf,
gint width, gint height)
{
GstFlowReturn ret;
gint fsize;
ret = GST_FLOW_ERROR;
*outbuf = NULL;
GST_LOG_OBJECT (ffmpegdec, "alloc output buffer");
/* see if we need renegotiation */
if (G_UNLIKELY (!gst_ffmpegdec_negotiate (ffmpegdec)))
goto negotiate_failed;
/* get the size of the gstreamer output buffer given a
* width/height/format */
fsize = gst_ffmpeg_avpicture_get_size (ffmpegdec->context->pix_fmt,
width, height);
if (!ffmpegdec->context->palctrl) {
/* no pallete, we can use the buffer size to alloc */
ret = gst_pad_alloc_buffer_and_set_caps (ffmpegdec->srcpad,
GST_BUFFER_OFFSET_NONE, fsize,
GST_PAD_CAPS (ffmpegdec->srcpad), outbuf);
if (G_UNLIKELY (ret != GST_FLOW_OK))
goto alloc_failed;
} else {
/* for paletted data we can't use pad_alloc_buffer(), because
* fsize contains the size of the palette, so the overall size
* is bigger than ffmpegcolorspace's unit size, which will
* prompt GstBaseTransform to complain endlessly ... */
*outbuf = gst_buffer_new_and_alloc (fsize);
gst_buffer_set_caps (*outbuf, GST_PAD_CAPS (ffmpegdec->srcpad));
ret = GST_FLOW_OK;
}
return ret;
/* special cases */
negotiate_failed:
{
GST_DEBUG_OBJECT (ffmpegdec, "negotiate failed");
return GST_FLOW_NOT_NEGOTIATED;
}
alloc_failed:
{
GST_DEBUG_OBJECT (ffmpegdec, "pad_alloc failed");
return ret;
}
}
static int static int
gst_ffmpegdec_get_buffer (AVCodecContext * context, AVFrame * picture) gst_ffmpegdec_get_buffer (AVCodecContext * context, AVFrame * picture)
{ {
GstBuffer *buf = NULL; GstBuffer *buf = NULL;
gulong bufsize = 0;
GstFFMpegDec *ffmpegdec; GstFFMpegDec *ffmpegdec;
int width;
int height;
ffmpegdec = (GstFFMpegDec *) context->opaque; ffmpegdec = (GstFFMpegDec *) context->opaque;
width = context->width;
height = context->height;
GST_DEBUG_OBJECT (ffmpegdec, "getting buffer, apply pts %"G_GINT64_FORMAT, GST_DEBUG_OBJECT (ffmpegdec, "getting buffer, apply pts %"G_GINT64_FORMAT,
ffmpegdec->in_ts); ffmpegdec->in_ts);
@ -719,7 +774,7 @@ gst_ffmpegdec_get_buffer (AVCodecContext * context, AVFrame * picture)
picture->opaque = NULL; picture->opaque = NULL;
if (!ffmpegdec->current_dr) { if (!ffmpegdec->current_dr) {
GST_LOG_OBJECT (ffmpegdec, "direct rendering disabled"); GST_LOG_OBJECT (ffmpegdec, "direct rendering disabled, fallback alloc");
return avcodec_default_get_buffer (context, picture); return avcodec_default_get_buffer (context, picture);
} }
@ -727,32 +782,45 @@ gst_ffmpegdec_get_buffer (AVCodecContext * context, AVFrame * picture)
case CODEC_TYPE_VIDEO: case CODEC_TYPE_VIDEO:
/* some ffmpeg video plugins don't see the point in setting codec_type ... */ /* some ffmpeg video plugins don't see the point in setting codec_type ... */
case CODEC_TYPE_UNKNOWN: case CODEC_TYPE_UNKNOWN:
avcodec_align_dimensions (context, &width, &height); {
GstFlowReturn ret;
gint width, height;
gint clip_width, clip_height;
bufsize = avpicture_get_size (context->pix_fmt, width, height); /* take width and height before clipping */
width = context->width;
height = context->height;
/* take final clipped output size */
if ((clip_width = ffmpegdec->format.video.clip_width) == -1)
clip_width = width;
if ((clip_height = ffmpegdec->format.video.clip_height) == -1)
clip_height = height;
if ((width != context->width) || (height != context->height) || 1) { /* this is the size ffmpeg needs for the buffer */
context->width = width; avcodec_align_dimensions(context, &width, &height);
context->height = height;
}
if (!gst_ffmpegdec_negotiate (ffmpegdec)) { GST_LOG_OBJECT (ffmpegdec, "aligned outsize %d/%d, clip %d/%d",
GST_ELEMENT_ERROR (ffmpegdec, CORE, NEGOTIATION, (NULL), width, height, clip_width, clip_height);
("Failed to link ffmpeg decoder to next element"));
if (width != clip_width || height != clip_height) {
/* We can't alloc if we need to clip the output buffer later */
GST_LOG_OBJECT (ffmpegdec, "we need clipping, fallback alloc");
return avcodec_default_get_buffer (context, picture); return avcodec_default_get_buffer (context, picture);
} }
if (gst_pad_alloc_buffer_and_set_caps (ffmpegdec->srcpad, /* alloc with aligned dimensions for ffmpeg */
GST_BUFFER_OFFSET_NONE, bufsize, GST_PAD_CAPS (ffmpegdec->srcpad), ret = alloc_output_buffer (ffmpegdec, &buf, width, height);
&buf) != GST_FLOW_OK) { if (G_UNLIKELY (ret != GST_FLOW_OK)) {
/* when allocation fails, still provide a default buffer */ /* alloc default buffer when we can't get one from downstream */
GST_LOG_OBJECT (ffmpegdec, "alloc failed, fallback alloc");
return avcodec_default_get_buffer (context, picture); return avcodec_default_get_buffer (context, picture);
} }
/* copy the right pointers and strides in the picture object */
gst_ffmpeg_avpicture_fill ((AVPicture *) picture, gst_ffmpeg_avpicture_fill ((AVPicture *) picture,
GST_BUFFER_DATA (buf), GST_BUFFER_DATA (buf), context->pix_fmt, width, height);
context->pix_fmt, context->width, context->height);
break; break;
}
case CODEC_TYPE_AUDIO: case CODEC_TYPE_AUDIO:
default: default:
g_assert_not_reached (); g_assert_not_reached ();
@ -765,6 +833,13 @@ gst_ffmpegdec_get_buffer (AVCodecContext * context, AVFrame * picture)
picture->age = 256*256*256*64; picture->age = 256*256*256*64;
picture->opaque = buf; picture->opaque = buf;
#ifdef EXTRA_REF
if (picture->reference != 0 || ffmpegdec->extra_ref) {
GST_DEBUG_OBJECT (ffmpegdec, "adding extra ref");
gst_buffer_ref (buf);
}
#endif
GST_LOG_OBJECT (ffmpegdec, "returned buffer %p", buf); GST_LOG_OBJECT (ffmpegdec, "returned buffer %p", buf);
return 0; return 0;
@ -779,10 +854,9 @@ gst_ffmpegdec_release_buffer (AVCodecContext * context, AVFrame * picture)
ffmpegdec = (GstFFMpegDec *) context->opaque; ffmpegdec = (GstFFMpegDec *) context->opaque;
GST_DEBUG_OBJECT (ffmpegdec, "release buffer");
/* check if it was our buffer */ /* check if it was our buffer */
if (picture->opaque == NULL) { if (picture->opaque == NULL) {
GST_DEBUG_OBJECT (ffmpegdec, "default release buffer");
avcodec_default_release_buffer (context, picture); avcodec_default_release_buffer (context, picture);
return; return;
} }
@ -790,9 +864,17 @@ gst_ffmpegdec_release_buffer (AVCodecContext * context, AVFrame * picture)
/* we remove the opaque data now */ /* we remove the opaque data now */
buf = GST_BUFFER_CAST (picture->opaque); buf = GST_BUFFER_CAST (picture->opaque);
GST_DEBUG_OBJECT (ffmpegdec, "release buffer %p", buf); GST_DEBUG_OBJECT (ffmpegdec, "release buffer %p", buf);
gst_buffer_unref (buf);
picture->opaque = NULL; picture->opaque = NULL;
#ifdef EXTRA_REF
if (picture->reference != 0 || ffmpegdec->extra_ref) {
GST_DEBUG_OBJECT (ffmpegdec, "remove extra ref");
gst_buffer_unref (buf);
}
#else
gst_buffer_unref (buf);
#endif
/* zero out the reference in ffmpeg */ /* zero out the reference in ffmpeg */
for (i = 0; i < 4; i++) { for (i = 0; i < 4; i++) {
picture->data[i] = NULL; picture->data[i] = NULL;
@ -936,6 +1018,8 @@ gst_ffmpegdec_negotiate (GstFFMpegDec * ffmpegdec)
height = ffmpegdec->format.video.clip_height; height = ffmpegdec->format.video.clip_height;
if (width != -1 && height != -1) { if (width != -1 && height != -1) {
/* overwrite the output size with the dimension of the
* clipping region */
gst_caps_set_simple (caps, gst_caps_set_simple (caps,
"width", G_TYPE_INT, width, "height", G_TYPE_INT, height, NULL); "width", G_TYPE_INT, width, "height", G_TYPE_INT, height, NULL);
} }
@ -1178,54 +1262,33 @@ get_output_buffer (GstFFMpegDec * ffmpegdec, GstBuffer ** outbuf)
{ {
GstFlowReturn ret; GstFlowReturn ret;
ret = GST_FLOW_ERROR; ret = GST_FLOW_OK;
*outbuf = NULL; *outbuf = NULL;
/* libavcodec constantly crashes on stupid buffer allocation
* errors inside. This drives me crazy, so we let it allocate
* its own buffers and copy to our own buffer afterwards... */
/* BUFFER CREATION */
if (ffmpegdec->picture->opaque != NULL) { if (ffmpegdec->picture->opaque != NULL) {
/* we allocated a picture already for ffmpeg to decode into, let's pick it
* up and use it now. */
GST_LOG_OBJECT (ffmpegdec, "using opaque buffer"); GST_LOG_OBJECT (ffmpegdec, "using opaque buffer");
*outbuf = (GstBuffer *) ffmpegdec->picture->opaque; *outbuf = (GstBuffer *) ffmpegdec->picture->opaque;
#ifndef EXTRA_REF
gst_buffer_ref (*outbuf); gst_buffer_ref (*outbuf);
ret = GST_FLOW_OK; #endif
} else { } else {
AVPicture pic; AVPicture pic;
gint fsize;
gint width, height; gint width, height;
GST_LOG_OBJECT (ffmpegdec, "get output buffer"); GST_LOG_OBJECT (ffmpegdec, "get output buffer");
/* see if we need renegotiation */ /* figure out size of output buffer, this is the clipped output size because
if (G_UNLIKELY (!gst_ffmpegdec_negotiate (ffmpegdec))) * we will copy the picture into it. */
goto negotiate_failed;
/* figure out size of output buffer */
if ((width = ffmpegdec->format.video.clip_width) == -1) if ((width = ffmpegdec->format.video.clip_width) == -1)
width = ffmpegdec->context->width; width = ffmpegdec->context->width;
if ((height = ffmpegdec->format.video.clip_height) == -1) if ((height = ffmpegdec->format.video.clip_height) == -1)
height = ffmpegdec->context->height; height = ffmpegdec->context->height;
fsize = gst_ffmpeg_avpicture_get_size (ffmpegdec->context->pix_fmt, ret = alloc_output_buffer (ffmpegdec, outbuf, width, height);
width, height); if (G_UNLIKELY (ret != GST_FLOW_OK))
goto alloc_failed;
if (!ffmpegdec->context->palctrl) {
ret = gst_pad_alloc_buffer_and_set_caps (ffmpegdec->srcpad,
GST_BUFFER_OFFSET_NONE, fsize,
GST_PAD_CAPS (ffmpegdec->srcpad), outbuf);
if (G_UNLIKELY (ret != GST_FLOW_OK))
goto alloc_failed;
} else {
/* for paletted data we can't use pad_alloc_buffer(), because
* fsize contains the size of the palette, so the overall size
* is bigger than ffmpegcolorspace's unit size, which will
* prompt GstBaseTransform to complain endlessly ... */
*outbuf = gst_buffer_new_and_alloc (fsize);
gst_buffer_set_caps (*outbuf, GST_PAD_CAPS (ffmpegdec->srcpad));
ret = GST_FLOW_OK;
}
/* original ffmpeg code does not handle odd sizes correctly. /* original ffmpeg code does not handle odd sizes correctly.
* This patched up version does */ * This patched up version does */
@ -1242,11 +1305,6 @@ get_output_buffer (GstFFMpegDec * ffmpegdec, GstBuffer ** outbuf)
return ret; return ret;
/* special cases */ /* special cases */
negotiate_failed:
{
GST_DEBUG_OBJECT (ffmpegdec, "negotiate failed");
return GST_FLOW_NOT_NEGOTIATED;
}
alloc_failed: alloc_failed:
{ {
GST_DEBUG_OBJECT (ffmpegdec, "pad_alloc failed"); GST_DEBUG_OBJECT (ffmpegdec, "pad_alloc failed");