summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--ChangeLog13
-rw-r--r--ext/faad/gstfaad.c250
-rw-r--r--ext/faad/gstfaad.h4
3 files changed, 174 insertions, 93 deletions
diff --git a/ChangeLog b/ChangeLog
index 14464a17..58ce5807 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,16 @@
+2006-04-28 Tim-Philipp Müller <tim at centricular dot net>
+
+ * ext/faad/gstfaad.c: (gst_faad_init), (aac_rate_idx),
+ (gst_faad_setcaps), (gst_faad_chain), (gst_faad_open_decoder),
+ (gst_faad_close_decoder), (gst_faad_change_state):
+ * ext/faad/gstfaad.h:
+ If we run into a decoding error, try re-opening the decoder
+ with faacDecInit2() using fake codec data created from the
+ data the demuxer gave us. Should fix a whole bunch of
+ GStreamer-faad problems incl. 'channel coupling not
+ implemented', 'maximum number of scalefactor bands exceeded'
+ etc. (#173007, #332892).
+
2006-04-26 Stefan Kost <ensonic@users.sf.net>
* ext/amrwb/gstamrwbdec.c:
diff --git a/ext/faad/gstfaad.c b/ext/faad/gstfaad.c
index ec20f633..85009de8 100644
--- a/ext/faad/gstfaad.c
+++ b/ext/faad/gstfaad.c
@@ -1,5 +1,6 @@
/* GStreamer FAAD (Free AAC Decoder) plugin
* Copyright (C) 2003 Ronald Bultje <rbultje@ronald.bitfreak.net>
+ * Copyright (C) 2006 Tim-Philipp Müller <tim centricular net>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
@@ -121,6 +122,8 @@ static GstStateChangeReturn gst_faad_change_state (GstElement * element,
GstStateChange transition);
static gboolean gst_faad_src_convert (GstFaad * faad, GstFormat src_format,
gint64 src_val, GstFormat dest_format, gint64 * dest_val);
+static gboolean gst_faad_open_decoder (GstFaad * faad);
+static void gst_faad_close_decoder (GstFaad * faad);
static GstElementClass *parent_class; /* NULL */
@@ -190,9 +193,7 @@ gst_faad_init (GstFaad * faad)
faad->sum_dur_out = 0;
faad->packetised = FALSE;
- faad->sinkpad =
- gst_pad_new_from_template (gst_static_pad_template_get (&sink_template),
- "sink");
+ faad->sinkpad = gst_pad_new_from_static_template (&sink_template, "sink");
gst_element_add_pad (GST_ELEMENT (faad), faad->sinkpad);
gst_pad_set_event_function (faad->sinkpad,
GST_DEBUG_FUNCPTR (gst_faad_sink_event));
@@ -201,9 +202,7 @@ gst_faad_init (GstFaad * faad)
gst_pad_set_chain_function (faad->sinkpad,
GST_DEBUG_FUNCPTR (gst_faad_chain));
- faad->srcpad =
- gst_pad_new_from_template (gst_static_pad_template_get (&src_template),
- "src");
+ faad->srcpad = gst_pad_new_from_static_template (&src_template, "src");
gst_pad_use_fixed_caps (faad->srcpad);
gst_pad_set_getcaps_function (faad->srcpad,
GST_DEBUG_FUNCPTR (gst_faad_srcgetcaps));
@@ -227,6 +226,35 @@ gst_faad_send_tags (GstFaad * faad)
gst_element_found_tags (GST_ELEMENT (faad), tags);
}
+static gint
+aac_rate_idx (gint rate)
+{
+ if (92017 <= rate)
+ return 0;
+ else if (75132 <= rate)
+ return 1;
+ else if (55426 <= rate)
+ return 2;
+ else if (46009 <= rate)
+ return 3;
+ else if (37566 <= rate)
+ return 4;
+ else if (27713 <= rate)
+ return 5;
+ else if (23004 <= rate)
+ return 6;
+ else if (18783 <= rate)
+ return 7;
+ else if (13856 <= rate)
+ return 8;
+ else if (11502 <= rate)
+ return 9;
+ else if (9391 <= rate)
+ return 10;
+ else
+ return 11;
+}
+
static gboolean
gst_faad_setcaps (GstPad * pad, GstCaps * caps)
{
@@ -239,8 +267,8 @@ gst_faad_setcaps (GstPad * pad, GstCaps * caps)
faad->packetised = FALSE;
if ((value = gst_structure_get_value (str, "codec_data"))) {
- guint samplerate;
- guchar channels;
+ guint32 samplerate;
+ guint8 channels;
/* We have codec data, means packetised stream */
faad->packetised = TRUE;
@@ -253,10 +281,9 @@ gst_faad_setcaps (GstPad * pad, GstCaps * caps)
GST_DEBUG ("faacDecInit2() failed");
return FALSE;
}
-#if 0
- faad->samplerate = samplerate;
- faad->channels = channels;
-#endif
+
+ GST_DEBUG_OBJECT (faad, "channels=%u, rate=%u", channels, samplerate);
+
/* not updating these here, so they are updated in the
* chain function, and new caps are created etc. */
faad->samplerate = 0;
@@ -275,6 +302,25 @@ gst_faad_setcaps (GstPad * pad, GstCaps * caps)
faad->init = FALSE;
}
+ faad->fake_codec_data[0] = 0;
+ faad->fake_codec_data[1] = 0;
+
+ if (faad->packetised) {
+ gint rate, channels;
+
+ if (gst_structure_get_int (str, "rate", &rate) &&
+ gst_structure_get_int (str, "channels", &channels)) {
+ gint rate_idx, profile;
+
+ profile = 3; /* 0=MAIN, 1=LC, 2=SSR, 3=? */
+ rate_idx = aac_rate_idx (rate);
+
+ faad->fake_codec_data[0] = ((profile + 1) << 3) | ((rate_idx & 0xE) >> 1);
+ faad->fake_codec_data[1] = ((rate_idx & 0x1) << 7) | (channels << 3);
+ GST_LOG_OBJECT (faad, "created fake codec data (%u,%u)", rate, channels);
+ }
+ }
+
faad->need_channel_setup = TRUE;
if (!faad->packetised)
@@ -407,51 +453,6 @@ gst_faad_chanpos_to_gst (guchar * fpos, guint num)
return pos;
}
-/*
-static GstPadLinkReturn
-gst_faad_sinkconnect (GstPad * pad, const GstCaps * caps)
-{
- GstFaad *faad = GST_FAAD (gst_pad_get_parent (pad));
- GstStructure *str = gst_caps_get_structure (caps, 0);
- const GValue *value;
- GstBuffer *buf;
-
- // Assume raw stream
- faad->packetised = FALSE;
-
- if ((value = gst_structure_get_value (str, "codec_data"))) {
- gulong samplerate;
- guchar channels;
-
- // We have codec data, means packetised stream
- faad->packetised = TRUE;
- buf = g_value_get_boxed (value);
-
- // someone forgot that char can be unsigned when writing the API
- if ((gint8) faacDecInit2 (faad->handle, GST_BUFFER_DATA (buf),
- GST_BUFFER_SIZE (buf), &samplerate, &channels) < 0)
- return GST_PAD_LINK_REFUSED;
-
- //faad->samplerate = samplerate;
- //faad->channels = channels;
- faad->init = TRUE;
-
- if (faad->tempbuf) {
- gst_buffer_unref (faad->tempbuf);
- faad->tempbuf = NULL;
- }
- } else {
- faad->init = FALSE;
- }
-
- faad->need_channel_setup = TRUE;
-
- // if there's no decoderspecificdata, it's all fine. We cannot know
- // * much more at this point...
- return GST_PAD_LINK_OK;
-}
-*/
-
static GstCaps *
gst_faad_srcgetcaps (GstPad * pad)
{
@@ -1079,7 +1080,7 @@ gst_faad_chain (GstPad * pad, GstBuffer * buffer)
faad->next_ts = GST_BUFFER_TIMESTAMP (buffer);
faad->prev_ts = GST_BUFFER_TIMESTAMP (buffer);
}
- GST_DEBUG ("Timestamp on incoming buffer: %" GST_TIME_FORMAT
+ GST_LOG_OBJECT (faad, "Timestamp on incoming buffer: %" GST_TIME_FORMAT
", next_ts: %" GST_TIME_FORMAT,
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buffer)),
GST_TIME_ARGS (faad->next_ts));
@@ -1103,19 +1104,16 @@ gst_faad_chain (GstPad * pad, GstBuffer * buffer)
/* init if not already done during capsnego */
if (!faad->init) {
- guint32 samplerate;
- guchar channels;
- glong init_res;
-
- init_res = faacDecInit (faad->handle, input_data, input_size,
- &samplerate, &channels);
- if (init_res < 0) {
- GST_ELEMENT_ERROR (faad, STREAM, DECODE, (NULL),
- ("Failed to init decoder from stream"));
- gst_object_unref (faad);
- return GST_FLOW_UNEXPECTED;
- }
- skip_bytes = 0; /* init_res; */
+ guint32 rate;
+ guint8 ch;
+
+ GST_DEBUG_OBJECT (faad, "initialising ...");
+ if (faacDecInit (faad->handle, input_data, input_size, &rate, &ch) < 0)
+ goto init_failed;
+
+ GST_DEBUG_OBJECT (faad, "faacDecInit() ok: rate=%u,channels=%u", rate, ch);
+
+ skip_bytes = 0;
faad->init = TRUE;
/* make sure we create new caps below */
@@ -1146,11 +1144,35 @@ gst_faad_chain (GstPad * pad, GstBuffer * buffer)
out = faacDecDecode (faad->handle, &info, input_data + skip_bytes,
input_size - skip_bytes);
+
if (info.error) {
- GST_ELEMENT_ERROR (faad, STREAM, DECODE, (NULL),
- ("Failed to decode buffer: %s", faacDecGetErrorMessage (info.error)));
- ret = GST_FLOW_ERROR;
- goto out;
+ guint32 rate;
+ guint8 ch;
+
+ if (!faad->packetised)
+ goto decode_error;
+
+ /* decode error? try again using faacDecInit2
+ * fabricated private codec data from sink caps */
+ gst_faad_close_decoder (faad);
+ if (!gst_faad_open_decoder (faad))
+ goto init2_failed;
+
+ GST_DEBUG_OBJECT (faad, "decoding error, reopening with faacDecInit2()");
+ if ((gint8) faacDecInit2 (faad->handle, faad->fake_codec_data, 2,
+ &rate, &ch) < 0) {
+ goto init2_failed;
+ }
+
+ GST_DEBUG_OBJECT (faad, "faacDecInit2(): rate=%d,channels=%d", rate, ch);
+
+ /* let's try again */
+ info.error = 0;
+ out = faacDecDecode (faad->handle, &info, input_data + skip_bytes,
+ input_size - skip_bytes);
+
+ if (info.error)
+ goto decode_error;
}
if (info.bytesconsumed > input_size)
@@ -1184,7 +1206,7 @@ gst_faad_chain (GstPad * pad, GstBuffer * buffer)
}
/* play decoded data */
- if (info.samples > 0 && GST_PAD_PEER (faad->srcpad)) {
+ if (info.samples > 0) {
guint bufsize = info.samples * faad->bps;
guint num_samples = info.samples / faad->channels;
@@ -1207,7 +1229,7 @@ gst_faad_chain (GstPad * pad, GstBuffer * buffer)
faad->sum_dur_out += GST_BUFFER_DURATION (outbuf);
GST_OBJECT_UNLOCK (faad);
- GST_DEBUG ("pushing buffer, off=%" G_GUINT64_FORMAT ", ts=%"
+ GST_LOG_OBJECT (faad, "pushing buffer, off=%" G_GUINT64_FORMAT ", ts=%"
GST_TIME_FORMAT, GST_BUFFER_OFFSET (outbuf),
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (outbuf)));
if ((ret = gst_pad_push (faad->srcpad, outbuf)) != GST_FLOW_OK &&
@@ -1239,6 +1261,63 @@ out:
gst_object_unref (faad);
return ret;
+
+/* ERRORS */
+init_failed:
+ {
+ GST_ELEMENT_ERROR (faad, STREAM, DECODE, (NULL),
+ ("Failed to init decoder from stream"));
+ ret = GST_FLOW_ERROR;
+ goto out;
+ }
+
+init2_failed:
+ {
+ GST_ELEMENT_ERROR (faad, STREAM, DECODE, (NULL),
+ ("%s() failed", (faad->handle) ? "faacDecInit2" : "faacDecOpen"));
+ ret = GST_FLOW_ERROR;
+ goto out;
+ }
+
+decode_error:
+ {
+ GST_ELEMENT_ERROR (faad, STREAM, DECODE, (NULL),
+ ("Failed to decode buffer: %s", faacDecGetErrorMessage (info.error)));
+ ret = GST_FLOW_ERROR;
+ goto out;
+ }
+}
+
+static gboolean
+gst_faad_open_decoder (GstFaad * faad)
+{
+ faacDecConfiguration *conf;
+
+ faad->handle = faacDecOpen ();
+
+ if (faad->handle == NULL) {
+ GST_WARNING_OBJECT (faad, "faacDecOpen() failed");
+ return FALSE;
+ }
+
+ conf = faacDecGetCurrentConfiguration (faad->handle);
+ conf->defObjectType = LC;
+ /* conf->dontUpSampleImplicitSBR = 1; */
+ conf->outputFormat = FAAD_FMT_16BIT;
+
+ if (faacDecSetConfiguration (faad->handle, conf) == 0) {
+ GST_WARNING_OBJECT (faad, "faacDecSetConfiguration() failed");
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+static void
+gst_faad_close_decoder (GstFaad * faad)
+{
+ faacDecClose (faad->handle);
+ faad->handle = NULL;
}
static GstStateChangeReturn
@@ -1249,21 +1328,9 @@ gst_faad_change_state (GstElement * element, GstStateChange transition)
switch (transition) {
case GST_STATE_CHANGE_NULL_TO_READY:
- {
- if (!(faad->handle = faacDecOpen ()))
+ if (!gst_faad_open_decoder (faad))
return GST_STATE_CHANGE_FAILURE;
- else {
- faacDecConfiguration *conf;
-
- conf = faacDecGetCurrentConfiguration (faad->handle);
- conf->defObjectType = LC;
- /* conf->dontUpSampleImplicitSBR = 1; */
- conf->outputFormat = FAAD_FMT_16BIT;
- if (faacDecSetConfiguration (faad->handle, conf) == 0)
- return GST_STATE_CHANGE_FAILURE;
- }
break;
- }
default:
break;
}
@@ -1285,8 +1352,7 @@ gst_faad_change_state (GstElement * element, GstStateChange transition)
faad->sum_dur_out = 0;
break;
case GST_STATE_CHANGE_READY_TO_NULL:
- faacDecClose (faad->handle);
- faad->handle = NULL;
+ gst_faad_close_decoder (faad);
if (faad->tempbuf) {
gst_buffer_unref (faad->tempbuf);
faad->tempbuf = NULL;
diff --git a/ext/faad/gstfaad.h b/ext/faad/gstfaad.h
index 419ce20d..160b8ba4 100644
--- a/ext/faad/gstfaad.h
+++ b/ext/faad/gstfaad.h
@@ -33,7 +33,7 @@ G_BEGIN_DECLS
(G_TYPE_CHECK_CLASS_CAST ((klass), GST_TYPE_FAAD, GstFaadClass))
#define GST_IS_FAAD(obj) \
(G_TYPE_CHECK_INSTANCE_TYPE ((obj), GST_TYPE_FAAD))
-#define GST_IS_FAAD_CLASS(obj) \
+#define GST_IS_FAAD_CLASS(klass) \
(G_TYPE_CHECK_CLASS_TYPE ((klass), GST_TYPE_FAAD))
typedef struct _GstFaad {
@@ -46,6 +46,8 @@ typedef struct _GstFaad {
guint channels; /* number of channels of the last frame */
guint bps; /* bytes per sample */
+ guint8 fake_codec_data[2];
+
GstBuffer *tempbuf; /* used to keep input leftovers */
/* FAAD object */