summaryrefslogtreecommitdiffstats
path: root/gst/real/gstrealvideodec.c
diff options
context:
space:
mode:
Diffstat (limited to 'gst/real/gstrealvideodec.c')
-rw-r--r--gst/real/gstrealvideodec.c422
1 files changed, 71 insertions, 351 deletions
diff --git a/gst/real/gstrealvideodec.c b/gst/real/gstrealvideodec.c
index 2678e699..076918fb 100644
--- a/gst/real/gstrealvideodec.c
+++ b/gst/real/gstrealvideodec.c
@@ -87,345 +87,110 @@ typedef struct
} RVOutData;
static GstFlowReturn
-gst_real_video_dec_alloc_buffer (GstRealVideoDec * dec, GstClockTime timestamp,
- GstBuffer ** buf)
+gst_real_video_dec_chain (GstPad * pad, GstBuffer * in)
{
+ GstRealVideoDec *dec;
+ guint8 *data;
+ guint size;
GstFlowReturn ret;
- const guint8 *b;
- guint8 frame_type;
- guint16 seq;
- GstClockTime ts = timestamp;
+ RVInData tin;
+ RVOutData tout;
+ GstClockTime timestamp, duration;
+ GstBuffer *out;
+ guint32 result;
+ guint frag_count;
- GST_LOG_OBJECT (dec, "timestamp %" GST_TIME_FORMAT, GST_TIME_ARGS (ts));
+ dec = GST_REAL_VIDEO_DEC (GST_PAD_PARENT (pad));
- /* Fix timestamp. */
- b = gst_adapter_peek (dec->adapter, 4);
- switch (dec->version) {
- case GST_REAL_VIDEO_DEC_VERSION_2:
- {
+ if (G_UNLIKELY (dec->hooks.transform == NULL || dec->hooks.module == NULL))
+ goto not_negotiated;
- /*
- * Bit 1- 2: frame type
- * Bit 3- 9: ?
- * Bit 10-22: sequence number
- * Bit 23-32: ?
- */
- frame_type = (b[0] >> 6) & 0x03;
- seq = ((b[1] & 0x7f) << 6) + ((b[2] & 0xfc) >> 2);
- break;
- }
- case GST_REAL_VIDEO_DEC_VERSION_3:
- {
- /*
- * Bit 1- 2: ?
- * Bit 3: skip packet if 1
- * Bit 4- 5: frame type
- * Bit 6-12: ?
- * Bit 13-25: sequence number
- * Bit 26-32: ?
- */
- frame_type = (b[0] >> 3) & 0x03;
- seq = ((b[1] & 0x0f) << 9) + (b[2] << 1) + ((b[3] & 0x80) >> 7);
- break;
- }
- case GST_REAL_VIDEO_DEC_VERSION_4:
- {
- /*
- * Bit 1: skip packet if 1
- * Bit 2- 3: frame type
- * Bit 4-13: ?
- * Bit 14-26: sequence number
- * Bit 27-32: ?
- */
- frame_type = (b[0] >> 5) & 0x03;
- seq = ((b[1] & 0x07) << 10) + (b[2] << 2) + ((b[3] & 0xc0) >> 6);
- break;
- }
- default:
- goto unknown_version;
- }
+ data = GST_BUFFER_DATA (in);
+ size = GST_BUFFER_SIZE (in);
+ timestamp = GST_BUFFER_TIMESTAMP (in);
+ duration = GST_BUFFER_DURATION (in);
- GST_LOG_OBJECT (dec, "frame_type:%d", frame_type);
+ /* alloc output buffer */
+ ret = gst_pad_alloc_buffer (dec->src, GST_BUFFER_OFFSET_NONE,
+ dec->width * dec->height * 3 / 2, GST_PAD_CAPS (dec->src), &out);
+ if (ret != GST_FLOW_OK)
+ goto alloc_failed;
- switch (frame_type) {
- case 0:
- case 1:
- {
- /* I frame */
- timestamp = dec->next_ts;
- dec->last_ts = dec->next_ts;
- dec->next_ts = ts;
- dec->last_seq = dec->next_seq;
- dec->next_seq = seq;
+ GST_BUFFER_TIMESTAMP (out) = timestamp;
+ GST_BUFFER_DURATION (out) = duration;
- break;
- }
- case 2:
- {
- /* P frame */
- timestamp = dec->last_ts = dec->next_ts;
- if (seq < dec->next_seq)
- dec->next_ts += (seq + 0x2000 - dec->next_seq) * GST_MSECOND;
- else
- dec->next_ts += (seq - dec->next_seq) * GST_MSECOND;
- dec->last_seq = dec->next_seq;
- dec->next_seq = seq;
- break;
- }
- case 3:
- {
- /* B frame */
- if (seq < dec->last_seq) {
- timestamp = (seq + 0x2000 - dec->last_seq) * GST_MSECOND + dec->last_ts;
- } else {
- timestamp = (seq - dec->last_seq) * GST_MSECOND + dec->last_ts;
- }
+ frag_count = *data++;
- break;
- }
- default:
- goto unknown_frame_type;
- }
+ /* Decode */
+ tin.datalen = size;
+ tin.interpolate = 0;
+ tin.nfragments = frag_count;
+ tin.fragments = data;
+ tin.flags = 0;
+ tin.timestamp = timestamp;
- ret = gst_pad_alloc_buffer (dec->src, GST_BUFFER_OFFSET_NONE,
- dec->width * dec->height * 3 / 2, GST_PAD_CAPS (dec->src), buf);
+ /* jump over the frag table to the fragments */
+ data += (frag_count + 1) * 8;
- if (ret == GST_FLOW_OK)
- GST_BUFFER_TIMESTAMP (*buf) = timestamp;
+ result = dec->hooks.transform (
+ (gchar *) data,
+ (gchar *) GST_BUFFER_DATA (out), &tin, &tout, dec->hooks.context);
+ if (result)
+ goto could_not_transform;
- return ret;
+ gst_buffer_unref (in);
- /* Errors */
-unknown_version:
- {
- GST_ELEMENT_ERROR (dec, STREAM, DECODE,
- ("Unknown version: %i.", dec->version), (NULL));
- return GST_FLOW_ERROR;
- }
+ /* Check for new dimensions */
+ if (tout.frames && ((dec->width != tout.width)
+ || (dec->height != tout.height))) {
+ GstCaps *caps = gst_caps_copy (GST_PAD_CAPS (dec->src));
+ GstStructure *s = gst_caps_get_structure (caps, 0);
-unknown_frame_type:
- {
- GST_ELEMENT_ERROR (dec, STREAM, DECODE, ("Unknown frame type."), (NULL));
- return GST_FLOW_ERROR;
- }
-}
+ GST_DEBUG_OBJECT (dec, "New dimensions: %"
+ G_GUINT32_FORMAT " x %" G_GUINT32_FORMAT, tout.width, tout.height);
-static GstFlowReturn
-gst_real_video_dec_decode (GstRealVideoDec * dec, GstBuffer * in, guint offset)
-{
- guint32 result;
- GstBuffer *out = NULL;
- GstFlowReturn ret;
- guint8 *data, hdr_subseq, hdr_seqnum;
- guint32 hdr_offset, hdr_length;
- guint16 n;
- gboolean bres;
- guint8 *buf = GST_BUFFER_DATA (in) + offset;
- guint len = GST_BUFFER_SIZE (in) - offset;
- GstClockTime timestamp = GST_BUFFER_TIMESTAMP (in);
-
- GST_LOG_OBJECT (dec,
- "Got buffer %p with timestamp %" GST_TIME_FORMAT " offset %d", in,
- GST_TIME_ARGS (timestamp), offset);
-
- /* Subsequence */
- if (len < 1)
- goto not_enough_data;
- if (*buf != 0x40 && *buf != 0x41 && *buf != 0x42 &&
- *buf != 0x43 && *buf != 0x44 && *buf != 0x45) {
- hdr_subseq = *buf & 0x7f;
- buf++;
- len--;
- if (hdr_subseq == 64)
- hdr_subseq = 1;
- } else {
- hdr_subseq = 1;
- }
+ gst_structure_set (s, "width", G_TYPE_LONG, tout.width,
+ "height", G_TYPE_LONG, tout.height, NULL);
- /* Length */
- if (len < 2)
- goto not_enough_data;
- hdr_length = GST_READ_UINT16_BE (buf);
- if (!(hdr_length & 0xc000)) {
- if (len < 4)
- goto not_enough_data;
- hdr_length = GST_READ_UINT32_BE (buf);
- buf += 4;
- len -= 4;
- } else {
- hdr_length &= 0x3fff;
- buf += 2;
- len -= 2;
- }
+ gst_pad_set_caps (dec->src, caps);
+ gst_buffer_set_caps (out, caps);
+ gst_caps_unref (caps);
- /* Offset */
- if (len < 2)
- goto not_enough_data;
- hdr_offset = GST_READ_UINT16_BE (buf);
- if (!(hdr_offset & 0xc000)) {
- if (len < 4)
- goto not_enough_data;
- hdr_offset = GST_READ_UINT32_BE (buf);
- buf += 4;
- len -= 4;
- } else {
- hdr_offset &= 0x3fff;
- buf += 2;
- len -= 2;
+ dec->width = tout.width;
+ dec->height = tout.height;
}
- /* Sequence number */
- if (len < 1)
- goto not_enough_data;
- hdr_seqnum = *buf;
- buf++;
- len--;
- if (len < 1)
- goto not_enough_data;
-
- /* Verify the sequence numbers. */
- if (hdr_subseq == 1) {
- n = gst_adapter_available_fast (dec->adapter);
- if (n > 0) {
- GST_DEBUG_OBJECT (dec, "Dropping data for sequence %i "
- "because we are already receiving data for sequence %i.",
- dec->seqnum, hdr_seqnum);
- gst_adapter_clear (dec->adapter);
- }
- dec->seqnum = hdr_seqnum;
- dec->length = hdr_length;
- dec->fragment_count = 1;
- } else {
- if (dec->seqnum != hdr_seqnum) {
- GST_DEBUG_OBJECT (dec, "Expected sequence %i, got sequence %i "
- "(subseq=%i). Dropping packet.", dec->seqnum, hdr_seqnum, hdr_subseq);
- return GST_FLOW_OK;
- } else if (dec->subseq + 1 != hdr_subseq) {
- GST_DEBUG_OBJECT (dec, "Expected subsequence %i, got subseqence %i. "
- "Dropping packet.", dec->subseq + 1, hdr_subseq);
- return GST_FLOW_OK;
- }
- dec->fragment_count++;
- }
- dec->subseq = hdr_subseq;
-
- /* Remember the offset */
- if (sizeof (dec->fragments) < 2 * (dec->fragment_count - 1) + 1)
- goto too_many_fragments;
- dec->fragments[2 * (dec->fragment_count - 1)] = 1;
- dec->fragments[2 * (dec->fragment_count - 1) + 1] =
- gst_adapter_available (dec->adapter);
-
- /* Some buffers need to be skipped. */
- if (((dec->version == GST_REAL_VIDEO_DEC_VERSION_3) && (*buf & 0x20)) ||
- ((dec->version == GST_REAL_VIDEO_DEC_VERSION_4) && (*buf & 0x80))) {
- dec->fragment_count--;
- dec->length -= len;
- } else {
- gst_adapter_push (dec->adapter,
- gst_buffer_create_sub (in, GST_BUFFER_SIZE (in) - len, len));
- }
-
- /* All bytes received? */
- n = gst_adapter_available (dec->adapter);
- GST_LOG_OBJECT (dec, "We know have %d bytes, and we need %d", n, dec->length);
- if (dec->length <= n) {
- RVInData tin;
- RVOutData tout;
-
- if ((ret =
- gst_real_video_dec_alloc_buffer (dec, timestamp,
- &out)) != GST_FLOW_OK)
- return ret;
-
- /* Decode */
- tin.datalen = dec->length;
- tin.interpolate = 0;
- tin.nfragments = dec->fragment_count - 1;
- tin.fragments = dec->fragments;
- tin.flags = 0;
- tin.timestamp = GST_BUFFER_TIMESTAMP (out);
- data = gst_adapter_take (dec->adapter, dec->length);
-
- result = dec->hooks.transform (
- (gchar *) data,
- (gchar *) GST_BUFFER_DATA (out), &tin, &tout, dec->hooks.context);
-
- g_free (data);
- if (result)
- goto could_not_transform;
-
- /* Check for new dimensions */
- if (tout.frames && ((dec->width != tout.width)
- || (dec->height != tout.height))) {
- GstCaps *caps = gst_caps_copy (GST_PAD_CAPS (dec->src));
- GstStructure *s = gst_caps_get_structure (caps, 0);
-
- GST_DEBUG_OBJECT (dec, "New dimensions: %"
- G_GUINT32_FORMAT " x %" G_GUINT32_FORMAT, tout.width, tout.height);
- gst_structure_set (s, "width", G_TYPE_LONG, tout.width,
- "height", G_TYPE_LONG, tout.height, NULL);
- bres = gst_pad_set_caps (dec->src, caps);
- gst_caps_unref (caps);
- if (!bres)
- goto new_dimensions_failed;
- dec->width = tout.width;
- dec->height = tout.height;
- }
+ GST_DEBUG_OBJECT (dec,
+ "Pushing out buffer with timestamp %" GST_TIME_FORMAT,
+ GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (out)));
- GST_DEBUG_OBJECT (dec,
- "Pushing out buffer with timestamp %" GST_TIME_FORMAT,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (out)));
-
- if ((ret = gst_pad_push (dec->src, out)) != GST_FLOW_OK)
- goto could_not_push;
-
- n = gst_adapter_available (dec->adapter);
- if (n > 0) {
- GST_LOG_OBJECT (dec, "Data left in the adapter: %d", n);
- in = gst_adapter_take_buffer (dec->adapter, n);
- ret = gst_real_video_dec_decode (dec, in, 0);
- gst_buffer_unref (in);
- if (ret != GST_FLOW_OK)
- return ret;
- }
- }
+ if ((ret = gst_pad_push (dec->src, out)) != GST_FLOW_OK)
+ goto could_not_push;
- return GST_FLOW_OK;
+ return ret;
/* Errors */
-not_enough_data:
+not_negotiated:
{
- GST_ELEMENT_ERROR (dec, STREAM, DECODE, ("Not enough data."), (NULL));
- return GST_FLOW_ERROR;
+ GST_WARNING_OBJECT (dec, "decoder not open, probably no input caps set "
+ "yet, caps on input buffer: %" GST_PTR_FORMAT, GST_BUFFER_CAPS (in));
+ gst_buffer_unref (in);
+ return GST_FLOW_NOT_NEGOTIATED;
}
-
-too_many_fragments:
+alloc_failed:
{
+ GST_DEBUG_OBJECT (dec, "buffer alloc failed: %s", gst_flow_get_name (ret));
gst_buffer_unref (in);
- GST_ELEMENT_ERROR (dec, STREAM, DECODE,
- ("Got more fragments (%u) than can be handled (%u)",
- dec->fragment_count, (guint) G_N_ELEMENTS (dec->fragments)),
- (NULL));
- return GST_FLOW_ERROR;
+ return ret;
}
-
could_not_transform:
{
gst_buffer_unref (out);
+ gst_buffer_unref (in);
GST_ELEMENT_ERROR (dec, STREAM, DECODE,
("Could not decode buffer: %" G_GUINT32_FORMAT, result), (NULL));
return GST_FLOW_ERROR;
}
-
-new_dimensions_failed:
- {
- gst_buffer_unref (out);
- GST_ELEMENT_ERROR (dec, STREAM, DECODE,
- ("Could not set new dimensions."), (NULL));
- return GST_FLOW_ERROR;
- }
-
could_not_push:
{
GST_DEBUG_OBJECT (dec, "Could not push buffer: %s",
@@ -434,52 +199,6 @@ could_not_push:
}
}
-static GstFlowReturn
-gst_real_video_dec_chain (GstPad * pad, GstBuffer * in)
-{
- GstRealVideoDec *dec = GST_REAL_VIDEO_DEC (GST_PAD_PARENT (pad));
- guint8 flags, *buf = GST_BUFFER_DATA (in);
- guint len = GST_BUFFER_SIZE (in);
- GstFlowReturn ret;
-
- if (G_UNLIKELY (dec->hooks.transform == NULL || dec->hooks.module == NULL))
- goto not_negotiated;
-
- /* Flags */
- if (len < 1)
- goto not_enough_data;
- flags = *buf;
- buf++;
- len--;
-
- if (flags == 0x40) {
- GST_DEBUG_OBJECT (dec, "Don't know how to handle buffer of type 0x40 "
- "(size=%i).", len);
- return GST_FLOW_OK;
- }
-
- ret = gst_real_video_dec_decode (dec, in, 1);
- gst_buffer_unref (in);
- if (ret != GST_FLOW_OK)
- return ret;
- return GST_FLOW_OK;
-
- /* Errors */
-not_enough_data:
- {
- GST_ELEMENT_ERROR (dec, STREAM, DECODE, ("Not enough data."), (NULL));
- gst_buffer_unref (in);
- return GST_FLOW_ERROR;
- }
-not_negotiated:
- {
- GST_WARNING_OBJECT (dec, "decoder not open, probably no input caps set "
- "yet, caps on input buffer: %" GST_PTR_FORMAT, GST_BUFFER_CAPS (in));
- gst_buffer_unref (in);
- return GST_FLOW_NOT_NEGOTIATED;
- }
-}
-
static gboolean
gst_real_video_dec_activate_push (GstPad * pad, gboolean active)
{
@@ -709,7 +428,8 @@ unknown_version:
could_not_open:
{
- GST_ERROR_OBJECT (dec, "Could not find library '%s' in '%s'", names, path);
+ GST_ERROR_OBJECT (dec, "Could not open library '%s' in '%s': %s", names,
+ path, g_module_error ());
return FALSE;
}