mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-20 00:31:13 +00:00
gstffmpegvidenc: Port to -base video classes
This commit is contained in:
parent
bdf7ebf411
commit
134f7d7058
3 changed files with 191 additions and 345 deletions
|
@ -23,7 +23,7 @@ libgstffmpeg_la_SOURCES = gstffmpeg.c \
|
||||||
# gstffmpegscale.c
|
# gstffmpegscale.c
|
||||||
|
|
||||||
libgstffmpeg_la_CFLAGS = $(FFMPEG_CFLAGS) $(GST_PLUGINS_BASE_CFLAGS) $(GST_CFLAGS)
|
libgstffmpeg_la_CFLAGS = $(FFMPEG_CFLAGS) $(GST_PLUGINS_BASE_CFLAGS) $(GST_CFLAGS)
|
||||||
libgstffmpeg_la_LIBADD = $(FFMPEG_LIBS) $(GST_BASE_LIBS) $(GST_PLUGINS_BASE_LIBS) -lgstaudio-$(GST_MAJORMINOR) -lgstpbutils-$(GST_MAJORMINOR) $(LIBM) $(WIN32_LIBS) -lz $(BZ2_LIBS)
|
libgstffmpeg_la_LIBADD = $(FFMPEG_LIBS) $(GST_BASE_LIBS) $(GST_PLUGINS_BASE_LIBS) -lgstvideo-$(GST_MAJORMINOR) -lgstaudio-$(GST_MAJORMINOR) -lgstpbutils-$(GST_MAJORMINOR) $(LIBM) $(WIN32_LIBS) -lz $(BZ2_LIBS)
|
||||||
libgstffmpeg_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS) $(DARWIN_LDFLAGS)
|
libgstffmpeg_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS) $(DARWIN_LDFLAGS)
|
||||||
libgstffmpeg_la_LIBTOOLFLAGS = --tag=disable-static
|
libgstffmpeg_la_LIBTOOLFLAGS = --tag=disable-static
|
||||||
|
|
||||||
|
|
|
@ -96,21 +96,20 @@ static void gst_ffmpegvidenc_base_init (GstFFMpegVidEncClass * klass);
|
||||||
static void gst_ffmpegvidenc_init (GstFFMpegVidEnc * ffmpegenc);
|
static void gst_ffmpegvidenc_init (GstFFMpegVidEnc * ffmpegenc);
|
||||||
static void gst_ffmpegvidenc_finalize (GObject * object);
|
static void gst_ffmpegvidenc_finalize (GObject * object);
|
||||||
|
|
||||||
static gboolean gst_ffmpegvidenc_setcaps (GstPad * pad, GstCaps * caps);
|
static gboolean gst_ffmpegvidenc_stop (GstVideoEncoder * encoder);
|
||||||
static GstCaps *gst_ffmpegvidenc_getcaps (GstPad * pad);
|
static GstFlowReturn gst_ffmpegvidenc_finish (GstVideoEncoder * encoder);
|
||||||
static GstFlowReturn gst_ffmpegvidenc_chain_video (GstPad * pad,
|
static gboolean gst_ffmpegvidenc_set_format (GstVideoEncoder * encoder,
|
||||||
GstBuffer * buffer);
|
GstVideoCodecState * state);
|
||||||
static gboolean gst_ffmpegvidenc_event_video (GstPad * pad, GstEvent * event);
|
|
||||||
static gboolean gst_ffmpegvidenc_event_src (GstPad * pad, GstEvent * event);
|
static GstCaps *gst_ffmpegvidenc_getcaps (GstVideoEncoder * encoder);
|
||||||
|
static GstFlowReturn gst_ffmpegvidenc_handle_frame (GstVideoEncoder * encoder,
|
||||||
|
GstVideoCodecFrame * frame);
|
||||||
|
|
||||||
static void gst_ffmpegvidenc_set_property (GObject * object,
|
static void gst_ffmpegvidenc_set_property (GObject * object,
|
||||||
guint prop_id, const GValue * value, GParamSpec * pspec);
|
guint prop_id, const GValue * value, GParamSpec * pspec);
|
||||||
static void gst_ffmpegvidenc_get_property (GObject * object,
|
static void gst_ffmpegvidenc_get_property (GObject * object,
|
||||||
guint prop_id, GValue * value, GParamSpec * pspec);
|
guint prop_id, GValue * value, GParamSpec * pspec);
|
||||||
|
|
||||||
static GstStateChangeReturn gst_ffmpegvidenc_change_state (GstElement * element,
|
|
||||||
GstStateChange transition);
|
|
||||||
|
|
||||||
#define GST_FFENC_PARAMS_QDATA g_quark_from_static_string("ffenc-params")
|
#define GST_FFENC_PARAMS_QDATA g_quark_from_static_string("ffenc-params")
|
||||||
|
|
||||||
static GstElementClass *parent_class = NULL;
|
static GstElementClass *parent_class = NULL;
|
||||||
|
@ -169,10 +168,10 @@ static void
|
||||||
gst_ffmpegvidenc_class_init (GstFFMpegVidEncClass * klass)
|
gst_ffmpegvidenc_class_init (GstFFMpegVidEncClass * klass)
|
||||||
{
|
{
|
||||||
GObjectClass *gobject_class;
|
GObjectClass *gobject_class;
|
||||||
GstElementClass *gstelement_class;
|
GstVideoEncoderClass *venc_class;
|
||||||
|
|
||||||
gobject_class = (GObjectClass *) klass;
|
gobject_class = (GObjectClass *) klass;
|
||||||
gstelement_class = (GstElementClass *) klass;
|
venc_class = (GstVideoEncoderClass *) klass;
|
||||||
|
|
||||||
parent_class = g_type_class_peek_parent (klass);
|
parent_class = g_type_class_peek_parent (klass);
|
||||||
|
|
||||||
|
@ -207,7 +206,11 @@ gst_ffmpegvidenc_class_init (GstFFMpegVidEncClass * klass)
|
||||||
/* register additional properties, possibly dependent on the exact CODEC */
|
/* register additional properties, possibly dependent on the exact CODEC */
|
||||||
gst_ffmpeg_cfg_install_property (klass, ARG_CFG_BASE);
|
gst_ffmpeg_cfg_install_property (klass, ARG_CFG_BASE);
|
||||||
|
|
||||||
gstelement_class->change_state = gst_ffmpegvidenc_change_state;
|
venc_class->stop = gst_ffmpegvidenc_stop;
|
||||||
|
venc_class->finish = gst_ffmpegvidenc_finish;
|
||||||
|
venc_class->handle_frame = gst_ffmpegvidenc_handle_frame;
|
||||||
|
venc_class->getcaps = gst_ffmpegvidenc_getcaps;
|
||||||
|
venc_class->set_format = gst_ffmpegvidenc_set_format;
|
||||||
|
|
||||||
gobject_class->finalize = gst_ffmpegvidenc_finalize;
|
gobject_class->finalize = gst_ffmpegvidenc_finalize;
|
||||||
}
|
}
|
||||||
|
@ -215,28 +218,12 @@ gst_ffmpegvidenc_class_init (GstFFMpegVidEncClass * klass)
|
||||||
static void
|
static void
|
||||||
gst_ffmpegvidenc_init (GstFFMpegVidEnc * ffmpegenc)
|
gst_ffmpegvidenc_init (GstFFMpegVidEnc * ffmpegenc)
|
||||||
{
|
{
|
||||||
GstFFMpegVidEncClass *oclass =
|
|
||||||
(GstFFMpegVidEncClass *) (G_OBJECT_GET_CLASS (ffmpegenc));
|
|
||||||
|
|
||||||
/* setup pads */
|
|
||||||
ffmpegenc->sinkpad = gst_pad_new_from_template (oclass->sinktempl, "sink");
|
|
||||||
gst_pad_set_setcaps_function (ffmpegenc->sinkpad, gst_ffmpegvidenc_setcaps);
|
|
||||||
gst_pad_set_getcaps_function (ffmpegenc->sinkpad, gst_ffmpegvidenc_getcaps);
|
|
||||||
ffmpegenc->srcpad = gst_pad_new_from_template (oclass->srctempl, "src");
|
|
||||||
gst_pad_use_fixed_caps (ffmpegenc->srcpad);
|
|
||||||
|
|
||||||
/* ffmpeg objects */
|
/* ffmpeg objects */
|
||||||
ffmpegenc->context = avcodec_alloc_context ();
|
ffmpegenc->context = avcodec_alloc_context ();
|
||||||
ffmpegenc->picture = avcodec_alloc_frame ();
|
ffmpegenc->picture = avcodec_alloc_frame ();
|
||||||
ffmpegenc->opened = FALSE;
|
ffmpegenc->opened = FALSE;
|
||||||
|
|
||||||
ffmpegenc->file = NULL;
|
ffmpegenc->file = NULL;
|
||||||
ffmpegenc->delay = g_queue_new ();
|
|
||||||
|
|
||||||
gst_pad_set_chain_function (ffmpegenc->sinkpad, gst_ffmpegvidenc_chain_video);
|
|
||||||
/* so we know when to flush the buffers on EOS */
|
|
||||||
gst_pad_set_event_function (ffmpegenc->sinkpad, gst_ffmpegvidenc_event_video);
|
|
||||||
gst_pad_set_event_function (ffmpegenc->srcpad, gst_ffmpegvidenc_event_src);
|
|
||||||
|
|
||||||
ffmpegenc->bitrate = DEFAULT_VIDEO_BITRATE;
|
ffmpegenc->bitrate = DEFAULT_VIDEO_BITRATE;
|
||||||
ffmpegenc->me_method = ME_EPZS;
|
ffmpegenc->me_method = ME_EPZS;
|
||||||
|
@ -249,9 +236,6 @@ gst_ffmpegvidenc_init (GstFFMpegVidEnc * ffmpegenc)
|
||||||
ffmpegenc->max_key_interval = 0;
|
ffmpegenc->max_key_interval = 0;
|
||||||
|
|
||||||
gst_ffmpeg_cfg_set_defaults (ffmpegenc);
|
gst_ffmpeg_cfg_set_defaults (ffmpegenc);
|
||||||
|
|
||||||
gst_element_add_pad (GST_ELEMENT (ffmpegenc), ffmpegenc->sinkpad);
|
|
||||||
gst_element_add_pad (GST_ELEMENT (ffmpegenc), ffmpegenc->srcpad);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -271,91 +255,27 @@ gst_ffmpegvidenc_finalize (GObject * object)
|
||||||
av_free (ffmpegenc->context);
|
av_free (ffmpegenc->context);
|
||||||
av_free (ffmpegenc->picture);
|
av_free (ffmpegenc->picture);
|
||||||
|
|
||||||
g_queue_free (ffmpegenc->delay);
|
|
||||||
g_free (ffmpegenc->filename);
|
g_free (ffmpegenc->filename);
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->finalize (object);
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
||||||
}
|
}
|
||||||
|
|
||||||
static GstCaps *
|
static GstCaps *
|
||||||
gst_ffmpegvidenc_get_possible_sizes (GstFFMpegVidEnc * ffmpegenc, GstPad * pad,
|
gst_ffmpegvidenc_getcaps (GstVideoEncoder * encoder)
|
||||||
const GstCaps * caps)
|
|
||||||
{
|
{
|
||||||
GstCaps *othercaps = NULL;
|
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) encoder;
|
||||||
GstCaps *tmpcaps = NULL;
|
|
||||||
GstCaps *intersect = NULL;
|
|
||||||
guint i;
|
|
||||||
|
|
||||||
othercaps = gst_pad_peer_get_caps (ffmpegenc->srcpad);
|
|
||||||
|
|
||||||
if (!othercaps)
|
|
||||||
return gst_caps_copy (caps);
|
|
||||||
|
|
||||||
intersect = gst_caps_intersect (othercaps,
|
|
||||||
gst_pad_get_pad_template_caps (ffmpegenc->srcpad));
|
|
||||||
gst_caps_unref (othercaps);
|
|
||||||
|
|
||||||
if (gst_caps_is_empty (intersect))
|
|
||||||
return intersect;
|
|
||||||
|
|
||||||
if (gst_caps_is_any (intersect))
|
|
||||||
return gst_caps_copy (caps);
|
|
||||||
|
|
||||||
tmpcaps = gst_caps_new_empty ();
|
|
||||||
|
|
||||||
for (i = 0; i < gst_caps_get_size (intersect); i++) {
|
|
||||||
GstStructure *s = gst_caps_get_structure (intersect, i);
|
|
||||||
const GValue *height = NULL;
|
|
||||||
const GValue *width = NULL;
|
|
||||||
const GValue *framerate = NULL;
|
|
||||||
GstStructure *tmps;
|
|
||||||
|
|
||||||
height = gst_structure_get_value (s, "height");
|
|
||||||
width = gst_structure_get_value (s, "width");
|
|
||||||
framerate = gst_structure_get_value (s, "framerate");
|
|
||||||
|
|
||||||
tmps = gst_structure_new ("video/x-raw-rgb", NULL);
|
|
||||||
if (width)
|
|
||||||
gst_structure_set_value (tmps, "width", width);
|
|
||||||
if (height)
|
|
||||||
gst_structure_set_value (tmps, "height", height);
|
|
||||||
if (framerate)
|
|
||||||
gst_structure_set_value (tmps, "framerate", framerate);
|
|
||||||
gst_caps_merge_structure (tmpcaps, gst_structure_copy (tmps));
|
|
||||||
|
|
||||||
gst_structure_set_name (tmps, "video/x-raw-yuv");
|
|
||||||
gst_caps_merge_structure (tmpcaps, gst_structure_copy (tmps));
|
|
||||||
|
|
||||||
gst_structure_set_name (tmps, "video/x-raw-gray");
|
|
||||||
gst_caps_merge_structure (tmpcaps, tmps);
|
|
||||||
}
|
|
||||||
gst_caps_unref (intersect);
|
|
||||||
|
|
||||||
intersect = gst_caps_intersect (caps, tmpcaps);
|
|
||||||
gst_caps_unref (tmpcaps);
|
|
||||||
|
|
||||||
return intersect;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static GstCaps *
|
|
||||||
gst_ffmpegvidenc_getcaps (GstPad * pad)
|
|
||||||
{
|
|
||||||
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) GST_PAD_PARENT (pad);
|
|
||||||
GstFFMpegVidEncClass *oclass =
|
GstFFMpegVidEncClass *oclass =
|
||||||
(GstFFMpegVidEncClass *) G_OBJECT_GET_CLASS (ffmpegenc);
|
(GstFFMpegVidEncClass *) G_OBJECT_GET_CLASS (ffmpegenc);
|
||||||
AVCodecContext *ctx = NULL;
|
AVCodecContext *ctx = NULL;
|
||||||
enum PixelFormat pixfmt;
|
enum PixelFormat pixfmt;
|
||||||
GstCaps *caps = NULL;
|
GstCaps *caps = NULL;
|
||||||
GstCaps *finalcaps = NULL;
|
|
||||||
gint i;
|
gint i;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "getting caps");
|
GST_DEBUG_OBJECT (ffmpegenc, "getting caps");
|
||||||
|
|
||||||
/* cached */
|
/* cached */
|
||||||
if (oclass->sinkcaps) {
|
if (oclass->sinkcaps) {
|
||||||
caps =
|
caps = gst_video_encoder_proxy_getcaps (encoder, oclass->sinkcaps);
|
||||||
gst_ffmpegvidenc_get_possible_sizes (ffmpegenc, pad, oclass->sinkcaps);
|
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "return cached caps %" GST_PTR_FORMAT, caps);
|
GST_DEBUG_OBJECT (ffmpegenc, "return cached caps %" GST_PTR_FORMAT, caps);
|
||||||
return caps;
|
return caps;
|
||||||
}
|
}
|
||||||
|
@ -452,32 +372,21 @@ gst_ffmpegvidenc_getcaps (GstPad * pad)
|
||||||
_shut_up_I_am_probing = FALSE;
|
_shut_up_I_am_probing = FALSE;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* make sure we have something */
|
oclass->sinkcaps = gst_video_encoder_proxy_getcaps (encoder, caps);
|
||||||
if (!caps) {
|
|
||||||
caps = gst_ffmpegvidenc_get_possible_sizes (ffmpegenc, pad,
|
|
||||||
gst_pad_get_pad_template_caps (pad));
|
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "probing gave nothing, "
|
|
||||||
"return template %" GST_PTR_FORMAT, caps);
|
|
||||||
return caps;
|
|
||||||
}
|
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "probed caps gave %" GST_PTR_FORMAT, caps);
|
return gst_caps_ref (oclass->sinkcaps);
|
||||||
oclass->sinkcaps = gst_caps_copy (caps);
|
|
||||||
|
|
||||||
finalcaps = gst_ffmpegvidenc_get_possible_sizes (ffmpegenc, pad, caps);
|
|
||||||
gst_caps_unref (caps);
|
|
||||||
|
|
||||||
return finalcaps;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_ffmpegvidenc_setcaps (GstPad * pad, GstCaps * caps)
|
gst_ffmpegvidenc_set_format (GstVideoEncoder * encoder,
|
||||||
|
GstVideoCodecState * state)
|
||||||
{
|
{
|
||||||
GstCaps *other_caps;
|
GstCaps *other_caps;
|
||||||
GstCaps *allowed_caps;
|
GstCaps *allowed_caps;
|
||||||
GstCaps *icaps;
|
GstCaps *icaps;
|
||||||
|
GstVideoCodecState *output_format;
|
||||||
enum PixelFormat pix_fmt;
|
enum PixelFormat pix_fmt;
|
||||||
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) GST_PAD_PARENT (pad);
|
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) encoder;
|
||||||
GstFFMpegVidEncClass *oclass =
|
GstFFMpegVidEncClass *oclass =
|
||||||
(GstFFMpegVidEncClass *) G_OBJECT_GET_CLASS (ffmpegenc);
|
(GstFFMpegVidEncClass *) G_OBJECT_GET_CLASS (ffmpegenc);
|
||||||
|
|
||||||
|
@ -485,9 +394,12 @@ gst_ffmpegvidenc_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
if (ffmpegenc->opened) {
|
if (ffmpegenc->opened) {
|
||||||
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
||||||
ffmpegenc->opened = FALSE;
|
ffmpegenc->opened = FALSE;
|
||||||
|
#if 0
|
||||||
|
/* FIXME : Is this still needed with GstVideoEncoder ?? */
|
||||||
/* fixed src caps;
|
/* fixed src caps;
|
||||||
* so clear src caps for proper (re-)negotiation */
|
* so clear src caps for proper (re-)negotiation */
|
||||||
gst_pad_set_caps (ffmpegenc->srcpad, NULL);
|
gst_pad_set_caps (ffmpegenc->srcpad, NULL);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/* set defaults */
|
/* set defaults */
|
||||||
|
@ -545,24 +457,16 @@ gst_ffmpegvidenc_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
/* we don't close when changing caps, fingers crossed */
|
/* we don't close when changing caps, fingers crossed */
|
||||||
if (!ffmpegenc->file)
|
if (!ffmpegenc->file)
|
||||||
ffmpegenc->file = g_fopen (ffmpegenc->filename, "w");
|
ffmpegenc->file = g_fopen (ffmpegenc->filename, "w");
|
||||||
if (!ffmpegenc->file) {
|
if (!ffmpegenc->file)
|
||||||
GST_ELEMENT_ERROR (ffmpegenc, RESOURCE, OPEN_WRITE,
|
goto open_file_err;
|
||||||
(("Could not open file \"%s\" for writing."), ffmpegenc->filename),
|
|
||||||
GST_ERROR_SYSTEM);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case CODEC_FLAG_PASS2:
|
case CODEC_FLAG_PASS2:
|
||||||
{ /* need to read the whole stats file ! */
|
{ /* need to read the whole stats file ! */
|
||||||
gsize size;
|
gsize size;
|
||||||
|
|
||||||
if (!g_file_get_contents (ffmpegenc->filename,
|
if (!g_file_get_contents (ffmpegenc->filename,
|
||||||
&ffmpegenc->context->stats_in, &size, NULL)) {
|
&ffmpegenc->context->stats_in, &size, NULL))
|
||||||
GST_ELEMENT_ERROR (ffmpegenc, RESOURCE, READ,
|
goto file_read_err;
|
||||||
(("Could not get contents of file \"%s\"."), ffmpegenc->filename),
|
|
||||||
GST_ERROR_SYSTEM);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -570,14 +474,11 @@ gst_ffmpegvidenc_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* fetch pix_fmt and so on */
|
GST_DEBUG_OBJECT (ffmpegenc, "Extracting common video information");
|
||||||
gst_ffmpeg_caps_with_codectype (oclass->in_plugin->type,
|
/* fetch pix_fmt, fps, par, width, height... */
|
||||||
caps, ffmpegenc->context);
|
gst_ffmpeg_videoinfo_to_context (&state->info, ffmpegenc->context);
|
||||||
if (!ffmpegenc->context->time_base.den) {
|
|
||||||
ffmpegenc->context->time_base.den = 25;
|
if ((oclass->in_plugin->id == CODEC_ID_MPEG4)
|
||||||
ffmpegenc->context->time_base.num = 1;
|
|
||||||
ffmpegenc->context->ticks_per_frame = 1;
|
|
||||||
} else if ((oclass->in_plugin->id == CODEC_ID_MPEG4)
|
|
||||||
&& (ffmpegenc->context->time_base.den > 65535)) {
|
&& (ffmpegenc->context->time_base.den > 65535)) {
|
||||||
/* MPEG4 Standards do not support time_base denominator greater than
|
/* MPEG4 Standards do not support time_base denominator greater than
|
||||||
* (1<<16) - 1 . We therefore scale them down.
|
* (1<<16) - 1 . We therefore scale them down.
|
||||||
|
@ -606,46 +507,33 @@ gst_ffmpegvidenc_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* open codec */
|
/* open codec */
|
||||||
if (gst_ffmpeg_avcodec_open (ffmpegenc->context, oclass->in_plugin) < 0) {
|
if (gst_ffmpeg_avcodec_open (ffmpegenc->context, oclass->in_plugin) < 0)
|
||||||
if (ffmpegenc->context->priv_data)
|
goto open_codec_fail;
|
||||||
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
|
||||||
if (ffmpegenc->context->stats_in)
|
|
||||||
g_free (ffmpegenc->context->stats_in);
|
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "ffenc_%s: Failed to open FFMPEG codec",
|
|
||||||
oclass->in_plugin->name);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* second pass stats buffer no longer needed */
|
/* second pass stats buffer no longer needed */
|
||||||
if (ffmpegenc->context->stats_in)
|
if (ffmpegenc->context->stats_in)
|
||||||
g_free (ffmpegenc->context->stats_in);
|
g_free (ffmpegenc->context->stats_in);
|
||||||
|
|
||||||
/* is the colourspace correct? */
|
/* is the colourspace correct? */
|
||||||
if (pix_fmt != ffmpegenc->context->pix_fmt) {
|
if (pix_fmt != ffmpegenc->context->pix_fmt)
|
||||||
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
goto pix_fmt_err;
|
||||||
GST_DEBUG_OBJECT (ffmpegenc,
|
|
||||||
"ffenc_%s: AV wants different colourspace (%d given, %d wanted)",
|
|
||||||
oclass->in_plugin->name, pix_fmt, ffmpegenc->context->pix_fmt);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
/* we may have failed mapping caps to a pixfmt,
|
/* we may have failed mapping caps to a pixfmt,
|
||||||
* and quite some codecs do not make up their own mind about that
|
* and quite some codecs do not make up their own mind about that
|
||||||
* in any case, _NONE can never work out later on */
|
* in any case, _NONE can never work out later on */
|
||||||
if (oclass->in_plugin->type == AVMEDIA_TYPE_VIDEO && pix_fmt == PIX_FMT_NONE) {
|
if (pix_fmt == PIX_FMT_NONE)
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "ffenc_%s: Failed to determine input format",
|
goto bad_input_fmt;
|
||||||
oclass->in_plugin->name);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* some codecs support more than one format, first auto-choose one */
|
/* some codecs support more than one format, first auto-choose one */
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "picking an output format ...");
|
GST_DEBUG_OBJECT (ffmpegenc, "picking an output format ...");
|
||||||
allowed_caps = gst_pad_get_allowed_caps (ffmpegenc->srcpad);
|
allowed_caps = gst_pad_get_allowed_caps (GST_VIDEO_ENCODER_SRC_PAD (encoder));
|
||||||
if (!allowed_caps) {
|
if (!allowed_caps) {
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "... but no peer, using template caps");
|
GST_DEBUG_OBJECT (ffmpegenc, "... but no peer, using template caps");
|
||||||
/* we need to copy because get_allowed_caps returns a ref, and
|
/* we need to copy because get_allowed_caps returns a ref, and
|
||||||
* get_pad_template_caps doesn't */
|
* get_pad_template_caps doesn't */
|
||||||
allowed_caps =
|
allowed_caps =
|
||||||
gst_caps_copy (gst_pad_get_pad_template_caps (ffmpegenc->srcpad));
|
gst_caps_copy (gst_pad_get_pad_template_caps (GST_VIDEO_ENCODER_SRC_PAD
|
||||||
|
(encoder)));
|
||||||
}
|
}
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "chose caps %" GST_PTR_FORMAT, allowed_caps);
|
GST_DEBUG_OBJECT (ffmpegenc, "chose caps %" GST_PTR_FORMAT, allowed_caps);
|
||||||
gst_ffmpeg_caps_with_codecid (oclass->in_plugin->id,
|
gst_ffmpeg_caps_with_codecid (oclass->in_plugin->id,
|
||||||
|
@ -655,11 +543,8 @@ gst_ffmpegvidenc_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
other_caps = gst_ffmpeg_codecid_to_caps (oclass->in_plugin->id,
|
other_caps = gst_ffmpeg_codecid_to_caps (oclass->in_plugin->id,
|
||||||
ffmpegenc->context, TRUE);
|
ffmpegenc->context, TRUE);
|
||||||
|
|
||||||
if (!other_caps) {
|
if (!other_caps)
|
||||||
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
goto unsupported_codec;
|
||||||
GST_DEBUG ("Unsupported codec - no caps found");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
icaps = gst_caps_intersect (allowed_caps, other_caps);
|
icaps = gst_caps_intersect (allowed_caps, other_caps);
|
||||||
gst_caps_unref (allowed_caps);
|
gst_caps_unref (allowed_caps);
|
||||||
|
@ -679,17 +564,68 @@ gst_ffmpegvidenc_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
icaps = newcaps;
|
icaps = newcaps;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!gst_pad_set_caps (ffmpegenc->srcpad, icaps)) {
|
/* Store input state and set output state */
|
||||||
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
if (ffmpegenc->input_state)
|
||||||
gst_caps_unref (icaps);
|
gst_video_codec_state_unref (ffmpegenc->input_state);
|
||||||
return FALSE;
|
ffmpegenc->input_state = gst_video_codec_state_ref (state);
|
||||||
}
|
|
||||||
gst_caps_unref (icaps);
|
output_format = gst_video_encoder_set_output_state (encoder, icaps, state);
|
||||||
|
gst_video_codec_state_unref (output_format);
|
||||||
|
|
||||||
/* success! */
|
/* success! */
|
||||||
ffmpegenc->opened = TRUE;
|
ffmpegenc->opened = TRUE;
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
|
/* ERRORS */
|
||||||
|
open_file_err:
|
||||||
|
{
|
||||||
|
GST_ELEMENT_ERROR (ffmpegenc, RESOURCE, OPEN_WRITE,
|
||||||
|
(("Could not open file \"%s\" for writing."), ffmpegenc->filename),
|
||||||
|
GST_ERROR_SYSTEM);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
file_read_err:
|
||||||
|
{
|
||||||
|
GST_ELEMENT_ERROR (ffmpegenc, RESOURCE, READ,
|
||||||
|
(("Could not get contents of file \"%s\"."), ffmpegenc->filename),
|
||||||
|
GST_ERROR_SYSTEM);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
open_codec_fail:
|
||||||
|
{
|
||||||
|
if (ffmpegenc->context->priv_data)
|
||||||
|
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
||||||
|
if (ffmpegenc->context->stats_in)
|
||||||
|
g_free (ffmpegenc->context->stats_in);
|
||||||
|
GST_DEBUG_OBJECT (ffmpegenc, "ffenc_%s: Failed to open FFMPEG codec",
|
||||||
|
oclass->in_plugin->name);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
pix_fmt_err:
|
||||||
|
{
|
||||||
|
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
||||||
|
GST_DEBUG_OBJECT (ffmpegenc,
|
||||||
|
"ffenc_%s: AV wants different colourspace (%d given, %d wanted)",
|
||||||
|
oclass->in_plugin->name, pix_fmt, ffmpegenc->context->pix_fmt);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
bad_input_fmt:
|
||||||
|
{
|
||||||
|
GST_DEBUG_OBJECT (ffmpegenc, "ffenc_%s: Failed to determine input format",
|
||||||
|
oclass->in_plugin->name);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsupported_codec:
|
||||||
|
{
|
||||||
|
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
||||||
|
GST_DEBUG ("Unsupported codec - no caps found");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -712,33 +648,31 @@ ffmpegenc_setup_working_buf (GstFFMpegVidEnc * ffmpegenc)
|
||||||
}
|
}
|
||||||
|
|
||||||
static GstFlowReturn
|
static GstFlowReturn
|
||||||
gst_ffmpegvidenc_chain_video (GstPad * pad, GstBuffer * inbuf)
|
gst_ffmpegvidenc_handle_frame (GstVideoEncoder * encoder,
|
||||||
|
GstVideoCodecFrame * frame)
|
||||||
{
|
{
|
||||||
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) (GST_PAD_PARENT (pad));
|
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) encoder;
|
||||||
GstBuffer *outbuf;
|
GstBuffer *outbuf;
|
||||||
gint ret_size = 0, frame_size;
|
gint ret_size = 0, c;
|
||||||
gboolean force_keyframe;
|
GstVideoInfo *info = &ffmpegenc->input_state->info;
|
||||||
|
guint8 *data = GST_BUFFER_DATA (frame->input_buffer);
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (ffmpegenc,
|
if (GST_VIDEO_CODEC_FRAME_IS_FORCE_KEYFRAME (frame))
|
||||||
"Received buffer of time %" GST_TIME_FORMAT,
|
|
||||||
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (inbuf)));
|
|
||||||
|
|
||||||
GST_OBJECT_LOCK (ffmpegenc);
|
|
||||||
force_keyframe = ffmpegenc->force_keyframe;
|
|
||||||
ffmpegenc->force_keyframe = FALSE;
|
|
||||||
GST_OBJECT_UNLOCK (ffmpegenc);
|
|
||||||
|
|
||||||
if (force_keyframe)
|
|
||||||
ffmpegenc->picture->pict_type = FF_I_TYPE;
|
ffmpegenc->picture->pict_type = FF_I_TYPE;
|
||||||
|
|
||||||
frame_size = gst_ffmpeg_avpicture_fill ((AVPicture *) ffmpegenc->picture,
|
/* Fill avpicture */
|
||||||
GST_BUFFER_DATA (inbuf),
|
for (c = 0; c < AV_NUM_DATA_POINTERS; c++) {
|
||||||
ffmpegenc->context->pix_fmt,
|
if (c < GST_VIDEO_INFO_N_COMPONENTS (info)) {
|
||||||
ffmpegenc->context->width, ffmpegenc->context->height);
|
ffmpegenc->picture->data[c] = data + GST_VIDEO_INFO_COMP_OFFSET (info, c);
|
||||||
g_return_val_if_fail (frame_size == GST_BUFFER_SIZE (inbuf), GST_FLOW_ERROR);
|
ffmpegenc->picture->linesize[c] = GST_VIDEO_INFO_COMP_STRIDE (info, c);
|
||||||
|
} else {
|
||||||
|
ffmpegenc->picture->data[c] = NULL;
|
||||||
|
ffmpegenc->picture->linesize[c] = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ffmpegenc->picture->pts =
|
ffmpegenc->picture->pts =
|
||||||
gst_ffmpeg_time_gst_to_ff (GST_BUFFER_TIMESTAMP (inbuf) /
|
gst_ffmpeg_time_gst_to_ff (frame->pts /
|
||||||
ffmpegenc->context->ticks_per_frame, ffmpegenc->context->time_base);
|
ffmpegenc->context->ticks_per_frame, ffmpegenc->context->time_base);
|
||||||
|
|
||||||
ffmpegenc_setup_working_buf (ffmpegenc);
|
ffmpegenc_setup_working_buf (ffmpegenc);
|
||||||
|
@ -746,24 +680,11 @@ gst_ffmpegvidenc_chain_video (GstPad * pad, GstBuffer * inbuf)
|
||||||
ret_size = avcodec_encode_video (ffmpegenc->context,
|
ret_size = avcodec_encode_video (ffmpegenc->context,
|
||||||
ffmpegenc->working_buf, ffmpegenc->working_buf_size, ffmpegenc->picture);
|
ffmpegenc->working_buf, ffmpegenc->working_buf_size, ffmpegenc->picture);
|
||||||
|
|
||||||
if (ret_size < 0) {
|
if (ret_size < 0)
|
||||||
#ifndef GST_DISABLE_GST_DEBUG
|
goto encode_fail;
|
||||||
GstFFMpegVidEncClass *oclass =
|
|
||||||
(GstFFMpegVidEncClass *) (G_OBJECT_GET_CLASS (ffmpegenc));
|
|
||||||
GST_ERROR_OBJECT (ffmpegenc,
|
|
||||||
"ffenc_%s: failed to encode buffer", oclass->in_plugin->name);
|
|
||||||
#endif /* GST_DISABLE_GST_DEBUG */
|
|
||||||
gst_buffer_unref (inbuf);
|
|
||||||
return GST_FLOW_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* handle b-frame delay when no output, so we don't output empty frames;
|
/* Encoder needs more data */
|
||||||
* timestamps and so can permute a bit between coding and display order
|
if (!ret_size)
|
||||||
* but keyframes should still end up with the proper metadata */
|
|
||||||
g_queue_push_tail (ffmpegenc->delay, inbuf);
|
|
||||||
if (ret_size)
|
|
||||||
inbuf = g_queue_pop_head (ffmpegenc->delay);
|
|
||||||
else
|
|
||||||
return GST_FLOW_OK;
|
return GST_FLOW_OK;
|
||||||
|
|
||||||
/* save stats info if there is some as well as a stats file */
|
/* save stats info if there is some as well as a stats file */
|
||||||
|
@ -773,48 +694,54 @@ gst_ffmpegvidenc_chain_video (GstPad * pad, GstBuffer * inbuf)
|
||||||
(("Could not write to file \"%s\"."), ffmpegenc->filename),
|
(("Could not write to file \"%s\"."), ffmpegenc->filename),
|
||||||
GST_ERROR_SYSTEM);
|
GST_ERROR_SYSTEM);
|
||||||
|
|
||||||
outbuf = gst_buffer_new_and_alloc (ret_size);
|
/* Get oldest frame */
|
||||||
|
frame = gst_video_encoder_get_oldest_frame (encoder);
|
||||||
|
|
||||||
|
/* Allocate output buffer */
|
||||||
|
frame->output_buffer = outbuf = gst_buffer_new_and_alloc (ret_size);
|
||||||
memcpy (GST_BUFFER_DATA (outbuf), ffmpegenc->working_buf, ret_size);
|
memcpy (GST_BUFFER_DATA (outbuf), ffmpegenc->working_buf, ret_size);
|
||||||
GST_BUFFER_TIMESTAMP (outbuf) = GST_BUFFER_TIMESTAMP (inbuf);
|
|
||||||
GST_BUFFER_DURATION (outbuf) = GST_BUFFER_DURATION (inbuf);
|
|
||||||
/* buggy codec may not set coded_frame */
|
/* buggy codec may not set coded_frame */
|
||||||
if (ffmpegenc->context->coded_frame) {
|
if (ffmpegenc->context->coded_frame) {
|
||||||
if (!ffmpegenc->context->coded_frame->key_frame)
|
if (ffmpegenc->context->coded_frame->key_frame)
|
||||||
GST_BUFFER_FLAG_SET (outbuf, GST_BUFFER_FLAG_DELTA_UNIT);
|
GST_VIDEO_CODEC_FRAME_SET_SYNC_POINT (frame);
|
||||||
} else
|
} else
|
||||||
GST_WARNING_OBJECT (ffmpegenc, "codec did not provide keyframe info");
|
GST_WARNING_OBJECT (ffmpegenc, "codec did not provide keyframe info");
|
||||||
gst_buffer_set_caps (outbuf, GST_PAD_CAPS (ffmpegenc->srcpad));
|
|
||||||
|
|
||||||
gst_buffer_unref (inbuf);
|
|
||||||
|
|
||||||
/* Reset frame type */
|
/* Reset frame type */
|
||||||
if (ffmpegenc->picture->pict_type)
|
if (ffmpegenc->picture->pict_type)
|
||||||
ffmpegenc->picture->pict_type = 0;
|
ffmpegenc->picture->pict_type = 0;
|
||||||
|
|
||||||
if (force_keyframe) {
|
return gst_video_encoder_finish_frame (encoder, frame);
|
||||||
gst_pad_push_event (ffmpegenc->srcpad,
|
|
||||||
gst_event_new_custom (GST_EVENT_CUSTOM_DOWNSTREAM,
|
|
||||||
gst_structure_new ("GstForceKeyUnit",
|
|
||||||
"timestamp", G_TYPE_UINT64, GST_BUFFER_TIMESTAMP (outbuf),
|
|
||||||
NULL)));
|
|
||||||
}
|
|
||||||
|
|
||||||
return gst_pad_push (ffmpegenc->srcpad, outbuf);
|
/* ERRORS */
|
||||||
|
encode_fail:
|
||||||
|
{
|
||||||
|
#ifndef GST_DISABLE_GST_DEBUG
|
||||||
|
GstFFMpegVidEncClass *oclass =
|
||||||
|
(GstFFMpegVidEncClass *) (G_OBJECT_GET_CLASS (ffmpegenc));
|
||||||
|
GST_ERROR_OBJECT (ffmpegenc,
|
||||||
|
"ffenc_%s: failed to encode buffer", oclass->in_plugin->name);
|
||||||
|
#endif /* GST_DISABLE_GST_DEBUG */
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_ffmpegvidenc_flush_buffers (GstFFMpegVidEnc * ffmpegenc, gboolean send)
|
gst_ffmpegvidenc_flush_buffers (GstFFMpegVidEnc * ffmpegenc, gboolean send)
|
||||||
{
|
{
|
||||||
GstBuffer *outbuf, *inbuf;
|
GstVideoCodecFrame *frame;
|
||||||
|
GstBuffer *outbuf;
|
||||||
gint ret_size;
|
gint ret_size;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (ffmpegenc, "flushing buffers with sending %d", send);
|
GST_DEBUG_OBJECT (ffmpegenc, "flushing buffers with sending %d", send);
|
||||||
|
|
||||||
/* no need to empty codec if there is none */
|
/* no need to empty codec if there is none */
|
||||||
if (!ffmpegenc->opened)
|
if (!ffmpegenc->opened)
|
||||||
goto flush;
|
return;
|
||||||
|
|
||||||
while (!g_queue_is_empty (ffmpegenc->delay)) {
|
while ((frame =
|
||||||
|
gst_video_encoder_get_oldest_frame (GST_VIDEO_ENCODER (ffmpegenc)))) {
|
||||||
|
|
||||||
ffmpegenc_setup_working_buf (ffmpegenc);
|
ffmpegenc_setup_working_buf (ffmpegenc);
|
||||||
|
|
||||||
|
@ -838,90 +765,16 @@ gst_ffmpegvidenc_flush_buffers (GstFFMpegVidEnc * ffmpegenc, gboolean send)
|
||||||
(("Could not write to file \"%s\"."), ffmpegenc->filename),
|
(("Could not write to file \"%s\"."), ffmpegenc->filename),
|
||||||
GST_ERROR_SYSTEM);
|
GST_ERROR_SYSTEM);
|
||||||
|
|
||||||
/* handle b-frame delay when no output, so we don't output empty frames */
|
frame->output_buffer = outbuf = gst_buffer_new_and_alloc (ret_size);
|
||||||
inbuf = g_queue_pop_head (ffmpegenc->delay);
|
|
||||||
|
|
||||||
outbuf = gst_buffer_new_and_alloc (ret_size);
|
|
||||||
memcpy (GST_BUFFER_DATA (outbuf), ffmpegenc->working_buf, ret_size);
|
memcpy (GST_BUFFER_DATA (outbuf), ffmpegenc->working_buf, ret_size);
|
||||||
GST_BUFFER_TIMESTAMP (outbuf) = GST_BUFFER_TIMESTAMP (inbuf);
|
|
||||||
GST_BUFFER_DURATION (outbuf) = GST_BUFFER_DURATION (inbuf);
|
|
||||||
|
|
||||||
if (!ffmpegenc->context->coded_frame->key_frame)
|
if (ffmpegenc->context->coded_frame->key_frame)
|
||||||
GST_BUFFER_FLAG_SET (outbuf, GST_BUFFER_FLAG_DELTA_UNIT);
|
GST_VIDEO_CODEC_FRAME_SET_SYNC_POINT (frame);
|
||||||
gst_buffer_set_caps (outbuf, GST_PAD_CAPS (ffmpegenc->srcpad));
|
|
||||||
|
|
||||||
gst_buffer_unref (inbuf);
|
gst_video_encoder_finish_frame (GST_VIDEO_ENCODER (ffmpegenc), frame);
|
||||||
|
|
||||||
if (send)
|
|
||||||
gst_pad_push (ffmpegenc->srcpad, outbuf);
|
|
||||||
else
|
|
||||||
gst_buffer_unref (outbuf);
|
|
||||||
}
|
|
||||||
|
|
||||||
flush:
|
|
||||||
{
|
|
||||||
/* make sure that we empty the queue, is still needed if we had to break */
|
|
||||||
while (!g_queue_is_empty (ffmpegenc->delay))
|
|
||||||
gst_buffer_unref (g_queue_pop_head (ffmpegenc->delay));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
|
||||||
gst_ffmpegvidenc_event_video (GstPad * pad, GstEvent * event)
|
|
||||||
{
|
|
||||||
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) (GST_PAD_PARENT (pad));
|
|
||||||
|
|
||||||
switch (GST_EVENT_TYPE (event)) {
|
|
||||||
case GST_EVENT_EOS:
|
|
||||||
gst_ffmpegvidenc_flush_buffers (ffmpegenc, TRUE);
|
|
||||||
break;
|
|
||||||
/* no flushing if flush received,
|
|
||||||
* buffers in encoder are considered (in the) past */
|
|
||||||
|
|
||||||
case GST_EVENT_CUSTOM_DOWNSTREAM:{
|
|
||||||
const GstStructure *s;
|
|
||||||
s = gst_event_get_structure (event);
|
|
||||||
if (gst_structure_has_name (s, "GstForceKeyUnit")) {
|
|
||||||
ffmpegenc->picture->pict_type = FF_I_TYPE;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return gst_pad_push_event (ffmpegenc->srcpad, event);
|
|
||||||
}
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
gst_ffmpegvidenc_event_src (GstPad * pad, GstEvent * event)
|
|
||||||
{
|
|
||||||
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) (GST_PAD_PARENT (pad));
|
|
||||||
gboolean forward = TRUE;
|
|
||||||
|
|
||||||
switch (GST_EVENT_TYPE (event)) {
|
|
||||||
case GST_EVENT_CUSTOM_UPSTREAM:{
|
|
||||||
const GstStructure *s;
|
|
||||||
s = gst_event_get_structure (event);
|
|
||||||
if (gst_structure_has_name (s, "GstForceKeyUnit")) {
|
|
||||||
GST_OBJECT_LOCK (ffmpegenc);
|
|
||||||
ffmpegenc->force_keyframe = TRUE;
|
|
||||||
GST_OBJECT_UNLOCK (ffmpegenc);
|
|
||||||
forward = FALSE;
|
|
||||||
gst_event_unref (event);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (forward)
|
|
||||||
return gst_pad_push_event (ffmpegenc->sinkpad, event);
|
|
||||||
else
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_ffmpegvidenc_set_property (GObject * object,
|
gst_ffmpegvidenc_set_property (GObject * object,
|
||||||
|
@ -994,39 +847,36 @@ gst_ffmpegvidenc_get_property (GObject * object,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static GstStateChangeReturn
|
static gboolean
|
||||||
gst_ffmpegvidenc_change_state (GstElement * element, GstStateChange transition)
|
gst_ffmpegvidenc_stop (GstVideoEncoder * encoder)
|
||||||
{
|
{
|
||||||
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) element;
|
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) encoder;
|
||||||
GstStateChangeReturn result;
|
|
||||||
|
|
||||||
switch (transition) {
|
gst_ffmpegvidenc_flush_buffers (ffmpegenc, FALSE);
|
||||||
default:
|
if (ffmpegenc->opened) {
|
||||||
break;
|
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
||||||
|
ffmpegenc->opened = FALSE;
|
||||||
|
}
|
||||||
|
if (ffmpegenc->file) {
|
||||||
|
fclose (ffmpegenc->file);
|
||||||
|
ffmpegenc->file = NULL;
|
||||||
|
}
|
||||||
|
if (ffmpegenc->working_buf) {
|
||||||
|
g_free (ffmpegenc->working_buf);
|
||||||
|
ffmpegenc->working_buf = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
result = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
switch (transition) {
|
static GstFlowReturn
|
||||||
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
gst_ffmpegvidenc_finish (GstVideoEncoder * encoder)
|
||||||
gst_ffmpegvidenc_flush_buffers (ffmpegenc, FALSE);
|
{
|
||||||
if (ffmpegenc->opened) {
|
GstFFMpegVidEnc *ffmpegenc = (GstFFMpegVidEnc *) encoder;
|
||||||
gst_ffmpeg_avcodec_close (ffmpegenc->context);
|
|
||||||
ffmpegenc->opened = FALSE;
|
gst_ffmpegvidenc_flush_buffers (ffmpegenc, TRUE);
|
||||||
}
|
|
||||||
if (ffmpegenc->file) {
|
return GST_FLOW_OK;
|
||||||
fclose (ffmpegenc->file);
|
|
||||||
ffmpegenc->file = NULL;
|
|
||||||
}
|
|
||||||
if (ffmpegenc->working_buf) {
|
|
||||||
g_free (ffmpegenc->working_buf);
|
|
||||||
ffmpegenc->working_buf = NULL;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
|
@ -1092,8 +942,7 @@ gst_ffmpegvidenc_register (GstPlugin * plugin)
|
||||||
GST_DEBUG ("Trying plugin %s [%s]", in_plugin->name, in_plugin->long_name);
|
GST_DEBUG ("Trying plugin %s [%s]", in_plugin->name, in_plugin->long_name);
|
||||||
|
|
||||||
/* no codecs for which we're GUARANTEED to have better alternatives */
|
/* no codecs for which we're GUARANTEED to have better alternatives */
|
||||||
if (!strcmp (in_plugin->name, "vorbis") ||
|
if (!strcmp (in_plugin->name, "gif")) {
|
||||||
!strcmp (in_plugin->name, "gif") || !strcmp (in_plugin->name, "flac")) {
|
|
||||||
GST_LOG ("Ignoring encoder %s", in_plugin->name);
|
GST_LOG ("Ignoring encoder %s", in_plugin->name);
|
||||||
goto next;
|
goto next;
|
||||||
}
|
}
|
||||||
|
@ -1106,7 +955,9 @@ gst_ffmpegvidenc_register (GstPlugin * plugin)
|
||||||
if (!type) {
|
if (!type) {
|
||||||
|
|
||||||
/* create the glib type now */
|
/* create the glib type now */
|
||||||
type = g_type_register_static (GST_TYPE_ELEMENT, type_name, &typeinfo, 0);
|
type =
|
||||||
|
g_type_register_static (GST_TYPE_VIDEO_ENCODER, type_name, &typeinfo,
|
||||||
|
0);
|
||||||
g_type_set_qdata (type, GST_FFENC_PARAMS_QDATA, (gpointer) in_plugin);
|
g_type_set_qdata (type, GST_FFENC_PARAMS_QDATA, (gpointer) in_plugin);
|
||||||
|
|
||||||
{
|
{
|
||||||
|
|
|
@ -26,15 +26,15 @@
|
||||||
|
|
||||||
G_BEGIN_DECLS
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
#include <gst/video/gstvideoencoder.h>
|
||||||
|
|
||||||
typedef struct _GstFFMpegVidEnc GstFFMpegVidEnc;
|
typedef struct _GstFFMpegVidEnc GstFFMpegVidEnc;
|
||||||
|
|
||||||
struct _GstFFMpegVidEnc
|
struct _GstFFMpegVidEnc
|
||||||
{
|
{
|
||||||
GstElement element;
|
GstVideoEncoder parent;
|
||||||
|
|
||||||
/* We need to keep track of our pads, so we do so here. */
|
GstVideoCodecState *input_state;
|
||||||
GstPad *srcpad;
|
|
||||||
GstPad *sinkpad;
|
|
||||||
|
|
||||||
AVCodecContext *context;
|
AVCodecContext *context;
|
||||||
AVFrame *picture;
|
AVFrame *picture;
|
||||||
|
@ -63,21 +63,16 @@ struct _GstFFMpegVidEnc
|
||||||
/* statistics file */
|
/* statistics file */
|
||||||
FILE *file;
|
FILE *file;
|
||||||
|
|
||||||
/* for b-frame delay handling */
|
|
||||||
GQueue *delay;
|
|
||||||
|
|
||||||
/* other settings are copied over straight,
|
/* other settings are copied over straight,
|
||||||
* include a context here, rather than copy-and-past it from avcodec.h */
|
* include a context here, rather than copy-and-past it from avcodec.h */
|
||||||
AVCodecContext config;
|
AVCodecContext config;
|
||||||
|
|
||||||
gboolean force_keyframe;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct _GstFFMpegVidEncClass GstFFMpegVidEncClass;
|
typedef struct _GstFFMpegVidEncClass GstFFMpegVidEncClass;
|
||||||
|
|
||||||
struct _GstFFMpegVidEncClass
|
struct _GstFFMpegVidEncClass
|
||||||
{
|
{
|
||||||
GstElementClass parent_class;
|
GstVideoEncoderClass parent_class;
|
||||||
|
|
||||||
AVCodec *in_plugin;
|
AVCodec *in_plugin;
|
||||||
GstPadTemplate *srctempl, *sinktempl;
|
GstPadTemplate *srctempl, *sinktempl;
|
||||||
|
|
Loading…
Reference in a new issue