diff --git a/ChangeLog b/ChangeLog index 26eb8395df..cf76b3c9cf 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,18 @@ +2008-01-11 Wim Taymans + + * gst/mpegvideoparse/mpegpacketiser.c: (mpeg_util_find_start_code): + Small meaningless cleanup. + + * gst/mpegvideoparse/mpegvideoparse.c: (gst_mpegvideoparse_flush), + (mpegvideoparse_drain_avail), (gst_mpegvideoparse_chain_forward), + (scan_keyframe), (gst_mpegvideoparse_flush_decode), + (gst_mpegvideoparse_chain_reverse), (gst_mpegvideoparse_chain), + (mpv_parse_sink_event), (gst_mpegvideoparse_change_state): + * gst/mpegvideoparse/mpegvideoparse.h: + Track segment events. + Do the first part of reverse playback by sending data between two + I-frames to the decoder. + 2008-01-10 Thijs Vermeir * gst/rtpmanager/gstrtpbin.c: diff --git a/common b/common index 49c2fc5c9b..bd02d78838 160000 --- a/common +++ b/common @@ -1 +1 @@ -Subproject commit 49c2fc5c9bff0e9858e89978bd98164a386de51d +Subproject commit bd02d788384b40ff511cac0e32aa77f51a68912d diff --git a/gst/mpegvideoparse/mpegpacketiser.c b/gst/mpegvideoparse/mpegpacketiser.c index 5685fa7b51..447c50aef6 100644 --- a/gst/mpegvideoparse/mpegpacketiser.c +++ b/gst/mpegvideoparse/mpegpacketiser.c @@ -89,11 +89,13 @@ mpeg_packetiser_flush (MPEGPacketiser * p) guint8 * mpeg_util_find_start_code (guint32 * sync_word, guint8 * cur, guint8 * end) { - guint32 code = *sync_word; + guint32 code; if (G_UNLIKELY (cur == NULL)) return NULL; + code = *sync_word; + while (cur < end) { code <<= 8; diff --git a/gst/mpegvideoparse/mpegvideoparse.c b/gst/mpegvideoparse/mpegvideoparse.c index 0d700afdef..017f6f2664 100644 --- a/gst/mpegvideoparse/mpegvideoparse.c +++ b/gst/mpegvideoparse/mpegvideoparse.c @@ -296,6 +296,13 @@ gst_mpegvideoparse_flush (MpegVideoParse * mpegvideoparse) GST_DEBUG_OBJECT (mpegvideoparse, "mpegvideoparse: flushing"); mpegvideoparse->next_offset = GST_BUFFER_OFFSET_NONE; + + g_list_foreach (mpegvideoparse->gather, (GFunc) gst_mini_object_unref, NULL); + g_list_free (mpegvideoparse->gather); + mpegvideoparse->gather = NULL; + g_list_foreach (mpegvideoparse->decode, (GFunc) gst_mini_object_unref, NULL); + g_list_free (mpegvideoparse->decode); + mpegvideoparse->decode = NULL; } static GstFlowReturn @@ -309,7 +316,7 @@ mpegvideoparse_drain_avail (MpegVideoParse * mpegvideoparse) while ((cur != NULL) && (res == GST_FLOW_OK)) { /* Handle the block */ GST_LOG_OBJECT (mpegvideoparse, - "Have block of size %u with pack_type 0x%02x and flags 0x%02x\n", + "Have block of size %u with pack_type 0x%02x and flags 0x%02x", cur->length, cur->first_pack_type, cur->flags); /* Don't start pushing out buffers until we've seen a sequence header */ @@ -375,19 +382,12 @@ mpegvideoparse_drain_avail (MpegVideoParse * mpegvideoparse) } static GstFlowReturn -gst_mpegvideoparse_chain (GstPad * pad, GstBuffer * buf) +gst_mpegvideoparse_chain_forward (MpegVideoParse * mpegvideoparse, + gboolean discont, GstBuffer * buf) { - MpegVideoParse *mpegvideoparse; GstFlowReturn res; - gboolean have_discont; guint64 next_offset = GST_BUFFER_OFFSET_NONE; - g_return_val_if_fail (pad != NULL, GST_FLOW_ERROR); - g_return_val_if_fail (buf != NULL, GST_FLOW_ERROR); - - mpegvideoparse = - GST_MPEGVIDEOPARSE (gst_object_get_parent (GST_OBJECT (pad))); - GST_DEBUG_OBJECT (mpegvideoparse, "mpegvideoparse: received buffer of %u bytes with ts %" GST_TIME_FORMAT " and offset %" G_GINT64_FORMAT, GST_BUFFER_SIZE (buf), @@ -396,12 +396,10 @@ gst_mpegvideoparse_chain (GstPad * pad, GstBuffer * buf) /* If we have an offset, and the incoming offset doesn't match, or we have a discont, handle it first by flushing out data we have collected. */ - have_discont = GST_BUFFER_IS_DISCONT (buf); - if (mpegvideoparse->next_offset != GST_BUFFER_OFFSET_NONE) { if (GST_BUFFER_OFFSET_IS_VALID (buf)) { if (mpegvideoparse->next_offset != GST_BUFFER_OFFSET (buf)) - have_discont = TRUE; + discont = TRUE; next_offset = GST_BUFFER_OFFSET (buf) + GST_BUFFER_SIZE (buf); } else { next_offset = mpegvideoparse->next_offset + GST_BUFFER_SIZE (buf); @@ -409,7 +407,7 @@ gst_mpegvideoparse_chain (GstPad * pad, GstBuffer * buf) } /* Clear out any existing stuff if the new buffer is discontinuous */ - if (have_discont) { + if (discont) { GST_DEBUG_OBJECT (mpegvideoparse, "Have discont packet, draining data"); mpegvideoparse->need_discont = TRUE; @@ -417,9 +415,8 @@ gst_mpegvideoparse_chain (GstPad * pad, GstBuffer * buf) res = mpegvideoparse_drain_avail (mpegvideoparse); mpeg_packetiser_flush (&mpegvideoparse->packer); if (res != GST_FLOW_OK) { - mpegvideoparse->next_offset = next_offset; gst_buffer_unref (buf); - return res; + goto done; } } @@ -429,10 +426,277 @@ gst_mpegvideoparse_chain (GstPad * pad, GstBuffer * buf) /* And push out what we can */ res = mpegvideoparse_drain_avail (mpegvideoparse); +done: /* Update our offset */ mpegvideoparse->next_offset = next_offset; + return res; +} + +/* scan the decode queue for a picture header with an I frame and return the + * index in the first buffer. We only scan the first buffer and possibly a + * couple of bytes of the next buffers to find the I frame. Scanning is done + * backwards because the first buffer could contain many picture start codes + * with I frames. */ +static guint +scan_keyframe (MpegVideoParse * mpegvideoparse) +{ + guint64 scanword; + guint count; + GList *walk; + GstBuffer *head; + guint8 *data; + guint size; + + /* we use an 8 byte buffer, this is enough to hold the picture start code and + * the picture header bits we are interested in. We init to 0xff so that when + * we have a valid picture start without the header bits, we will be able to + * detect this because it will generate an invalid picture type. */ + scanword = ~0LL; + + GST_LOG_OBJECT (mpegvideoparse, "scan keyframe"); + + /* move to the second buffer if we can, we should have at least one buffer */ + walk = mpegvideoparse->decode; + g_return_val_if_fail (walk != NULL, -1); + + head = GST_BUFFER_CAST (walk->data); + + count = 0; + walk = g_list_next (walk); + while (walk) { + GstBuffer *buf = GST_BUFFER_CAST (walk->data); + + data = GST_BUFFER_DATA (buf); + size = GST_BUFFER_SIZE (buf); + + GST_LOG_OBJECT (mpegvideoparse, "collect remaining %d bytes from %p", + 6 - count, buf); + + while (size > 0 && count < 6) { + scanword = (scanword << 8) | *data++; + size--; + count++; + } + if (count == 6) + break; + + walk = g_list_next (walk); + } + /* move bits to the beginning of the word now */ + if (count) + scanword = (scanword << (8 * (8 - count))); + + GST_LOG_OBJECT (mpegvideoparse, "scanword 0x%016llx", scanword); + + data = GST_BUFFER_DATA (head); + size = GST_BUFFER_SIZE (head); + + while (size > 0) { + scanword = (((guint64) data[size - 1]) << 56) | (scanword >> 8); + + GST_LOG_OBJECT (mpegvideoparse, "scanword at %d 0x%016llx", size - 1, + scanword); + + /* check picture start and picture type */ + if ((scanword & 0xffffffff00380000LL) == 0x0000010000080000LL) + break; + + size--; + } + return size - 1; +} + +/* For reverse playback we use a technique that can be used for + * any keyframe based video codec. + * + * Input: + * Buffer decoding order: 7 8 9 4 5 6 1 2 3 EOS + * Keyframe flag: K K + * Discont flag: D D D + * + * - Each Discont marks a discont in the decoding order. + * - The keyframes mark where we can start decoding. For mpeg they are either + * set by the demuxer or we have to scan the buffers for a syncword and + * picture header with an I frame. + * + * First we prepend incomming buffers to the gather queue, whenever we receive + * a discont, we flush out the gather queue. + * + * The above data will be accumulated in the gather queue like this: + * + * gather queue: 9 8 7 + * D + * + * Whe buffer 4 is received (with a DISCONT), we flush the gather queue like + * this: + * + * while (gather) + * take head of queue and prepend to decode queue. + * if we copied a keyframe, decode the decode queue. + * + * After we flushed the gather queue, we add 4 to the (now empty) gather queue. + * We get the following situation: + * + * gather queue: 4 + * decode queue: 7 8 9 + * + * After we received 5 (Keyframe) and 6: + * + * gather queue: 6 5 4 + * decode queue: 7 8 9 + * + * When we receive 1 (DISCONT) which triggers a flush of the gather queue: + * + * Copy head of the gather queue (6) to decode queue: + * + * gather queue: 5 4 + * decode queue: 6 7 8 9 + * + * Copy head of the gather queue (5) to decode queue. This is a keyframe so we + * can start decoding. + * + * gather queue: 4 + * decode queue: 5 6 7 8 9 + * + * Decode frames in decode queue, we don't actually do the decoding but we + * will send the decode queue to the downstream element. This will empty the + * decoding queue again. + * + * Copy head of the gather queue (4) to decode queue, we flushed the gather + * queue and can now store input buffer in the gather queue: + * + * gather queue: 1 + * decode queue: 4 + * + * When we receive EOS, the queue looks like: + * + * gather queue: 3 2 1 + * decode queue: 4 + * + * Fill decode queue, first keyframe we copy is 2: + * + * gather queue: 1 + * decode queue: 2 3 4 + * + * After pushing the decode queue: + * + * gather queue: 1 + * decode queue: + * + * Leftover buffer 1 cannot be decoded and must be discarded. + */ +static GstFlowReturn +gst_mpegvideoparse_flush_decode (MpegVideoParse * mpegvideoparse, guint idx) +{ + GstFlowReturn res = GST_FLOW_OK; + GstBuffer *head = NULL; + + while (mpegvideoparse->decode) { + GstBuffer *buf; + + buf = GST_BUFFER_CAST (mpegvideoparse->decode->data); + + if (idx != -1) { + GstBuffer *temp; + + if (idx > 0) { + /* first buffer, split at the point where the picture start was + * detected and store as the new head of the decoding list. */ + head = gst_buffer_create_sub (buf, 0, idx); + /* push the remainder after the picture sync point downstream, this is the + * first DISCONT buffer we push. */ + temp = gst_buffer_create_sub (buf, idx, GST_BUFFER_SIZE (buf) - idx); + /* don't need old buffer anymore and swap new buffer */ + gst_buffer_unref (buf); + buf = temp; + } + GST_BUFFER_FLAG_SET (buf, GST_BUFFER_FLAG_DISCONT); + idx = -1; + } else { + /* next buffers are not discont */ + GST_BUFFER_FLAG_UNSET (buf, GST_BUFFER_FLAG_DISCONT); + } + + GST_DEBUG_OBJECT (mpegvideoparse, "pushing buffer %p, ts %" GST_TIME_FORMAT, + buf, GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf))); + + res = gst_pad_push (mpegvideoparse->srcpad, buf); + + mpegvideoparse->decode = + g_list_delete_link (mpegvideoparse->decode, mpegvideoparse->decode); + } + if (head) { + /* store remainder of the buffer */ + mpegvideoparse->decode = g_list_prepend (mpegvideoparse->decode, head); + } + return res; +} + +static GstFlowReturn +gst_mpegvideoparse_chain_reverse (MpegVideoParse * mpegvideoparse, + gboolean discont, GstBuffer * buf) +{ + GstFlowReturn res = GST_FLOW_OK; + + /* if we have a discont, move buffers to the decode list */ + if (G_UNLIKELY (discont)) { + GST_DEBUG_OBJECT (mpegvideoparse, "received discont,gathering buffers"); + + while (mpegvideoparse->gather) { + GstBuffer *gbuf; + guint keyframeidx; + + gbuf = GST_BUFFER_CAST (mpegvideoparse->gather->data); + /* remove from the gather list */ + mpegvideoparse->gather = + g_list_delete_link (mpegvideoparse->gather, mpegvideoparse->gather); + /* copy to decode queue */ + mpegvideoparse->decode = g_list_prepend (mpegvideoparse->decode, gbuf); + + GST_DEBUG_OBJECT (mpegvideoparse, "copied decoding buffer %p, len %d", + gbuf, g_list_length (mpegvideoparse->decode)); + + /* check if we copied a keyframe, we scan the buffers on the decode queue. + * We only need to scan the first buffer and at most 3 bytes of the second + * buffer. We return the index of the keyframe (or -1 when nothing was + * found) */ + while ((keyframeidx = scan_keyframe (mpegvideoparse)) != -1) { + GST_DEBUG_OBJECT (mpegvideoparse, "copied keyframe at %u", keyframeidx); + res = gst_mpegvideoparse_flush_decode (mpegvideoparse, keyframeidx); + } + } + } + + if (buf) { + /* add buffer to gather queue */ + GST_DEBUG_OBJECT (mpegvideoparse, "gathering buffer %p, size %u", buf, + GST_BUFFER_SIZE (buf)); + mpegvideoparse->gather = g_list_prepend (mpegvideoparse->gather, buf); + } + + return res; +} + +static GstFlowReturn +gst_mpegvideoparse_chain (GstPad * pad, GstBuffer * buf) +{ + MpegVideoParse *mpegvideoparse; + GstFlowReturn res; + gboolean discont; + + mpegvideoparse = + GST_MPEGVIDEOPARSE (gst_object_get_parent (GST_OBJECT (pad))); + + discont = GST_BUFFER_IS_DISCONT (buf); + + if (mpegvideoparse->segment.rate > 0.0) + res = gst_mpegvideoparse_chain_forward (mpegvideoparse, discont, buf); + else + res = gst_mpegvideoparse_chain_reverse (mpegvideoparse, discont, buf); + gst_object_unref (mpegvideoparse); + return res; } @@ -483,35 +747,49 @@ mpv_parse_sink_event (GstPad * pad, GstEvent * event) /* Unknown incoming segment format. Output a default open-ended * TIME segment */ gst_event_unref (event); + + /* set new values */ + format = GST_FORMAT_TIME; + start = 0; + stop = GST_CLOCK_TIME_NONE; + pos = 0; + /* create a new segment with these values */ event = gst_event_new_new_segment_full (update, rate, applied_rate, - GST_FORMAT_TIME, 0, GST_CLOCK_TIME_NONE, 0); + format, start, stop, pos); } - gst_event_parse_new_segment_full (event, &update, &rate, &applied_rate, - &format, &start, &stop, &pos); + /* now configure the values */ + gst_segment_set_newsegment_full (&mpegvideoparse->segment, update, + rate, applied_rate, format, start, stop, pos); GST_DEBUG_OBJECT (mpegvideoparse, "Pushing newseg rate %g, applied rate %g, format %d, start %" G_GINT64_FORMAT ", stop %" G_GINT64_FORMAT ", pos %" G_GINT64_FORMAT, rate, applied_rate, format, start, stop, pos); - res = gst_pad_event_default (pad, event); + res = gst_pad_push_event (mpegvideoparse->srcpad, event); break; } case GST_EVENT_FLUSH_STOP: + GST_DEBUG_OBJECT (mpegvideoparse, "flush stop"); gst_mpegvideoparse_flush (mpegvideoparse); - res = gst_pad_event_default (pad, event); + res = gst_pad_push_event (mpegvideoparse->srcpad, event); break; case GST_EVENT_EOS: /* Push any remaining buffers out, then flush. */ - mpeg_packetiser_handle_eos (&mpegvideoparse->packer); - mpegvideoparse_drain_avail (mpegvideoparse); - gst_mpegvideoparse_flush (mpegvideoparse); - - res = gst_pad_event_default (pad, event); + GST_DEBUG_OBJECT (mpegvideoparse, "received EOS"); + if (mpegvideoparse->segment.rate >= 0.0) { + mpeg_packetiser_handle_eos (&mpegvideoparse->packer); + mpegvideoparse_drain_avail (mpegvideoparse); + gst_mpegvideoparse_flush (mpegvideoparse); + } else { + gst_mpegvideoparse_chain_reverse (mpegvideoparse, TRUE, NULL); + gst_mpegvideoparse_flush_decode (mpegvideoparse, 0); + } + res = gst_pad_push_event (mpegvideoparse->srcpad, event); break; default: - res = gst_pad_event_default (pad, event); + res = gst_pad_push_event (mpegvideoparse->srcpad, event); break; } @@ -531,6 +809,14 @@ gst_mpegvideoparse_change_state (GstElement * element, mpegvideoparse = GST_MPEGVIDEOPARSE (element); + switch (transition) { + case GST_STATE_CHANGE_READY_TO_PAUSED: + gst_segment_init (&mpegvideoparse->segment, GST_FORMAT_UNDEFINED); + break; + default: + break; + } + ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition); switch (transition) { diff --git a/gst/mpegvideoparse/mpegvideoparse.h b/gst/mpegvideoparse/mpegvideoparse.h index c021265844..8495adf1bb 100644 --- a/gst/mpegvideoparse/mpegvideoparse.h +++ b/gst/mpegvideoparse/mpegvideoparse.h @@ -43,6 +43,7 @@ struct _MpegVideoParse { GstElement element; GstPad *sinkpad, *srcpad; + GstSegment segment; gint64 next_offset; gboolean need_discont; @@ -53,6 +54,10 @@ struct _MpegVideoParse { /* Packetise helper */ MPEGPacketiser packer; + + /* gather/decode queues for reverse playback */ + GList *gather; + GList *decode; }; struct _MpegVideoParseClass {