mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-24 16:18:16 +00:00
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:
parent
914b108db9
commit
4162c36f17
2 changed files with 459 additions and 246 deletions
|
@ -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->essence_tracks->len; i++) {
|
||||||
|
GstMXFDemuxEssenceTrack *tmp =
|
||||||
|
&g_array_index (demux->essence_tracks, GstMXFDemuxEssenceTrack, i);
|
||||||
|
|
||||||
|
if (tmp->body_sid == demux->current_partition->partition.body_sid &&
|
||||||
|
tmp->track_number == track_number) {
|
||||||
|
etrack = tmp;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!etrack) {
|
||||||
|
GST_WARNING_OBJECT (demux,
|
||||||
|
"No essence track for this essence element found");
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* TODO update essence tracks offsets, position, etc... */
|
||||||
|
|
||||||
for (i = 0; i < demux->src->len; i++) {
|
for (i = 0; i < demux->src->len; i++) {
|
||||||
GstMXFDemuxPad *p = g_ptr_array_index (demux->src, i);
|
GstMXFDemuxPad *tmp = g_ptr_array_index (demux->src, i);
|
||||||
MXFMetadataContentStorage *content_storage =
|
|
||||||
demux->preface->content_storage;
|
|
||||||
|
|
||||||
if (p->source_track->parent.track_number == track_number ||
|
if (tmp->current_essence_track == etrack) {
|
||||||
(p->source_track->parent.track_number == 0 &&
|
pad = tmp;
|
||||||
demux->src->len == 1 &&
|
break;
|
||||||
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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
pad = p;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (pad)
|
|
||||||
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Loading…
Reference in a new issue