From 2965dbac47a7d08a7294919aa0a1e7e24ca1252b Mon Sep 17 00:00:00 2001 From: Stefan Kost Date: Fri, 3 Jun 2011 11:35:55 +0300 Subject: [PATCH 01/11] synaesthesia: fix wrong debug log string (copy'n'paste) --- gst/synaesthesia/gstsynaesthesia.c | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/gst/synaesthesia/gstsynaesthesia.c b/gst/synaesthesia/gstsynaesthesia.c index 910ffb2932..79ca45fef3 100644 --- a/gst/synaesthesia/gstsynaesthesia.c +++ b/gst/synaesthesia/gstsynaesthesia.c @@ -312,7 +312,8 @@ done: /* Errors */ missing_caps_details: { - GST_WARNING_OBJECT (synaesthesia, "missing channels or rate in the caps"); + GST_WARNING_OBJECT (synaesthesia, + "missing width, height or framerate in the caps"); res = FALSE; goto done; } From 3ba6d1588fa3643aa5c0ff61535050997e64cf83 Mon Sep 17 00:00:00 2001 From: Mark Nauwelaerts Date: Mon, 6 Jun 2011 12:41:03 +0200 Subject: [PATCH 02/11] rtpasfdepay: fix fragmented packet handling and packet padding Also remove a bogus assert. --- gst/asfdemux/gstrtpasfdepay.c | 16 +++++++--------- 1 file changed, 7 insertions(+), 9 deletions(-) diff --git a/gst/asfdemux/gstrtpasfdepay.c b/gst/asfdemux/gstrtpasfdepay.c index 12ce8afdd3..7138be2c26 100644 --- a/gst/asfdemux/gstrtpasfdepay.c +++ b/gst/asfdemux/gstrtpasfdepay.c @@ -456,21 +456,19 @@ gst_rtp_asf_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf) return NULL; /* we need to pad with zeroes to packet_size if it's smaller */ - g_assert (packet_len == GST_BUFFER_SIZE (outbuf)); - packet_len = GST_BUFFER_SIZE (outbuf); - if (packet_len < depay->packet_size) { + if (GST_BUFFER_SIZE (outbuf) < depay->packet_size) { GstBuffer *tmp; + gint plen = GST_BUFFER_SIZE (outbuf); GST_LOG_OBJECT (depay, "padding buffer size %d to packet size %d", - packet_len, depay->packet_size); + plen, depay->packet_size); tmp = gst_buffer_new_and_alloc (depay->packet_size); - memcpy (GST_BUFFER_DATA (tmp), GST_BUFFER_DATA (outbuf), packet_len); + memcpy (GST_BUFFER_DATA (tmp), GST_BUFFER_DATA (outbuf), plen); + gst_buffer_copy_metadata (tmp, outbuf, GST_BUFFER_COPY_ALL); gst_buffer_unref (outbuf); outbuf = tmp; - memset (GST_BUFFER_DATA (outbuf) + packet_len, 0, - depay->packet_size - packet_len); - gst_rtp_asf_depay_set_padding (depay, outbuf, - depay->packet_size - packet_len); + memset (GST_BUFFER_DATA (outbuf) + plen, 0, depay->packet_size - plen); + gst_rtp_asf_depay_set_padding (depay, outbuf, depay->packet_size - plen); } gst_buffer_set_caps (outbuf, GST_PAD_CAPS (depayload->srcpad)); From 50f1f519302e06763d60ebc038b2cfeb40edcb29 Mon Sep 17 00:00:00 2001 From: David Schleef Date: Thu, 23 Jun 2011 11:28:58 -0700 Subject: [PATCH 03/11] Automatic update of common submodule From 69b981f to 605cd9a --- common | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/common b/common index 69b981f10c..605cd9a65e 160000 --- a/common +++ b/common @@ -1 +1 @@ -Subproject commit 69b981f10caa234ad0ff639179d0fda8505bd94b +Subproject commit 605cd9a65ed61505f24b840d3fe8e252be72b151 From 48f899257db49b5289f754cbd92fce246854dd0f Mon Sep 17 00:00:00 2001 From: Brian Gitonga Marete Date: Sat, 25 Jun 2011 06:29:50 +0300 Subject: [PATCH 04/11] x264enc: fix subme property annotation - subme maximum is 10, not 6. Although the element accepts subme values > 6, the annotation which is visible through gst-inspect (for example) erroneously indicates 6 as the maximum. Fix this by indicating 10 (which is the x264 max) as the maximum. https://bugzilla.gnome.org/show_bug.cgi?id=653473 --- ext/x264/gstx264enc.c | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/ext/x264/gstx264enc.c b/ext/x264/gstx264enc.c index 66e764797e..23d76f704d 100644 --- a/ext/x264/gstx264enc.c +++ b/ext/x264/gstx264enc.c @@ -673,8 +673,9 @@ gst_x264_enc_class_init (GstX264EncClass * klass) x264_motion_est_names[ARG_ME_DEFAULT]); g_object_class_install_property (gobject_class, ARG_SUBME, g_param_spec_uint ("subme", "Subpixel Motion Estimation", - "Subpixel motion estimation and partition decision quality: 1=fast, 6=best", - 1, 6, ARG_SUBME_DEFAULT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); + "Subpixel motion estimation and partition decision quality: 1=fast, 10=best", + 1, 10, ARG_SUBME_DEFAULT, + G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); g_string_append_printf (x264enc_defaults, ":subme=%d", ARG_SUBME_DEFAULT); g_object_class_install_property (gobject_class, ARG_ANALYSE, g_param_spec_flags ("analyse", "Analyse", "Partitions to consider", From e27dda7c62574d130a2e02edf3825062f3033fcd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Olivier=20Cr=C3=AAte?= Date: Mon, 7 Mar 2011 17:58:34 -0500 Subject: [PATCH 05/11] x264: Use profile and level from caps Enforces the profile and level from the downstream caps, also sets them on the fixated caps https://bugzilla.gnome.org/show_bug.cgi?id=644233 --- ext/x264/Makefile.am | 4 +- ext/x264/gstx264enc.c | 216 +++++++++++++++++++++++++++++++++++++++--- ext/x264/gstx264enc.h | 5 + 3 files changed, 211 insertions(+), 14 deletions(-) diff --git a/ext/x264/Makefile.am b/ext/x264/Makefile.am index f5590139fe..92691d1535 100644 --- a/ext/x264/Makefile.am +++ b/ext/x264/Makefile.am @@ -6,7 +6,9 @@ libgstx264_la_CFLAGS = \ $(GST_CFLAGS) \ $(X264_CFLAGS) libgstx264_la_LIBADD = \ - $(GST_PLUGINS_BASE_LIBS) -lgstvideo-$(GST_MAJORMINOR) \ + $(GST_PLUGINS_BASE_LIBS) \ + -lgstvideo-$(GST_MAJORMINOR) \ + -lgstpbutils-$(GST_MAJORMINOR) \ $(GST_LIBS) \ $(X264_LIBS) libgstx264_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS) diff --git a/ext/x264/gstx264enc.c b/ext/x264/gstx264enc.c index 23d76f704d..15dd190f19 100644 --- a/ext/x264/gstx264enc.c +++ b/ext/x264/gstx264enc.c @@ -37,9 +37,11 @@ * If #GstX264Enc:dct8x8 is enabled, then High profile is used. * Otherwise, if #GstX264Enc:cabac entropy coding is enabled or #GstX264Enc:bframes * are allowed, then Main Profile is in effect, and otherwise Baseline profile - * applies. No profile is imposed by default, which is fine for most software - * players and settings, but in some cases (e.g. hardware platforms) a more - * restricted profile/level may be necessary. + * applies. The main profile is imposed by default, + * which is fine for most software players and settings, + * but in some cases (e.g. hardware platforms) a more restricted profile/level + * may be necessary. The recommended way to set a profile is to set it in the + * downstream caps. * * If a preset/tuning are specified then these will define the default values and * the property defaults will be ignored. After this the option-string property is @@ -71,7 +73,7 @@ * ]| This example pipeline will encode a test video source to H264 using fixed * quantization, and muxes it in a Matroska container. * |[ - * gst-launch -v videotestsrc num-buffers=1000 ! x264enc pass=5 quantizer=25 speed-preset=6 profile=1 ! \ + * gst-launch -v videotestsrc num-buffers=1000 ! x264enc pass=5 quantizer=25 speed-preset=6 ! video/x-h264, profile=baseline ! \ * qtmux ! filesink location=videotestsrc.mov * ]| This example pipeline will encode a test video source to H264 using * constant quality at around Q25 using the 'medium' speed/quality preset and @@ -94,6 +96,8 @@ #include "gstx264enc.h" +#include + #if X264_BUILD >= 71 #define X264_DELAYED_FRAMES_API #endif @@ -461,7 +465,9 @@ static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src", "framerate = (fraction) [0/1, MAX], " "width = (int) [ 1, MAX ], " "height = (int) [ 1, MAX ], " "stream-format = (string) { byte-stream, avc }, " - "alignment = (string) { au }") + "alignment = (string) { au }, " + "profile = (string) { high-10, high, main, constrained-baseline, " + "high-10-intra }") ); static void gst_x264_enc_finalize (GObject * object); @@ -604,7 +610,8 @@ gst_x264_enc_class_init (GstX264EncClass * klass) g_object_class_install_property (gobject_class, ARG_PROFILE, g_param_spec_enum ("profile", "H.264 profile", "Apply restrictions to meet H.264 Profile constraints. This will " - "override other properties if necessary.", + "override other properties if necessary. This will only be used " + "if downstream elements do not specify a profile in their caps (DEPRECATED)", GST_X264_ENC_PROFILE_TYPE, ARG_PROFILE_DEFAULT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); #endif /* X264_PRESETS */ @@ -1150,8 +1157,8 @@ gst_x264_enc_init_encoder (GstX264Enc * encoder) encoder->x264param.rc.i_rc_method = X264_RC_ABR; encoder->x264param.rc.i_bitrate = encoder->bitrate; encoder->x264param.rc.i_vbv_max_bitrate = encoder->bitrate; - encoder->x264param.rc.i_vbv_buffer_size - = encoder->x264param.rc.i_vbv_max_bitrate + encoder->x264param.rc.i_vbv_buffer_size = + encoder->x264param.rc.i_vbv_max_bitrate * encoder->vbv_buf_capacity / 1000; pass = encoder->pass & 0xF; break; @@ -1207,14 +1214,43 @@ gst_x264_enc_init_encoder (GstX264Enc * encoder) #endif #ifdef X264_PRESETS - if (encoder->profile - && x264_param_apply_profile (&encoder->x264param, - x264_profile_names[encoder->profile - 1])) { - GST_WARNING_OBJECT (encoder, "Bad profile name: %s", - x264_profile_names[encoder->profile - 1]); + if (encoder->peer_profile) { + if (x264_param_apply_profile (&encoder->x264param, encoder->peer_profile)) + GST_WARNING_OBJECT (encoder, "Bad downstream profile name: %s", + encoder->peer_profile); + } else if (encoder->profile) { + if (x264_param_apply_profile (&encoder->x264param, + x264_profile_names[encoder->profile - 1])) + GST_WARNING_OBJECT (encoder, "Bad profile name: %s", + x264_profile_names[encoder->profile - 1]); } #endif /* X264_PRESETS */ + /* If using an intra profile, all frames are intra frames */ + if (encoder->peer_intra_profile) + encoder->x264param.i_keyint_max = encoder->x264param.i_keyint_min = 1; + + /* Enforce level limits if they were in the caps */ + if (encoder->peer_level) { + encoder->x264param.i_level_idc = encoder->peer_level->level_idc; + + encoder->x264param.rc.i_bitrate = MIN (encoder->x264param.rc.i_bitrate, + encoder->peer_level->bitrate); + encoder->x264param.rc.i_vbv_max_bitrate = + MIN (encoder->x264param.rc.i_vbv_max_bitrate, + encoder->peer_level->bitrate); + encoder->x264param.rc.i_vbv_buffer_size = + MIN (encoder->x264param.rc.i_vbv_buffer_size, encoder->peer_level->cpb); + encoder->x264param.analyse.i_mv_range = + MIN (encoder->x264param.analyse.i_mv_range, + encoder->peer_level->mv_range); + + if (encoder->peer_level->frame_only) { + encoder->x264param.b_interlaced = FALSE; + encoder->x264param.b_fake_interlaced = FALSE; + } + } + encoder->reconfig = FALSE; GST_OBJECT_UNLOCK (encoder); @@ -1247,6 +1283,42 @@ gst_x264_enc_close_encoder (GstX264Enc * encoder) } } +static gboolean +gst_x264_enc_set_profile_and_level (GstX264Enc * encoder, GstCaps * caps) +{ + x264_nal_t *nal; + int i_nal; + int header_return; + gint sps_ni = 0; + guint8 *sps; + + + header_return = x264_encoder_headers (encoder->x264enc, &nal, &i_nal); + if (header_return < 0) { + GST_ELEMENT_ERROR (encoder, STREAM, ENCODE, ("Encode x264 header failed."), + ("x264_encoder_headers return code=%d", header_return)); + return FALSE; + } + + /* old x264 returns SEI, SPS and PPS, newer one has SEI last */ + if (i_nal == 3 && nal[sps_ni].i_type != 7) + sps_ni = 1; + + /* old style API: nal's are not encapsulated, and have no sync/size prefix, + * new style API: nal's are encapsulated, and have 4-byte size prefix */ +#ifndef X264_ENC_NALS + sps = nal[sps_ni].p_payload; +#else + sps = nal[sps_ni].p_payload + 4; + /* skip NAL unit type */ + sps++; +#endif + + gst_codec_utils_h264_caps_set_level_and_profile (caps, sps, 3); + + return TRUE; +} + /* * Returns: Buffer with the stream headers. */ @@ -1386,6 +1458,11 @@ gst_x264_enc_set_src_caps (GstX264Enc * encoder, GstPad * pad, GstCaps * caps) } gst_structure_set (structure, "alignment", G_TYPE_STRING, "au", NULL); + if (!gst_x264_enc_set_profile_and_level (encoder, outcaps)) { + gst_caps_unref (outcaps); + return FALSE; + } + res = gst_pad_set_caps (pad, outcaps); gst_caps_unref (outcaps); @@ -1401,6 +1478,10 @@ gst_x264_enc_sink_set_caps (GstPad * pad, GstCaps * caps) gint fps_num, fps_den; gint par_num, par_den; gint i; + GstCaps *peer_caps; + const GstCaps *template_caps; + GstCaps *allowed_caps = NULL; + gboolean level_ok = TRUE; /* get info from caps */ if (!gst_video_format_parse_caps (caps, &format, &width, &height)) @@ -1447,6 +1528,115 @@ gst_x264_enc_sink_set_caps (GstPad * pad, GstCaps * caps) i, width); } + encoder->peer_profile = NULL; + encoder->peer_intra_profile = FALSE; + encoder->peer_level = NULL; + + /* FIXME: Remove THIS bit in 0.11 when the profile property is removed */ + peer_caps = gst_pad_peer_get_caps_reffed (encoder->srcpad); + if (peer_caps) { + gint i; + gboolean has_profile_or_level = FALSE; + + for (i = 0; i < gst_caps_get_size (peer_caps); i++) { + GstStructure *s = gst_caps_get_structure (peer_caps, i); + + if (gst_structure_has_name (s, "video/x-h264") && + (gst_structure_has_field (s, "profile") || + gst_structure_has_field (s, "level"))) { + has_profile_or_level = TRUE; + break; + } + } + + if (has_profile_or_level) { + template_caps = gst_pad_get_pad_template_caps (encoder->srcpad); + + allowed_caps = gst_caps_intersect (peer_caps, template_caps); + } + + gst_caps_unref (peer_caps); + } + + /* Replace the bit since FIXME with this + * allowed_caps = gst_pad_get_allowed_caps (encoder->srcpad); + */ + + if (allowed_caps) { + GstStructure *s; + const gchar *profile; + const gchar *level; + + if (gst_caps_is_empty (allowed_caps)) { + gst_caps_unref (allowed_caps); + return FALSE; + } + + allowed_caps = gst_caps_make_writable (allowed_caps); + gst_pad_fixate_caps (encoder->srcpad, allowed_caps); + s = gst_caps_get_structure (allowed_caps, 0); + + profile = gst_structure_get_string (s, "profile"); + if (profile) { + if (!strcmp (profile, "constrained-baseline")) { + encoder->peer_profile = "baseline"; + } else if (!strcmp (profile, "high-10-intra")) { + encoder->peer_intra_profile = TRUE; + encoder->peer_profile = "high10"; + } else if (!strcmp (profile, "high-10")) { + encoder->peer_profile = "high10"; + } else if (!strcmp (profile, "high")) { + encoder->peer_profile = "high"; + } else if (!strcmp (profile, "main")) { + encoder->peer_profile = "main"; + } else { + g_assert_not_reached (); + } + } + + level = gst_structure_get_string (s, "level"); + if (level) { + int level_idc = gst_codec_utils_h264_get_level_idc (level); + + if (level_idc) { + gint i; + + for (i = 0; x264_levels[i].level_idc; i++) { + if (level_idc == x264_levels[i].level_idc) { + int mb_width = (width + 15) / 16; + int mb_height = (height + 15) / 16; + int mbs = mb_width * mb_height; + + if (x264_levels[i].frame_size < mbs || + x264_levels[i].frame_size * 8 < mb_width * mb_width || + x264_levels[i].frame_size * 8 < mb_height * mb_height) { + GST_WARNING_OBJECT (encoder, + "Frame size larger than level %s allows", level); + level_ok = FALSE; + break; + } + + if (fps_den && + x264_levels[i].mbps < (gint64) mbs * fps_num / fps_den) { + GST_WARNING_OBJECT (encoder, + "Macroblock rate higher than level %s allows", level); + level_ok = FALSE; + break; + } + + encoder->peer_level = &x264_levels[i]; + break; + } + } + } + } + + gst_caps_unref (allowed_caps); + } + + if (!level_ok) + return FALSE; + if (!gst_x264_enc_init_encoder (encoder)) return FALSE; diff --git a/ext/x264/gstx264enc.h b/ext/x264/gstx264enc.h index 01cd27de1f..e779a5e893 100644 --- a/ext/x264/gstx264enc.h +++ b/ext/x264/gstx264enc.h @@ -115,6 +115,11 @@ struct _GstX264Enc /* configuration changed while playing */ gboolean reconfig; + + /* from the downstream caps */ + const gchar *peer_profile; + gboolean peer_intra_profile; + const x264_level_t *peer_level; }; struct _GstX264EncClass From e595cdc3112680b3b66c6676f12ee7bc60bd11c7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Olivier=20Cr=C3=AAte?= Date: Thu, 9 Jun 2011 20:20:27 -0400 Subject: [PATCH 06/11] tests: Test x264enc profiles from the caps https://bugzilla.gnome.org/show_bug.cgi?id=644233 --- tests/check/elements/x264enc.c | 103 +++++++++++++++++++++++++++++++++ 1 file changed, 103 insertions(+) diff --git a/tests/check/elements/x264enc.c b/tests/check/elements/x264enc.c index ecfa4d4105..777a4725bb 100644 --- a/tests/check/elements/x264enc.c +++ b/tests/check/elements/x264enc.c @@ -211,6 +211,108 @@ GST_START_TEST (test_video_pad) GST_END_TEST; +GstCaps *pad_caps; + +GstCaps * +getcaps_test (GstPad * pad) +{ + return gst_caps_ref (pad_caps); +} + +GST_START_TEST (test_profile_in_caps) +{ + GstElement *x264enc; + GstPad *srcpad; + GstPad *sinkpad; + GstStructure *s; + + pad_caps = gst_caps_from_string (MPEG_CAPS_STRING); + + x264enc = setup_x264enc (); + gst_pad_set_getcaps_function (mysinkpad, getcaps_test); + srcpad = gst_element_get_static_pad (x264enc, "src"); + sinkpad = gst_element_get_static_pad (x264enc, "sink"); + fail_unless (gst_element_set_state (x264enc, + GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS, + "could not set to playing"); + + fail_unless (gst_pad_set_caps (sinkpad, + (GstCaps *) gst_pad_get_pad_template_caps (mysrcpad))); + s = gst_caps_get_structure (GST_PAD_CAPS (srcpad), 0); + fail_unless (!g_strcmp0 (gst_structure_get_string (s, "profile"), "main")); + + fail_unless (gst_element_set_state (x264enc, + GST_STATE_READY) == GST_STATE_CHANGE_SUCCESS, + "could not set to ready"); + g_object_set (x264enc, "profile", 1, NULL); + fail_unless (gst_element_set_state (x264enc, + GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS, + "could not set to playing"); + fail_unless (gst_pad_set_caps (sinkpad, + (GstCaps *) gst_pad_get_pad_template_caps (mysrcpad))); + s = gst_caps_get_structure (GST_PAD_CAPS (srcpad), 0); + fail_unless (!g_strcmp0 (gst_structure_get_string (s, "profile"), + "constrained-baseline")); + + fail_unless (gst_element_set_state (x264enc, + GST_STATE_READY) == GST_STATE_CHANGE_SUCCESS, + "could not set to ready"); + g_object_set (x264enc, "profile", 3, NULL); + fail_unless (gst_element_set_state (x264enc, + GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS, + "could not set to playing"); + fail_unless (gst_pad_set_caps (sinkpad, + (GstCaps *) gst_pad_get_pad_template_caps (mysrcpad))); + s = gst_caps_get_structure (GST_PAD_CAPS (srcpad), 0); + fail_unless (!g_strcmp0 (gst_structure_get_string (s, "profile"), "high")); + + fail_unless (gst_element_set_state (x264enc, + GST_STATE_READY) == GST_STATE_CHANGE_SUCCESS, + "could not set to ready"); + g_object_set (x264enc, "profile", 2, NULL); + fail_unless (gst_element_set_state (x264enc, + GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS, + "could not set to playing"); + fail_unless (gst_pad_set_caps (sinkpad, + (GstCaps *) gst_pad_get_pad_template_caps (mysrcpad))); + s = gst_caps_get_structure (GST_PAD_CAPS (srcpad), 0); + fail_unless (!g_strcmp0 (gst_structure_get_string (s, "profile"), "main")); + + s = gst_caps_get_structure (pad_caps, 0); + gst_structure_set (s, "profile", G_TYPE_STRING, "constrained-baseline", NULL); + fail_unless (gst_element_set_state (x264enc, + GST_STATE_READY) == GST_STATE_CHANGE_SUCCESS, + "could not set to ready"); + fail_unless (gst_element_set_state (x264enc, + GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS, + "could not set to playing"); + fail_unless (gst_pad_set_caps (sinkpad, + (GstCaps *) gst_pad_get_pad_template_caps (mysrcpad))); + s = gst_caps_get_structure (GST_PAD_CAPS (srcpad), 0); + fail_unless (!g_strcmp0 (gst_structure_get_string (s, "profile"), + "constrained-baseline")); + + s = gst_caps_get_structure (pad_caps, 0); + gst_structure_set (s, "profile", G_TYPE_STRING, "high", NULL); + fail_unless (gst_element_set_state (x264enc, + GST_STATE_READY) == GST_STATE_CHANGE_SUCCESS, + "could not set to ready"); + fail_unless (gst_element_set_state (x264enc, + GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS, + "could not set to playing"); + fail_unless (gst_pad_set_caps (sinkpad, + (GstCaps *) gst_pad_get_pad_template_caps (mysrcpad))); + s = gst_caps_get_structure (GST_PAD_CAPS (srcpad), 0); + fail_unless (!g_strcmp0 (gst_structure_get_string (s, "profile"), "high")); + + gst_object_unref (srcpad); + gst_object_unref (sinkpad); + cleanup_x264enc (x264enc); + gst_caps_unref (pad_caps); +} + +GST_END_TEST; + Suite * x264enc_suite (void) { @@ -219,6 +321,7 @@ x264enc_suite (void) suite_add_tcase (s, tc_chain); tcase_add_test (tc_chain, test_video_pad); + tcase_add_test (tc_chain, test_profile_in_caps); return s; } From ac47d20faeba32339e8c7b7d45facb287f54abd2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Olivier=20Cr=C3=AAte?= Date: Mon, 4 Jul 2011 18:03:49 -0400 Subject: [PATCH 07/11] x264: Allow renegotiation but prefer current caps --- ext/x264/gstx264enc.c | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/ext/x264/gstx264enc.c b/ext/x264/gstx264enc.c index 15dd190f19..f3377bda98 100644 --- a/ext/x264/gstx264enc.c +++ b/ext/x264/gstx264enc.c @@ -1655,10 +1655,6 @@ gst_x264_enc_sink_get_caps (GstPad * pad) GstPad *peer; GstCaps *caps; - /* If we already have caps return them */ - if (GST_PAD_CAPS (pad)) - return gst_caps_ref (GST_PAD_CAPS (pad)); - encoder = GST_X264_ENC (gst_pad_get_parent (pad)); if (!encoder) return gst_caps_new_empty (); @@ -1692,6 +1688,14 @@ gst_x264_enc_sink_get_caps (GstPad * pad) caps = gst_caps_copy (gst_pad_get_pad_template_caps (pad)); } + /* If we already have caps return them */ + if (GST_PAD_CAPS (pad) && gst_caps_can_intersect (GST_PAD_CAPS (pad), caps)) { + GstCaps *tmpcaps = gst_caps_copy (GST_PAD_CAPS (pad)); + + gst_caps_merge (tmpcaps, caps); + caps = tmpcaps; + } + gst_object_unref (encoder); return caps; From 7aafba6f825d7704dddfb58db180dd02e5bea479 Mon Sep 17 00:00:00 2001 From: Thiago Santos Date: Mon, 13 Jun 2011 23:24:27 -0300 Subject: [PATCH 08/11] x264enc: Select stream-format based on caps Makes x264 select its stream-format based on what's available on caps, the user selected option will be chosen as a fallback when both options are available. https://bugzilla.gnome.org/show_bug.cgi?id=644233 --- ext/x264/gstx264enc.c | 34 ++++++++++++++++++++++++++++++++-- ext/x264/gstx264enc.h | 1 + 2 files changed, 33 insertions(+), 2 deletions(-) diff --git a/ext/x264/gstx264enc.c b/ext/x264/gstx264enc.c index f3377bda98..f8cde40f8a 100644 --- a/ext/x264/gstx264enc.c +++ b/ext/x264/gstx264enc.c @@ -218,6 +218,13 @@ static GString *x264enc_defaults; #define ARG_PSY_TUNE_DEFAULT 0 /* no psy tuning */ #define ARG_TUNE_DEFAULT 0 /* no tuning */ +enum +{ + GST_X264_ENC_STREAM_FORMAT_FROM_PROPERTY, + GST_X264_ENC_STREAM_FORMAT_AVC, + GST_X264_ENC_STREAM_FORMAT_BYTE_STREAM +}; + enum { GST_X264_ENC_PASS_CBR = 0, @@ -940,6 +947,7 @@ gst_x264_enc_reset (GstX264Enc * encoder) encoder->x264enc = NULL; encoder->width = 0; encoder->height = 0; + encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_FROM_PROPERTY; GST_OBJECT_LOCK (encoder); encoder->i_type = X264_TYPE_AUTO; @@ -1445,7 +1453,14 @@ gst_x264_enc_set_src_caps (GstX264Enc * encoder, GstPad * pad, GstCaps * caps) structure = gst_caps_get_structure (outcaps, 0); - if (!encoder->byte_stream) { + if (encoder->current_byte_stream == GST_X264_ENC_STREAM_FORMAT_FROM_PROPERTY) { + if (encoder->byte_stream) { + encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_BYTE_STREAM; + } else { + encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_AVC; + } + } + if (encoder->current_byte_stream == GST_X264_ENC_STREAM_FORMAT_AVC) { buf = gst_x264_enc_header_buf (encoder); if (buf != NULL) { gst_caps_set_simple (outcaps, "codec_data", GST_TYPE_BUFFER, buf, NULL); @@ -1566,6 +1581,7 @@ gst_x264_enc_sink_set_caps (GstPad * pad, GstCaps * caps) GstStructure *s; const gchar *profile; const gchar *level; + const gchar *bytestream; if (gst_caps_is_empty (allowed_caps)) { gst_caps_unref (allowed_caps); @@ -1631,6 +1647,18 @@ gst_x264_enc_sink_set_caps (GstPad * pad, GstCaps * caps) } } + bytestream = gst_structure_get_string (s, "stream-format"); + encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_FROM_PROPERTY; + if (bytestream) { + if (!strcmp (bytestream, "avc")) { + encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_AVC; + } else if (!strcmp (profile, "byte-stream")) { + encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_BYTE_STREAM; + } else { + /* means we have both in caps and _FROM_PROPERTY should be the option */ + } + } + gst_caps_unref (allowed_caps); } @@ -1904,7 +1932,9 @@ gst_x264_enc_encode_frame (GstX264Enc * encoder, x264_picture_t * pic_in, nal_size = x264_nal_encode (encoder->buffer + i_size + 4, &i_data, 0, &nal[i]); - if (encoder->byte_stream) + g_assert (encoder->current_byte_stream != + GST_X264_ENC_STREAM_FORMAT_FROM_PROPERTY); + if (encoder->current_byte_stream == GST_X264_ENC_STREAM_FORMAT_BYTE_STREAM) GST_WRITE_UINT32_BE (encoder->buffer + i_size, 1); else GST_WRITE_UINT32_BE (encoder->buffer + i_size, nal_size); diff --git a/ext/x264/gstx264enc.h b/ext/x264/gstx264enc.h index e779a5e893..a2f1145b97 100644 --- a/ext/x264/gstx264enc.h +++ b/ext/x264/gstx264enc.h @@ -52,6 +52,7 @@ struct _GstX264Enc x264_t *x264enc; x264_param_t x264param; + gint current_byte_stream; /* properties */ guint threads; From 3cb99f46b7b76bc68b77638a00d154f1a2ee9adf Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Olivier=20Cr=C3=AAte?= Date: Mon, 18 Jul 2011 20:41:20 -0400 Subject: [PATCH 09/11] x264enc: Read stream-format from the right place Read the stream-format from "stream-format" and not from profile, also rename the "bytestream" variable to "stream_format" so it's easier to understand. --- ext/x264/gstx264enc.c | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/ext/x264/gstx264enc.c b/ext/x264/gstx264enc.c index f8cde40f8a..30d0b906dc 100644 --- a/ext/x264/gstx264enc.c +++ b/ext/x264/gstx264enc.c @@ -1581,7 +1581,7 @@ gst_x264_enc_sink_set_caps (GstPad * pad, GstCaps * caps) GstStructure *s; const gchar *profile; const gchar *level; - const gchar *bytestream; + const gchar *stream_format; if (gst_caps_is_empty (allowed_caps)) { gst_caps_unref (allowed_caps); @@ -1647,12 +1647,12 @@ gst_x264_enc_sink_set_caps (GstPad * pad, GstCaps * caps) } } - bytestream = gst_structure_get_string (s, "stream-format"); + stream_format = gst_structure_get_string (s, "stream-format"); encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_FROM_PROPERTY; - if (bytestream) { - if (!strcmp (bytestream, "avc")) { + if (stream_format) { + if (!strcmp (stream_format, "avc")) { encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_AVC; - } else if (!strcmp (profile, "byte-stream")) { + } else if (!strcmp (stream_format, "byte-stream")) { encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_BYTE_STREAM; } else { /* means we have both in caps and _FROM_PROPERTY should be the option */ From 3f8a245796acfa7b26d22e2e93a19fc2ad2b83de Mon Sep 17 00:00:00 2001 From: Stefan Sauer Date: Tue, 19 Jul 2011 15:06:49 +0200 Subject: [PATCH 10/11] x264enc: fix the build for older x264enc b_fake_interlaced was introduced in x264 commit 1b48874d06 = X264_BUILD 96. --- ext/x264/gstx264enc.c | 2 ++ 1 file changed, 2 insertions(+) diff --git a/ext/x264/gstx264enc.c b/ext/x264/gstx264enc.c index 30d0b906dc..08c930b43b 100644 --- a/ext/x264/gstx264enc.c +++ b/ext/x264/gstx264enc.c @@ -1255,7 +1255,9 @@ gst_x264_enc_init_encoder (GstX264Enc * encoder) if (encoder->peer_level->frame_only) { encoder->x264param.b_interlaced = FALSE; +#if X264_BUILD >= 95 encoder->x264param.b_fake_interlaced = FALSE; +#endif } } From 19f1c38d4ea16c8c199a1b4cd8d21beef3a70b6c Mon Sep 17 00:00:00 2001 From: Thiago Santos Date: Sat, 30 Jul 2011 18:47:44 -0300 Subject: [PATCH 11/11] x264enc: Properly set the stream format in setcaps Do not forget to set the selected format to the options string when getting it from caps https://bugzilla.gnome.org/show_bug.cgi?id=655223 --- ext/x264/gstx264enc.c | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/ext/x264/gstx264enc.c b/ext/x264/gstx264enc.c index 08c930b43b..3aa02feadc 100644 --- a/ext/x264/gstx264enc.c +++ b/ext/x264/gstx264enc.c @@ -1553,20 +1553,21 @@ gst_x264_enc_sink_set_caps (GstPad * pad, GstCaps * caps) peer_caps = gst_pad_peer_get_caps_reffed (encoder->srcpad); if (peer_caps) { gint i; - gboolean has_profile_or_level = FALSE; + gboolean has_profile_or_level_or_format = FALSE; for (i = 0; i < gst_caps_get_size (peer_caps); i++) { GstStructure *s = gst_caps_get_structure (peer_caps, i); if (gst_structure_has_name (s, "video/x-h264") && (gst_structure_has_field (s, "profile") || - gst_structure_has_field (s, "level"))) { - has_profile_or_level = TRUE; + gst_structure_has_field (s, "level") || + gst_structure_has_field (s, "stream-format"))) { + has_profile_or_level_or_format = TRUE; break; } } - if (has_profile_or_level) { + if (has_profile_or_level_or_format) { template_caps = gst_pad_get_pad_template_caps (encoder->srcpad); allowed_caps = gst_caps_intersect (peer_caps, template_caps); @@ -1654,8 +1655,10 @@ gst_x264_enc_sink_set_caps (GstPad * pad, GstCaps * caps) if (stream_format) { if (!strcmp (stream_format, "avc")) { encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_AVC; + g_string_append_printf (encoder->option_string, ":annexb=0"); } else if (!strcmp (stream_format, "byte-stream")) { encoder->current_byte_stream = GST_X264_ENC_STREAM_FORMAT_BYTE_STREAM; + g_string_append_printf (encoder->option_string, ":annexb=1"); } else { /* means we have both in caps and _FROM_PROPERTY should be the option */ }