diff options
Diffstat (limited to 'ext/metadata')
-rw-r--r-- | ext/metadata/gstmetadataparse.c | 200 | ||||
-rw-r--r-- | ext/metadata/gstmetadataparse.h | 1 |
2 files changed, 190 insertions, 11 deletions
diff --git a/ext/metadata/gstmetadataparse.c b/ext/metadata/gstmetadataparse.c index 43dbe6ad..fec1cd60 100644 --- a/ext/metadata/gstmetadataparse.c +++ b/ext/metadata/gstmetadataparse.c @@ -168,6 +168,18 @@ gst_metadata_parse_get_strip_range (gint64 * boffset, guint32 * bsize, const gint64 seg_offset, const guint32 seg_size, gint64 * toffset, guint32 * tsize, gint64 * ioffset); +static gboolean +gst_metadata_parse_strip_buffer (GstMetadataParse * filter, gint64 offset, + GstBuffer ** buf); + +static void +gst_metadata_parse_translate_pos (GstMetadataParse * filter, gint64 pos, + gint64 * orig_pos); + +static const GstQueryType *gst_metadata_parse_get_query_types (GstPad * pad); + +static gboolean gst_metadata_parse_src_query (GstPad * pad, GstQuery * query); + static void gst_metadata_parse_base_init (gpointer gclass) { @@ -253,6 +265,10 @@ gst_metadata_parse_init (GstMetadataParse * filter, gst_pad_set_getcaps_function (filter->srcpad, GST_DEBUG_FUNCPTR (gst_metadata_parse_get_caps)); gst_pad_set_event_function (filter->srcpad, gst_metadata_parse_src_event); + gst_pad_set_query_function (filter->srcpad, + GST_DEBUG_FUNCPTR (gst_metadata_parse_src_query)); + gst_pad_set_query_type_function (filter->srcpad, + GST_DEBUG_FUNCPTR (gst_metadata_parse_get_query_types)); gst_pad_use_fixed_caps (filter->srcpad); gst_pad_set_getrange_function (filter->srcpad, gst_metadata_parse_get_range); gst_pad_set_activatepull_function (filter->srcpad, @@ -406,11 +422,60 @@ gst_metadata_parse_src_event (GstPad * pad, GstEvent * event) switch (GST_EVENT_TYPE (event)) { case GST_EVENT_SEEK: - if (filter->state != MT_STATE_PARSED) { - /* FIXME: What to do here */ - ret = TRUE; + { + gdouble rate; + GstFormat format; + GstSeekFlags flags; + GstSeekType start_type; + gint64 start; + GstSeekType stop_type; + gint64 stop; + + /* we don't know where are the chunks to be stripped before parse */ + if (filter->state != MT_STATE_PARSED) goto done; + + gst_event_parse_seek (event, &rate, &format, &flags, + &start_type, &start, &stop_type, &stop); + + switch (format) { + case GST_FORMAT_BYTES: + break; + case GST_FORMAT_PERCENT: + if (filter->duration < 0) + goto done; + break; + default: + goto done; + } + + if (start_type == GST_SEEK_TYPE_CUR) + start = filter->offset + start; + else if (start_type == GST_SEEK_TYPE_END) { + if (filter->duration < 0) + goto done; + start = filter->duration + start; } + start_type == GST_SEEK_TYPE_SET; + + gst_metadata_parse_translate_pos (filter, start, &start); + + if (stop_type == GST_SEEK_TYPE_CUR) + stop = filter->offset + stop; + else if (stop_type == GST_SEEK_TYPE_END) { + if (filter->duration < 0) + goto done; + stop = filter->duration + stop; + } + stop_type == GST_SEEK_TYPE_SET; + + gst_metadata_parse_translate_pos (filter, stop, &stop); + + gst_event_unref (event); + event = gst_event_new_seek (rate, format, flags, + start_type, start, stop_type, stop); + + } break; default: break; @@ -516,6 +581,7 @@ gst_metadata_parse_init_members (GstMetadataParse * filter) filter->next_size = 0; filter->img_type = IMG_NONE; filter->duration = -1; + filter->offset_orig = 0; filter->offset = 0; filter->state = MT_STATE_NULL; filter->need_more_data = FALSE; @@ -698,6 +764,59 @@ gst_metadata_parse_send_tags (GstMetadataParse * filter) filter->need_send_tag = FALSE; } +static const GstQueryType * +gst_metadata_parse_get_query_types (GstPad * pad) +{ + static const GstQueryType gst_metadata_parse_src_query_types[] = { + GST_QUERY_POSITION, + GST_QUERY_DURATION, + GST_QUERY_FORMATS, + 0 + }; + + return gst_metadata_parse_src_query_types; +} + +static gboolean +gst_metadata_parse_src_query (GstPad * pad, GstQuery * query) +{ + gboolean ret = FALSE; + GstFormat format; + GstMetadataParse *filter = GST_METADATA_PARSE (gst_pad_get_parent (pad)); + + switch (GST_QUERY_TYPE (query)) { + case GST_QUERY_POSITION: + gst_query_parse_position (query, &format, NULL); + + if (format == GST_FORMAT_BYTES) { + gst_query_set_position (query, GST_FORMAT_BYTES, filter->offset); + ret = TRUE; + } + break; + case GST_QUERY_DURATION: + gst_query_parse_duration (query, &format, NULL); + + if (format == GST_FORMAT_BYTES) { + if (filter->duration >= 0) { + gst_query_set_duration (query, GST_FORMAT_BYTES, filter->duration); + ret = TRUE; + } + } + break; + case GST_QUERY_FORMATS: + gst_query_set_formats (query, 1, GST_FORMAT_BYTES); + ret = TRUE; + break; + default: + break; + } + + gst_object_unref (filter); + + return ret; + +} + /* * return: * -1 -> error @@ -818,12 +937,19 @@ done: * this function does the actual processing */ +/* FIXME */ +/* Current parse is just done before is pull mode could be activated */ +/* may be it is possible to parse in chain mode by doing some trick with gst-adapter */ +/* the pipeline below would be a test for that case */ +/* gst-launch-0.10 filesrc location=Exif.jpg ! queue ! metadataparse ! filesink location=gen3.jpg */ + static GstFlowReturn gst_metadata_parse_chain (GstPad * pad, GstBuffer * buf) { GstMetadataParse *filter = NULL; GstFlowReturn ret = GST_FLOW_ERROR; - guint32 buf_size; + guint32 buf_size = 0; + guint32 new_buf_size = 0; filter = GST_METADATA_PARSE (gst_pad_get_parent (pad)); @@ -879,11 +1005,11 @@ gst_metadata_parse_chain (GstPad * pad, GstBuffer * buf) } buf_size = GST_BUFFER_SIZE (buf); - gst_metadata_parse_strip_buffer (filter, &buf); - filter->offset += buf_size; + gst_metadata_parse_strip_buffer (filter, filter->offset_orig, &buf); if (buf) { /* may be all buffer has been striped */ gst_buffer_set_caps (buf, GST_PAD_CAPS (filter->srcpad)); + new_buf_size = GST_BUFFER_SIZE (buf); ret = gst_pad_push (filter->srcpad, buf); buf = NULL; /* this function don't owner it anymore */ @@ -893,6 +1019,9 @@ gst_metadata_parse_chain (GstPad * pad, GstBuffer * buf) done: + filter->offset_orig += buf_size; + filter->offset += new_buf_size; + if (buf) { /* there was an error and buffer wasn't pushed */ gst_buffer_unref (buf); @@ -960,6 +1089,19 @@ gst_metadata_parse_pull_range_parse (GstMetadataParse * filter) } while (res > 0); + if (res == 0) { + int i; + + filter->duration = duration; + + for (i = 0; i < filter->num_segs; ++i) { + + filter->duration -= (filter->seg_size[i] - filter->seg_inject_size[i]); + + } + + } + done: return ret; @@ -977,6 +1119,7 @@ gst_metadata_parse_activate (GstPad * pad) if (!gst_pad_check_pull_range (pad) || !gst_pad_activate_pull (filter->sinkpad, TRUE)) { + /* FIXME: currently it is not possible to parse in chain. Fail here ? */ /* nothing to be done by now, activate push mode */ return gst_pad_activate_push (pad, TRUE); } @@ -1005,7 +1148,7 @@ done: * Returns FALSE if nothing has stripped, TRUE if otherwise */ -gboolean +static gboolean gst_metadata_parse_get_strip_range (gint64 * boffset, guint32 * bsize, const gint64 seg_offset, const guint32 seg_size, gint64 * toffset, guint32 * tsize, gint64 * ioffset) @@ -1076,8 +1219,9 @@ done: * FALSE -> buffer unmodified */ -gboolean -gst_metadata_parse_strip_buffer (GstMetadataParse * filter, GstBuffer ** buf) +static gboolean +gst_metadata_parse_strip_buffer (GstMetadataParse * filter, gint64 offset, + GstBuffer ** buf) { gint64 boffset[3]; @@ -1095,7 +1239,6 @@ gst_metadata_parse_strip_buffer (GstMetadataParse * filter, GstBuffer ** buf) int i; gboolean ret = FALSE; guint32 new_size = 0; - const gint64 offset = filter->offset; if (filter->num_segs == 0) goto done; @@ -1192,11 +1335,35 @@ done: } +/* + * pos - position in stream striped + * orig_pos - position in original stream + */ +static void +gst_metadata_parse_translate_pos (GstMetadataParse * filter, gint64 pos, + gint64 * orig_pos) +{ + int i; + + *orig_pos = 0; + for (i = 0; i < filter->num_segs; ++i) { + if (filter->seg_offset[i] > pos) { + break; + } + *orig_pos += filter->seg_size[i]; + } + *orig_pos += pos; + +} + static gboolean gst_metadata_parse_get_range (GstPad * pad, guint64 offset, guint size, GstBuffer ** buf) { GstMetadataParse *filter = NULL; + gboolean ret; + gint64 pos; + const gint64 saved_offset = offset + size; filter = GST_METADATA_PARSE (GST_PAD_PARENT (pad)); @@ -1204,7 +1371,18 @@ gst_metadata_parse_get_range (GstPad * pad, gst_metadata_parse_send_tags (filter); } - return gst_pad_pull_range (filter->sinkpad, offset, size, buf); + gst_metadata_parse_translate_pos (filter, offset + size - 1, &pos); + gst_metadata_parse_translate_pos (filter, offset, &offset); + + ret = gst_pad_pull_range (filter->sinkpad, offset, 1 + pos - offset, buf); + + if (ret == GST_FLOW_OK && *buf) { + gst_metadata_parse_strip_buffer (filter, offset, buf); + filter->offset_orig = pos; + filter->offset = saved_offset; + } + + return ret; } diff --git a/ext/metadata/gstmetadataparse.h b/ext/metadata/gstmetadataparse.h index dd5a031c..11e6a52e 100644 --- a/ext/metadata/gstmetadataparse.h +++ b/ext/metadata/gstmetadataparse.h @@ -96,6 +96,7 @@ struct _GstMetadataParse guint32 next_size; ImageType img_type; + gint64 offset_orig; gint64 duration; gint64 offset; |