Add essence track abstraction

Add an abstraction to represent essence tracks and
use this everywhere. This will later be used to keep
track of positions and to generate/handle seek tables.

Some random cleanup and renaming.
This commit is contained in:
Sebastian Dröge 2009-01-26 16:20:34 +01:00
parent 914b108db9
commit 4162c36f17
2 changed files with 459 additions and 246 deletions

View file

@ -84,48 +84,6 @@ GST_STATIC_PAD_TEMPLATE ("track_%u",
GST_DEBUG_CATEGORY_STATIC (mxfdemux_debug); GST_DEBUG_CATEGORY_STATIC (mxfdemux_debug);
#define GST_CAT_DEFAULT mxfdemux_debug #define GST_CAT_DEFAULT mxfdemux_debug
#define GST_TYPE_MXF_DEMUX_PAD (gst_mxf_demux_pad_get_type())
#define GST_MXF_DEMUX_PAD(pad) (G_TYPE_CHECK_INSTANCE_CAST((pad),GST_TYPE_MXF_DEMUX_PAD,GstMXFDemuxPad))
#define GST_MXF_DEMUX_PAD_CAST(pad) ((GstMXFDemuxPad *) pad)
#define GST_IS_MXF_DEMUX_PAD(pad) (G_TYPE_CHECK_INSTANCE_TYPE((pad),GST_TYPE_MXF_DEMUX_PAD))
typedef struct
{
GstPad parent;
guint32 track_id;
gboolean need_segment;
GstClockTime last_stop;
GstFlowReturn last_flow;
gboolean eos, discont;
gpointer mapping_data;
const MXFEssenceElementHandler *handler;
MXFEssenceElementHandleFunc handle_func;
GstTagList *tags;
MXFMetadataGenericPackage *material_package;
MXFMetadataTimelineTrack *material_track;
MXFMetadataSourceClip *component;
guint current_component;
gint64 current_component_position;
gint64 current_component_drop;
MXFMetadataSourcePackage *source_package;
MXFMetadataTimelineTrack *source_track;
GstCaps *caps;
} GstMXFDemuxPad;
typedef struct
{
GstPadClass parent;
} GstMXFDemuxPadClass;
G_DEFINE_TYPE (GstMXFDemuxPad, gst_mxf_demux_pad, GST_TYPE_PAD); G_DEFINE_TYPE (GstMXFDemuxPad, gst_mxf_demux_pad, GST_TYPE_PAD);
static void static void
@ -133,11 +91,6 @@ gst_mxf_demux_pad_finalize (GObject * object)
{ {
GstMXFDemuxPad *pad = GST_MXF_DEMUX_PAD (object); GstMXFDemuxPad *pad = GST_MXF_DEMUX_PAD (object);
gst_caps_replace (&pad->caps, NULL);
g_free (pad->mapping_data);
pad->mapping_data = NULL;
if (pad->tags) { if (pad->tags) {
gst_tag_list_free (pad->tags); gst_tag_list_free (pad->tags);
pad->tags = NULL; pad->tags = NULL;
@ -186,7 +139,7 @@ gst_mxf_demux_flush (GstMXFDemux * demux, gboolean discont)
/* Only in push mode */ /* Only in push mode */
if (!demux->random_access) { if (!demux->random_access) {
/* We reset the offset and will get one from first push */ /* We reset the offset and will get one from first push */
demux->offset = 0; demux->offset = -1;
} }
} }
@ -227,10 +180,30 @@ gst_mxf_demux_reset_mxf_state (GstMXFDemux * demux)
demux->partitions = NULL; demux->partitions = NULL;
demux->current_partition = NULL; demux->current_partition = NULL;
if (demux->essence_tracks) {
guint i;
for (i = 0; i < demux->essence_tracks->len; i++) {
GstMXFDemuxEssenceTrack *t =
&g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack, i);
g_free (t->offsets);
g_free (t->mapping_data);
if (t->tags)
gst_tag_list_free (t->tags);
if (t->caps)
gst_caps_unref (t->caps);
}
g_array_free (demux->essence_tracks, TRUE);
demux->essence_tracks = NULL;
}
} }
static void static void
gst_mxf_demux_reset_track_metadata (GstMXFDemux * demux) gst_mxf_demux_reset_linked_metadata (GstMXFDemux * demux)
{ {
guint i; guint i;
@ -238,16 +211,23 @@ gst_mxf_demux_reset_track_metadata (GstMXFDemux * demux)
for (i = 0; i < demux->src->len; i++) { for (i = 0; i < demux->src->len; i++) {
GstMXFDemuxPad *pad = g_ptr_array_index (demux->src, i); GstMXFDemuxPad *pad = g_ptr_array_index (demux->src, i);
pad->handler = NULL;
pad->handle_func = NULL;
pad->material_track = NULL; pad->material_track = NULL;
pad->material_package = NULL; pad->material_package = NULL;
pad->component = NULL; pad->current_component = NULL;
pad->source_track = NULL;
pad->source_package = NULL;
} }
} }
if (demux->essence_tracks) {
for (i = 0; i < demux->essence_tracks->len; i++) {
GstMXFDemuxEssenceTrack *track =
&g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack, i);
track->source_package = NULL;
track->source_track = NULL;
}
}
demux->current_package = NULL;
} }
static void static void
@ -258,9 +238,7 @@ gst_mxf_demux_reset_metadata (GstMXFDemux * demux)
demux->update_metadata = TRUE; demux->update_metadata = TRUE;
demux->metadata_resolved = FALSE; demux->metadata_resolved = FALSE;
gst_mxf_demux_reset_track_metadata (demux); gst_mxf_demux_reset_linked_metadata (demux);
demux->current_package = NULL;
demux->preface = NULL; demux->preface = NULL;
@ -305,15 +283,16 @@ gst_mxf_demux_reset (GstMXFDemux * demux)
demux->random_index_pack = NULL; demux->random_index_pack = NULL;
} }
if (demux->index_table) { if (demux->pending_index_table_segments) {
guint i; GList *l;
for (i = 0; i < demux->index_table->len; i++) for (l = demux->pending_index_table_segments; l; l = l->next) {
mxf_index_table_segment_reset (&g_array_index (demux->index_table, MXFIndexTableSegment *s = l->data;
MXFIndexTableSegment, i)); mxf_index_table_segment_reset (s);
g_free (s);
g_array_free (demux->index_table, TRUE); }
demux->index_table = NULL; g_list_free (demux->pending_index_table_segments);
demux->pending_index_table_segments = NULL;
} }
gst_mxf_demux_reset_mxf_state (demux); gst_mxf_demux_reset_mxf_state (demux);
@ -496,7 +475,7 @@ gst_mxf_demux_handle_primer_pack (GstMXFDemux * demux, const MXFUL * key,
} }
static GstFlowReturn static GstFlowReturn
gst_mxf_demux_handle_header_metadata_resolve_references (GstMXFDemux * demux) gst_mxf_demux_resolve_references (GstMXFDemux * demux)
{ {
GstFlowReturn ret = GST_FLOW_OK; GstFlowReturn ret = GST_FLOW_OK;
GHashTableIter iter; GHashTableIter iter;
@ -607,11 +586,11 @@ gst_mxf_demux_choose_package (GstMXFDemux * demux)
for (i = 0; i < demux->preface->content_storage->n_packages; i++) { for (i = 0; i < demux->preface->content_storage->n_packages; i++) {
if (demux->preface->content_storage->packages[i] && if (demux->preface->content_storage->packages[i] &&
MXF_IS_METADATA_MATERIAL_PACKAGE (demux->preface->content_storage-> MXF_IS_METADATA_MATERIAL_PACKAGE (demux->preface->
packages[i])) { content_storage->packages[i])) {
ret = ret =
MXF_METADATA_GENERIC_PACKAGE (demux->preface->content_storage-> MXF_METADATA_GENERIC_PACKAGE (demux->preface->
packages[i]); content_storage->packages[i]);
break; break;
} }
} }
@ -630,14 +609,190 @@ gst_mxf_demux_choose_package (GstMXFDemux * demux)
} }
static GstFlowReturn static GstFlowReturn
gst_mxf_demux_handle_header_metadata_update_streams (GstMXFDemux * demux) gst_mxf_demux_update_essence_tracks (GstMXFDemux * demux)
{
guint i, j, k;
g_return_val_if_fail (demux->preface->content_storage, GST_FLOW_ERROR);
g_return_val_if_fail (demux->preface->content_storage->essence_container_data,
GST_FLOW_ERROR);
for (i = 0; i < demux->preface->content_storage->n_essence_container_data;
i++) {
MXFMetadataEssenceContainerData *edata;
MXFMetadataSourcePackage *package;
if (demux->preface->content_storage->essence_container_data[i] == NULL)
continue;
edata = demux->preface->content_storage->essence_container_data[i];
if (!edata->linked_package) {
GST_WARNING_OBJECT (demux, "Linked package not resolved");
continue;
}
package = edata->linked_package;
if (!package->parent.tracks) {
GST_WARNING_OBJECT (demux, "Linked package with no resolved tracks");
continue;
}
for (j = 0; j < package->parent.n_tracks; j++) {
MXFMetadataTimelineTrack *track;
GstMXFDemuxEssenceTrack *etrack = NULL;
GstCaps *caps = NULL;
gboolean new = FALSE;
if (!package->parent.tracks[j]
|| !MXF_IS_METADATA_TIMELINE_TRACK (package->parent.tracks[j]))
continue;
track = MXF_METADATA_TIMELINE_TRACK (package->parent.tracks[j]);
if ((track->parent.type & 0xf0) != 0x30)
continue;
if (demux->essence_tracks) {
for (k = 0; k < demux->essence_tracks->len; k++) {
GstMXFDemuxEssenceTrack *tmp =
&g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack,
k);
if (tmp->track_number == track->parent.track_number &&
tmp->body_sid == edata->body_sid) {
etrack = tmp;
break;
}
}
}
if (!etrack) {
GstMXFDemuxEssenceTrack tmp;
memset (&tmp, 0, sizeof (tmp));
tmp.body_sid = edata->body_sid;
tmp.track_number = track->parent.track_number;
if (!demux->essence_tracks)
demux->essence_tracks =
g_array_new (FALSE, FALSE, sizeof (GstMXFDemuxEssenceTrack));
g_array_append_val (demux->essence_tracks, tmp);
etrack =
&g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack,
demux->essence_tracks->len - 1);
new = TRUE;
}
etrack->source_package = NULL;
etrack->source_track = NULL;
if (!track->parent.sequence) {
GST_WARNING_OBJECT (demux, "Source track has no sequence");
goto next;
}
if (track->parent.sequence->duration > etrack->duration) {
guint64 old_duration = etrack->duration;
etrack->duration = track->parent.sequence->duration;
if (etrack->offsets) {
etrack->offsets =
g_realloc (etrack->offsets,
etrack->duration * sizeof (GstMXFDemuxIndex));
memset (&etrack->offsets[old_duration - 1], 0,
etrack->duration - old_duration);
} else {
etrack->offsets = g_new0 (GstMXFDemuxIndex, etrack->duration);
}
}
g_free (etrack->mapping_data);
etrack->mapping_data = NULL;
etrack->handler = NULL;
etrack->handle_func = NULL;
if (etrack->tags)
gst_tag_list_free (etrack->tags);
etrack->tags = NULL;
etrack->handler = mxf_essence_element_handler_find (track);
if (!etrack->handler) {
GST_WARNING_OBJECT (demux,
"No essence element handler for track found");
goto next;
}
caps =
etrack->handler->create_caps (track, &etrack->tags,
&etrack->handle_func, &etrack->mapping_data);
GST_DEBUG_OBJECT (demux, "Created caps %" GST_PTR_FORMAT, caps);
if (!caps && new) {
GST_WARNING_OBJECT (demux, "No caps created, ignoring stream");
g_free (etrack->mapping_data);
if (etrack->tags)
gst_tag_list_free (etrack->tags);
g_array_remove_index (demux->essence_tracks,
demux->essence_tracks->len - 1);
goto next;
} else if (!caps) {
GST_WARNING_OBJECT (demux, "Couldn't create updated caps for stream");
} else if (!etrack->caps || !gst_caps_is_equal (etrack->caps, caps)) {
gst_caps_replace (&etrack->caps, caps);
}
etrack->source_package = package;
etrack->source_track = track;
continue;
next:
if (new) {
g_free (etrack->mapping_data);
if (etrack->tags)
gst_tag_list_free (etrack->tags);
if (etrack->caps)
gst_caps_unref (etrack->caps);
g_array_remove_index (demux->essence_tracks,
demux->essence_tracks->len - 1);
}
}
}
if (!demux->essence_tracks || demux->essence_tracks->len == 0) {
GST_ERROR_OBJECT (demux, "No valid essence tracks in this file");
return GST_FLOW_ERROR;
}
for (i = 0; i < demux->essence_tracks->len; i++) {
GstMXFDemuxEssenceTrack *etrack =
&g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack, i);
if (!etrack->source_package || !etrack->source_track || !etrack->caps) {
GST_ERROR_OBJECT (demux, "Failed to update essence track");
return GST_FLOW_ERROR;
}
}
return GST_FLOW_OK;
}
static GstFlowReturn
gst_mxf_demux_update_tracks (GstMXFDemux * demux)
{ {
MXFMetadataGenericPackage *current_package = NULL; MXFMetadataGenericPackage *current_package = NULL;
guint i, j, k; guint i, j, k;
gboolean first_run; gboolean first_run;
guint component_index; guint component_index;
GstFlowReturn ret;
GST_DEBUG_OBJECT (demux, "Updating streams"); GST_DEBUG_OBJECT (demux, "Updating tracks");
if ((ret = gst_mxf_demux_update_essence_tracks (demux))) {
return ret;
}
current_package = gst_mxf_demux_choose_package (demux); current_package = gst_mxf_demux_choose_package (demux);
@ -661,8 +816,8 @@ gst_mxf_demux_handle_header_metadata_update_streams (GstMXFDemux * demux)
MXFMetadataSourceClip *component = NULL; MXFMetadataSourceClip *component = NULL;
MXFMetadataSourcePackage *source_package = NULL; MXFMetadataSourcePackage *source_package = NULL;
MXFMetadataTimelineTrack *source_track = NULL; MXFMetadataTimelineTrack *source_track = NULL;
GstMXFDemuxEssenceTrack *etrack = NULL;
GstMXFDemuxPad *pad = NULL; GstMXFDemuxPad *pad = NULL;
GstCaps *caps = NULL;
GST_DEBUG_OBJECT (demux, "Handling track %u", i); GST_DEBUG_OBJECT (demux, "Handling track %u", i);
@ -691,7 +846,7 @@ gst_mxf_demux_handle_header_metadata_update_streams (GstMXFDemux * demux)
} }
if (pad) if (pad)
component_index = pad->current_component; component_index = pad->current_component_index;
else else
component_index = 0; component_index = 0;
@ -748,6 +903,22 @@ gst_mxf_demux_handle_header_metadata_update_streams (GstMXFDemux * demux)
continue; continue;
} }
for (k = 0; k < demux->essence_tracks->len; k++) {
GstMXFDemuxEssenceTrack *tmp =
&g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack, k);
if (tmp->source_package == source_package &&
tmp->source_track == source_track) {
etrack = tmp;
break;
}
}
if (!etrack) {
GST_WARNING_OBJECT (demux, "No essence track for this track found");
continue;
}
if (track->edit_rate.n <= 0 || track->edit_rate.d <= 0 || if (track->edit_rate.n <= 0 || track->edit_rate.d <= 0 ||
source_track->edit_rate.n <= 0 || source_track->edit_rate.d <= 0) { source_track->edit_rate.n <= 0 || source_track->edit_rate.d <= 0) {
GST_WARNING_OBJECT (demux, "Track has an invalid edit rate"); GST_WARNING_OBJECT (demux, "Track has an invalid edit rate");
@ -803,58 +974,37 @@ gst_mxf_demux_handle_header_metadata_update_streams (GstMXFDemux * demux)
/* If we just added the pad initialize for the current component */ /* If we just added the pad initialize for the current component */
if (first_run && MXF_IS_METADATA_MATERIAL_PACKAGE (current_package)) { if (first_run && MXF_IS_METADATA_MATERIAL_PACKAGE (current_package)) {
pad->current_component = 0; pad->current_component_index = 0;
pad->current_component_position = 0; pad->current_component_position = 0;
pad->current_component_drop = source_track->origin; pad->current_component_start = source_track->origin;
if (track->edit_rate.n != source_track->edit_rate.n || if (track->edit_rate.n != source_track->edit_rate.n ||
track->edit_rate.n != source_track->edit_rate.n) { track->edit_rate.n != source_track->edit_rate.n) {
pad->current_component_drop += pad->current_component_start +=
gst_util_uint64_scale (component->start_position, gst_util_uint64_scale (component->start_position,
source_track->edit_rate.n * track->edit_rate.d, source_track->edit_rate.n * track->edit_rate.d,
source_track->edit_rate.d * track->edit_rate.n); source_track->edit_rate.d * track->edit_rate.n);
} else { } else {
pad->current_component_drop += component->start_position; pad->current_component_start += component->start_position;
} }
} }
/* NULL iff playing a source package */ /* NULL iff playing a source package */
pad->component = component; pad->current_component = component;
pad->source_package = source_package; pad->current_essence_track = etrack;
pad->source_track = source_track;
pad->handler = NULL; if (pad->tags)
g_free (pad->mapping_data); gst_tag_list_free (pad->tags);
pad->handle_func = NULL; pad->tags = NULL;
pad->mapping_data = NULL; if (etrack->tags)
pad->tags = gst_tag_list_copy (etrack->tags);
pad->handler = mxf_essence_element_handler_find (source_track); if (GST_PAD_CAPS (pad)
&& !gst_caps_is_equal (GST_PAD_CAPS (pad), etrack->caps)) {
if (!pad->handler) { gst_pad_set_caps (GST_PAD_CAST (pad), etrack->caps);
GST_WARNING_OBJECT (demux, "No essence element handler for track found"); } else if (!GST_PAD_CAPS (pad)) {
gst_object_unref (pad); gst_pad_set_caps (GST_PAD_CAST (pad), etrack->caps);
continue;
}
caps =
pad->handler->create_caps (source_track, &pad->tags, &pad->handle_func,
&pad->mapping_data);
if (!caps) {
GST_WARNING_OBJECT (demux, "No caps created, ignoring stream");
gst_object_unref (pad);
continue;
}
GST_DEBUG_OBJECT (demux, "Created caps %" GST_PTR_FORMAT, caps);
if (pad->caps && !gst_caps_is_equal (pad->caps, caps)) {
gst_pad_set_caps (GST_PAD_CAST (pad), caps);
gst_caps_replace (&pad->caps, gst_caps_ref (caps));
} else if (!pad->caps) {
gst_pad_set_caps (GST_PAD_CAST (pad), caps);
pad->caps = gst_caps_ref (caps);
gst_pad_set_event_function (GST_PAD_CAST (pad), gst_pad_set_event_function (GST_PAD_CAST (pad),
GST_DEBUG_FUNCPTR (gst_mxf_demux_src_event)); GST_DEBUG_FUNCPTR (gst_mxf_demux_src_event));
@ -874,7 +1024,6 @@ gst_mxf_demux_handle_header_metadata_update_streams (GstMXFDemux * demux)
g_ptr_array_add (demux->src, pad); g_ptr_array_add (demux->src, pad);
pad->discont = TRUE; pad->discont = TRUE;
} }
gst_caps_unref (caps);
} }
gst_element_no_more_pads (GST_ELEMENT_CAST (demux)); gst_element_no_more_pads (GST_ELEMENT_CAST (demux));
@ -961,7 +1110,7 @@ gst_mxf_demux_handle_metadata (GstMXFDemux * demux, const MXFUL * key,
demux->preface = MXF_METADATA_PREFACE (metadata); demux->preface = MXF_METADATA_PREFACE (metadata);
} }
gst_mxf_demux_reset_track_metadata (demux); gst_mxf_demux_reset_linked_metadata (demux);
g_hash_table_replace (demux->metadata, g_hash_table_replace (demux->metadata,
&MXF_METADATA_BASE (metadata)->instance_uid, metadata); &MXF_METADATA_BASE (metadata)->instance_uid, metadata);
@ -1047,7 +1196,7 @@ gst_mxf_demux_handle_descriptive_metadata (GstMXFDemux * demux,
return GST_FLOW_OK; return GST_FLOW_OK;
} }
gst_mxf_demux_reset_track_metadata (demux); gst_mxf_demux_reset_linked_metadata (demux);
g_hash_table_replace (demux->metadata, &MXF_METADATA_BASE (m)->instance_uid, g_hash_table_replace (demux->metadata, &MXF_METADATA_BASE (m)->instance_uid,
m); m);
@ -1072,112 +1221,104 @@ gst_mxf_demux_pad_next_component (GstMXFDemux * demux, GstMXFDemuxPad * pad)
{ {
MXFMetadataSequence *sequence; MXFMetadataSequence *sequence;
guint k; guint k;
GstCaps *caps = NULL; MXFMetadataSourcePackage *source_package = NULL;
MXFMetadataTimelineTrack *source_track = NULL;
pad->current_component++; pad->current_component_index++;
sequence = pad->material_track->parent.sequence; sequence = pad->material_track->parent.sequence;
if (pad->current_component >= sequence->n_structural_components) { if (pad->current_component_index >= sequence->n_structural_components) {
GST_DEBUG_OBJECT (demux, "After last structural component"); GST_DEBUG_OBJECT (demux, "After last structural component");
return GST_FLOW_UNEXPECTED; return GST_FLOW_UNEXPECTED;
} }
GST_DEBUG_OBJECT (demux, "Switching to component %u", pad->current_component); GST_DEBUG_OBJECT (demux, "Switching to component %u",
pad->current_component_index);
pad->component = pad->current_component =
MXF_METADATA_SOURCE_CLIP (sequence->structural_components[pad-> MXF_METADATA_SOURCE_CLIP (sequence->
current_component]); structural_components[pad->current_component_index]);
if (pad->component == NULL) { if (pad->current_component == NULL) {
GST_ERROR_OBJECT (demux, "No such structural component"); GST_ERROR_OBJECT (demux, "No such structural component");
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
} }
if (!pad->component->source_package if (!pad->current_component->source_package
|| !pad->component->source_package->top_level || !pad->current_component->source_package->top_level
|| !MXF_METADATA_GENERIC_PACKAGE (pad->component-> || !MXF_METADATA_GENERIC_PACKAGE (pad->
source_package)->tracks) { current_component->source_package)->tracks) {
GST_ERROR_OBJECT (demux, "Invalid component"); GST_ERROR_OBJECT (demux, "Invalid component");
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
} }
pad->source_package = pad->component->source_package; source_package = pad->current_component->source_package;
pad->source_track = NULL;
for (k = 0; k < pad->source_package->parent.n_tracks; k++) { for (k = 0; k < source_package->parent.n_tracks; k++) {
MXFMetadataTrack *tmp = pad->source_package->parent.tracks[k]; MXFMetadataTrack *tmp = source_package->parent.tracks[k];
if (tmp->track_id == pad->component->source_track_id) { if (tmp->track_id == pad->current_component->source_track_id) {
pad->source_track = MXF_METADATA_TIMELINE_TRACK (tmp); source_track = MXF_METADATA_TIMELINE_TRACK (tmp);
break; break;
} }
} }
if (!pad->source_track) { if (!source_track) {
GST_ERROR_OBJECT (demux, "No source track found"); GST_ERROR_OBJECT (demux, "No source track found");
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
} }
if (!pad->source_package->descriptors) { pad->current_essence_track = NULL;
for (k = 0; k < demux->essence_tracks->len; k++) {
GstMXFDemuxEssenceTrack *tmp =
&g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack, k);
if (tmp->source_package == source_package &&
tmp->source_track == source_track) {
pad->current_essence_track = tmp;
break;
}
}
if (!pad->current_essence_track) {
GST_ERROR_OBJECT (demux, "No corresponding essence track found");
return GST_FLOW_ERROR;
}
if (!source_package->descriptors) {
GST_ERROR_OBJECT (demux, "Source package has no descriptors"); GST_ERROR_OBJECT (demux, "Source package has no descriptors");
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
} }
if (!pad->source_track->parent.descriptor) { if (!source_track->parent.descriptor) {
GST_ERROR_OBJECT (demux, "No descriptor found for track"); GST_ERROR_OBJECT (demux, "No descriptor found for track");
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
} }
if (pad->source_track->edit_rate.n <= 0 || if (source_track->edit_rate.n <= 0 || source_track->edit_rate.d <= 0) {
pad->source_track->edit_rate.d <= 0) {
GST_ERROR_OBJECT (demux, "Source track has invalid edit rate"); GST_ERROR_OBJECT (demux, "Source track has invalid edit rate");
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
} }
pad->current_component_position = 0; pad->current_component_position = 0;
pad->current_component_drop = pad->source_track->origin; pad->current_component_start = source_track->origin;
if (pad->material_track->edit_rate.n != pad->source_track->edit_rate.n || if (pad->material_track->edit_rate.n != source_track->edit_rate.n ||
pad->material_track->edit_rate.n != pad->source_track->edit_rate.n) { pad->material_track->edit_rate.n != source_track->edit_rate.n) {
pad->current_component_drop += pad->current_component_start +=
gst_util_uint64_scale (pad->component->start_position, gst_util_uint64_scale (pad->current_component->start_position,
pad->source_track->edit_rate.n * pad->material_track->edit_rate.d, source_track->edit_rate.n * pad->material_track->edit_rate.d,
pad->source_track->edit_rate.d * pad->material_track->edit_rate.n); source_track->edit_rate.d * pad->material_track->edit_rate.n);
} else { } else {
pad->current_component_drop += pad->component->start_position; pad->current_component_start += pad->current_component->start_position;
} }
pad->handler = NULL; if (!gst_caps_is_equal (GST_PAD_CAPS (pad), pad->current_essence_track->caps)) {
g_free (pad->mapping_data); gst_pad_set_caps (GST_PAD_CAST (pad), pad->current_essence_track->caps);
pad->handle_func = NULL;
pad->mapping_data = NULL;
pad->handler = mxf_essence_element_handler_find (pad->source_track);
if (!pad->handler) {
GST_ERROR_OBJECT (demux, "No essence element handler for track found");
return GST_FLOW_ERROR;
} }
caps =
pad->handler->create_caps (pad->source_track, &pad->tags,
&pad->handle_func, &pad->mapping_data);
if (!caps) {
GST_ERROR_OBJECT (demux, "No caps created");
return GST_FLOW_ERROR;
}
GST_DEBUG_OBJECT (demux, "Created caps %" GST_PTR_FORMAT, caps);
if (!gst_caps_is_equal (pad->caps, caps)) {
gst_pad_set_caps (GST_PAD_CAST (pad), caps);
gst_caps_replace (&pad->caps, gst_caps_ref (caps));
}
gst_caps_unref (caps);
return GST_FLOW_OK; return GST_FLOW_OK;
} }
@ -1187,10 +1328,11 @@ gst_mxf_demux_handle_generic_container_essence_element (GstMXFDemux * demux,
{ {
GstFlowReturn ret = GST_FLOW_OK; GstFlowReturn ret = GST_FLOW_OK;
guint32 track_number; guint32 track_number;
guint i, j; guint i;
GstMXFDemuxPad *pad = NULL;
GstBuffer *inbuf; GstBuffer *inbuf;
GstBuffer *outbuf = NULL; GstBuffer *outbuf = NULL;
GstMXFDemuxPad *pad = NULL;
GstMXFDemuxEssenceTrack *etrack = NULL;
GST_DEBUG_OBJECT (demux, GST_DEBUG_OBJECT (demux,
"Handling generic container essence element of size %u" "Handling generic container essence element of size %u"
@ -1211,6 +1353,11 @@ gst_mxf_demux_handle_generic_container_essence_element (GstMXFDemux * demux,
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
} }
if (!demux->essence_tracks || demux->essence_tracks->len == 0) {
GST_ERROR_OBJECT (demux, "No essence streams found in the metadata");
return GST_FLOW_ERROR;
}
if (GST_BUFFER_SIZE (buffer) == 0) { if (GST_BUFFER_SIZE (buffer) == 0) {
GST_DEBUG_OBJECT (demux, "Zero sized essence element, ignoring"); GST_DEBUG_OBJECT (demux, "Zero sized essence element, ignoring");
return GST_FLOW_OK; return GST_FLOW_OK;
@ -1218,76 +1365,68 @@ gst_mxf_demux_handle_generic_container_essence_element (GstMXFDemux * demux,
track_number = GST_READ_UINT32_BE (&key->u[12]); track_number = GST_READ_UINT32_BE (&key->u[12]);
for (i = 0; i < demux->src->len; i++) { for (i = 0; i < demux->essence_tracks->len; i++) {
GstMXFDemuxPad *p = g_ptr_array_index (demux->src, i); GstMXFDemuxEssenceTrack *tmp =
MXFMetadataContentStorage *content_storage = &g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack, i);
demux->preface->content_storage;
if (p->source_track->parent.track_number == track_number || if (tmp->body_sid == demux->current_partition->partition.body_sid &&
(p->source_track->parent.track_number == 0 && tmp->track_number == track_number) {
demux->src->len == 1 && etrack = tmp;
demux->current_package->n_essence_tracks == 1)) {
if (content_storage->essence_container_data) {
for (j = 0; j < content_storage->n_essence_container_data; j++) {
MXFMetadataEssenceContainerData *edata =
content_storage->essence_container_data[j];
if (edata && p->source_package == edata->linked_package
&& demux->current_partition->partition.body_sid ==
edata->body_sid) {
pad = p;
break; break;
} }
} }
} else {
pad = p; if (!etrack) {
GST_WARNING_OBJECT (demux,
"No essence track for this essence element found");
return GST_FLOW_OK;
} }
if (pad) /* TODO update essence tracks offsets, position, etc... */
for (i = 0; i < demux->src->len; i++) {
GstMXFDemuxPad *tmp = g_ptr_array_index (demux->src, i);
if (tmp->current_essence_track == etrack) {
pad = tmp;
break; break;
} }
} }
if (!pad) { if (!pad) {
GST_WARNING_OBJECT (demux, "No corresponding pad found"); GST_DEBUG_OBJECT (demux, "No pad for essence track found");
return GST_FLOW_OK; return GST_FLOW_OK;
} }
if (pad->component && pad->current_component_drop > 0) { if (pad->eos) {
GST_DEBUG_OBJECT (demux, "Before component start, dropping"); GST_DEBUG_OBJECT (demux, "Pad is already EOS");
pad->current_component_drop--;
return GST_FLOW_OK; return GST_FLOW_OK;
} }
if (pad->need_segment) { if (pad->current_component &&
gst_pad_push_event (GST_PAD_CAST (pad), pad->current_component_position < pad->current_component_start) {
gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_TIME, 0, -1, 0)); GST_DEBUG_OBJECT (demux, "Before current component's start position");
pad->need_segment = FALSE; pad->current_component_position++;
} return GST_FLOW_OK;
if (pad->tags) {
gst_element_found_tags_for_pad (GST_ELEMENT_CAST (demux),
GST_PAD_CAST (pad), pad->tags);
pad->tags = NULL;
} }
/* Create subbuffer to be able to change metadata */ /* Create subbuffer to be able to change metadata */
inbuf = gst_buffer_create_sub (buffer, 0, GST_BUFFER_SIZE (buffer)); inbuf = gst_buffer_create_sub (buffer, 0, GST_BUFFER_SIZE (buffer));
//FIXME broken!
GST_BUFFER_TIMESTAMP (inbuf) = pad->last_stop; GST_BUFFER_TIMESTAMP (inbuf) = pad->last_stop;
GST_BUFFER_DURATION (inbuf) = GST_BUFFER_DURATION (inbuf) =
gst_util_uint64_scale (GST_SECOND, pad->material_track->edit_rate.d, gst_util_uint64_scale (GST_SECOND, pad->material_track->edit_rate.d,
pad->material_track->edit_rate.n); pad->material_track->edit_rate.n);
GST_BUFFER_OFFSET (inbuf) = GST_BUFFER_OFFSET_NONE; GST_BUFFER_OFFSET (inbuf) = GST_BUFFER_OFFSET_NONE;
GST_BUFFER_OFFSET_END (inbuf) = GST_BUFFER_OFFSET_NONE; GST_BUFFER_OFFSET_END (inbuf) = GST_BUFFER_OFFSET_NONE;
gst_buffer_set_caps (inbuf, pad->caps); gst_buffer_set_caps (inbuf, etrack->caps);
if (pad->handle_func) { if (etrack->handle_func) {
/* Takes ownership of inbuf */ /* Takes ownership of inbuf */
ret = ret =
pad->handle_func (key, inbuf, pad->caps, etrack->handle_func (key, inbuf, etrack->caps,
pad->source_track, pad->component, pad->mapping_data, &outbuf); etrack->source_track, pad->current_component, etrack->mapping_data,
&outbuf);
inbuf = NULL; inbuf = NULL;
} else { } else {
outbuf = inbuf; outbuf = inbuf;
@ -1303,11 +1442,23 @@ gst_mxf_demux_handle_generic_container_essence_element (GstMXFDemux * demux,
} }
} }
if (pad->need_segment) {
gst_pad_push_event (GST_PAD_CAST (pad),
gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_TIME, 0, -1, 0));
pad->need_segment = FALSE;
}
if (pad->tags) {
gst_element_found_tags_for_pad (GST_ELEMENT_CAST (demux),
GST_PAD_CAST (pad), pad->tags);
pad->tags = NULL;
}
if (outbuf) if (outbuf)
pad->last_stop += GST_BUFFER_DURATION (outbuf); pad->last_stop += GST_BUFFER_DURATION (outbuf);
if (outbuf) { if (outbuf) {
/* TODO: handle timestamp gaps */
GST_DEBUG_OBJECT (demux, GST_DEBUG_OBJECT (demux,
"Pushing buffer of size %u for track %u: timestamp %" GST_TIME_FORMAT "Pushing buffer of size %u for track %u: timestamp %" GST_TIME_FORMAT
" duration %" GST_TIME_FORMAT, GST_BUFFER_SIZE (outbuf), " duration %" GST_TIME_FORMAT, GST_BUFFER_SIZE (outbuf),
@ -1330,10 +1481,11 @@ gst_mxf_demux_handle_generic_container_essence_element (GstMXFDemux * demux,
if (ret != GST_FLOW_OK) if (ret != GST_FLOW_OK)
return ret; return ret;
if (pad->component) { if (pad->current_component) {
pad->current_component_position++; pad->current_component_position++;
if (pad->component->parent.duration != -1 && if (pad->current_component->parent.duration != -1 &&
pad->current_component_position >= pad->component->parent.duration) { pad->current_component_position - pad->current_component_start
>= pad->current_component->parent.duration) {
GST_DEBUG_OBJECT (demux, "Switching to next component"); GST_DEBUG_OBJECT (demux, "Switching to next component");
if ((ret = gst_mxf_demux_pad_next_component (demux, pad)) != GST_FLOW_OK) { if ((ret = gst_mxf_demux_pad_next_component (demux, pad)) != GST_FLOW_OK) {
@ -1422,9 +1574,7 @@ static GstFlowReturn
gst_mxf_demux_handle_index_table_segment (GstMXFDemux * demux, gst_mxf_demux_handle_index_table_segment (GstMXFDemux * demux,
const MXFUL * key, GstBuffer * buffer) const MXFUL * key, GstBuffer * buffer)
{ {
MXFIndexTableSegment segment; MXFIndexTableSegment *segment;
memset (&segment, 0, sizeof (segment));
GST_DEBUG_OBJECT (demux, GST_DEBUG_OBJECT (demux,
"Handling index table segment of size %u at offset %" "Handling index table segment of size %u at offset %"
@ -1435,7 +1585,9 @@ gst_mxf_demux_handle_index_table_segment (GstMXFDemux * demux,
return GST_FLOW_OK; return GST_FLOW_OK;
} }
if (!mxf_index_table_segment_parse (key, &segment, segment = g_new0 (MXFIndexTableSegment, 1);
if (!mxf_index_table_segment_parse (key, segment,
&demux->current_partition->primer, GST_BUFFER_DATA (buffer), &demux->current_partition->primer, GST_BUFFER_DATA (buffer),
GST_BUFFER_SIZE (buffer))) { GST_BUFFER_SIZE (buffer))) {
@ -1443,11 +1595,9 @@ gst_mxf_demux_handle_index_table_segment (GstMXFDemux * demux,
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
} }
if (!demux->index_table) demux->pending_index_table_segments =
demux->index_table = g_list_prepend (demux->pending_index_table_segments, segment);
g_array_new (FALSE, FALSE, sizeof (MXFIndexTableSegment));
g_array_append_val (demux->index_table, segment);
return GST_FLOW_OK; return GST_FLOW_OK;
} }
@ -1744,10 +1894,8 @@ next_try:
/* resolve references etc */ /* resolve references etc */
if (gst_mxf_demux_handle_header_metadata_resolve_references (demux) != if (gst_mxf_demux_resolve_references (demux) !=
GST_FLOW_OK GST_FLOW_OK || gst_mxf_demux_update_tracks (demux) != GST_FLOW_OK) {
|| gst_mxf_demux_handle_header_metadata_update_streams (demux) !=
GST_FLOW_OK) {
demux->current_partition->parsed_metadata = TRUE; demux->current_partition->parsed_metadata = TRUE;
demux->offset = demux->offset =
demux->run_in + demux->current_partition->partition.this_partition - demux->run_in + demux->current_partition->partition.this_partition -
@ -1778,18 +1926,12 @@ gst_mxf_demux_handle_klv_packet (GstMXFDemux * demux, const MXFUL * key,
demux->run_in + demux->current_partition->primer.offset + demux->run_in + demux->current_partition->primer.offset +
demux->current_partition->partition.header_byte_count) { demux->current_partition->partition.header_byte_count) {
demux->current_partition->parsed_metadata = TRUE; demux->current_partition->parsed_metadata = TRUE;
if ((ret = if ((ret = gst_mxf_demux_resolve_references (demux)) != GST_FLOW_OK)
gst_mxf_demux_handle_header_metadata_resolve_references (demux)) !=
GST_FLOW_OK)
goto beach; goto beach;
if ((ret = if ((ret = gst_mxf_demux_update_tracks (demux)) != GST_FLOW_OK)
gst_mxf_demux_handle_header_metadata_update_streams (demux)) !=
GST_FLOW_OK)
goto beach; goto beach;
} else if (demux->metadata_resolved && demux->requested_package_string) { } else if (demux->metadata_resolved && demux->requested_package_string) {
if ((ret = if ((ret = gst_mxf_demux_update_tracks (demux)) != GST_FLOW_OK)
gst_mxf_demux_handle_header_metadata_update_streams (demux)) !=
GST_FLOW_OK)
goto beach; goto beach;
} }

View file

@ -28,6 +28,7 @@
#include "mxfmetadata.h" #include "mxfmetadata.h"
G_BEGIN_DECLS G_BEGIN_DECLS
#define GST_TYPE_MXF_DEMUX \ #define GST_TYPE_MXF_DEMUX \
(gst_mxf_demux_get_type()) (gst_mxf_demux_get_type())
#define GST_MXF_DEMUX(obj) \ #define GST_MXF_DEMUX(obj) \
@ -41,6 +42,13 @@ G_BEGIN_DECLS
typedef struct _GstMXFDemux GstMXFDemux; typedef struct _GstMXFDemux GstMXFDemux;
typedef struct _GstMXFDemuxClass GstMXFDemuxClass; typedef struct _GstMXFDemuxClass GstMXFDemuxClass;
#define GST_TYPE_MXF_DEMUX_PAD (gst_mxf_demux_pad_get_type())
#define GST_MXF_DEMUX_PAD(pad) (G_TYPE_CHECK_INSTANCE_CAST((pad),GST_TYPE_MXF_DEMUX_PAD,GstMXFDemuxPad))
#define GST_MXF_DEMUX_PAD_CAST(pad) ((GstMXFDemuxPad *) pad)
#define GST_IS_MXF_DEMUX_PAD(pad) (G_TYPE_CHECK_INSTANCE_TYPE((pad),GST_TYPE_MXF_DEMUX_PAD))
typedef struct _GstMXFDemuxPad GstMXFDemuxPad;
typedef struct _GstMXFDemuxPadClass GstMXFDemuxPadClass;
typedef struct typedef struct
{ {
MXFPartitionPack partition; MXFPartitionPack partition;
@ -48,6 +56,67 @@ typedef struct
gboolean parsed_metadata; gboolean parsed_metadata;
} GstMXFDemuxPartition; } GstMXFDemuxPartition;
typedef struct
{
guint64 offset;
gboolean keyframe;
} GstMXFDemuxIndex;
typedef struct
{
guint32 body_sid;
guint32 track_number;
guint64 position;
guint64 duration;
GstMXFDemuxIndex *offsets;
guint64 last_offset;
guint64 last_indexed_offset;
MXFMetadataSourcePackage *source_package;
MXFMetadataTimelineTrack *source_track;
gpointer mapping_data;
const MXFEssenceElementHandler *handler;
MXFEssenceElementHandleFunc handle_func;
GstTagList *tags;
GstCaps *caps;
} GstMXFDemuxEssenceTrack;
struct _GstMXFDemuxPad
{
GstPad parent;
guint32 track_id;
gboolean need_segment;
GstClockTime last_stop;
GstFlowReturn last_flow;
gboolean eos, discont;
GstTagList *tags;
MXFMetadataGenericPackage *material_package;
MXFMetadataTimelineTrack *material_track;
guint current_component_index;
MXFMetadataSourceClip *current_component;
gint64 current_component_start;
gint64 current_component_position;
GstMXFDemuxEssenceTrack *current_essence_track;
};
struct _GstMXFDemuxPadClass
{
GstPadClass parent;
};
struct _GstMXFDemux struct _GstMXFDemux
{ {
GstElement element; GstElement element;
@ -78,8 +147,10 @@ struct _GstMXFDemux
GList *partitions; GList *partitions;
GstMXFDemuxPartition *current_partition; GstMXFDemuxPartition *current_partition;
GArray *essence_tracks;
GList *pending_index_table_segments;
GArray *random_index_pack; GArray *random_index_pack;
GArray *index_table;
/* Structural metadata */ /* Structural metadata */
gboolean update_metadata; gboolean update_metadata;